191 lines
5.2 KiB
TypeScript
191 lines
5.2 KiB
TypeScript
import * as plugins from './docker.plugins';
|
|
import * as interfaces from './interfaces';
|
|
|
|
import { DockerHost } from './docker.classes.host';
|
|
import { DockerImage } from './docker.classes.image';
|
|
import { DockerSecret } from './docker.classes.secret';
|
|
|
|
export class DockerService {
|
|
// STATIC
|
|
public static async getServices(dockerHost: DockerHost) {
|
|
const services: DockerService[] = [];
|
|
const response = await dockerHost.request('GET', '/services');
|
|
for (const serviceObject of response.body) {
|
|
const dockerService = new DockerService(dockerHost);
|
|
Object.assign(dockerService, serviceObject);
|
|
services.push(dockerService);
|
|
}
|
|
return services;
|
|
}
|
|
|
|
public static async getServiceByName(
|
|
dockerHost: DockerHost,
|
|
networkName: string
|
|
): Promise<DockerService> {
|
|
const allServices = await DockerService.getServices(dockerHost);
|
|
const wantedService = allServices.find(service => {
|
|
return service.Spec.Name === networkName;
|
|
});
|
|
return wantedService;
|
|
}
|
|
|
|
/**
|
|
* creates a service
|
|
*/
|
|
public static async createService(
|
|
dockerHost: DockerHost,
|
|
serviceCreationDescriptor: interfaces.IServiceCreationDescriptor
|
|
): Promise<DockerService> {
|
|
// lets get the image
|
|
plugins.smartlog.defaultLogger.log(
|
|
'info',
|
|
`now creating service ${serviceCreationDescriptor.name}`
|
|
);
|
|
|
|
// await serviceCreationDescriptor.image.pullLatestImageFromRegistry();
|
|
const serviceVersion = await serviceCreationDescriptor.image.getVersion();
|
|
|
|
const labels: interfaces.TLabels = {
|
|
...serviceCreationDescriptor.labels,
|
|
version: serviceVersion
|
|
};
|
|
|
|
const networkArray: any[] = [];
|
|
for (const network of serviceCreationDescriptor.networks) {
|
|
networkArray.push({
|
|
Target: network.Name,
|
|
Aliases: [serviceCreationDescriptor.networkAlias]
|
|
});
|
|
}
|
|
|
|
const secretArray: any[] = [];
|
|
for (const secret of serviceCreationDescriptor.secrets) {
|
|
secretArray.push({
|
|
File: {
|
|
Name: 'secret.json',
|
|
UID: '33',
|
|
GID: '33',
|
|
Mode: 384
|
|
},
|
|
SecretID: secret.ID,
|
|
SecretName: secret.Spec.Name
|
|
});
|
|
}
|
|
|
|
const response = await dockerHost.request('POST', '/services/create', {
|
|
Name: serviceCreationDescriptor.name,
|
|
TaskTemplate: {
|
|
ContainerSpec: {
|
|
Image: serviceCreationDescriptor.image.RepoTags[0],
|
|
Labels: labels,
|
|
Secrets: secretArray
|
|
},
|
|
UpdateConfig: {
|
|
Parallelism: 0,
|
|
Delay: 0,
|
|
FailureAction: 'pause',
|
|
Monitor: 15000000000,
|
|
MaxFailureRatio: 0.15
|
|
},
|
|
ForceUpdate: 1
|
|
},
|
|
Labels: serviceCreationDescriptor.labels,
|
|
Networks: networkArray
|
|
});
|
|
|
|
const createdService = await DockerService.getServiceByName(
|
|
dockerHost,
|
|
serviceCreationDescriptor.name
|
|
);
|
|
return createdService;
|
|
}
|
|
|
|
// INSTANCE
|
|
public dockerHostRef: DockerHost;
|
|
|
|
public ID: string;
|
|
public Version: { Index: number };
|
|
public CreatedAt: string;
|
|
public UpdatedAt: string;
|
|
public Spec: {
|
|
Name: string;
|
|
Labels: interfaces.TLabels;
|
|
TaskTemplate: {
|
|
ContainerSpec: {
|
|
Image: string;
|
|
Isolation: string;
|
|
Secrets: Array<{
|
|
File: {
|
|
Name: string;
|
|
UID: string;
|
|
GID: string;
|
|
Mode: number;
|
|
};
|
|
SecretID: string;
|
|
SecretName: string;
|
|
}>;
|
|
};
|
|
ForceUpdate: 0;
|
|
};
|
|
Mode: {};
|
|
Networks: [any[]];
|
|
};
|
|
public Endpoint: { Spec: {}; VirtualIPs: [any[]] };
|
|
|
|
constructor(dockerHostArg: DockerHost) {
|
|
this.dockerHostRef = dockerHostArg;
|
|
}
|
|
|
|
public async update() {
|
|
const labels: interfaces.TLabels = {
|
|
...this.Spec.Labels,
|
|
version: 'x.x.x'
|
|
};
|
|
|
|
const dockerData = await this.dockerHostRef.request(
|
|
'POST',
|
|
`/services/${this.ID}/update?version=${this.Version.Index}`,
|
|
{
|
|
Name: this.Spec.Name,
|
|
TaskTemplate: this.Spec.TaskTemplate,
|
|
Labels: labels,
|
|
Networks: this.Spec.Networks
|
|
}
|
|
);
|
|
Object.assign(this, dockerData);
|
|
}
|
|
|
|
public async remove() {
|
|
await this.dockerHostRef.request('DELETE', `/services/${this.ID}`);
|
|
}
|
|
|
|
public async reReadFromDockerEngine() {
|
|
const dockerData = await this.dockerHostRef.request('GET', `/services/${this.ID}`);
|
|
Object.assign(this, dockerData);
|
|
}
|
|
|
|
public async needsUpdate(): Promise<boolean> {
|
|
// TODO: implement digest based update recognition
|
|
|
|
await this.reReadFromDockerEngine();
|
|
const dockerImage = await DockerImage.createFromRegistry(this.dockerHostRef, {
|
|
imageUrl: this.Spec.TaskTemplate.ContainerSpec.Image
|
|
});
|
|
|
|
const imageVersion = new plugins.smartversion.SmartVersion(dockerImage.Labels.version);
|
|
const serviceVersion = new plugins.smartversion.SmartVersion(this.Spec.Labels.version);
|
|
if (imageVersion.greaterThan(serviceVersion)) {
|
|
console.log(`service ${this.Spec.Name} needs to be updated`);
|
|
return true;
|
|
} else {
|
|
console.log(`service ${this.Spec.Name} is up to date.`);
|
|
}
|
|
}
|
|
|
|
public async updateFromRegistry() {
|
|
if (await this.needsUpdate()) {
|
|
this.update();
|
|
}
|
|
}
|
|
}
|