2016-06-04 21:20:39 +00:00
|
|
|
"use strict";
|
|
|
|
var plugins = require("./npmci.plugins");
|
2016-06-07 01:59:47 +00:00
|
|
|
var paths = require("./npmci.paths");
|
2016-06-05 02:48:39 +00:00
|
|
|
var NpmciEnv = require("./npmci.env");
|
2016-06-05 12:55:08 +00:00
|
|
|
var npmci_bash_1 = require("./npmci.bash");
|
2016-06-04 21:20:39 +00:00
|
|
|
exports.build = function () {
|
|
|
|
var done = plugins.q.defer();
|
2016-06-05 12:27:56 +00:00
|
|
|
exports.readDockerfiles()
|
|
|
|
.then(exports.sortDockerfiles)
|
|
|
|
.then(exports.mapDockerfiles)
|
2016-06-05 14:43:27 +00:00
|
|
|
.then(exports.buildDockerfiles)
|
2016-06-05 15:17:15 +00:00
|
|
|
.then(exports.pushDockerfiles)
|
2016-06-05 14:43:27 +00:00
|
|
|
.then(function () {
|
|
|
|
done.resolve();
|
|
|
|
});
|
2016-06-04 21:20:39 +00:00
|
|
|
return done.promise;
|
|
|
|
};
|
2016-06-05 11:01:45 +00:00
|
|
|
exports.readDockerfiles = function () {
|
|
|
|
var done = plugins.q.defer();
|
|
|
|
var readDockerfilesArray = [];
|
|
|
|
plugins.gulp.src("./Dockerfile*")
|
|
|
|
.pipe(plugins.through2.obj(function (file, enc, cb) {
|
2016-06-05 03:16:14 +00:00
|
|
|
var myDockerfile = new Dockerfile({
|
2016-06-05 02:48:39 +00:00
|
|
|
filePath: file.path,
|
|
|
|
read: true
|
2016-06-05 03:16:14 +00:00
|
|
|
});
|
2016-06-05 11:01:45 +00:00
|
|
|
readDockerfilesArray.push(myDockerfile);
|
2016-06-05 03:16:14 +00:00
|
|
|
cb(null, file);
|
2016-06-05 11:01:45 +00:00
|
|
|
}, function () {
|
|
|
|
done.resolve(readDockerfilesArray);
|
|
|
|
}));
|
|
|
|
return done.promise;
|
2016-06-05 03:16:14 +00:00
|
|
|
};
|
2016-06-05 11:01:45 +00:00
|
|
|
exports.sortDockerfiles = function (sortableArrayArg) {
|
2016-06-05 04:20:05 +00:00
|
|
|
var done = plugins.q.defer();
|
2016-06-05 09:08:20 +00:00
|
|
|
var sortedArray = [];
|
2016-06-07 04:05:13 +00:00
|
|
|
var cleanTagsOriginal = exports.cleanTagsArrayFunction(sortableArrayArg, sortedArray);
|
2016-06-05 09:08:20 +00:00
|
|
|
var sorterFunctionCounter = 0;
|
|
|
|
var sorterFunction = function () {
|
2016-06-05 11:01:45 +00:00
|
|
|
sortableArrayArg.forEach(function (dockerfileArg) {
|
2016-06-07 04:05:13 +00:00
|
|
|
var cleanTags = exports.cleanTagsArrayFunction(sortableArrayArg, sortedArray);
|
|
|
|
if (cleanTags.indexOf(dockerfileArg.baseImage) == -1 && sortedArray.indexOf(dockerfileArg) == -1) {
|
2016-06-05 09:08:20 +00:00
|
|
|
sortedArray.push(dockerfileArg);
|
|
|
|
}
|
2016-06-07 04:05:13 +00:00
|
|
|
;
|
|
|
|
if (cleanTagsOriginal.indexOf(dockerfileArg.baseImage) != -1) {
|
2016-06-05 12:27:56 +00:00
|
|
|
dockerfileArg.localBaseImageDependent = true;
|
|
|
|
}
|
|
|
|
;
|
2016-06-05 04:20:05 +00:00
|
|
|
});
|
2016-06-05 11:50:45 +00:00
|
|
|
if (sortableArrayArg.length == sortedArray.length) {
|
2016-06-05 11:01:45 +00:00
|
|
|
done.resolve(sortedArray);
|
2016-06-05 04:20:05 +00:00
|
|
|
}
|
2016-06-05 11:50:45 +00:00
|
|
|
else if (sorterFunctionCounter < 10) {
|
2016-06-05 09:08:20 +00:00
|
|
|
sorterFunctionCounter++;
|
|
|
|
sorterFunction();
|
2016-06-05 04:20:05 +00:00
|
|
|
}
|
2016-06-05 09:08:20 +00:00
|
|
|
;
|
|
|
|
};
|
|
|
|
sorterFunction();
|
2016-06-05 04:20:05 +00:00
|
|
|
return done.promise;
|
|
|
|
};
|
2016-06-05 12:27:56 +00:00
|
|
|
exports.mapDockerfiles = function (sortedArray) {
|
2016-06-05 04:20:05 +00:00
|
|
|
var done = plugins.q.defer();
|
2016-06-05 12:27:56 +00:00
|
|
|
sortedArray.forEach(function (dockerfileArg) {
|
|
|
|
if (dockerfileArg.localBaseImageDependent) {
|
|
|
|
sortedArray.forEach(function (dockfile2) {
|
|
|
|
if (dockfile2.cleanTag == dockerfileArg.baseImage) {
|
|
|
|
dockerfileArg.localBaseDockerfile = dockfile2;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
;
|
|
|
|
});
|
|
|
|
done.resolve(sortedArray);
|
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
exports.buildDockerfiles = function (sortedArrayArg) {
|
|
|
|
var done = plugins.q.defer();
|
|
|
|
sortedArrayArg.forEach(function (dockerfileArg) {
|
2016-06-05 04:20:05 +00:00
|
|
|
dockerfileArg.build();
|
|
|
|
});
|
2016-06-05 15:17:15 +00:00
|
|
|
done.resolve(sortedArrayArg);
|
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
exports.pushDockerfiles = function (sortedArrayArg) {
|
|
|
|
var done = plugins.q.defer();
|
|
|
|
sortedArrayArg.forEach(function (dockerfileArg) {
|
|
|
|
dockerfileArg.push();
|
|
|
|
});
|
|
|
|
done.resolve(sortedArrayArg);
|
2016-06-05 04:20:05 +00:00
|
|
|
return done.promise;
|
2016-06-05 02:48:39 +00:00
|
|
|
};
|
2016-06-07 01:59:47 +00:00
|
|
|
exports.pullDockerfileImages = function (sortableArrayArg) {
|
|
|
|
var done = plugins.q.defer();
|
2016-06-07 02:31:25 +00:00
|
|
|
sortableArrayArg.forEach(function (dockerfileArg) {
|
|
|
|
dockerfileArg.pull();
|
|
|
|
});
|
|
|
|
done.resolve(sortableArrayArg);
|
2016-06-07 01:59:47 +00:00
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
exports.testDockerfiles = function (sortedArrayArg) {
|
|
|
|
var done = plugins.q.defer();
|
|
|
|
sortedArrayArg.forEach(function (dockerfileArg) {
|
|
|
|
dockerfileArg.test();
|
|
|
|
});
|
|
|
|
done.resolve(sortedArrayArg);
|
|
|
|
return done.promise;
|
|
|
|
};
|
|
|
|
exports.releaseDockerfiles = function (sortedArrayArg) {
|
|
|
|
var done = plugins.q.defer();
|
|
|
|
sortedArrayArg.forEach(function (dockerfileArg) {
|
|
|
|
dockerfileArg.release();
|
|
|
|
});
|
|
|
|
done.resolve(sortedArrayArg);
|
|
|
|
return done.promise;
|
|
|
|
};
|
2016-06-05 02:48:39 +00:00
|
|
|
var Dockerfile = (function () {
|
|
|
|
function Dockerfile(options) {
|
|
|
|
this.filePath = options.filePath;
|
|
|
|
this.repo = NpmciEnv.repo.user + "/" + NpmciEnv.repo.repo;
|
2016-06-05 11:01:45 +00:00
|
|
|
this.version = exports.dockerFileVersion(plugins.path.parse(options.filePath).base);
|
2016-06-05 04:20:05 +00:00
|
|
|
this.cleanTag = this.repo + ":" + this.version;
|
2016-06-07 01:59:47 +00:00
|
|
|
this.buildTag = exports.dockerTag(this.repo, this.version, "build");
|
|
|
|
this.releaseTag = exports.dockerTag(this.repo, this.version, "release");
|
|
|
|
this.containerName = "dockerfile-" + this.version;
|
2016-06-05 02:48:39 +00:00
|
|
|
if (options.filePath && options.read) {
|
|
|
|
this.content = plugins.smartfile.local.toStringSync(plugins.path.resolve(options.filePath));
|
|
|
|
}
|
|
|
|
;
|
2016-06-05 11:01:45 +00:00
|
|
|
this.baseImage = exports.dockerBaseImage(this.content);
|
2016-06-05 12:27:56 +00:00
|
|
|
this.localBaseImageDependent = false;
|
2016-06-05 02:48:39 +00:00
|
|
|
}
|
|
|
|
;
|
|
|
|
Dockerfile.prototype.build = function () {
|
2016-06-07 04:35:10 +00:00
|
|
|
plugins.beautylog.info("now building Dockerfile for " + this.cleanTag);
|
2016-06-05 20:51:59 +00:00
|
|
|
var done = plugins.q.defer();
|
|
|
|
this.patchContents();
|
2016-06-07 01:59:47 +00:00
|
|
|
npmci_bash_1.bashBare("docker build -t " + this.buildTag + " -f " + this.filePath + " .");
|
2016-06-05 20:51:59 +00:00
|
|
|
NpmciEnv.dockerFilesBuilt.push(this);
|
|
|
|
this.restoreContents();
|
|
|
|
done.resolve();
|
|
|
|
return done.promise;
|
2016-06-05 02:48:39 +00:00
|
|
|
};
|
|
|
|
;
|
|
|
|
Dockerfile.prototype.push = function () {
|
2016-06-05 20:51:59 +00:00
|
|
|
var done = plugins.q.defer();
|
2016-06-05 02:48:39 +00:00
|
|
|
if (this.buildTag) {
|
2016-06-05 12:55:08 +00:00
|
|
|
npmci_bash_1.bashBare("docker push " + this.buildTag);
|
2016-06-05 02:48:39 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
plugins.beautylog.error("Dockerfile hasn't been built yet!");
|
|
|
|
}
|
2016-06-05 20:51:59 +00:00
|
|
|
done.resolve();
|
|
|
|
return done.promise;
|
2016-06-05 02:48:39 +00:00
|
|
|
};
|
2016-06-07 01:59:47 +00:00
|
|
|
Dockerfile.prototype.pull = function () {
|
|
|
|
npmci_bash_1.bashBare("docker pull " + this.buildTag);
|
|
|
|
};
|
|
|
|
;
|
|
|
|
Dockerfile.prototype.test = function () {
|
2016-06-07 08:21:56 +00:00
|
|
|
var testFile = plugins.path.join(paths.NpmciTestDir, "test_" + this.version + ".sh");
|
2016-06-07 10:12:19 +00:00
|
|
|
var testFileExists = plugins.smartfile.checks.fileExistsSync(testFile);
|
|
|
|
if (testFileExists) {
|
|
|
|
npmci_bash_1.bashBare("docker run --name npmci_test_container " + this.buildTag + " mkdir /npmci_test");
|
|
|
|
npmci_bash_1.bashBare("docker cp " + testFile + " npmci_test_container:/npmci_test/test.sh");
|
|
|
|
npmci_bash_1.bashBare("docker commit npmci_test_container npmci_test_image");
|
|
|
|
npmci_bash_1.bashBare("docker run npmci_test_image sh /npmci_test/test.sh");
|
|
|
|
npmci_bash_1.bashBare("docker rm npmci_test_container");
|
|
|
|
npmci_bash_1.bashBare("docker rmi --force npmci_test_image");
|
2016-06-07 03:20:04 +00:00
|
|
|
}
|
|
|
|
else {
|
|
|
|
plugins.beautylog.warn("skipping tests for " + this.cleanTag + " because no testfile was found!");
|
|
|
|
}
|
2016-06-07 01:59:47 +00:00
|
|
|
};
|
|
|
|
;
|
|
|
|
Dockerfile.prototype.release = function () {
|
|
|
|
npmci_bash_1.bashBare("docker tag " + this.getId() + " " + this.releaseTag);
|
|
|
|
npmci_bash_1.bashBare("docker push " + this.releaseTag);
|
|
|
|
};
|
|
|
|
Dockerfile.prototype.getId = function () {
|
2016-06-07 04:14:36 +00:00
|
|
|
var containerId = npmci_bash_1.bashBare("docker inspect --type=image --format=\"{{.Id}}\" " + this.buildTag);
|
2016-06-07 01:59:47 +00:00
|
|
|
return containerId;
|
|
|
|
};
|
|
|
|
;
|
2016-06-05 12:27:56 +00:00
|
|
|
Dockerfile.prototype.patchContents = function () {
|
2016-06-05 20:51:59 +00:00
|
|
|
var done = plugins.q.defer();
|
|
|
|
if (this.localBaseImageDependent == true) {
|
2016-06-07 04:35:10 +00:00
|
|
|
plugins.beautylog.info("patching Dockerfile due to local build dependency!");
|
2016-06-05 20:51:59 +00:00
|
|
|
this.patchedContent = this.content.replace(/FROM\s[a-zA-Z0-9\/\-\:]*/, 'FROM ' + this.localBaseDockerfile.buildTag);
|
|
|
|
plugins.smartfile.memory.toFsSync(this.patchedContent, {
|
|
|
|
fileName: plugins.path.parse(this.filePath).name,
|
|
|
|
filePath: plugins.path.parse(this.filePath).dir
|
|
|
|
});
|
|
|
|
}
|
|
|
|
done.resolve();
|
|
|
|
return done.promise;
|
2016-06-05 12:27:56 +00:00
|
|
|
};
|
|
|
|
;
|
|
|
|
Dockerfile.prototype.restoreContents = function () {
|
2016-06-05 20:51:59 +00:00
|
|
|
var done = plugins.q.defer();
|
|
|
|
if (this.localBaseImageDependent == true) {
|
|
|
|
plugins.smartfile.memory.toFsSync(this.content, {
|
|
|
|
fileName: plugins.path.parse(this.filePath).name,
|
|
|
|
filePath: plugins.path.parse(this.filePath).dir
|
|
|
|
});
|
|
|
|
}
|
|
|
|
done.resolve();
|
|
|
|
return done.promise;
|
2016-06-05 12:27:56 +00:00
|
|
|
};
|
|
|
|
;
|
2016-06-05 02:48:39 +00:00
|
|
|
return Dockerfile;
|
|
|
|
}());
|
|
|
|
exports.Dockerfile = Dockerfile;
|
2016-06-05 11:01:45 +00:00
|
|
|
exports.dockerFileVersion = function (dockerfileNameArg) {
|
2016-06-05 02:48:39 +00:00
|
|
|
var versionString;
|
|
|
|
var versionRegex = /Dockerfile_([a-zA-Z0-9\.]*)$/;
|
|
|
|
var regexResultArray = versionRegex.exec(dockerfileNameArg);
|
2016-06-05 06:06:32 +00:00
|
|
|
if (regexResultArray && regexResultArray.length == 2) {
|
2016-06-05 02:48:39 +00:00
|
|
|
versionString = regexResultArray[1];
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
versionString = "latest";
|
|
|
|
}
|
|
|
|
return versionString;
|
|
|
|
};
|
2016-06-05 11:01:45 +00:00
|
|
|
exports.dockerBaseImage = function (dockerfileContentArg) {
|
|
|
|
var baseImageRegex = /FROM\s([a-zA-z0-9\/\-\:]*)\n?/;
|
2016-06-05 02:48:39 +00:00
|
|
|
var regexResultArray = baseImageRegex.exec(dockerfileContentArg);
|
|
|
|
return regexResultArray[1];
|
|
|
|
};
|
2016-06-07 01:59:47 +00:00
|
|
|
exports.dockerTag = function (repoArg, versionArg, stageArg) {
|
2016-06-05 02:48:39 +00:00
|
|
|
var tagString;
|
|
|
|
var registry = NpmciEnv.dockerRegistry;
|
2016-06-07 01:59:47 +00:00
|
|
|
if (stageArg == "build" || stageArg == "test") {
|
2016-06-05 02:48:39 +00:00
|
|
|
registry = "registry.gitlab.com";
|
|
|
|
}
|
|
|
|
var repo = repoArg;
|
|
|
|
var version = versionArg;
|
2016-06-07 10:36:00 +00:00
|
|
|
if (stageArg == "build" || stageArg == "test") {
|
2016-06-05 02:48:39 +00:00
|
|
|
version = version + "_test";
|
|
|
|
}
|
|
|
|
tagString = registry + "/" + repo + ":" + version;
|
|
|
|
return tagString;
|
|
|
|
};
|
2016-06-05 12:27:56 +00:00
|
|
|
exports.cleanTagsArrayFunction = function (dockerfileArrayArg, trackingArrayArg) {
|
|
|
|
var cleanTagsArray = [];
|
|
|
|
dockerfileArrayArg.forEach(function (dockerfileArg) {
|
|
|
|
if (trackingArrayArg.indexOf(dockerfileArg) == -1) {
|
|
|
|
cleanTagsArray.push(dockerfileArg.cleanTag);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
return cleanTagsArray;
|
|
|
|
};
|
2016-06-04 21:20:39 +00:00
|
|
|
|
2016-06-07 10:36:00 +00:00
|
|
|
//# sourceMappingURL=data:application/json;charset=utf8;base64,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
|