Previously, Travis pushed the build artitfacts to the preview server. This required us to use JWT to secure the POST request from Travis, to ensure we couldn't receive malicious builds. JWT has been deprecated and we are moving our builds to CircleCI. This commit rewrites the TypeScript part of the preview server that handles converting build artifact into hosted previews of the docs.
141 lines
5.1 KiB
TypeScript
141 lines
5.1 KiB
TypeScript
// Imports
|
|
import * as cp from 'child_process';
|
|
import {EventEmitter} from 'events';
|
|
import * as fs from 'fs';
|
|
import * as path from 'path';
|
|
import * as shell from 'shelljs';
|
|
import {HIDDEN_DIR_PREFIX} from '../common/constants';
|
|
import {assertNotMissingOrEmpty, computeShortSha} from '../common/utils';
|
|
import {ChangedPrVisibilityEvent, CreatedBuildEvent} from './build-events';
|
|
import {UploadError} from './upload-error';
|
|
|
|
// Classes
|
|
export class BuildCreator extends EventEmitter {
|
|
// Constructor
|
|
constructor(protected buildsDir: string) {
|
|
super();
|
|
assertNotMissingOrEmpty('buildsDir', buildsDir);
|
|
}
|
|
|
|
// Methods - Public
|
|
public create(pr: number, sha: string, archivePath: string, isPublic: boolean): Promise<void> {
|
|
// Use only part of the SHA for more readable URLs.
|
|
sha = computeShortSha(sha);
|
|
|
|
const {newPrDir: prDir} = this.getCandidatePrDirs(pr, isPublic);
|
|
const shaDir = path.join(prDir, sha);
|
|
let dirToRemoveOnError: string;
|
|
|
|
return Promise.resolve().
|
|
// If the same PR exists with different visibility, update the visibility first.
|
|
then(() => this.updatePrVisibility(pr, isPublic)).
|
|
then(() => Promise.all([this.exists(prDir), this.exists(shaDir)])).
|
|
then(([prDirExisted, shaDirExisted]) => {
|
|
if (shaDirExisted) {
|
|
const publicOrNot = isPublic ? 'public' : 'non-public';
|
|
throw new UploadError(409, `Request to overwrite existing ${publicOrNot} directory: ${shaDir}`);
|
|
}
|
|
|
|
dirToRemoveOnError = prDirExisted ? shaDir : prDir;
|
|
|
|
return Promise.resolve().
|
|
then(() => shell.mkdir('-p', shaDir)).
|
|
then(() => this.extractArchive(archivePath, shaDir)).
|
|
then(() => this.emit(CreatedBuildEvent.type, new CreatedBuildEvent(+pr, sha, isPublic))).
|
|
then(() => undefined);
|
|
}).
|
|
catch(err => {
|
|
if (dirToRemoveOnError) {
|
|
shell.rm('-rf', dirToRemoveOnError);
|
|
}
|
|
|
|
if (!(err instanceof UploadError)) {
|
|
err = new UploadError(500, `Error while uploading to directory: ${shaDir}\n${err}`);
|
|
}
|
|
|
|
throw err;
|
|
});
|
|
}
|
|
|
|
public updatePrVisibility(pr: number, makePublic: boolean): Promise<boolean> {
|
|
const {oldPrDir: otherVisPrDir, newPrDir: targetVisPrDir} = this.getCandidatePrDirs(pr, makePublic);
|
|
|
|
return Promise.
|
|
all([this.exists(otherVisPrDir), this.exists(targetVisPrDir)]).
|
|
then(([otherVisPrDirExisted, targetVisPrDirExisted]) => {
|
|
if (!otherVisPrDirExisted) {
|
|
// No visibility change: Either the visibility is up-to-date or the PR does not exist.
|
|
return false;
|
|
} else if (targetVisPrDirExisted) {
|
|
// Error: Directories for both visibilities exist.
|
|
throw new UploadError(409, `Request to move '${otherVisPrDir}' to existing directory '${targetVisPrDir}'.`);
|
|
}
|
|
|
|
// Visibility change: Moving `otherVisPrDir` to `targetVisPrDir`.
|
|
return Promise.resolve().
|
|
then(() => shell.mv(otherVisPrDir, targetVisPrDir)).
|
|
then(() => this.listShasByDate(targetVisPrDir)).
|
|
then(shas => this.emit(ChangedPrVisibilityEvent.type, new ChangedPrVisibilityEvent(+pr, shas, makePublic))).
|
|
then(() => true);
|
|
}).
|
|
catch(err => {
|
|
if (!(err instanceof UploadError)) {
|
|
err = new UploadError(500, `Error while making PR ${pr} ${makePublic ? 'public' : 'hidden'}.\n${err}`);
|
|
}
|
|
|
|
throw err;
|
|
});
|
|
}
|
|
|
|
// Methods - Protected
|
|
protected exists(fileOrDir: string): Promise<boolean> {
|
|
return new Promise(resolve => fs.access(fileOrDir, err => resolve(!err)));
|
|
}
|
|
|
|
protected extractArchive(inputFile: string, outputDir: string): Promise<void> {
|
|
return new Promise<void>((resolve, reject) => {
|
|
const cmd = `tar --extract --gzip --directory "${outputDir}" --file "${inputFile}"`;
|
|
|
|
cp.exec(cmd, (err, _stdout, stderr) => {
|
|
if (err) {
|
|
return reject(err);
|
|
}
|
|
|
|
if (stderr) {
|
|
console.warn(stderr);
|
|
}
|
|
|
|
try {
|
|
shell.chmod('-R', 'a-w', outputDir);
|
|
shell.rm('-f', inputFile);
|
|
resolve();
|
|
} catch (err) {
|
|
reject(err);
|
|
}
|
|
});
|
|
});
|
|
}
|
|
|
|
protected getCandidatePrDirs(pr: number, isPublic: boolean) {
|
|
const hiddenPrDir = path.join(this.buildsDir, HIDDEN_DIR_PREFIX + pr);
|
|
const publicPrDir = path.join(this.buildsDir, `${pr}`);
|
|
|
|
const oldPrDir = isPublic ? hiddenPrDir : publicPrDir;
|
|
const newPrDir = isPublic ? publicPrDir : hiddenPrDir;
|
|
|
|
return {oldPrDir, newPrDir};
|
|
}
|
|
|
|
protected listShasByDate(inputDir: string): Promise<string[]> {
|
|
return Promise.resolve().
|
|
then(() => shell.ls('-l', inputDir) as any as Promise<(fs.Stats & {name: string})[]>).
|
|
// Keep directories only.
|
|
// (Also, convert to standard Array - ShellJS provides custom `sort()` method for sorting file contents.)
|
|
then(items => items.filter(item => item.isDirectory())).
|
|
// Sort by modification date.
|
|
then(items => items.sort((a, b) => a.mtime.getTime() - b.mtime.getTime())).
|
|
// Return directory names.
|
|
then(items => items.map(item => item.name));
|
|
}
|
|
}
|