Skip to content

Commit 6ffd42a

Browse files
kylecarbscode-asher
authored andcommitted
Improve CI caching (#416)
* Adjust linux distro to ubuntu 14.04 * Cache lib directory for speedy builds * Fix path linking for default extensions * Update reset * Reset to head * Improve caching * Still run yarn in CI * Update yarn before install * Increase cache timeout * Install vscode from vstar * Undo data-dir changes to CLI, add back clean, remove unused CI func * Remove additional flags added * Remove unused dependency * Reset vscode install dir so patching always works
1 parent 0a68a12 commit 6ffd42a

File tree

7 files changed

+145
-84
lines changed

7 files changed

+145
-84
lines changed

.gitignore

+1
Original file line numberDiff line numberDiff line change
@@ -4,3 +4,4 @@ dist
44
out
55
.DS_Store
66
release
7+
.cache

.travis.yml

+7-2
Original file line numberDiff line numberDiff line change
@@ -6,11 +6,12 @@ env:
66
matrix:
77
include:
88
- os: linux
9-
dist: ubuntu
9+
dist: trusty
1010
- os: osx
1111
before_install:
1212
- if [[ "$TRAVIS_OS_NAME" == "linux" ]]; then sudo apt-get install libxkbfile-dev
1313
libsecret-1-dev; fi
14+
- npm install -g yarn@1.12.3
1415
script:
1516
- scripts/build.sh
1617
before_deploy:
@@ -35,4 +36,8 @@ deploy:
3536
on:
3637
repo: codercom/code-server
3738
branch: master
38-
cache: yarn
39+
cache:
40+
yarn: true
41+
timeout: 1000
42+
directories:
43+
- .cache

build/tasks.ts

+34-80
Original file line numberDiff line numberDiff line change
@@ -4,21 +4,22 @@ import * as fse from "fs-extra";
44
import * as os from "os";
55
import * as path from "path";
66
import * as zlib from "zlib";
7+
import * as https from "https";
8+
import * as tar from "tar";
79

810
const isWin = os.platform() === "win32";
911
const libPath = path.join(__dirname, "../lib");
1012
const vscodePath = path.join(libPath, "vscode");
13+
const defaultExtensionsPath = path.join(libPath, "extensions");
1114
const pkgsPath = path.join(__dirname, "../packages");
12-
const defaultExtensionsPath = path.join(libPath, "VSCode-linux-x64/resources/app/extensions");
1315
const vscodeVersion = process.env.VSCODE_VERSION || "1.32.0";
16+
const vsSourceUrl = `https://codesrv-ci.cdr.sh/vstar-${vscodeVersion}.tar.gz`;
1417

1518
const buildServerBinary = register("build:server:binary", async (runner) => {
1619
await ensureInstalled();
17-
await copyForDefaultExtensions();
1820
await Promise.all([
1921
buildBootstrapFork(),
2022
buildWeb(),
21-
buildDefaultExtensions(),
2223
buildServerBundle(),
2324
buildAppBrowser(),
2425
]);
@@ -129,97 +130,50 @@ const buildWeb = register("build:web", async (runner) => {
129130
await runner.execute(isWin ? "npm.cmd" : "npm", ["run", "build"]);
130131
});
131132

132-
const extDirPath = path.join("lib", "vscode-default-extensions");
133-
const copyForDefaultExtensions = register("build:copy-vscode", async (runner) => {
134-
if (!fs.existsSync(defaultExtensionsPath)) {
135-
await ensureClean();
136-
await ensureInstalled();
137-
await new Promise((resolve, reject): void => {
138-
fse.remove(extDirPath, (err) => {
139-
if (err) {
140-
return reject(err);
141-
}
142-
143-
resolve();
144-
});
145-
});
146-
await new Promise((resolve, reject): void => {
147-
fse.copy(vscodePath, extDirPath, (err) => {
148-
if (err) {
149-
return reject(err);
150-
}
151-
152-
resolve();
153-
});
154-
});
155-
}
156-
});
157-
158-
const buildDefaultExtensions = register("build:default-extensions", async (runner) => {
159-
if (!fs.existsSync(defaultExtensionsPath)) {
160-
await copyForDefaultExtensions();
161-
runner.cwd = extDirPath;
162-
const resp = await runner.execute(isWin ? "npx.cmd" : "npx", [isWin ? "gulp.cmd" : "gulp", "vscode-linux-x64", "--max-old-space-size=32384"]);
163-
if (resp.exitCode !== 0) {
164-
throw new Error(`Failed to build default extensions: ${resp.stderr}`);
165-
}
166-
}
167-
});
168-
169133
const ensureInstalled = register("vscode:install", async (runner) => {
170-
await ensureCloned();
134+
runner.cwd = libPath;
171135

172-
runner.cwd = vscodePath;
173-
const install = await runner.execute(isWin ? "yarn.cmd" : "yarn", []);
174-
if (install.exitCode !== 0) {
175-
throw new Error(`Failed to install vscode dependencies: ${install.stderr}`);
176-
}
177-
});
136+
if (fs.existsSync(vscodePath) && fs.existsSync(defaultExtensionsPath)) {
137+
const pkgVersion = JSON.parse(fs.readFileSync(path.join(vscodePath, "package.json")).toString("utf8")).version;
138+
if (pkgVersion === vscodeVersion) {
139+
runner.cwd = vscodePath;
178140

179-
const ensureCloned = register("vscode:clone", async (runner) => {
180-
if (fs.existsSync(vscodePath)) {
181-
await ensureClean();
182-
} else {
183-
fse.mkdirpSync(libPath);
184-
runner.cwd = libPath;
185-
const clone = await runner.execute("git", ["clone", "https://github.com/microsoft/vscode", "--branch", vscodeVersion, "--single-branch", "--depth=1"]);
186-
if (clone.exitCode !== 0) {
187-
throw new Error(`Failed to clone: ${clone.exitCode}`);
141+
const reset = await runner.execute("git", ["reset", "--hard"]);
142+
if (reset.exitCode !== 0) {
143+
throw new Error(`Failed to clean git repository: ${reset.stderr}`);
144+
}
145+
146+
return;
188147
}
189148
}
190149

191-
runner.cwd = vscodePath;
192-
const checkout = await runner.execute("git", ["checkout", vscodeVersion]);
193-
if (checkout.exitCode !== 0) {
194-
throw new Error(`Failed to checkout: ${checkout.stderr}`);
195-
}
196-
});
150+
fse.removeSync(libPath);
151+
fse.mkdirpSync(libPath);
197152

198-
const ensureClean = register("vscode:clean", async (runner) => {
199-
runner.cwd = vscodePath;
153+
await new Promise<void>((resolve, reject): void => {
154+
https.get(vsSourceUrl, (res) => {
155+
if (res.statusCode !== 200) {
156+
return reject(res.statusMessage);
157+
}
200158

201-
const status = await runner.execute("git", ["status", "--porcelain"]);
202-
if (status.stdout.trim() !== "") {
203-
const clean = await runner.execute("git", ["clean", "-f", "-d", "-X"]);
204-
if (clean.exitCode !== 0) {
205-
throw new Error(`Failed to clean git repository: ${clean.stderr}`);
206-
}
207-
const removeUnstaged = await runner.execute("git", ["checkout", "--", "."]);
208-
if (removeUnstaged.exitCode !== 0) {
209-
throw new Error(`Failed to remove unstaged files: ${removeUnstaged.stderr}`);
210-
}
211-
}
212-
const fetch = await runner.execute("git", ["fetch", "--prune"]);
213-
if (fetch.exitCode !== 0) {
214-
throw new Error(`Failed to fetch latest changes: ${fetch.stderr}`);
215-
}
159+
res.pipe(tar.x({
160+
C: libPath,
161+
}).on("finish", () => {
162+
resolve();
163+
}).on("error", (err: Error) => {
164+
reject(err);
165+
}));
166+
}).on("error", (err) => {
167+
reject(err);
168+
});
169+
});
216170
});
217171

218172
const ensurePatched = register("vscode:patch", async (runner) => {
219173
if (!fs.existsSync(vscodePath)) {
220174
throw new Error("vscode must be cloned to patch");
221175
}
222-
await ensureClean();
176+
await ensureInstalled();
223177

224178
runner.cwd = vscodePath;
225179
const patchPath = path.join(__dirname, "../scripts/vscode.patch");

package.json

+4
Original file line numberDiff line numberDiff line change
@@ -15,7 +15,9 @@
1515
"devDependencies": {
1616
"@types/fs-extra": "^5.0.4",
1717
"@types/node": "^10.12.18",
18+
"@types/tar": "^4.0.0",
1819
"@types/trash": "^4.3.1",
20+
"cache-loader": "^2.0.1",
1921
"cross-env": "^5.2.0",
2022
"crypto-browserify": "^3.12.0",
2123
"css-loader": "^2.1.0",
@@ -34,6 +36,8 @@
3436
"sass-loader": "^7.1.0",
3537
"string-replace-loader": "^2.1.1",
3638
"style-loader": "^0.23.1",
39+
"tar": "^4.4.8",
40+
"terser-webpack-plugin": "^1.2.3",
3741
"ts-loader": "^5.3.3",
3842
"ts-node": "^7.0.1",
3943
"tsconfig-paths": "^3.8.0",

scripts/vstar.sh

+24
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
#!/bin/bash
2+
set -euxo pipefail
3+
4+
# Builds a tarfile containing vscode sourcefiles neccessary for CI.
5+
# Done outside the CI and uploaded to object storage to reduce CI time.
6+
7+
branch=1.32.0
8+
dir=/tmp/vstar
9+
outfile=/tmp/vstar-$branch.tar.gz
10+
rm -rf $dir
11+
mkdir -p $dir
12+
13+
cd $dir
14+
git clone https://github.com/microsoft/vscode --branch $branch --single-branch --depth=1
15+
cd vscode
16+
17+
yarn
18+
19+
npx gulp vscode-linux-x64 --max-old-space-size=32384
20+
rm -rf extensions build out* test
21+
cd ..
22+
mv *-x64/resources/app/extensions ./extensions
23+
rm -rf *-x64
24+
tar -czvf $outfile .

scripts/webpack.general.config.js

+14
Original file line numberDiff line numberDiff line change
@@ -3,6 +3,7 @@ const os = require("os");
33
const environment = process.env.NODE_ENV || "development";
44
const HappyPack = require("happypack");
55
const webpack = require("webpack");
6+
const TerserPlugin = require("terser-webpack-plugin");
67

78
const root = path.join(__dirname, "..");
89

@@ -107,6 +108,11 @@ module.exports = (options = {}) => ({
107108
id: "ts",
108109
threads: Math.max(os.cpus().length - 1, 1),
109110
loaders: [{
111+
path: "cache-loader",
112+
query: {
113+
cacheDirectory: path.join(__dirname, "..", ".cache"),
114+
},
115+
}, {
110116
path: "ts-loader",
111117
query: {
112118
happyPackMode: true,
@@ -121,6 +127,14 @@ module.exports = (options = {}) => ({
121127
"process.env.VERSION": `"${process.env.VERSION || ""}"`,
122128
}),
123129
],
130+
optimization: {
131+
minimizer: [
132+
new TerserPlugin({
133+
cache: path.join(__dirname, "..", ".cache", "terser"),
134+
parallel: true,
135+
}),
136+
],
137+
},
124138
stats: {
125139
all: false, // Fallback for options not defined.
126140
errors: true,

yarn.lock

+61-2
Original file line numberDiff line numberDiff line change
@@ -24,6 +24,13 @@
2424
resolved "https://registry.yarnpkg.com/@types/node/-/node-10.12.18.tgz#1d3ca764718915584fcd9f6344621b7672665c67"
2525
integrity sha512-fh+pAqt4xRzPfqA6eh3Z2y6fyZavRIumvjhaCL753+TVkGKGhpPeyrJG2JftD0T9q4GF00KjefsQ+PQNDdWQaQ==
2626

27+
"@types/tar@^4.0.0":
28+
version "4.0.0"
29+
resolved "https://registry.yarnpkg.com/@types/tar/-/tar-4.0.0.tgz#e3239d969eeb693a012200613860d0eb871c94f0"
30+
integrity sha512-YybbEHNngcHlIWVCYsoj7Oo1JU9JqONuAlt1LlTH/lmL8BMhbzdFUgReY87a05rY1j8mfK47Del+TCkaLAXwLw==
31+
dependencies:
32+
"@types/node" "*"
33+
2734
"@types/trash@^4.3.1":
2835
version "4.3.1"
2936
resolved "https://registry.yarnpkg.com/@types/trash/-/trash-4.3.1.tgz#4880ff17c4eb467f1a26774ea6328428403b5c57"
@@ -723,6 +730,17 @@ cache-base@^1.0.1:
723730
union-value "^1.0.0"
724731
unset-value "^1.0.0"
725732

733+
cache-loader@^2.0.1:
734+
version "2.0.1"
735+
resolved "https://registry.yarnpkg.com/cache-loader/-/cache-loader-2.0.1.tgz#5758f41a62d7c23941e3c3c7016e6faeb03acb07"
736+
integrity sha512-V99T3FOynmGx26Zom+JrVBytLBsmUCzVG2/4NnUKgvXN4bEV42R1ERl1IyiH/cvFIDA1Ytq2lPZ9tXDSahcQpQ==
737+
dependencies:
738+
loader-utils "^1.1.0"
739+
mkdirp "^0.5.1"
740+
neo-async "^2.6.0"
741+
normalize-path "^3.0.0"
742+
schema-utils "^1.0.0"
743+
726744
camel-case@3.0.x:
727745
version "3.0.0"
728746
resolved "https://registry.yarnpkg.com/camel-case/-/camel-case-3.0.0.tgz#ca3c3688a4e9cf3a4cda777dc4dcbc713249cf73"
@@ -916,6 +934,11 @@ commander@^2.12.1, commander@^2.18.0:
916934
resolved "https://registry.yarnpkg.com/commander/-/commander-2.19.0.tgz#f6198aa84e5b83c46054b94ddedbfed5ee9ff12a"
917935
integrity sha512-6tvAOO+D6OENvRAh524Dh9jcfKTYDQAqvqezbCW82xj5X0pSrcpxtvRKHLG0yBY6SD7PSDrJaj+0AiOcKVd1Xg==
918936

937+
commander@^2.19.0:
938+
version "2.20.0"
939+
resolved "https://registry.yarnpkg.com/commander/-/commander-2.20.0.tgz#d58bb2b5c1ee8f87b0d340027e9e94e222c5a422"
940+
integrity sha512-7j2y+40w61zy6YC2iRNpUe/NwhNyoXrYpHMrSunaMG64nRnaf96zO/KMQR4OyN/UnE5KLyEBnKHd4aG3rskjpQ==
941+
919942
commondir@^1.0.1:
920943
version "1.0.1"
921944
resolved "https://registry.yarnpkg.com/commondir/-/commondir-1.0.1.tgz#ddd800da0c66127393cca5950ea968a3aaf1253b"
@@ -3305,7 +3328,7 @@ negotiator@0.6.1:
33053328
resolved "https://registry.yarnpkg.com/negotiator/-/negotiator-0.6.1.tgz#2b327184e8992101177b28563fb5e7102acd0ca9"
33063329
integrity sha1-KzJxhOiZIQEXeyhWP7XnECrNDKk=
33073330

3308-
neo-async@^2.5.0:
3331+
neo-async@^2.5.0, neo-async@^2.6.0:
33093332
version "2.6.0"
33103333
resolved "https://registry.yarnpkg.com/neo-async/-/neo-async-2.6.0.tgz#b9d15e4d71c6762908654b5183ed38b753340835"
33113334
integrity sha512-MFh0d/Wa7vkKO3Y3LlacqAEeHK0mckVqzDieUKTT+KGxi+zIpeVsFxymkIiRpbpDziHc290Xr9A1O4Om7otoRA==
@@ -3452,6 +3475,11 @@ normalize-path@^2.1.1:
34523475
dependencies:
34533476
remove-trailing-separator "^1.0.1"
34543477

3478+
normalize-path@^3.0.0:
3479+
version "3.0.0"
3480+
resolved "https://registry.yarnpkg.com/normalize-path/-/normalize-path-3.0.0.tgz#0dcd69ff23a1c9b11fd0978316644a0388216a65"
3481+
integrity sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==
3482+
34553483
npm-bundled@^1.0.1:
34563484
version "1.0.5"
34573485
resolved "https://registry.yarnpkg.com/npm-bundled/-/npm-bundled-1.0.5.tgz#3c1732b7ba936b3a10325aef616467c0ccbcc979"
@@ -4625,6 +4653,14 @@ source-map-support@^0.5.6, source-map-support@~0.5.6:
46254653
buffer-from "^1.0.0"
46264654
source-map "^0.6.0"
46274655

4656+
source-map-support@~0.5.10:
4657+
version "0.5.11"
4658+
resolved "https://registry.yarnpkg.com/source-map-support/-/source-map-support-0.5.11.tgz#efac2ce0800355d026326a0ca23e162aeac9a4e2"
4659+
integrity sha512-//sajEx/fGL3iw6fltKMdPvy8kL3kJ2O3iuYlRoT3k9Kb4BjOoZ+BZzaNHeuaruSt+Kf3Zk9tnfAQg9/AJqUVQ==
4660+
dependencies:
4661+
buffer-from "^1.0.0"
4662+
source-map "^0.6.0"
4663+
46284664
source-map-url@^0.4.0:
46294665
version "0.4.0"
46304666
resolved "https://registry.yarnpkg.com/source-map-url/-/source-map-url-0.4.0.tgz#3e935d7ddd73631b97659956d55128e87b5084a3"
@@ -4924,7 +4960,7 @@ tar@^2.0.0:
49244960
fstream "^1.0.2"
49254961
inherits "2"
49264962

4927-
tar@^4:
4963+
tar@^4, tar@^4.4.8:
49284964
version "4.4.8"
49294965
resolved "https://registry.yarnpkg.com/tar/-/tar-4.4.8.tgz#b19eec3fde2a96e64666df9fdb40c5ca1bc3747d"
49304966
integrity sha512-LzHF64s5chPQQS0IYBn9IN5h3i98c12bo4NCO7e0sGM2llXQ3p2FGC5sdENN4cTW48O915Sh+x+EXx7XW96xYQ==
@@ -4951,6 +4987,29 @@ terser-webpack-plugin@^1.1.0:
49514987
webpack-sources "^1.1.0"
49524988
worker-farm "^1.5.2"
49534989

4990+
terser-webpack-plugin@^1.2.3:
4991+
version "1.2.3"
4992+
resolved "https://registry.yarnpkg.com/terser-webpack-plugin/-/terser-webpack-plugin-1.2.3.tgz#3f98bc902fac3e5d0de730869f50668561262ec8"
4993+
integrity sha512-GOK7q85oAb/5kE12fMuLdn2btOS9OBZn4VsecpHDywoUC/jLhSAKOiYo0ezx7ss2EXPMzyEWFoE0s1WLE+4+oA==
4994+
dependencies:
4995+
cacache "^11.0.2"
4996+
find-cache-dir "^2.0.0"
4997+
schema-utils "^1.0.0"
4998+
serialize-javascript "^1.4.0"
4999+
source-map "^0.6.1"
5000+
terser "^3.16.1"
5001+
webpack-sources "^1.1.0"
5002+
worker-farm "^1.5.2"
5003+
5004+
terser@^3.16.1:
5005+
version "3.17.0"
5006+
resolved "https://registry.yarnpkg.com/terser/-/terser-3.17.0.tgz#f88ffbeda0deb5637f9d24b0da66f4e15ab10cb2"
5007+
integrity sha512-/FQzzPJmCpjAH9Xvk2paiWrFq+5M6aVOf+2KRbwhByISDX/EujxsK+BAvrhb6H+2rtrLCHK9N01wO014vrIwVQ==
5008+
dependencies:
5009+
commander "^2.19.0"
5010+
source-map "~0.6.1"
5011+
source-map-support "~0.5.10"
5012+
49545013
terser@^3.8.1:
49555014
version "3.14.1"
49565015
resolved "https://registry.yarnpkg.com/terser/-/terser-3.14.1.tgz#cc4764014af570bc79c79742358bd46926018a32"

0 commit comments

Comments
 (0)