2024-06-05 14:10:44 +02:00
|
|
|
|
import * as plugins from './plugins.js';
|
2022-10-17 09:36:35 +02:00
|
|
|
|
import * as interfaces from './interfaces/index.js';
|
2024-06-05 14:10:44 +02:00
|
|
|
|
import { DockerHost } from './classes.host.js';
|
2024-06-08 15:03:19 +02:00
|
|
|
|
import { logger } from './logger.js';
|
2018-07-16 23:52:50 +02:00
|
|
|
|
|
2024-06-08 15:03:19 +02:00
|
|
|
|
/**
|
|
|
|
|
* represents a docker image on the remote docker host
|
|
|
|
|
*/
|
2018-07-16 23:52:50 +02:00
|
|
|
|
export class DockerImage {
|
2019-08-14 14:19:45 +02:00
|
|
|
|
// STATIC
|
2019-08-15 18:50:13 +02:00
|
|
|
|
public static async getImages(dockerHost: DockerHost) {
|
|
|
|
|
const images: DockerImage[] = [];
|
|
|
|
|
const response = await dockerHost.request('GET', '/images/json');
|
|
|
|
|
for (const imageObject of response.body) {
|
|
|
|
|
images.push(new DockerImage(dockerHost, imageObject));
|
|
|
|
|
}
|
|
|
|
|
return images;
|
|
|
|
|
}
|
|
|
|
|
|
2024-06-08 15:03:19 +02:00
|
|
|
|
public static async getImageByName(dockerHost: DockerHost, imageNameArg: string) {
|
2019-08-15 19:00:17 +02:00
|
|
|
|
const images = await this.getImages(dockerHost);
|
2020-09-30 16:35:24 +00:00
|
|
|
|
const result = images.find((image) => {
|
2019-09-13 14:40:38 +02:00
|
|
|
|
if (image.RepoTags) {
|
|
|
|
|
return image.RepoTags.includes(imageNameArg);
|
|
|
|
|
} else {
|
|
|
|
|
return false;
|
|
|
|
|
}
|
2019-08-15 19:00:17 +02:00
|
|
|
|
});
|
2019-09-13 14:40:38 +02:00
|
|
|
|
return result;
|
2019-08-15 19:00:17 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-08-14 20:56:57 +02:00
|
|
|
|
public static async createFromRegistry(
|
|
|
|
|
dockerHostArg: DockerHost,
|
2024-06-05 14:10:44 +02:00
|
|
|
|
optionsArg: {
|
|
|
|
|
creationObject: interfaces.IImageCreationDescriptor
|
|
|
|
|
}
|
2019-08-14 20:56:57 +02:00
|
|
|
|
): Promise<DockerImage> {
|
2019-08-16 14:46:48 +02:00
|
|
|
|
// lets create a sanatized imageUrlObject
|
|
|
|
|
const imageUrlObject: {
|
|
|
|
|
imageUrl: string;
|
|
|
|
|
imageTag: string;
|
|
|
|
|
imageOriginTag: string;
|
|
|
|
|
} = {
|
2024-06-05 14:10:44 +02:00
|
|
|
|
imageUrl: optionsArg.creationObject.imageUrl,
|
|
|
|
|
imageTag: optionsArg.creationObject.imageTag,
|
2020-09-30 16:35:24 +00:00
|
|
|
|
imageOriginTag: null,
|
2019-08-16 14:46:48 +02:00
|
|
|
|
};
|
|
|
|
|
if (imageUrlObject.imageUrl.includes(':')) {
|
|
|
|
|
const imageUrl = imageUrlObject.imageUrl.split(':')[0];
|
|
|
|
|
const imageTag = imageUrlObject.imageUrl.split(':')[1];
|
|
|
|
|
if (imageUrlObject.imageTag) {
|
|
|
|
|
throw new Error(
|
2019-09-13 18:20:12 +02:00
|
|
|
|
`imageUrl ${imageUrlObject.imageUrl} can't be tagged with ${imageUrlObject.imageTag} because it is already tagged with ${imageTag}`
|
2019-08-16 14:46:48 +02:00
|
|
|
|
);
|
|
|
|
|
} else {
|
2019-09-12 14:45:36 +02:00
|
|
|
|
imageUrlObject.imageUrl = imageUrl;
|
2019-08-16 14:46:48 +02:00
|
|
|
|
imageUrlObject.imageTag = imageTag;
|
|
|
|
|
}
|
2019-08-16 18:21:55 +02:00
|
|
|
|
} else if (!imageUrlObject.imageTag) {
|
|
|
|
|
imageUrlObject.imageTag = 'latest';
|
2019-08-16 14:46:48 +02:00
|
|
|
|
}
|
|
|
|
|
imageUrlObject.imageOriginTag = `${imageUrlObject.imageUrl}:${imageUrlObject.imageTag}`;
|
|
|
|
|
|
|
|
|
|
// lets actually create the image
|
2019-08-15 18:50:13 +02:00
|
|
|
|
const response = await dockerHostArg.request(
|
|
|
|
|
'POST',
|
|
|
|
|
`/images/create?fromImage=${encodeURIComponent(
|
2019-08-16 14:46:48 +02:00
|
|
|
|
imageUrlObject.imageUrl
|
|
|
|
|
)}&tag=${encodeURIComponent(imageUrlObject.imageTag)}`
|
2019-08-15 18:50:13 +02:00
|
|
|
|
);
|
|
|
|
|
if (response.statusCode < 300) {
|
2020-09-30 16:35:24 +00:00
|
|
|
|
logger.log('info', `Successfully pulled image ${imageUrlObject.imageUrl} from the registry`);
|
2024-06-08 15:03:19 +02:00
|
|
|
|
const image = await DockerImage.getImageByName(dockerHostArg, imageUrlObject.imageOriginTag);
|
2019-08-15 18:50:13 +02:00
|
|
|
|
return image;
|
|
|
|
|
} else {
|
2020-09-30 16:27:43 +00:00
|
|
|
|
logger.log('error', `Failed at the attempt of creating a new image`);
|
2019-08-15 18:50:13 +02:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-08-14 14:19:45 +02:00
|
|
|
|
|
2024-06-05 14:10:44 +02:00
|
|
|
|
/**
|
|
|
|
|
*
|
|
|
|
|
* @param dockerHostArg
|
|
|
|
|
* @param tarStreamArg
|
|
|
|
|
*/
|
2024-12-23 00:30:00 +01:00
|
|
|
|
public static async createFromTarStream(
|
|
|
|
|
dockerHostArg: DockerHost,
|
|
|
|
|
optionsArg: {
|
|
|
|
|
creationObject: interfaces.IImageCreationDescriptor;
|
|
|
|
|
tarStream: plugins.smartstream.stream.Readable;
|
|
|
|
|
}
|
|
|
|
|
): Promise<DockerImage> {
|
|
|
|
|
// Start the request for importing an image
|
|
|
|
|
const response = await dockerHostArg.requestStreaming(
|
|
|
|
|
'POST',
|
|
|
|
|
'/images/load',
|
|
|
|
|
optionsArg.tarStream
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* Docker typically returns lines like:
|
|
|
|
|
* {"stream":"Loaded image: myrepo/myimage:latest"}
|
|
|
|
|
*
|
|
|
|
|
* So we will collect those lines and parse out the final image name.
|
|
|
|
|
*/
|
|
|
|
|
let rawOutput = '';
|
|
|
|
|
response.on('data', (chunk) => {
|
|
|
|
|
rawOutput += chunk.toString();
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Wrap the end event in a Promise for easier async/await usage
|
|
|
|
|
await new Promise<void>((resolve, reject) => {
|
|
|
|
|
response.on('end', () => {
|
|
|
|
|
resolve();
|
|
|
|
|
});
|
|
|
|
|
response.on('error', (err) => {
|
|
|
|
|
reject(err);
|
|
|
|
|
});
|
|
|
|
|
});
|
|
|
|
|
|
|
|
|
|
// Attempt to parse each line to find something like "Loaded image: ..."
|
|
|
|
|
let loadedImageTag: string | undefined;
|
|
|
|
|
const lines = rawOutput.trim().split('\n').filter(Boolean);
|
|
|
|
|
|
|
|
|
|
for (const line of lines) {
|
|
|
|
|
try {
|
|
|
|
|
const jsonLine = JSON.parse(line);
|
|
|
|
|
if (
|
|
|
|
|
jsonLine.stream &&
|
|
|
|
|
(jsonLine.stream.startsWith('Loaded image:') ||
|
|
|
|
|
jsonLine.stream.startsWith('Loaded image ID:'))
|
|
|
|
|
) {
|
|
|
|
|
// Examples:
|
|
|
|
|
// "Loaded image: your-image:latest"
|
|
|
|
|
// "Loaded image ID: sha256:...."
|
|
|
|
|
loadedImageTag = jsonLine.stream
|
|
|
|
|
.replace('Loaded image: ', '')
|
|
|
|
|
.replace('Loaded image ID: ', '')
|
|
|
|
|
.trim();
|
|
|
|
|
}
|
|
|
|
|
} catch {
|
|
|
|
|
// not valid JSON, ignore
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
if (!loadedImageTag) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Could not parse the loaded image info from Docker response.\nResponse was:\n${rawOutput}`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
// Now try to look up that image by the "loadedImageTag".
|
|
|
|
|
// Depending on Docker’s response, it might be something like:
|
|
|
|
|
// "myrepo/myimage:latest" OR "sha256:someHash..."
|
|
|
|
|
// If Docker gave you an ID (e.g. "sha256:..."), you may need a separate
|
|
|
|
|
// DockerImage.getImageById method; or if you prefer, you can treat it as a name.
|
|
|
|
|
const newlyImportedImage = await DockerImage.getImageByName(dockerHostArg, loadedImageTag);
|
|
|
|
|
|
|
|
|
|
if (!newlyImportedImage) {
|
|
|
|
|
throw new Error(
|
|
|
|
|
`Image load succeeded, but no local reference found for "${loadedImageTag}".`
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
logger.log(
|
|
|
|
|
'info',
|
|
|
|
|
`Successfully imported image "${loadedImageTag}".`
|
|
|
|
|
);
|
|
|
|
|
|
|
|
|
|
return newlyImportedImage;
|
2024-06-05 14:10:44 +02:00
|
|
|
|
}
|
|
|
|
|
|
2024-12-23 00:30:00 +01:00
|
|
|
|
|
2019-08-15 18:50:13 +02:00
|
|
|
|
public static async tagImageByIdOrName(
|
|
|
|
|
dockerHost: DockerHost,
|
|
|
|
|
idOrNameArg: string,
|
|
|
|
|
newTagArg: string
|
|
|
|
|
) {
|
|
|
|
|
const response = await dockerHost.request(
|
|
|
|
|
'POST',
|
|
|
|
|
`/images/${encodeURIComponent(idOrNameArg)}/${encodeURIComponent(newTagArg)}`
|
|
|
|
|
);
|
2024-12-23 00:30:00 +01:00
|
|
|
|
|
|
|
|
|
|
2019-08-14 14:19:45 +02:00
|
|
|
|
}
|
|
|
|
|
|
2019-08-15 18:50:13 +02:00
|
|
|
|
public static async buildImage(dockerHostArg: DockerHost, dockerImageTag) {
|
|
|
|
|
// TODO: implement building an image
|
|
|
|
|
}
|
2019-08-14 14:19:45 +02:00
|
|
|
|
|
|
|
|
|
// INSTANCE
|
2019-08-15 18:50:13 +02:00
|
|
|
|
// references
|
|
|
|
|
public dockerHost: DockerHost;
|
|
|
|
|
|
|
|
|
|
// properties
|
2018-07-16 23:52:50 +02:00
|
|
|
|
/**
|
|
|
|
|
* the tags for an image
|
|
|
|
|
*/
|
2019-08-15 18:50:13 +02:00
|
|
|
|
public Containers: number;
|
|
|
|
|
public Created: number;
|
|
|
|
|
public Id: string;
|
|
|
|
|
public Labels: interfaces.TLabels;
|
|
|
|
|
public ParentId: string;
|
|
|
|
|
public RepoDigests: string[];
|
|
|
|
|
public RepoTags: string[];
|
|
|
|
|
public SharedSize: number;
|
|
|
|
|
public Size: number;
|
|
|
|
|
public VirtualSize: number;
|
|
|
|
|
|
|
|
|
|
constructor(dockerHostArg, dockerImageObjectArg: any) {
|
|
|
|
|
this.dockerHost = dockerHostArg;
|
2020-09-30 16:35:24 +00:00
|
|
|
|
Object.keys(dockerImageObjectArg).forEach((keyArg) => {
|
2019-08-15 18:50:13 +02:00
|
|
|
|
this[keyArg] = dockerImageObjectArg[keyArg];
|
|
|
|
|
});
|
|
|
|
|
}
|
2018-07-16 23:52:50 +02:00
|
|
|
|
|
2019-09-13 16:57:21 +02:00
|
|
|
|
/**
|
|
|
|
|
* tag an image
|
|
|
|
|
* @param newTag
|
|
|
|
|
*/
|
|
|
|
|
public async tagImage(newTag) {
|
|
|
|
|
throw new Error('.tagImage is not yet implemented');
|
|
|
|
|
}
|
2019-08-15 19:00:17 +02:00
|
|
|
|
|
2019-08-14 14:19:45 +02:00
|
|
|
|
/**
|
2019-08-15 18:50:13 +02:00
|
|
|
|
* pulls the latest version from the registry
|
2019-08-14 14:19:45 +02:00
|
|
|
|
*/
|
2019-08-14 20:56:57 +02:00
|
|
|
|
public async pullLatestImageFromRegistry(): Promise<boolean> {
|
2019-08-15 18:50:13 +02:00
|
|
|
|
const updatedImage = await DockerImage.createFromRegistry(this.dockerHost, {
|
2024-06-05 14:10:44 +02:00
|
|
|
|
creationObject: {
|
|
|
|
|
imageUrl: this.RepoTags[0],
|
|
|
|
|
},
|
2019-08-15 18:50:13 +02:00
|
|
|
|
});
|
|
|
|
|
Object.assign(this, updatedImage);
|
|
|
|
|
// TODO: Compare image digists before and after
|
2019-08-14 14:19:45 +02:00
|
|
|
|
return true;
|
2018-07-16 23:52:50 +02:00
|
|
|
|
}
|
2019-09-13 14:45:35 +02:00
|
|
|
|
|
|
|
|
|
// get stuff
|
|
|
|
|
public async getVersion() {
|
2019-09-20 16:29:43 +02:00
|
|
|
|
if (this.Labels && this.Labels.version) {
|
|
|
|
|
return this.Labels.version;
|
|
|
|
|
} else {
|
2019-09-24 20:20:37 +02:00
|
|
|
|
return '0.0.0';
|
2019-09-20 16:29:43 +02:00
|
|
|
|
}
|
2019-09-13 14:45:35 +02:00
|
|
|
|
}
|
2024-06-05 14:10:44 +02:00
|
|
|
|
|
|
|
|
|
/**
|
|
|
|
|
* exports an image to a tar ball
|
|
|
|
|
*/
|
|
|
|
|
public async exportToTarStream(): Promise<plugins.smartstream.stream.Readable> {
|
2024-06-10 00:15:01 +02:00
|
|
|
|
logger.log('info', `Exporting image ${this.RepoTags[0]} to tar stream.`);
|
2024-06-05 14:10:44 +02:00
|
|
|
|
const response = await this.dockerHost.requestStreaming('GET', `/images/${encodeURIComponent(this.RepoTags[0])}/get`);
|
|
|
|
|
let counter = 0;
|
|
|
|
|
const webduplexStream = new plugins.smartstream.SmartDuplex({
|
|
|
|
|
writeFunction: async (chunk, tools) => {
|
|
|
|
|
if (counter % 1000 === 0)
|
|
|
|
|
console.log(`Got chunk: ${counter}`);
|
|
|
|
|
counter++;
|
|
|
|
|
return chunk;
|
|
|
|
|
}
|
|
|
|
|
});
|
|
|
|
|
response.on('data', (chunk) => {
|
|
|
|
|
if (!webduplexStream.write(chunk)) {
|
|
|
|
|
response.pause();
|
|
|
|
|
webduplexStream.once('drain', () => {
|
|
|
|
|
response.resume();
|
|
|
|
|
})
|
|
|
|
|
};
|
|
|
|
|
});
|
|
|
|
|
response.on('end', () => {
|
|
|
|
|
webduplexStream.end();
|
|
|
|
|
})
|
|
|
|
|
return webduplexStream;
|
|
|
|
|
}
|
2018-07-17 08:39:37 +02:00
|
|
|
|
}
|