Compare commits
101 Commits
Author | SHA1 | Date | |
---|---|---|---|
015ccfad48 | |||
06d2fcb750 | |||
f3e4bc0350 | |||
6de3abe3bf | |||
eaa4140f2f | |||
b21fe80109 | |||
96a2992432 | |||
870b5f2c07 | |||
212edf1db7 | |||
46dbd81bcc | |||
8f5678502d | |||
959d7aaed1 | |||
5aa10653b6 | |||
e120d6527e | |||
c80da05fbb | |||
b9c3475b86 | |||
de2d7e647b | |||
d9348bd016 | |||
034fbc3994 | |||
a33a6a1f7f | |||
9dd403821b | |||
601d82ea74 | |||
784bb22511 | |||
71c89ac9bc | |||
0b3e3b68c9 | |||
f3779faaaf | |||
73476c2c39 | |||
942f65268d | |||
a965647c1f | |||
db88c7f86c | |||
3f18cb68bf | |||
dae3b59e3b | |||
53062e70d4 | |||
3e70dc465b | |||
49445d93c6 | |||
4f838837f8 | |||
c76968bbe8 | |||
6c5e5644b1 | |||
5cf80944fe | |||
cdb69c5f17 | |||
178c1d2df1 | |||
43d9da808b | |||
15f5c38eb0 | |||
225c1be14c | |||
44f2aab2f6 | |||
b69315f1d3 | |||
7d20804986 | |||
0aab639fbd | |||
794bb60dfc | |||
b182a379af | |||
5c6c06dee6 | |||
a48e1e035e | |||
8836c06b56 | |||
7af8e0739b | |||
684185e951 | |||
21e6fff3fb | |||
83c49a6234 | |||
ad67849d45 | |||
0e4e07a912 | |||
1fbc09f557 | |||
d6201b864c | |||
ea5e552192 | |||
1afe5c6e16 | |||
eb0dc96dbd | |||
55f45b1c3a | |||
87ff0f01bb | |||
dd1939d7b2 | |||
5a2a5f1248 | |||
9767b8767a | |||
546e139b46 | |||
28d70bb49f | |||
b71f134abd | |||
968b3c7449 | |||
d9558f7843 | |||
19e67ffdcc | |||
e983b66c28 | |||
c79f6a698f | |||
0ef098e9c8 | |||
d56350ff28 | |||
a4dc4e7950 | |||
424e911804 | |||
b5c4727bae | |||
b6f3fbf8a9 | |||
7241e7a8fd | |||
ae37148ece | |||
65c37bdd6f | |||
6acbe30e2e | |||
eb6f7889d0 | |||
e39da5fee9 | |||
b07628bb0b | |||
5815f9b202 | |||
846ea9997e | |||
de54db33ad | |||
314cb692ac | |||
73f8ded3fe | |||
a28b10ac51 | |||
927e2e0acc | |||
c496405818 | |||
020737e21b | |||
fe3560caac | |||
b2a7e67868 |
4
.gitignore
vendored
4
.gitignore
vendored
@ -15,8 +15,6 @@ node_modules/
|
|||||||
|
|
||||||
# builds
|
# builds
|
||||||
dist/
|
dist/
|
||||||
dist_web/
|
dist_*/
|
||||||
dist_serve/
|
|
||||||
dist_ts_web/
|
|
||||||
|
|
||||||
# custom
|
# custom
|
@ -3,14 +3,14 @@ image: registry.gitlab.com/hosttoday/ht-docker-node:npmci
|
|||||||
|
|
||||||
cache:
|
cache:
|
||||||
paths:
|
paths:
|
||||||
- .npmci_cache/
|
- .npmci_cache/
|
||||||
key: "$CI_BUILD_STAGE"
|
key: '$CI_BUILD_STAGE'
|
||||||
|
|
||||||
stages:
|
stages:
|
||||||
- security
|
- security
|
||||||
- test
|
- test
|
||||||
- release
|
- release
|
||||||
- metadata
|
- metadata
|
||||||
|
|
||||||
# ====================
|
# ====================
|
||||||
# security stage
|
# security stage
|
||||||
@ -18,60 +18,66 @@ stages:
|
|||||||
mirror:
|
mirror:
|
||||||
stage: security
|
stage: security
|
||||||
script:
|
script:
|
||||||
- npmci git mirror
|
- npmci git mirror
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- notpriv
|
- docker
|
||||||
|
- notpriv
|
||||||
|
|
||||||
snyk:
|
audit:
|
||||||
|
image: registry.gitlab.com/hosttoday/ht-docker-node:npmci
|
||||||
stage: security
|
stage: security
|
||||||
script:
|
script:
|
||||||
- npmci npm prepare
|
- npmci npm prepare
|
||||||
- npmci command npm install -g snyk
|
|
||||||
- npmci command npm install --ignore-scripts
|
- npmci command npm install --ignore-scripts
|
||||||
- npmci command snyk test
|
- npmci command npm config set registry https://registry.npmjs.org
|
||||||
|
- npmci command npm audit --audit-level=high
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- notpriv
|
- docker
|
||||||
|
- notpriv
|
||||||
|
|
||||||
# ====================
|
# ====================
|
||||||
# test stage
|
# test stage
|
||||||
# ====================
|
# ====================
|
||||||
|
|
||||||
testLTS:
|
testStable:
|
||||||
stage: test
|
stage: test
|
||||||
script:
|
script:
|
||||||
- npmci npm prepare
|
- npmci npm prepare
|
||||||
- npmci node install lts
|
- npmci node install stable
|
||||||
- npmci npm install
|
- npmci npm install
|
||||||
- npmci npm test
|
- npmci npm test
|
||||||
coverage: /\d+.?\d+?\%\s*coverage/
|
coverage: /\d+.?\d+?\%\s*coverage/
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- notpriv
|
- docker
|
||||||
|
- priv
|
||||||
|
|
||||||
testBuild:
|
testBuild:
|
||||||
stage: test
|
stage: test
|
||||||
script:
|
script:
|
||||||
- npmci npm prepare
|
- npmci npm prepare
|
||||||
- npmci node install lts
|
- npmci node install stable
|
||||||
- npmci npm install
|
- npmci npm install
|
||||||
- npmci command npm run build
|
- npmci command npm run build
|
||||||
coverage: /\d+.?\d+?\%\s*coverage/
|
coverage: /\d+.?\d+?\%\s*coverage/
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- notpriv
|
- docker
|
||||||
|
- notpriv
|
||||||
|
|
||||||
release:
|
release:
|
||||||
stage: release
|
stage: release
|
||||||
script:
|
script:
|
||||||
- npmci node install lts
|
- npmci node install stable
|
||||||
- npmci npm publish
|
- npmci npm publish
|
||||||
only:
|
only:
|
||||||
- tags
|
- tags
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- notpriv
|
- docker
|
||||||
|
- notpriv
|
||||||
|
|
||||||
# ====================
|
# ====================
|
||||||
# metadata stage
|
# metadata stage
|
||||||
@ -81,33 +87,35 @@ codequality:
|
|||||||
allow_failure: true
|
allow_failure: true
|
||||||
script:
|
script:
|
||||||
- npmci command npm install -g tslint typescript
|
- npmci command npm install -g tslint typescript
|
||||||
|
- npmci npm prepare
|
||||||
- npmci npm install
|
- npmci npm install
|
||||||
- npmci command "tslint -c tslint.json ./ts/**/*.ts"
|
- npmci command "tslint -c tslint.json ./ts/**/*.ts"
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- priv
|
- docker
|
||||||
|
- priv
|
||||||
|
|
||||||
trigger:
|
trigger:
|
||||||
stage: metadata
|
stage: metadata
|
||||||
script:
|
script:
|
||||||
- npmci trigger
|
- npmci trigger
|
||||||
only:
|
only:
|
||||||
- tags
|
- tags
|
||||||
tags:
|
tags:
|
||||||
- docker
|
- lossless
|
||||||
- notpriv
|
- docker
|
||||||
|
- notpriv
|
||||||
|
|
||||||
pages:
|
pages:
|
||||||
image: hosttoday/ht-docker-dbase:npmci
|
|
||||||
services:
|
|
||||||
- docker:18-dind
|
|
||||||
stage: metadata
|
stage: metadata
|
||||||
script:
|
script:
|
||||||
|
- npmci node install lts
|
||||||
- npmci command npm install -g @gitzone/tsdoc
|
- npmci command npm install -g @gitzone/tsdoc
|
||||||
- npmci npm prepare
|
- npmci npm prepare
|
||||||
- npmci npm install
|
- npmci npm install
|
||||||
- npmci command tsdoc
|
- npmci command tsdoc
|
||||||
tags:
|
tags:
|
||||||
|
- lossless
|
||||||
- docker
|
- docker
|
||||||
- notpriv
|
- notpriv
|
||||||
only:
|
only:
|
||||||
@ -115,5 +123,5 @@ pages:
|
|||||||
artifacts:
|
artifacts:
|
||||||
expire_in: 1 week
|
expire_in: 1 week
|
||||||
paths:
|
paths:
|
||||||
- public
|
- public
|
||||||
allow_failure: true
|
allow_failure: true
|
||||||
|
9
.snyk
Normal file
9
.snyk
Normal file
@ -0,0 +1,9 @@
|
|||||||
|
# Snyk (https://snyk.io) policy file, patches or ignores known vulnerabilities.
|
||||||
|
version: v1.13.5
|
||||||
|
# ignores vulnerabilities until expiry date; change duration by modifying expiry date
|
||||||
|
ignore:
|
||||||
|
SNYK-JS-HTTPSPROXYAGENT-469131:
|
||||||
|
- '@pushrocks/smartnetwork > speedtest-net > https-proxy-agent':
|
||||||
|
reason: None given
|
||||||
|
expires: '2019-11-04T13:59:28.695Z'
|
||||||
|
patch: {}
|
26
.vscode/settings.json
vendored
Normal file
26
.vscode/settings.json
vendored
Normal file
@ -0,0 +1,26 @@
|
|||||||
|
{
|
||||||
|
"json.schemas": [
|
||||||
|
{
|
||||||
|
"fileMatch": ["/npmextra.json"],
|
||||||
|
"schema": {
|
||||||
|
"type": "object",
|
||||||
|
"properties": {
|
||||||
|
"npmci": {
|
||||||
|
"type": "object",
|
||||||
|
"description": "settings for npmci"
|
||||||
|
},
|
||||||
|
"gitzone": {
|
||||||
|
"type": "object",
|
||||||
|
"description": "settings for gitzone",
|
||||||
|
"properties": {
|
||||||
|
"projectType": {
|
||||||
|
"type": "string",
|
||||||
|
"enum": ["website", "element", "service", "npm"]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
]
|
||||||
|
}
|
@ -10,6 +10,7 @@
|
|||||||
"npmRegistryUrl": "registry.npmjs.org"
|
"npmRegistryUrl": "registry.npmjs.org"
|
||||||
},
|
},
|
||||||
"gitzone": {
|
"gitzone": {
|
||||||
|
"projectType": "npm",
|
||||||
"module": {
|
"module": {
|
||||||
"githost": "gitlab.com",
|
"githost": "gitlab.com",
|
||||||
"gitscope": "mojoio",
|
"gitscope": "mojoio",
|
||||||
|
1768
package-lock.json
generated
1768
package-lock.json
generated
File diff suppressed because it is too large
Load Diff
46
package.json
46
package.json
@ -1,10 +1,10 @@
|
|||||||
{
|
{
|
||||||
"name": "@mojoio/docker",
|
"name": "@mojoio/docker",
|
||||||
"version": "1.0.41",
|
"version": "1.0.92",
|
||||||
"description": "easy communication with docker remote api from node, TypeScript ready",
|
"description": "easy communication with docker remote api from node, TypeScript ready",
|
||||||
"private": false,
|
"private": false,
|
||||||
"main": "dist/index.js",
|
"main": "dist_ts/index.js",
|
||||||
"typings": "dist/index.d.ts",
|
"typings": "dist_ts/index.d.ts",
|
||||||
"scripts": {
|
"scripts": {
|
||||||
"test": "tstest test/",
|
"test": "tstest test/",
|
||||||
"build": "tsbuild"
|
"build": "tsbuild"
|
||||||
@ -27,28 +27,36 @@
|
|||||||
},
|
},
|
||||||
"homepage": "https://gitlab.com/pushrocks/dockersock#README",
|
"homepage": "https://gitlab.com/pushrocks/dockersock#README",
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@pushrocks/lik": "^3.0.10",
|
"@pushrocks/lik": "^3.0.19",
|
||||||
"@pushrocks/smartlog": "^2.0.19",
|
"@pushrocks/smartfile": "^7.0.11",
|
||||||
"@pushrocks/smartpromise": "^3.0.2",
|
"@pushrocks/smartjson": "^3.0.10",
|
||||||
"@pushrocks/smartrequest": "^1.1.16",
|
"@pushrocks/smartlog": "^2.0.21",
|
||||||
"rxjs": "^6.5.2"
|
"@pushrocks/smartnetwork": "^1.1.18",
|
||||||
|
"@pushrocks/smartpath": "^4.0.1",
|
||||||
|
"@pushrocks/smartpromise": "^3.0.6",
|
||||||
|
"@pushrocks/smartrequest": "^1.1.47",
|
||||||
|
"@pushrocks/smartstring": "^3.0.18",
|
||||||
|
"@pushrocks/smartversion": "^2.0.4",
|
||||||
|
"@tsclass/tsclass": "^3.0.13",
|
||||||
|
"rxjs": "^6.5.4"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@gitzone/tsbuild": "^2.1.11",
|
"@gitzone/tsbuild": "^2.1.22",
|
||||||
"@gitzone/tsrun": "^1.2.8",
|
"@gitzone/tsrun": "^1.2.8",
|
||||||
"@gitzone/tstest": "^1.0.24",
|
"@gitzone/tstest": "^1.0.28",
|
||||||
"@pushrocks/tapbundle": "^3.0.11",
|
"@pushrocks/tapbundle": "^3.2.1",
|
||||||
"@types/node": "^12.7.1",
|
"@types/node": "^13.9.3",
|
||||||
"tslint": "^5.18.0",
|
"tslint": "^6.1.0",
|
||||||
"tslint-config-prettier": "^1.18.0"
|
"tslint-config-prettier": "^1.18.0"
|
||||||
},
|
},
|
||||||
"files": [
|
"files": [
|
||||||
"ts/*",
|
"ts/**/*",
|
||||||
"ts_web/*",
|
"ts_web/**/*",
|
||||||
"dist/*",
|
"dist/**/*",
|
||||||
"dist_web/*",
|
"dist_*/**/*",
|
||||||
"dist_ts_web/*",
|
"dist_ts/**/*",
|
||||||
"assets/*",
|
"dist_ts_web/**/*",
|
||||||
|
"assets/**/*",
|
||||||
"cli.js",
|
"cli.js",
|
||||||
"npmextra.json",
|
"npmextra.json",
|
||||||
"readme.md"
|
"readme.md"
|
||||||
|
@ -8,7 +8,7 @@ unofficial docker engine api abstraction package written in TypeScript
|
|||||||
* [docs (typedoc)](https://mojoio.gitlab.io/docker/)
|
* [docs (typedoc)](https://mojoio.gitlab.io/docker/)
|
||||||
|
|
||||||
## Status for master
|
## Status for master
|
||||||
[](https://gitlab.com/mojoio/docker/commits/master)
|
[](https://gitlab.com/mojoio/docker/commits/master)
|
||||||
[](https://gitlab.com/mojoio/docker/commits/master)
|
[](https://gitlab.com/mojoio/docker/commits/master)
|
||||||
[](https://www.npmjs.com/package/@mojoio/docker)
|
[](https://www.npmjs.com/package/@mojoio/docker)
|
||||||
[](https://snyk.io/test/npm/@mojoio/docker)
|
[](https://snyk.io/test/npm/@mojoio/docker)
|
||||||
@ -30,6 +30,10 @@ const run = async () => {
|
|||||||
};
|
};
|
||||||
```
|
```
|
||||||
|
|
||||||
|
## Contribution
|
||||||
|
|
||||||
|
We are always happy for code contributions. If you are not the code contributing type that is ok. Still, maintaining Open Source repositories takes considerable time and thought. If you like the quality of what we do and our modules are useful to you we would appreciate a little monthly contribution: You can [contribute one time](https://lossless.link/contribute-onetime) or [contribute monthly](https://lossless.link/contribute). :)
|
||||||
|
|
||||||
For further information read the linked docs at the top of this readme.
|
For further information read the linked docs at the top of this readme.
|
||||||
|
|
||||||
> MIT licensed | **©** [Lossless GmbH](https://lossless.gmbh)
|
> MIT licensed | **©** [Lossless GmbH](https://lossless.gmbh)
|
11
scripts/testauth.ts
Normal file
11
scripts/testauth.ts
Normal file
@ -0,0 +1,11 @@
|
|||||||
|
import * as docker from '../ts';
|
||||||
|
import * as smartstring from '@pushrocks/smartstring';
|
||||||
|
|
||||||
|
const run = async () => {
|
||||||
|
const dockerHost = new docker.DockerHost();
|
||||||
|
await docker.DockerImage.createFromRegistry(dockerHost, {
|
||||||
|
imageUrl: 'registry.gitlab.com/servezone/private/cloudly:latest'
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
run();
|
62
test/test.ts
62
test/test.ts
@ -4,10 +4,14 @@ import * as docker from '../ts/index';
|
|||||||
let testDockerHost: docker.DockerHost;
|
let testDockerHost: docker.DockerHost;
|
||||||
|
|
||||||
tap.test('should create a new Dockersock instance', async () => {
|
tap.test('should create a new Dockersock instance', async () => {
|
||||||
testDockerHost = new docker.DockerHost();
|
testDockerHost = new docker.DockerHost('http://unix:/var/run/docker.sock:');
|
||||||
return expect(testDockerHost).to.be.instanceof(docker.DockerHost);
|
return expect(testDockerHost).to.be.instanceof(docker.DockerHost);
|
||||||
});
|
});
|
||||||
|
|
||||||
|
tap.test('should create a docker swarm', async () => {
|
||||||
|
await testDockerHost.activateSwarm();
|
||||||
|
});
|
||||||
|
|
||||||
// Containers
|
// Containers
|
||||||
tap.test('should list containers', async () => {
|
tap.test('should list containers', async () => {
|
||||||
const containers = await testDockerHost.getContainers();
|
const containers = await testDockerHost.getContainers();
|
||||||
@ -37,7 +41,7 @@ tap.test('should remove a network', async () => {
|
|||||||
tap.test('should pull an image from imagetag', async () => {
|
tap.test('should pull an image from imagetag', async () => {
|
||||||
const image = await docker.DockerImage.createFromRegistry(testDockerHost, {
|
const image = await docker.DockerImage.createFromRegistry(testDockerHost, {
|
||||||
imageUrl: 'hosttoday/ht-docker-node',
|
imageUrl: 'hosttoday/ht-docker-node',
|
||||||
tag: 'alpine'
|
imageTag: 'alpine'
|
||||||
});
|
});
|
||||||
expect(image).to.be.instanceOf(docker.DockerImage);
|
expect(image).to.be.instanceOf(docker.DockerImage);
|
||||||
console.log(image);
|
console.log(image);
|
||||||
@ -52,4 +56,58 @@ tap.test('should return a change Observable', async tools => {
|
|||||||
subscription.unsubscribe();
|
subscription.unsubscribe();
|
||||||
});
|
});
|
||||||
|
|
||||||
|
// SECRETS
|
||||||
|
tap.test('should create a secret', async () => {
|
||||||
|
const mySecret = await docker.DockerSecret.createSecret(testDockerHost, {
|
||||||
|
name: 'testSecret',
|
||||||
|
version: '1.0.3',
|
||||||
|
contentArg: `{ "hi": "wow"}`,
|
||||||
|
labels: {}
|
||||||
|
});
|
||||||
|
console.log(mySecret);
|
||||||
|
});
|
||||||
|
|
||||||
|
tap.test('should remove a secret by name', async () => {
|
||||||
|
const mySecret = await docker.DockerSecret.getSecretByName(testDockerHost, 'testSecret');
|
||||||
|
await mySecret.remove();
|
||||||
|
});
|
||||||
|
|
||||||
|
// SERVICES
|
||||||
|
tap.test('should activate swarm mode', async () => {
|
||||||
|
await testDockerHost.activateSwarm();
|
||||||
|
});
|
||||||
|
|
||||||
|
tap.test('should list all services', async tools => {
|
||||||
|
const services = await testDockerHost.getServices();
|
||||||
|
console.log(services);
|
||||||
|
});
|
||||||
|
|
||||||
|
tap.test('should create a service', async () => {
|
||||||
|
const testNetwork = await docker.DockerNetwork.createNetwork(testDockerHost, {
|
||||||
|
Name: 'testNetwork'
|
||||||
|
});
|
||||||
|
const testSecret = await docker.DockerSecret.createSecret(testDockerHost, {
|
||||||
|
name: 'testSecret',
|
||||||
|
version: '0.0.1',
|
||||||
|
labels: {},
|
||||||
|
contentArg: '{"hi": "wow"}'
|
||||||
|
});
|
||||||
|
const testImage = await docker.DockerImage.createFromRegistry(testDockerHost, {
|
||||||
|
imageUrl: 'registry.gitlab.com/hosttoday/ht-docker-static'
|
||||||
|
});
|
||||||
|
const testService = await docker.DockerService.createService(testDockerHost, {
|
||||||
|
image: testImage,
|
||||||
|
labels: {},
|
||||||
|
name: 'testService',
|
||||||
|
networks: [testNetwork],
|
||||||
|
networkAlias: 'testService',
|
||||||
|
secrets: [testSecret],
|
||||||
|
ports: ['3000:80']
|
||||||
|
});
|
||||||
|
|
||||||
|
await testService.remove();
|
||||||
|
await testNetwork.remove();
|
||||||
|
await testSecret.remove();
|
||||||
|
});
|
||||||
|
|
||||||
tap.start();
|
tap.start();
|
||||||
|
@ -49,7 +49,10 @@ export class DockerContainer {
|
|||||||
if (response.statusCode < 300) {
|
if (response.statusCode < 300) {
|
||||||
plugins.smartlog.defaultLogger.log('info', 'Container created successfully');
|
plugins.smartlog.defaultLogger.log('info', 'Container created successfully');
|
||||||
} else {
|
} else {
|
||||||
plugins.smartlog.defaultLogger.log('error', 'There has been a problem when creating the container');
|
plugins.smartlog.defaultLogger.log(
|
||||||
|
'error',
|
||||||
|
'There has been a problem when creating the container'
|
||||||
|
);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -1,19 +1,35 @@
|
|||||||
import * as plugins from './docker.plugins';
|
import * as plugins from './docker.plugins';
|
||||||
import { DockerContainer } from './docker.classes.container';
|
import { DockerContainer } from './docker.classes.container';
|
||||||
import { DockerNetwork } from './docker.classes.network';
|
import { DockerNetwork } from './docker.classes.network';
|
||||||
|
import { DockerService } from './docker.classes.service';
|
||||||
|
|
||||||
|
export interface IAuthData {
|
||||||
|
serveraddress: string;
|
||||||
|
username: string;
|
||||||
|
password: string;
|
||||||
|
}
|
||||||
|
|
||||||
export class DockerHost {
|
export class DockerHost {
|
||||||
/**
|
/**
|
||||||
* the path where the docker sock can be found
|
* the path where the docker sock can be found
|
||||||
*/
|
*/
|
||||||
public socketPath: string;
|
public socketPath: string;
|
||||||
|
private registryToken: string = '';
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* the constructor to instantiate a new docker sock instance
|
* the constructor to instantiate a new docker sock instance
|
||||||
* @param pathArg
|
* @param pathArg
|
||||||
*/
|
*/
|
||||||
constructor(pathArg: string = 'http://unix:/var/run/docker.sock:') {
|
constructor(pathArg?: string) {
|
||||||
this.socketPath = pathArg;
|
let pathToUse: string;
|
||||||
|
if (pathArg) {
|
||||||
|
pathToUse = pathArg;
|
||||||
|
} else if (process.env.CI) {
|
||||||
|
pathToUse = 'http://docker:2375/';
|
||||||
|
} else {
|
||||||
|
pathToUse = 'http://unix:/var/run/docker.sock:';
|
||||||
|
}
|
||||||
|
this.socketPath = pathToUse;
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -21,9 +37,32 @@ export class DockerHost {
|
|||||||
* @param userArg
|
* @param userArg
|
||||||
* @param passArg
|
* @param passArg
|
||||||
*/
|
*/
|
||||||
public async auth(registryArg: string, userArg: string, passArg: string) {
|
public async auth(authData: IAuthData) {
|
||||||
// TODO: implement Docker Registry authentication
|
const response = await this.request('POST', '/auth', authData);
|
||||||
await this.request('POST', '');
|
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);
|
||||||
|
this.registryToken = plugins.smartstring.base64.encode(
|
||||||
|
plugins.smartjson.Smartjson.stringify(authData, {})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gets the token from the .docker/config.json file for GitLab registry
|
||||||
|
*/
|
||||||
|
public async getGitlabComTokenFromDockerConfig() {
|
||||||
|
const dockerConfigPath = plugins.smartpath.get.home('~/.docker/config.json');
|
||||||
|
const configObject = plugins.smartfile.fs.toObjectSync(dockerConfigPath);
|
||||||
|
const gitlabAuthBase64 = configObject.auths['registry.gitlab.com'].auth;
|
||||||
|
const gitlabAuth: string = plugins.smartstring.base64.decode(gitlabAuthBase64);
|
||||||
|
const gitlabAuthArray = gitlabAuth.split(':');
|
||||||
|
await this.auth({
|
||||||
|
username: gitlabAuthArray[0],
|
||||||
|
password: gitlabAuthArray[1],
|
||||||
|
serveraddress: 'registry.gitlab.com'
|
||||||
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -33,6 +72,10 @@ export class DockerHost {
|
|||||||
return await DockerNetwork.getNetworks(this);
|
return await DockerNetwork.getNetworks(this);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
*
|
||||||
|
*/
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* gets all containers
|
* gets all containers
|
||||||
*/
|
*/
|
||||||
@ -41,6 +84,14 @@ export class DockerHost {
|
|||||||
return containerArray;
|
return containerArray;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* gets all services
|
||||||
|
*/
|
||||||
|
public async getServices() {
|
||||||
|
const serviceArray = await DockerService.getServices(this);
|
||||||
|
return serviceArray;
|
||||||
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
*
|
*
|
||||||
*/
|
*/
|
||||||
@ -65,10 +116,22 @@ export class DockerHost {
|
|||||||
/**
|
/**
|
||||||
* activates docker swarm
|
* activates docker swarm
|
||||||
*/
|
*/
|
||||||
public async activateSwarm(addvertisementIpArg: string) {
|
public async activateSwarm(addvertisementIpArg?: string) {
|
||||||
|
// 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;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
const response = await this.request('POST', '/swarm/init', {
|
const response = await this.request('POST', '/swarm/init', {
|
||||||
ListenAddr: '0.0.0.0:2377',
|
ListenAddr: '0.0.0.0:2377',
|
||||||
AdvertiseAddr: `${addvertisementIpArg}:2377`,
|
AdvertiseAddr: addvertisementIp,
|
||||||
DataPathPort: 4789,
|
DataPathPort: 4789,
|
||||||
DefaultAddrPool: ['10.10.0.0/8', '20.20.0.0/8'],
|
DefaultAddrPool: ['10.10.0.0/8', '20.20.0.0/8'],
|
||||||
SubnetSize: 24,
|
SubnetSize: 24,
|
||||||
@ -90,9 +153,11 @@ export class DockerHost {
|
|||||||
method: methodArg,
|
method: methodArg,
|
||||||
headers: {
|
headers: {
|
||||||
'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
|
'X-Registry-Auth': this.registryToken,
|
||||||
Host: 'docker.sock'
|
Host: 'docker.sock'
|
||||||
},
|
},
|
||||||
requestBody: dataArg
|
requestBody: dataArg,
|
||||||
|
keepAlive: false
|
||||||
});
|
});
|
||||||
if (response.statusCode !== 200) {
|
if (response.statusCode !== 200) {
|
||||||
console.log(response.body);
|
console.log(response.body);
|
||||||
@ -107,10 +172,12 @@ export class DockerHost {
|
|||||||
{
|
{
|
||||||
method: methodArg,
|
method: methodArg,
|
||||||
headers: {
|
headers: {
|
||||||
// 'Content-Type': 'application/json',
|
'Content-Type': 'application/json',
|
||||||
|
'X-Registry-Auth': this.registryToken,
|
||||||
Host: 'docker.sock'
|
Host: 'docker.sock'
|
||||||
},
|
},
|
||||||
requestBody: null
|
requestBody: null,
|
||||||
|
keepAlive: false
|
||||||
},
|
},
|
||||||
true
|
true
|
||||||
);
|
);
|
||||||
|
@ -13,31 +13,61 @@ export class DockerImage {
|
|||||||
return images;
|
return images;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async findImageByName (dockerHost: DockerHost, imageNameArg: string) {
|
public static async findImageByName(dockerHost: DockerHost, imageNameArg: string) {
|
||||||
const images = await this.getImages(dockerHost);
|
const images = await this.getImages(dockerHost);
|
||||||
return images.find(image => {
|
const result = images.find(image => {
|
||||||
return image.RepoTags.includes(imageNameArg);
|
if (image.RepoTags) {
|
||||||
|
return image.RepoTags.includes(imageNameArg);
|
||||||
|
} else {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
public static async createFromRegistry(
|
public static async createFromRegistry(
|
||||||
dockerHostArg: DockerHost,
|
dockerHostArg: DockerHost,
|
||||||
creationObject: interfaces.IImageCreationDescriptor
|
creationObject: interfaces.IImageCreationDescriptor
|
||||||
): Promise<DockerImage> {
|
): Promise<DockerImage> {
|
||||||
|
// lets create a sanatized imageUrlObject
|
||||||
|
const imageUrlObject: {
|
||||||
|
imageUrl: string;
|
||||||
|
imageTag: string;
|
||||||
|
imageOriginTag: string;
|
||||||
|
} = {
|
||||||
|
imageUrl: creationObject.imageUrl,
|
||||||
|
imageTag: creationObject.imageTag,
|
||||||
|
imageOriginTag: null
|
||||||
|
};
|
||||||
|
if (imageUrlObject.imageUrl.includes(':')) {
|
||||||
|
const imageUrl = imageUrlObject.imageUrl.split(':')[0];
|
||||||
|
const imageTag = imageUrlObject.imageUrl.split(':')[1];
|
||||||
|
if (imageUrlObject.imageTag) {
|
||||||
|
throw new Error(
|
||||||
|
`imageUrl ${imageUrlObject.imageUrl} can't be tagged with ${imageUrlObject.imageTag} because it is already tagged with ${imageTag}`
|
||||||
|
);
|
||||||
|
} else {
|
||||||
|
imageUrlObject.imageUrl = imageUrl;
|
||||||
|
imageUrlObject.imageTag = imageTag;
|
||||||
|
}
|
||||||
|
} else if (!imageUrlObject.imageTag) {
|
||||||
|
imageUrlObject.imageTag = 'latest';
|
||||||
|
}
|
||||||
|
imageUrlObject.imageOriginTag = `${imageUrlObject.imageUrl}:${imageUrlObject.imageTag}`;
|
||||||
|
|
||||||
|
// lets actually create the image
|
||||||
const response = await dockerHostArg.request(
|
const response = await dockerHostArg.request(
|
||||||
'POST',
|
'POST',
|
||||||
`/images/create?fromImage=${encodeURIComponent(
|
`/images/create?fromImage=${encodeURIComponent(
|
||||||
creationObject.imageUrl
|
imageUrlObject.imageUrl
|
||||||
)}&tag=${encodeURIComponent(creationObject.tag)}`
|
)}&tag=${encodeURIComponent(imageUrlObject.imageTag)}`
|
||||||
);
|
);
|
||||||
if (response.statusCode < 300) {
|
if (response.statusCode < 300) {
|
||||||
plugins.smartlog.defaultLogger.log(
|
plugins.smartlog.defaultLogger.log(
|
||||||
'info',
|
'info',
|
||||||
`Successfully pulled image ${creationObject.imageUrl} from the registry`
|
`Successfully pulled image ${imageUrlObject.imageUrl} from the registry`
|
||||||
);
|
);
|
||||||
const originTag = `${creationObject.imageUrl}:${creationObject.tag}`;
|
const image = await DockerImage.findImageByName(dockerHostArg, imageUrlObject.imageOriginTag);
|
||||||
console.log(originTag)
|
|
||||||
const image = await DockerImage.findImageByName(dockerHostArg, originTag);
|
|
||||||
return image;
|
return image;
|
||||||
} else {
|
} else {
|
||||||
plugins.smartlog.defaultLogger.log('error', `Failed at the attempt of creating a new image`);
|
plugins.smartlog.defaultLogger.log('error', `Failed at the attempt of creating a new image`);
|
||||||
@ -86,31 +116,31 @@ export class DockerImage {
|
|||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* returns a boolean wether the image has a upstream image
|
* tag an image
|
||||||
|
* @param newTag
|
||||||
*/
|
*/
|
||||||
public isUpstreamImage(): boolean {
|
public async tagImage(newTag) {
|
||||||
// TODO: implement isUpastreamImage
|
throw new Error('.tagImage is not yet implemented');
|
||||||
return this.RepoTags.length > 0;
|
|
||||||
}
|
|
||||||
|
|
||||||
public tagImage(newTag) {
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* pulls the latest version from the registry
|
* pulls the latest version from the registry
|
||||||
*/
|
*/
|
||||||
public async pullLatestImageFromRegistry(): Promise<boolean> {
|
public async pullLatestImageFromRegistry(): Promise<boolean> {
|
||||||
const dockerImageUrl = this.RepoTags[0].split(':')[0];
|
|
||||||
const dockerImageTag = this.RepoTags[0].split(':')[1];
|
|
||||||
const updatedImage = await DockerImage.createFromRegistry(this.dockerHost, {
|
const updatedImage = await DockerImage.createFromRegistry(this.dockerHost, {
|
||||||
imageUrl: dockerImageUrl,
|
imageUrl: this.RepoTags[0]
|
||||||
tag: dockerImageTag
|
|
||||||
});
|
});
|
||||||
Object.assign(this, updatedImage);
|
Object.assign(this, updatedImage);
|
||||||
// TODO: Compare image digists before and after
|
// TODO: Compare image digists before and after
|
||||||
return true;
|
return true;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// get stuff
|
||||||
|
public async getVersion() {
|
||||||
|
if (this.Labels && this.Labels.version) {
|
||||||
|
return this.Labels.version;
|
||||||
|
} else {
|
||||||
|
return '0.0.0';
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
@ -2,13 +2,16 @@ import * as plugins from './docker.plugins';
|
|||||||
import * as interfaces from './interfaces';
|
import * as interfaces from './interfaces';
|
||||||
|
|
||||||
import { DockerHost } from './docker.classes.host';
|
import { DockerHost } from './docker.classes.host';
|
||||||
|
import { DockerService } from './docker.classes.service';
|
||||||
|
|
||||||
export class DockerNetwork {
|
export class DockerNetwork {
|
||||||
public static async getNetworks(dockerHost: DockerHost): Promise<DockerNetwork[]> {
|
public static async getNetworks(dockerHost: DockerHost): Promise<DockerNetwork[]> {
|
||||||
const dockerNetworks: DockerNetwork[] = [];
|
const dockerNetworks: DockerNetwork[] = [];
|
||||||
const response = await dockerHost.request('GET', '/networks');
|
const response = await dockerHost.request('GET', '/networks');
|
||||||
for (const networkObject of response.body) {
|
for (const networkObject of response.body) {
|
||||||
dockerNetworks.push(new DockerNetwork(dockerHost, networkObject));
|
const dockerNetwork = new DockerNetwork(dockerHost);
|
||||||
|
Object.assign(dockerNetwork, networkObject);
|
||||||
|
dockerNetworks.push(dockerNetwork);
|
||||||
}
|
}
|
||||||
return dockerNetworks;
|
return dockerNetworks;
|
||||||
}
|
}
|
||||||
@ -27,26 +30,29 @@ export class DockerNetwork {
|
|||||||
CheckDuplicate: true,
|
CheckDuplicate: true,
|
||||||
Driver: 'overlay',
|
Driver: 'overlay',
|
||||||
EnableIPv6: false,
|
EnableIPv6: false,
|
||||||
IPAM: {
|
/* IPAM: {
|
||||||
Driver: 'default',
|
Driver: 'default',
|
||||||
Config: [
|
Config: [
|
||||||
{
|
{
|
||||||
Subnet: '172.20.10.0/16',
|
Subnet: `172.20.${networkCreationDescriptor.NetworkNumber}.0/16`,
|
||||||
IPRange: '172.20.10.0/24',
|
IPRange: `172.20.${networkCreationDescriptor.NetworkNumber}.0/24`,
|
||||||
Gateway: '172.20.10.11'
|
Gateway: `172.20.${networkCreationDescriptor.NetworkNumber}.11`
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
},
|
}, */
|
||||||
Internal: true,
|
Internal: false,
|
||||||
Attachable: true,
|
Attachable: true,
|
||||||
Ingress: false
|
Ingress: false
|
||||||
});
|
});
|
||||||
if (response.statusCode < 300 ) {
|
if (response.statusCode < 300) {
|
||||||
plugins.smartlog.defaultLogger.log('info', 'Created network successfully');
|
plugins.smartlog.defaultLogger.log('info', 'Created network successfully');
|
||||||
return await DockerNetwork.getNetworkByName(dockerHost, networkCreationDescriptor.Name);
|
return await DockerNetwork.getNetworkByName(dockerHost, networkCreationDescriptor.Name);
|
||||||
} else {
|
} else {
|
||||||
plugins.smartlog.defaultLogger.log('error', 'There has been an error creating the wanted network');
|
plugins.smartlog.defaultLogger.log(
|
||||||
return null
|
'error',
|
||||||
|
'There has been an error creating the wanted network'
|
||||||
|
);
|
||||||
|
return null;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -75,11 +81,8 @@ export class DockerNetwork {
|
|||||||
];
|
];
|
||||||
};
|
};
|
||||||
|
|
||||||
constructor(dockerHostArg: DockerHost, dockerNetworkObjectArg: any) {
|
constructor(dockerHostArg: DockerHost) {
|
||||||
this.dockerHost = dockerHostArg;
|
this.dockerHost = dockerHostArg;
|
||||||
Object.keys(dockerNetworkObjectArg).forEach(keyArg => {
|
|
||||||
this[keyArg] = dockerNetworkObjectArg[keyArg];
|
|
||||||
});
|
|
||||||
}
|
}
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -88,4 +91,30 @@ export class DockerNetwork {
|
|||||||
public async remove() {
|
public async remove() {
|
||||||
const response = await this.dockerHost.request('DELETE', `/networks/${this.Id}`);
|
const response = await this.dockerHost.request('DELETE', `/networks/${this.Id}`);
|
||||||
}
|
}
|
||||||
|
|
||||||
|
public async getContainersOnNetwork(): Promise<
|
||||||
|
Array<{
|
||||||
|
Name: string;
|
||||||
|
EndpointID: string;
|
||||||
|
MacAddress: string;
|
||||||
|
IPv4Address: string;
|
||||||
|
IPv6Address: string;
|
||||||
|
}>
|
||||||
|
> {
|
||||||
|
const returnArray = [];
|
||||||
|
const response = await this.dockerHost.request('GET', `/networks/${this.Id}`);
|
||||||
|
for (const key of Object.keys(response.body.Containers)) {
|
||||||
|
returnArray.push(response.body.Containers[key]);
|
||||||
|
}
|
||||||
|
|
||||||
|
return returnArray;
|
||||||
|
}
|
||||||
|
|
||||||
|
public async getContainersOnNetworkForService(serviceArg: DockerService) {
|
||||||
|
const containersOnNetwork = await this.getContainersOnNetwork();
|
||||||
|
const containersOfService = containersOnNetwork.filter(container => {
|
||||||
|
return container.Name.startsWith(serviceArg.Spec.Name);
|
||||||
|
});
|
||||||
|
return containersOfService;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
92
ts/docker.classes.secret.ts
Normal file
92
ts/docker.classes.secret.ts
Normal file
@ -0,0 +1,92 @@
|
|||||||
|
import * as plugins from './docker.plugins';
|
||||||
|
import { DockerHost } from './docker.classes.host';
|
||||||
|
|
||||||
|
// interfaces
|
||||||
|
import * as interfaces from './interfaces';
|
||||||
|
|
||||||
|
export class DockerSecret {
|
||||||
|
// STATIC
|
||||||
|
public static async getSecrets(dockerHostArg: DockerHost) {
|
||||||
|
const response = await dockerHostArg.request('GET', '/secrets');
|
||||||
|
const secrets: DockerSecret[] = [];
|
||||||
|
for (const secret of response.body) {
|
||||||
|
const dockerSecretInstance = new DockerSecret(dockerHostArg);
|
||||||
|
Object.assign(dockerSecretInstance, secret);
|
||||||
|
secrets.push(dockerSecretInstance);
|
||||||
|
}
|
||||||
|
return secrets;
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async getSecretByID(dockerHostArg: DockerHost, idArg: string) {
|
||||||
|
const secrets = await this.getSecrets(dockerHostArg);
|
||||||
|
return secrets.find(secret => secret.ID === idArg);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async getSecretByName(dockerHostArg: DockerHost, nameArg: string) {
|
||||||
|
const secrets = await this.getSecrets(dockerHostArg);
|
||||||
|
return secrets.find(secret => secret.Spec.Name === nameArg);
|
||||||
|
}
|
||||||
|
|
||||||
|
public static async createSecret(
|
||||||
|
dockerHostArg: DockerHost,
|
||||||
|
secretDescriptor: interfaces.ISecretCreationDescriptor
|
||||||
|
) {
|
||||||
|
const labels: interfaces.TLabels = {
|
||||||
|
...secretDescriptor.labels,
|
||||||
|
version: secretDescriptor.version
|
||||||
|
};
|
||||||
|
const response = await dockerHostArg.request('POST', '/secrets/create', {
|
||||||
|
Name: secretDescriptor.name,
|
||||||
|
Labels: labels,
|
||||||
|
Data: plugins.smartstring.base64.encode(secretDescriptor.contentArg)
|
||||||
|
});
|
||||||
|
|
||||||
|
const newSecretInstance = new DockerSecret(dockerHostArg);
|
||||||
|
Object.assign(newSecretInstance, response.body);
|
||||||
|
Object.assign(
|
||||||
|
newSecretInstance,
|
||||||
|
await DockerSecret.getSecretByID(dockerHostArg, newSecretInstance.ID)
|
||||||
|
);
|
||||||
|
return newSecretInstance;
|
||||||
|
}
|
||||||
|
|
||||||
|
// INSTANCE
|
||||||
|
public ID: string;
|
||||||
|
public Spec: {
|
||||||
|
Name: string;
|
||||||
|
Labels: interfaces.TLabels;
|
||||||
|
};
|
||||||
|
public Version: {
|
||||||
|
Index: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
public dockerHost: DockerHost;
|
||||||
|
constructor(dockerHostArg: DockerHost) {
|
||||||
|
this.dockerHost = dockerHostArg;
|
||||||
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* updates a secret
|
||||||
|
*/
|
||||||
|
public async update(contentArg: string) {
|
||||||
|
const route = `/secrets/${this.ID}/update?=version=${this.Version.Index}`;
|
||||||
|
const response = await this.dockerHost.request(
|
||||||
|
'POST',
|
||||||
|
`/secrets/${this.ID}/update?version=${this.Version.Index}`,
|
||||||
|
{
|
||||||
|
Name: this.Spec.Name,
|
||||||
|
Labels: this.Spec.Labels,
|
||||||
|
Data: plugins.smartstring.base64.encode(contentArg)
|
||||||
|
}
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async remove() {
|
||||||
|
await this.dockerHost.request('DELETE', `/secrets/${this.ID}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
// get things
|
||||||
|
public async getVersion() {
|
||||||
|
return this.Spec.Labels.version;
|
||||||
|
}
|
||||||
|
}
|
@ -2,5 +2,248 @@ import * as plugins from './docker.plugins';
|
|||||||
import * as interfaces from './interfaces';
|
import * as interfaces from './interfaces';
|
||||||
|
|
||||||
import { DockerHost } from './docker.classes.host';
|
import { DockerHost } from './docker.classes.host';
|
||||||
|
import { DockerImage } from './docker.classes.image';
|
||||||
|
import { DockerSecret } from './docker.classes.secret';
|
||||||
|
|
||||||
export class DockerService {}
|
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 mounts: Array<{
|
||||||
|
/**
|
||||||
|
* the target inside the container
|
||||||
|
*/
|
||||||
|
Target: string;
|
||||||
|
/**
|
||||||
|
* The Source from which to mount the data (Volume or host path)
|
||||||
|
*/
|
||||||
|
Source: string;
|
||||||
|
Type: 'bind' | 'volume' | 'tmpfs' | 'npipe';
|
||||||
|
ReadOnly: boolean;
|
||||||
|
Consistency: 'default' | 'consistent' | 'cached' | 'delegated';
|
||||||
|
}> = [];
|
||||||
|
if (serviceCreationDescriptor.accessHostDockerSock) {
|
||||||
|
mounts.push({
|
||||||
|
Target: '/var/run/docker.sock',
|
||||||
|
Source: '/var/run/docker.sock',
|
||||||
|
Consistency: 'default',
|
||||||
|
ReadOnly: false,
|
||||||
|
Type: 'bind'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (serviceCreationDescriptor.resources && serviceCreationDescriptor.resources.volumeMounts) {
|
||||||
|
for(const volumeMount of serviceCreationDescriptor.resources.volumeMounts) {
|
||||||
|
mounts.push({
|
||||||
|
Target: volumeMount.containerFsPath,
|
||||||
|
Source: volumeMount.hostFsPath,
|
||||||
|
Consistency: 'default',
|
||||||
|
ReadOnly: false,
|
||||||
|
Type: 'bind'
|
||||||
|
});
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const networkArray: Array<{
|
||||||
|
Target: string;
|
||||||
|
Aliases: string[];
|
||||||
|
}> = [];
|
||||||
|
|
||||||
|
for (const network of serviceCreationDescriptor.networks) {
|
||||||
|
networkArray.push({
|
||||||
|
Target: network.Name,
|
||||||
|
Aliases: [serviceCreationDescriptor.networkAlias]
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const ports = [];
|
||||||
|
for (const port of serviceCreationDescriptor.ports) {
|
||||||
|
const portArray = port.split(':');
|
||||||
|
const hostPort = portArray[0];
|
||||||
|
const containerPort = portArray[1];
|
||||||
|
ports.push({
|
||||||
|
Protocol: 'tcp',
|
||||||
|
PublishedPort: parseInt(hostPort, 10),
|
||||||
|
TargetPort: parseInt(containerPort, 10)
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// lets configure secrets
|
||||||
|
const secretArray: any[] = [];
|
||||||
|
for (const secret of serviceCreationDescriptor.secrets) {
|
||||||
|
secretArray.push({
|
||||||
|
File: {
|
||||||
|
Name: 'secret.json', // TODO: make sure that works with multiple secrets
|
||||||
|
UID: '33',
|
||||||
|
GID: '33',
|
||||||
|
Mode: 384
|
||||||
|
},
|
||||||
|
SecretID: secret.ID,
|
||||||
|
SecretName: secret.Spec.Name
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
// lets configure limits
|
||||||
|
|
||||||
|
const memoryLimitMB =
|
||||||
|
serviceCreationDescriptor.resources && serviceCreationDescriptor.resources.memorySizeMB
|
||||||
|
? serviceCreationDescriptor.resources.memorySizeMB
|
||||||
|
: 1000;
|
||||||
|
|
||||||
|
const limits = {
|
||||||
|
MemoryBytes: memoryLimitMB * 1000000
|
||||||
|
};
|
||||||
|
|
||||||
|
if (serviceCreationDescriptor.resources) {
|
||||||
|
limits.MemoryBytes = serviceCreationDescriptor.resources.memorySizeMB * 1000000;
|
||||||
|
}
|
||||||
|
|
||||||
|
const response = await dockerHost.request('POST', '/services/create', {
|
||||||
|
Name: serviceCreationDescriptor.name,
|
||||||
|
TaskTemplate: {
|
||||||
|
ContainerSpec: {
|
||||||
|
Image: serviceCreationDescriptor.image.RepoTags[0],
|
||||||
|
Labels: labels,
|
||||||
|
Secrets: secretArray,
|
||||||
|
Mounts: mounts
|
||||||
|
/* DNSConfig: {
|
||||||
|
Nameservers: ['1.1.1.1']
|
||||||
|
} */
|
||||||
|
},
|
||||||
|
UpdateConfig: {
|
||||||
|
Parallelism: 0,
|
||||||
|
Delay: 0,
|
||||||
|
FailureAction: 'pause',
|
||||||
|
Monitor: 15000000000,
|
||||||
|
MaxFailureRatio: 0.15
|
||||||
|
},
|
||||||
|
ForceUpdate: 1,
|
||||||
|
Resources: {
|
||||||
|
Limits: limits
|
||||||
|
},
|
||||||
|
LogDriver: {
|
||||||
|
Name: 'json-file',
|
||||||
|
Options: {
|
||||||
|
'max-file': '3',
|
||||||
|
'max-size': '10M'
|
||||||
|
}
|
||||||
|
}
|
||||||
|
},
|
||||||
|
Labels: labels,
|
||||||
|
Networks: networkArray,
|
||||||
|
EndpointSpec: {
|
||||||
|
Ports: ports
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
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 remove() {
|
||||||
|
await this.dockerHostRef.request('DELETE', `/services/${this.ID}`);
|
||||||
|
}
|
||||||
|
|
||||||
|
public async reReadFromDockerEngine() {
|
||||||
|
const dockerData = await this.dockerHostRef.request('GET', `/services/${this.ID}`);
|
||||||
|
// TODO: Better assign: 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.`);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
@ -1,12 +1,41 @@
|
|||||||
|
// node native path
|
||||||
|
import * as path from 'path';
|
||||||
|
|
||||||
|
export { path };
|
||||||
|
|
||||||
// @pushrocks scope
|
// @pushrocks scope
|
||||||
import * as lik from '@pushrocks/lik';
|
import * as lik from '@pushrocks/lik';
|
||||||
|
import * as smartfile from '@pushrocks/smartfile';
|
||||||
|
import * as smartjson from '@pushrocks/smartjson';
|
||||||
import * as smartlog from '@pushrocks/smartlog';
|
import * as smartlog from '@pushrocks/smartlog';
|
||||||
|
import * as smartnetwork from '@pushrocks/smartnetwork';
|
||||||
|
import * as smartpath from '@pushrocks/smartpath';
|
||||||
import * as smartpromise from '@pushrocks/smartpromise';
|
import * as smartpromise from '@pushrocks/smartpromise';
|
||||||
import * as smartrequest from '@pushrocks/smartrequest';
|
import * as smartrequest from '@pushrocks/smartrequest';
|
||||||
|
import * as smartstring from '@pushrocks/smartstring';
|
||||||
|
import * as smartversion from '@pushrocks/smartversion';
|
||||||
|
|
||||||
smartlog.defaultLogger.enableConsole();
|
smartlog.defaultLogger.enableConsole();
|
||||||
|
|
||||||
export { lik, smartlog, smartpromise, smartrequest };
|
export {
|
||||||
|
lik,
|
||||||
|
smartfile,
|
||||||
|
smartjson,
|
||||||
|
smartlog,
|
||||||
|
smartnetwork,
|
||||||
|
smartpath,
|
||||||
|
smartpromise,
|
||||||
|
smartrequest,
|
||||||
|
smartstring,
|
||||||
|
smartversion
|
||||||
|
};
|
||||||
|
|
||||||
|
// @tsclass scope
|
||||||
|
import * as tsclass from '@tsclass/tsclass';
|
||||||
|
|
||||||
|
export {
|
||||||
|
tsclass
|
||||||
|
};
|
||||||
|
|
||||||
// third party
|
// third party
|
||||||
import * as rxjs from 'rxjs';
|
import * as rxjs from 'rxjs';
|
||||||
|
@ -2,4 +2,5 @@ export * from './docker.classes.host';
|
|||||||
export * from './docker.classes.container';
|
export * from './docker.classes.container';
|
||||||
export * from './docker.classes.image';
|
export * from './docker.classes.image';
|
||||||
export * from './docker.classes.network';
|
export * from './docker.classes.network';
|
||||||
|
export * from './docker.classes.secret';
|
||||||
export * from './docker.classes.service';
|
export * from './docker.classes.service';
|
||||||
|
@ -1,4 +1,4 @@
|
|||||||
export interface IImageCreationDescriptor {
|
export interface IImageCreationDescriptor {
|
||||||
imageUrl: string;
|
imageUrl: string;
|
||||||
tag: string;
|
imageTag?: string;
|
||||||
}
|
}
|
||||||
|
@ -3,3 +3,5 @@ export * from './image';
|
|||||||
export * from './label';
|
export * from './label';
|
||||||
export * from './network';
|
export * from './network';
|
||||||
export * from './port';
|
export * from './port';
|
||||||
|
export * from './secret';
|
||||||
|
export * from './service';
|
||||||
|
8
ts/interfaces/secret.ts
Normal file
8
ts/interfaces/secret.ts
Normal file
@ -0,0 +1,8 @@
|
|||||||
|
import * as interfaces from './';
|
||||||
|
|
||||||
|
export interface ISecretCreationDescriptor {
|
||||||
|
name: string;
|
||||||
|
version: string;
|
||||||
|
contentArg: any;
|
||||||
|
labels: interfaces.TLabels;
|
||||||
|
}
|
21
ts/interfaces/service.ts
Normal file
21
ts/interfaces/service.ts
Normal file
@ -0,0 +1,21 @@
|
|||||||
|
import * as plugins from '../docker.plugins';
|
||||||
|
|
||||||
|
import * as interfaces from './';
|
||||||
|
import { DockerNetwork } from '../docker.classes.network';
|
||||||
|
import { DockerSecret } from '../docker.classes.secret';
|
||||||
|
import { DockerImage } from '../docker.classes.image';
|
||||||
|
|
||||||
|
export interface IServiceCreationDescriptor {
|
||||||
|
name: string;
|
||||||
|
image: DockerImage;
|
||||||
|
labels: interfaces.TLabels;
|
||||||
|
networks: DockerNetwork[];
|
||||||
|
networkAlias: string;
|
||||||
|
secrets: DockerSecret[];
|
||||||
|
ports: string[];
|
||||||
|
accessHostDockerSock?: boolean;
|
||||||
|
resources?: {
|
||||||
|
memorySizeMB?: number;
|
||||||
|
volumeMounts: plugins.tsclass.container.IVolumeMount[]
|
||||||
|
};
|
||||||
|
}
|
Reference in New Issue
Block a user