2024-06-05 14:10:44 +02:00
|
|
|
import * as plugins from './plugins.js';
|
2024-06-08 15:03:19 +02:00
|
|
|
import * as paths from './paths.js';
|
2024-06-05 14:10:44 +02:00
|
|
|
import { DockerContainer } from './classes.container.js';
|
|
|
|
import { DockerNetwork } from './classes.network.js';
|
|
|
|
import { DockerService } from './classes.service.js';
|
2024-06-08 15:03:19 +02:00
|
|
|
import { logger } from './logger.js';
|
2024-02-02 16:54:07 +01:00
|
|
|
import path from 'path';
|
2024-06-08 15:03:19 +02:00
|
|
|
import { DockerImageStore } from './classes.imagestore.js';
|
|
|
|
import { DockerImage } from './classes.image.js';
|
2018-07-16 23:52:50 +02:00
|
|
|
|
2019-09-13 22:09:35 +02:00
|
|
|
export interface IAuthData {
|
|
|
|
serveraddress: string;
|
|
|
|
username: string;
|
|
|
|
password: string;
|
|
|
|
}
|
|
|
|
|
2024-06-05 23:56:02 +02:00
|
|
|
export interface IDockerHostConstructorOptions {
|
|
|
|
dockerSockPath?: string;
|
|
|
|
imageStoreDir?: string;
|
|
|
|
}
|
|
|
|
|
2018-07-16 23:52:50 +02:00
|
|
|
export class DockerHost {
|
2024-06-08 15:03:19 +02:00
|
|
|
public options: IDockerHostConstructorOptions;
|
|
|
|
|
2018-07-16 23:52:50 +02:00
|
|
|
/**
|
|
|
|
* the path where the docker sock can be found
|
|
|
|
*/
|
2019-08-14 23:21:54 +02:00
|
|
|
public socketPath: string;
|
2019-09-13 17:54:17 +02:00
|
|
|
private registryToken: string = '';
|
2024-06-05 23:56:02 +02:00
|
|
|
public imageStore: DockerImageStore;
|
2024-06-10 00:15:01 +02:00
|
|
|
public smartBucket: plugins.smartbucket.SmartBucket;
|
2018-07-16 23:52:50 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* the constructor to instantiate a new docker sock instance
|
|
|
|
* @param pathArg
|
|
|
|
*/
|
2024-06-05 23:56:02 +02:00
|
|
|
constructor(optionsArg: IDockerHostConstructorOptions) {
|
2024-06-08 15:03:19 +02:00
|
|
|
this.options = {
|
|
|
|
...{
|
|
|
|
imageStoreDir: plugins.path.join(paths.nogitDir, 'temp-docker-image-store'),
|
|
|
|
},
|
|
|
|
...optionsArg,
|
|
|
|
}
|
2019-08-16 21:21:30 +02:00
|
|
|
let pathToUse: string;
|
2024-06-05 23:56:02 +02:00
|
|
|
if (optionsArg.dockerSockPath) {
|
|
|
|
pathToUse = optionsArg.dockerSockPath;
|
2024-02-02 16:54:07 +01:00
|
|
|
} else if (process.env.DOCKER_HOST) {
|
|
|
|
pathToUse = process.env.DOCKER_HOST;
|
2019-08-16 21:21:30 +02:00
|
|
|
} else if (process.env.CI) {
|
2019-08-16 21:34:35 +02:00
|
|
|
pathToUse = 'http://docker:2375/';
|
2019-08-16 21:21:30 +02:00
|
|
|
} else {
|
|
|
|
pathToUse = 'http://unix:/var/run/docker.sock:';
|
|
|
|
}
|
2024-02-02 16:54:07 +01:00
|
|
|
if (pathToUse.startsWith('unix:///')) {
|
2024-02-02 16:55:51 +01:00
|
|
|
pathToUse = pathToUse.replace('unix://', 'http://unix:');
|
|
|
|
}
|
|
|
|
if (pathToUse.endsWith('.sock')) {
|
|
|
|
pathToUse = pathToUse.replace('.sock', '.sock:');
|
2024-02-02 16:54:07 +01:00
|
|
|
}
|
|
|
|
console.log(`using docker sock at ${pathToUse}`);
|
2019-08-16 21:21:30 +02:00
|
|
|
this.socketPath = pathToUse;
|
2024-10-13 13:14:35 +02:00
|
|
|
this.imageStore = new DockerImageStore({
|
2024-06-08 15:03:19 +02:00
|
|
|
bucketDir: null,
|
|
|
|
localDirPath: this.options.imageStoreDir,
|
|
|
|
})
|
|
|
|
}
|
|
|
|
|
|
|
|
public async start() {
|
|
|
|
await this.imageStore.start();
|
|
|
|
}
|
|
|
|
public async stop() {
|
|
|
|
await this.imageStore.stop();
|
2018-07-16 23:52:50 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* authenticate against a registry
|
|
|
|
* @param userArg
|
|
|
|
* @param passArg
|
|
|
|
*/
|
2019-09-13 22:09:35 +02:00
|
|
|
public async auth(authData: IAuthData) {
|
|
|
|
const response = await this.request('POST', '/auth', authData);
|
2019-09-13 17:54:17 +02:00
|
|
|
if (response.body.Status !== 'Login Succeeded') {
|
|
|
|
console.log(`Login failed with ${response.body.Status}`);
|
|
|
|
throw new Error(response.body.Status);
|
|
|
|
}
|
|
|
|
console.log(response.body.Status);
|
2022-10-17 11:00:42 +02:00
|
|
|
this.registryToken = plugins.smartstring.base64.encode(plugins.smartjson.stringify(authData));
|
2019-08-14 23:21:54 +02:00
|
|
|
}
|
|
|
|
|
2019-09-13 18:15:45 +02:00
|
|
|
/**
|
|
|
|
* gets the token from the .docker/config.json file for GitLab registry
|
|
|
|
*/
|
2024-05-08 19:58:09 +02:00
|
|
|
public async getAuthTokenFromDockerConfig(registryUrlArg: string) {
|
2019-09-13 18:15:45 +02:00
|
|
|
const dockerConfigPath = plugins.smartpath.get.home('~/.docker/config.json');
|
|
|
|
const configObject = plugins.smartfile.fs.toObjectSync(dockerConfigPath);
|
2024-05-08 19:58:09 +02:00
|
|
|
const gitlabAuthBase64 = configObject.auths[registryUrlArg].auth;
|
2019-09-13 22:09:35 +02:00
|
|
|
const gitlabAuth: string = plugins.smartstring.base64.decode(gitlabAuthBase64);
|
|
|
|
const gitlabAuthArray = gitlabAuth.split(':');
|
|
|
|
await this.auth({
|
|
|
|
username: gitlabAuthArray[0],
|
|
|
|
password: gitlabAuthArray[1],
|
2024-06-05 14:10:44 +02:00
|
|
|
serveraddress: registryUrlArg,
|
2019-09-18 17:29:43 +02:00
|
|
|
});
|
2019-09-13 18:15:45 +02:00
|
|
|
}
|
|
|
|
|
2024-06-08 15:03:19 +02:00
|
|
|
// ==============
|
|
|
|
// NETWORKS
|
|
|
|
// ==============
|
2019-08-14 23:21:54 +02:00
|
|
|
/**
|
|
|
|
* gets all networks
|
|
|
|
*/
|
|
|
|
public async getNetworks() {
|
2019-08-15 18:50:13 +02:00
|
|
|
return await DockerNetwork.getNetworks(this);
|
2018-07-16 23:52:50 +02:00
|
|
|
}
|
|
|
|
|
2019-09-13 18:15:45 +02:00
|
|
|
/**
|
2024-06-08 15:03:19 +02:00
|
|
|
* create a network
|
2019-09-13 18:15:45 +02:00
|
|
|
*/
|
2024-06-08 15:03:19 +02:00
|
|
|
public async createNetwork(optionsArg: Parameters<typeof DockerNetwork.createNetwork>[1]) {
|
|
|
|
return await DockerNetwork.createNetwork(this, optionsArg);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get a network by name
|
|
|
|
*/
|
|
|
|
public async getNetworkByName(networkNameArg: string) {
|
|
|
|
return await DockerNetwork.getNetworkByName(this, networkNameArg);
|
|
|
|
}
|
2019-09-13 18:15:45 +02:00
|
|
|
|
2024-06-08 15:03:19 +02:00
|
|
|
|
|
|
|
// ==============
|
|
|
|
// CONTAINERS
|
|
|
|
// ==============
|
2018-07-16 23:52:50 +02:00
|
|
|
/**
|
2019-08-14 23:21:54 +02:00
|
|
|
* gets all containers
|
2018-07-16 23:52:50 +02:00
|
|
|
*/
|
2019-08-14 23:21:54 +02:00
|
|
|
public async getContainers() {
|
2018-07-16 23:52:50 +02:00
|
|
|
const containerArray = await DockerContainer.getContainers(this);
|
|
|
|
return containerArray;
|
2019-01-10 00:28:12 +01:00
|
|
|
}
|
2019-01-10 00:24:35 +01:00
|
|
|
|
2024-06-08 15:03:19 +02:00
|
|
|
// ==============
|
|
|
|
// SERVICES
|
|
|
|
// ==============
|
|
|
|
|
2019-09-08 19:22:20 +02:00
|
|
|
/**
|
|
|
|
* gets all services
|
|
|
|
*/
|
|
|
|
public async getServices() {
|
|
|
|
const serviceArray = await DockerService.getServices(this);
|
|
|
|
return serviceArray;
|
|
|
|
}
|
|
|
|
|
2024-06-08 15:03:19 +02:00
|
|
|
// ==============
|
|
|
|
// IMAGES
|
|
|
|
// ==============
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get all images
|
|
|
|
*/
|
|
|
|
public async getImages() {
|
|
|
|
return await DockerImage.getImages(this);
|
|
|
|
}
|
|
|
|
|
|
|
|
/**
|
|
|
|
* get an image by name
|
|
|
|
*/
|
|
|
|
public async getImageByName(imageNameArg: string) {
|
|
|
|
return await DockerImage.getImageByName(this, imageNameArg);
|
|
|
|
}
|
|
|
|
|
2019-08-15 18:50:13 +02:00
|
|
|
/**
|
|
|
|
*
|
|
|
|
*/
|
2019-08-14 23:21:54 +02:00
|
|
|
public async getEventObservable(): Promise<plugins.rxjs.Observable<any>> {
|
2019-01-10 00:24:35 +01:00
|
|
|
const response = await this.requestStreaming('GET', '/events');
|
2020-09-30 16:35:24 +00:00
|
|
|
return plugins.rxjs.Observable.create((observer) => {
|
|
|
|
response.on('data', (data) => {
|
2019-01-18 02:42:13 +01:00
|
|
|
const eventString = data.toString();
|
|
|
|
try {
|
|
|
|
const eventObject = JSON.parse(eventString);
|
|
|
|
observer.next(eventObject);
|
|
|
|
} catch (e) {
|
|
|
|
console.log(e);
|
|
|
|
}
|
2019-01-10 00:24:35 +01:00
|
|
|
});
|
|
|
|
return () => {
|
|
|
|
response.emit('end');
|
|
|
|
};
|
|
|
|
});
|
2018-07-16 23:52:50 +02:00
|
|
|
}
|
|
|
|
|
2019-08-15 18:50:13 +02:00
|
|
|
/**
|
|
|
|
* activates docker swarm
|
|
|
|
*/
|
2019-08-16 12:48:40 +02:00
|
|
|
public async activateSwarm(addvertisementIpArg?: string) {
|
2019-09-08 16:34:26 +02:00
|
|
|
// determine advertisement address
|
|
|
|
let addvertisementIp: string;
|
|
|
|
if (addvertisementIpArg) {
|
|
|
|
addvertisementIp = addvertisementIpArg;
|
|
|
|
} else {
|
|
|
|
const smartnetworkInstance = new plugins.smartnetwork.SmartNetwork();
|
|
|
|
const defaultGateway = await smartnetworkInstance.getDefaultGateway();
|
|
|
|
if (defaultGateway) {
|
|
|
|
addvertisementIp = defaultGateway.ipv4.address;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-08-15 18:50:13 +02:00
|
|
|
const response = await this.request('POST', '/swarm/init', {
|
|
|
|
ListenAddr: '0.0.0.0:2377',
|
2019-09-08 16:34:26 +02:00
|
|
|
AdvertiseAddr: addvertisementIp,
|
2019-08-15 18:50:13 +02:00
|
|
|
DataPathPort: 4789,
|
|
|
|
DefaultAddrPool: ['10.10.0.0/8', '20.20.0.0/8'],
|
|
|
|
SubnetSize: 24,
|
2020-09-30 16:35:24 +00:00
|
|
|
ForceNewCluster: false,
|
2019-08-15 18:50:13 +02:00
|
|
|
});
|
|
|
|
if (response.statusCode === 200) {
|
2020-09-30 16:27:43 +00:00
|
|
|
logger.log('info', 'created Swam succesfully');
|
2019-08-15 18:50:13 +02:00
|
|
|
} else {
|
2020-09-30 16:27:43 +00:00
|
|
|
logger.log('error', 'could not initiate swarm');
|
2019-08-15 18:50:13 +02:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-07-16 23:52:50 +02:00
|
|
|
/**
|
|
|
|
* fire a request
|
|
|
|
*/
|
2019-08-14 23:21:54 +02:00
|
|
|
public async request(methodArg: string, routeArg: string, dataArg = {}) {
|
|
|
|
const requestUrl = `${this.socketPath}${routeArg}`;
|
2018-07-16 23:52:50 +02:00
|
|
|
const response = await plugins.smartrequest.request(requestUrl, {
|
|
|
|
method: methodArg,
|
|
|
|
headers: {
|
2018-07-17 08:39:37 +02:00
|
|
|
'Content-Type': 'application/json',
|
2019-09-13 17:54:17 +02:00
|
|
|
'X-Registry-Auth': this.registryToken,
|
2020-09-30 16:35:24 +00:00
|
|
|
Host: 'docker.sock',
|
2018-07-16 23:52:50 +02:00
|
|
|
},
|
2019-09-08 19:22:20 +02:00
|
|
|
requestBody: dataArg,
|
2020-09-30 16:35:24 +00:00
|
|
|
keepAlive: false,
|
2018-07-16 23:52:50 +02:00
|
|
|
});
|
2019-08-15 18:50:13 +02:00
|
|
|
if (response.statusCode !== 200) {
|
|
|
|
console.log(response.body);
|
|
|
|
}
|
2018-07-17 08:39:37 +02:00
|
|
|
return response;
|
2018-07-16 23:52:50 +02:00
|
|
|
}
|
2019-01-10 00:24:35 +01:00
|
|
|
|
2024-06-05 14:10:44 +02:00
|
|
|
public async requestStreaming(methodArg: string, routeArg: string, readStream?: plugins.smartstream.stream.Readable) {
|
2019-08-14 23:21:54 +02:00
|
|
|
const requestUrl = `${this.socketPath}${routeArg}`;
|
2019-01-10 00:24:35 +01:00
|
|
|
const response = await plugins.smartrequest.request(
|
2019-01-10 00:28:12 +01:00
|
|
|
requestUrl,
|
|
|
|
{
|
|
|
|
method: methodArg,
|
|
|
|
headers: {
|
2019-09-08 19:22:20 +02:00
|
|
|
'Content-Type': 'application/json',
|
2019-09-13 22:09:35 +02:00
|
|
|
'X-Registry-Auth': this.registryToken,
|
2020-09-30 16:35:24 +00:00
|
|
|
Host: 'docker.sock',
|
2019-01-10 00:28:12 +01:00
|
|
|
},
|
2019-09-08 19:22:20 +02:00
|
|
|
requestBody: null,
|
2020-09-30 16:35:24 +00:00
|
|
|
keepAlive: false,
|
2019-01-10 00:24:35 +01:00
|
|
|
},
|
2024-06-05 14:10:44 +02:00
|
|
|
true,
|
|
|
|
(readStream ? reqArg => {
|
|
|
|
let counter = 0;
|
|
|
|
const smartduplex = new plugins.smartstream.SmartDuplex({
|
|
|
|
writeFunction: async (chunkArg) => {
|
|
|
|
if (counter % 1000 === 0) {
|
|
|
|
console.log(`posting chunk ${counter}`);
|
|
|
|
}
|
|
|
|
counter++;
|
|
|
|
return chunkArg;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
readStream.pipe(smartduplex).pipe(reqArg);
|
|
|
|
} : null),
|
2019-01-10 00:24:35 +01:00
|
|
|
);
|
|
|
|
console.log(response.statusCode);
|
|
|
|
console.log(response.body);
|
|
|
|
return response;
|
|
|
|
}
|
2024-06-10 00:15:01 +02:00
|
|
|
|
|
|
|
/**
|
|
|
|
* add s3 storage
|
|
|
|
* @param optionsArg
|
|
|
|
*/
|
|
|
|
public async addS3Storage(optionsArg: plugins.tsclass.storage.IS3Descriptor) {
|
|
|
|
this.smartBucket = new plugins.smartbucket.SmartBucket(optionsArg);
|
|
|
|
if (!optionsArg.bucketName) {
|
|
|
|
throw new Error('bucketName is required');
|
|
|
|
}
|
|
|
|
const bucket = await this.smartBucket.getBucketByName(optionsArg.bucketName);
|
|
|
|
let wantedDirectory = await bucket.getBaseDirectory();
|
|
|
|
if (optionsArg.directoryPath) {
|
|
|
|
wantedDirectory = await wantedDirectory.getSubDirectoryByName(optionsArg.directoryPath);
|
|
|
|
}
|
|
|
|
this.imageStore.options.bucketDir = wantedDirectory;
|
|
|
|
}
|
2018-07-16 23:52:50 +02:00
|
|
|
}
|