now supports qenv
This commit is contained in:
parent
30b88d36ae
commit
8778324e5b
@ -2,4 +2,4 @@ FROM hosttoday/ht-docker-node:npmts
|
|||||||
COPY ./buildContextDir /workspace
|
COPY ./buildContextDir /workspace
|
||||||
WORKDIR /workspace
|
WORKDIR /workspace
|
||||||
ENV CI=true
|
ENV CI=true
|
||||||
CMD ["npm","test"];
|
CMD ["npm","run","npmdocker"];
|
||||||
|
2
dist/npmdocker.config.d.ts
vendored
2
dist/npmdocker.config.d.ts
vendored
@ -1,9 +1,11 @@
|
|||||||
/// <reference types="q" />
|
/// <reference types="q" />
|
||||||
import * as plugins from "./npmdocker.plugins";
|
import * as plugins from "./npmdocker.plugins";
|
||||||
|
import { IKeyValueObject } from "qenv";
|
||||||
export interface IConfig {
|
export interface IConfig {
|
||||||
baseImage: string;
|
baseImage: string;
|
||||||
command: string;
|
command: string;
|
||||||
dockerSock: boolean;
|
dockerSock: boolean;
|
||||||
exitCode?: number;
|
exitCode?: number;
|
||||||
|
keyValueObjectArray: IKeyValueObject[];
|
||||||
}
|
}
|
||||||
export declare let run: () => plugins.q.Promise<{}>;
|
export declare let run: () => plugins.q.Promise<{}>;
|
||||||
|
33
dist/npmdocker.config.js
vendored
33
dist/npmdocker.config.js
vendored
@ -1,16 +1,39 @@
|
|||||||
"use strict";
|
"use strict";
|
||||||
const plugins = require("./npmdocker.plugins");
|
const plugins = require("./npmdocker.plugins");
|
||||||
|
const paths = require("./npmdocker.paths");
|
||||||
|
;
|
||||||
|
let getQenvKeyValueObject = () => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
|
let qenvKeyValueObjectArray;
|
||||||
|
if (plugins.smartfile.fs.fileExistsSync(plugins.path.join(paths.cwd, "qenv.yml"))) {
|
||||||
|
qenvKeyValueObjectArray = new plugins.qenv.Qenv(paths.cwd, ".nogit/").keyValueObjectArray;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
qenvKeyValueObjectArray = [];
|
||||||
|
}
|
||||||
|
;
|
||||||
|
done.resolve(qenvKeyValueObjectArray);
|
||||||
|
return done.promise;
|
||||||
|
};
|
||||||
|
let buildConfig = (qenvKeyValueObjectArrayArg) => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
let config = plugins.npmextra.dataFor({
|
let config = plugins.npmextra.dataFor({
|
||||||
toolName: "npmdocker",
|
toolName: "npmdocker",
|
||||||
defaultSettings: {
|
defaultSettings: {
|
||||||
baseImage: "hosttoday/ht-docker-node:npmts",
|
baseImage: "hosttoday/ht-docker-node:npmts",
|
||||||
command: "npm test",
|
command: "npm run npmdocker",
|
||||||
dockerSock: false
|
dockerSock: false,
|
||||||
|
keyValueObjectArray: qenvKeyValueObjectArrayArg
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
exports.run = () => {
|
|
||||||
let done = plugins.q.defer();
|
|
||||||
done.resolve(config);
|
done.resolve(config);
|
||||||
return done.promise;
|
return done.promise;
|
||||||
};
|
};
|
||||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnBtZG9ja2VyLmNvbmZpZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3RzL25wbWRvY2tlci5jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE1BQVksT0FBTyxXQUFNLHFCQUFxQixDQUFDLENBQUE7QUFVL0MsSUFBSSxNQUFNLEdBQVcsT0FBTyxDQUFDLFFBQVEsQ0FBQyxPQUFPLENBQUM7SUFDMUMsUUFBUSxFQUFDLFdBQVc7SUFDcEIsZUFBZSxFQUFFO1FBQ2IsU0FBUyxFQUFDLGdDQUFnQztRQUMxQyxPQUFPLEVBQUMsVUFBVTtRQUNsQixVQUFVLEVBQUUsS0FBSztLQUNwQjtDQUNKLENBQUMsQ0FBQztBQUVRLFdBQUcsR0FBRztJQUNiLElBQUksSUFBSSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDN0IsSUFBSSxDQUFDLE9BQU8sQ0FBQyxNQUFNLENBQUMsQ0FBQztJQUNyQixNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQztBQUN4QixDQUFDLENBQUEifQ==
|
exports.run = () => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
|
getQenvKeyValueObject()
|
||||||
|
.then(buildConfig)
|
||||||
|
.then(done.resolve);
|
||||||
|
return done.promise;
|
||||||
|
};
|
||||||
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnBtZG9ja2VyLmNvbmZpZy5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3RzL25wbWRvY2tlci5jb25maWcudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IjtBQUFBLE1BQVksT0FBTyxXQUFNLHFCQUFxQixDQUFDLENBQUE7QUFDL0MsTUFBWSxLQUFLLFdBQU0sbUJBQW1CLENBQUMsQ0FBQTtBQVcxQyxDQUFDO0FBRUYsSUFBSSxxQkFBcUIsR0FBRztJQUN4QixJQUFJLElBQUksR0FBRyxPQUFPLENBQUMsQ0FBQyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQzdCLElBQUksdUJBQXlDLENBQUM7SUFDOUMsRUFBRSxDQUFBLENBQUMsT0FBTyxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUMsY0FBYyxDQUFDLE9BQU8sQ0FBQyxJQUFJLENBQUMsSUFBSSxDQUFDLEtBQUssQ0FBQyxHQUFHLEVBQUMsVUFBVSxDQUFDLENBQUMsQ0FBQyxDQUFBLENBQUM7UUFDN0UsdUJBQXVCLEdBQUcsSUFBSSxPQUFPLENBQUMsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLENBQUMsR0FBRyxFQUFFLFNBQVMsQ0FBQyxDQUFDLG1CQUFtQixDQUFDO0lBQzlGLENBQUM7SUFBQyxJQUFJLENBQUMsQ0FBQztRQUNILHVCQUF1QixHQUFHLEVBQUUsQ0FBQztJQUNsQyxDQUFDO0lBQUEsQ0FBQztJQUNGLElBQUksQ0FBQyxPQUFPLENBQUMsdUJBQXVCLENBQUMsQ0FBQztJQUN0QyxNQUFNLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQztBQUN4QixDQUFDLENBQUM7QUFFRixJQUFJLFdBQVcsR0FBRyxDQUFDLDBCQUEwQztJQUN6RCxJQUFJLElBQUksR0FBRyxPQUFPLENBQUMsQ0FBQyxDQUFDLEtBQUssRUFBRSxDQUFDO0lBQzdCLElBQUksTUFBTSxHQUFZLE9BQU8sQ0FBQyxRQUFRLENBQUMsT0FBTyxDQUFDO1FBQzNDLFFBQVEsRUFBRSxXQUFXO1FBQ3JCLGVBQWUsRUFBRTtZQUNiLFNBQVMsRUFBRSxnQ0FBZ0M7WUFDM0MsT0FBTyxFQUFFLG1CQUFtQjtZQUM1QixVQUFVLEVBQUUsS0FBSztZQUNqQixtQkFBbUIsRUFBRSwwQkFBMEI7U0FDbEQ7S0FDSixDQUFDLENBQUM7SUFDSCxJQUFJLENBQUMsT0FBTyxDQUFDLE1BQU0sQ0FBQyxDQUFDO0lBQ3JCLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFBO0FBQ3ZCLENBQUMsQ0FBQztBQUVTLFdBQUcsR0FBRztJQUNiLElBQUksSUFBSSxHQUFHLE9BQU8sQ0FBQyxDQUFDLENBQUMsS0FBSyxFQUFFLENBQUM7SUFDN0IscUJBQXFCLEVBQUU7U0FDbEIsSUFBSSxDQUFDLFdBQVcsQ0FBQztTQUNqQixJQUFJLENBQUMsSUFBSSxDQUFDLE9BQU8sQ0FBQyxDQUFDO0lBQ3hCLE1BQU0sQ0FBQyxJQUFJLENBQUMsT0FBTyxDQUFDO0FBQ3hCLENBQUMsQ0FBQSJ9
|
54
dist/npmdocker.docker.js
vendored
54
dist/npmdocker.docker.js
vendored
File diff suppressed because one or more lines are too long
2
dist/npmdocker.plugins.d.ts
vendored
2
dist/npmdocker.plugins.d.ts
vendored
@ -2,7 +2,9 @@ import "typings-global";
|
|||||||
export import beautylog = require("beautylog");
|
export import beautylog = require("beautylog");
|
||||||
export import npmextra = require("npmextra");
|
export import npmextra = require("npmextra");
|
||||||
export import path = require("path");
|
export import path = require("path");
|
||||||
|
export import projectinfo = require("projectinfo");
|
||||||
export import q = require("q");
|
export import q = require("q");
|
||||||
|
export import qenv = require("qenv");
|
||||||
export import shelljs = require("shelljs");
|
export import shelljs = require("shelljs");
|
||||||
export import smartfile = require("smartfile");
|
export import smartfile = require("smartfile");
|
||||||
export import smartstring = require("smartstring");
|
export import smartstring = require("smartstring");
|
||||||
|
4
dist/npmdocker.plugins.js
vendored
4
dist/npmdocker.plugins.js
vendored
@ -3,8 +3,10 @@ require("typings-global");
|
|||||||
exports.beautylog = require("beautylog");
|
exports.beautylog = require("beautylog");
|
||||||
exports.npmextra = require("npmextra");
|
exports.npmextra = require("npmextra");
|
||||||
exports.path = require("path");
|
exports.path = require("path");
|
||||||
|
exports.projectinfo = require("projectinfo");
|
||||||
exports.q = require("q");
|
exports.q = require("q");
|
||||||
|
exports.qenv = require("qenv");
|
||||||
exports.shelljs = require("shelljs");
|
exports.shelljs = require("shelljs");
|
||||||
exports.smartfile = require("smartfile");
|
exports.smartfile = require("smartfile");
|
||||||
exports.smartstring = require("smartstring");
|
exports.smartstring = require("smartstring");
|
||||||
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnBtZG9ja2VyLnBsdWdpbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90cy9ucG1kb2NrZXIucGx1Z2lucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsUUFBTyxnQkFBZ0IsQ0FBQyxDQUFBO0FBQ1YsaUJBQVMsV0FBVyxXQUFXLENBQUMsQ0FBQztBQUNqQyxnQkFBUSxXQUFXLFVBQVUsQ0FBQyxDQUFDO0FBQy9CLFlBQUksV0FBVyxNQUFNLENBQUMsQ0FBQztBQUN2QixTQUFDLFdBQVcsR0FBRyxDQUFDLENBQUM7QUFDakIsZUFBTyxXQUFXLFNBQVMsQ0FBQyxDQUFDO0FBQzdCLGlCQUFTLFdBQVcsV0FBVyxDQUFDLENBQUM7QUFDakMsbUJBQVcsV0FBVyxhQUFhLENBQUMsQ0FBQyJ9
|
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoibnBtZG9ja2VyLnBsdWdpbnMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi90cy9ucG1kb2NrZXIucGx1Z2lucy50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUEsUUFBTyxnQkFBZ0IsQ0FBQyxDQUFBO0FBQ1YsaUJBQVMsV0FBVyxXQUFXLENBQUMsQ0FBQztBQUNqQyxnQkFBUSxXQUFXLFVBQVUsQ0FBQyxDQUFDO0FBQy9CLFlBQUksV0FBVyxNQUFNLENBQUMsQ0FBQztBQUN2QixtQkFBVyxXQUFXLGFBQWEsQ0FBQyxDQUFDO0FBQ3JDLFNBQUMsV0FBVyxHQUFHLENBQUMsQ0FBQztBQUNqQixZQUFJLFdBQVcsTUFBTSxDQUFDLENBQUM7QUFDdkIsZUFBTyxXQUFXLFNBQVMsQ0FBQyxDQUFDO0FBQzdCLGlCQUFTLFdBQVcsV0FBVyxDQUFDLENBQUM7QUFDakMsbUJBQVcsV0FBVyxhQUFhLENBQUMsQ0FBQyJ9
|
16
package.json
16
package.json
@ -31,20 +31,22 @@
|
|||||||
},
|
},
|
||||||
"homepage": "https://gitlab.com/pushrocks/npmdocker#README",
|
"homepage": "https://gitlab.com/pushrocks/npmdocker#README",
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"npmts-g": "^5.2.6",
|
"npmts-g": "^5.2.8",
|
||||||
"should": "^10.0.0",
|
"should": "^10.0.0",
|
||||||
"typings-test": "^1.0.1"
|
"typings-test": "^1.0.1"
|
||||||
},
|
},
|
||||||
"dependencies": {
|
"dependencies": {
|
||||||
"@types/q": "^0.0.27",
|
"@types/q": "0.x.x",
|
||||||
"@types/shelljs": "^0.3.27",
|
"@types/shelljs": "0.x.x",
|
||||||
"beautylog": "^5.0.14",
|
"beautylog": "^5.0.20",
|
||||||
"npmextra": "^1.0.8",
|
"npmextra": "^1.0.8",
|
||||||
|
"projectinfo": "^1.0.3",
|
||||||
"q": "^1.4.1",
|
"q": "^1.4.1",
|
||||||
|
"qenv": "^1.1.1",
|
||||||
"rxjs": "^5.0.0-beta.10",
|
"rxjs": "^5.0.0-beta.10",
|
||||||
"shelljs": "^0.7.0",
|
"shelljs": "^0.7.3",
|
||||||
"smartfile": "^4.0.11",
|
"smartfile": "^4.0.13",
|
||||||
"smartstring": "^2.0.15",
|
"smartstring": "^2.0.17",
|
||||||
"typings-global": "^1.0.6"
|
"typings-global": "^1.0.6"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -1,24 +1,48 @@
|
|||||||
import * as plugins from "./npmdocker.plugins";
|
import * as plugins from "./npmdocker.plugins";
|
||||||
import * as paths from "./npmdocker.paths";
|
import * as paths from "./npmdocker.paths";
|
||||||
|
|
||||||
|
// interfaces
|
||||||
|
import { IKeyValueObject } from "qenv";
|
||||||
|
|
||||||
export interface IConfig {
|
export interface IConfig {
|
||||||
baseImage: string;
|
baseImage: string;
|
||||||
command: string;
|
command: string;
|
||||||
dockerSock: boolean;
|
dockerSock: boolean;
|
||||||
exitCode?:number
|
exitCode?: number;
|
||||||
}
|
keyValueObjectArray: IKeyValueObject[];
|
||||||
|
};
|
||||||
|
|
||||||
|
let getQenvKeyValueObject = () => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
|
let qenvKeyValueObjectArray:IKeyValueObject[];
|
||||||
|
if(plugins.smartfile.fs.fileExistsSync(plugins.path.join(paths.cwd,"qenv.yml"))){
|
||||||
|
qenvKeyValueObjectArray = new plugins.qenv.Qenv(paths.cwd, ".nogit/").keyValueObjectArray;
|
||||||
|
} else {
|
||||||
|
qenvKeyValueObjectArray = [];
|
||||||
|
};
|
||||||
|
done.resolve(qenvKeyValueObjectArray);
|
||||||
|
return done.promise;
|
||||||
|
};
|
||||||
|
|
||||||
|
let buildConfig = (qenvKeyValueObjectArrayArg:IKeyValueObject) => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
let config: IConfig = plugins.npmextra.dataFor({
|
let config: IConfig = plugins.npmextra.dataFor({
|
||||||
toolName: "npmdocker",
|
toolName: "npmdocker",
|
||||||
defaultSettings: {
|
defaultSettings: {
|
||||||
baseImage: "hosttoday/ht-docker-node:npmts",
|
baseImage: "hosttoday/ht-docker-node:npmts",
|
||||||
command:"npm test",
|
command: "npm run npmdocker",
|
||||||
dockerSock: false
|
dockerSock: false,
|
||||||
|
keyValueObjectArray: qenvKeyValueObjectArrayArg
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
done.resolve(config);
|
||||||
|
return done.promise
|
||||||
|
};
|
||||||
|
|
||||||
export let run = () => {
|
export let run = () => {
|
||||||
let done = plugins.q.defer();
|
let done = plugins.q.defer();
|
||||||
done.resolve(config);
|
getQenvKeyValueObject()
|
||||||
|
.then(buildConfig)
|
||||||
|
.then(done.resolve);
|
||||||
return done.promise;
|
return done.promise;
|
||||||
}
|
}
|
@ -4,10 +4,16 @@ import * as snippets from "./npmdocker.snippets";
|
|||||||
|
|
||||||
import { npmdockerOra } from "./npmdocker.promisechain";
|
import { npmdockerOra } from "./npmdocker.promisechain";
|
||||||
|
|
||||||
let config;
|
// interfaces
|
||||||
|
import { IConfig } from "./npmdocker.config";
|
||||||
|
|
||||||
|
let config: IConfig;
|
||||||
let dockerData = {
|
let dockerData = {
|
||||||
imageTag: "npmdocker-temp-image:latest",
|
imageTag: "npmdocker-temp-image:latest",
|
||||||
containerName: "npmdocker-temp-container"
|
containerName: "npmdocker-temp-container",
|
||||||
|
dockerProjectMountString: "",
|
||||||
|
dockerSockString: "",
|
||||||
|
dockerEnvString: ""
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -68,6 +74,34 @@ let buildDockerImage = () => {
|
|||||||
return done.promise
|
return done.promise
|
||||||
};
|
};
|
||||||
|
|
||||||
|
let buildDockerProjectMountString = () => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
|
if (process.env.CI != "true") {
|
||||||
|
dockerData.dockerProjectMountString = `-v ${paths.cwd}:/workspace`;
|
||||||
|
};
|
||||||
|
done.resolve();
|
||||||
|
return done.promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
let buildDockerEnvString = () => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
|
console.log(config.keyValueObjectArray);
|
||||||
|
for (let keyValueObjectArg of config.keyValueObjectArray) {
|
||||||
|
let envString = dockerData.dockerEnvString = dockerData.dockerEnvString + `-e ${keyValueObjectArg.key}=${keyValueObjectArg.value} `
|
||||||
|
};
|
||||||
|
done.resolve();
|
||||||
|
return done.promise;
|
||||||
|
}
|
||||||
|
|
||||||
|
let buildDockerSockString = () => {
|
||||||
|
let done = plugins.q.defer();
|
||||||
|
if (config.dockerSock) {
|
||||||
|
dockerData.dockerSockString = `-v /var/run/docker.sock:/var/run/docker.sock`
|
||||||
|
};
|
||||||
|
done.resolve()
|
||||||
|
return done;
|
||||||
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* creates a container by running the built Dockerimage
|
* creates a container by running the built Dockerimage
|
||||||
*/
|
*/
|
||||||
@ -75,18 +109,8 @@ let runDockerImage = () => {
|
|||||||
let done = plugins.q.defer();
|
let done = plugins.q.defer();
|
||||||
npmdockerOra.text("starting Container...");
|
npmdockerOra.text("starting Container...");
|
||||||
npmdockerOra.end();
|
npmdockerOra.end();
|
||||||
// Are we mounting the project directory?
|
|
||||||
let dockerProjectMountString:string = "";
|
|
||||||
if(process.env.CI != "true"){
|
|
||||||
dockerProjectMountString = `-v ${paths.cwd}:/workspace`
|
|
||||||
};
|
|
||||||
// Are we mounting docker.sock?
|
|
||||||
let dockerSockString:string = "";
|
|
||||||
if(config.dockerSock){
|
|
||||||
dockerSockString = `-v /var/run/docker.sock:/var/run/docker.sock`
|
|
||||||
};
|
|
||||||
plugins.beautylog.log("now running Dockerimage");
|
plugins.beautylog.log("now running Dockerimage");
|
||||||
config.exitCode = plugins.shelljs.exec(`docker run ${dockerProjectMountString} ${dockerSockString} --name ${dockerData.containerName} ${dockerData.imageTag}`).code;
|
config.exitCode = plugins.shelljs.exec(`docker run ${dockerData.dockerProjectMountString} ${dockerData.dockerSockString} ${dockerData.dockerEnvString} --name ${dockerData.containerName} ${dockerData.imageTag}`).code;
|
||||||
done.resolve();
|
done.resolve();
|
||||||
return done.promise;
|
return done.promise;
|
||||||
};
|
};
|
||||||
@ -146,10 +170,13 @@ export let run = (configArg) => {
|
|||||||
.then(preClean)
|
.then(preClean)
|
||||||
.then(buildDockerFile)
|
.then(buildDockerFile)
|
||||||
.then(buildDockerImage)
|
.then(buildDockerImage)
|
||||||
|
.then(buildDockerProjectMountString)
|
||||||
|
.then(buildDockerEnvString)
|
||||||
|
.then(buildDockerSockString)
|
||||||
.then(runDockerImage)
|
.then(runDockerImage)
|
||||||
.then(postClean)
|
.then(postClean)
|
||||||
.then(() => {
|
.then(() => {
|
||||||
done.resolve(config);
|
done.resolve(config);
|
||||||
})
|
}).catch(err => {console.log(err)});
|
||||||
return done.promise;
|
return done.promise;
|
||||||
}
|
}
|
@ -2,7 +2,9 @@ import "typings-global";
|
|||||||
export import beautylog = require("beautylog");
|
export import beautylog = require("beautylog");
|
||||||
export import npmextra = require("npmextra");
|
export import npmextra = require("npmextra");
|
||||||
export import path = require("path");
|
export import path = require("path");
|
||||||
|
export import projectinfo = require("projectinfo");
|
||||||
export import q = require("q");
|
export import q = require("q");
|
||||||
|
export import qenv = require("qenv");
|
||||||
export import shelljs = require("shelljs");
|
export import shelljs = require("shelljs");
|
||||||
export import smartfile = require("smartfile");
|
export import smartfile = require("smartfile");
|
||||||
export import smartstring = require("smartstring");
|
export import smartstring = require("smartstring");
|
Loading…
Reference in New Issue
Block a user