docker/ts/classes.host.ts

296 lines
8.2 KiB
TypeScript
Raw Normal View History

import * as plugins from './plugins.js';
import * as paths from './paths.js';
import { DockerContainer } from './classes.container.js';
import { DockerNetwork } from './classes.network.js';
import { DockerService } from './classes.service.js';
import { logger } from './logger.js';
2024-02-02 16:54:07 +01:00
import path from 'path';
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 {
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) {
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;
this.imageStore = new DockerImageStore({
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],
serveraddress: registryUrlArg,
2019-09-18 17:29:43 +02:00
});
2019-09-13 18:15:45 +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
/**
* create a network
2019-09-13 18:15:45 +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
// ==============
// 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
// ==============
// SERVICES
// ==============
2019-09-08 19:22:20 +02:00
/**
* gets all services
*/
public async getServices() {
const serviceArray = await DockerService.getServices(this);
return serviceArray;
}
// ==============
// 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
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
},
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
}