2019-06-23 03:35:23 -07:00
|
|
|
import { ChildProcess, spawn } from 'child_process';
|
2019-09-22 02:33:53 -07:00
|
|
|
const copyfiles = require('copyfiles');
|
2019-09-19 06:22:22 -07:00
|
|
|
import {
|
|
|
|
readFile as fsReadFile,
|
|
|
|
write as fsWrite,
|
|
|
|
} from 'fs';
|
|
|
|
import { join } from 'path';
|
|
|
|
import { file } from 'tmp-promise';
|
|
|
|
import { promisify } from 'util';
|
|
|
|
|
|
|
|
const fsReadFileAsync = promisify(fsReadFile);
|
|
|
|
const fsWriteAsync = promisify(fsWrite);
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
import { IBuildOptions } from '.';
|
|
|
|
|
|
|
|
import {
|
|
|
|
UserSettings,
|
2019-09-19 05:14:37 -07:00
|
|
|
} from 'n8n-core';
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
|
2019-09-19 06:22:22 -07:00
|
|
|
/**
|
|
|
|
* Create a custom tsconfig file as tsc currently has no way to define a base
|
|
|
|
* directory:
|
|
|
|
* https://github.com/Microsoft/TypeScript/issues/25430
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @returns
|
|
|
|
*/
|
|
|
|
export async function createCustomTsconfig () {
|
|
|
|
|
|
|
|
// Get path to simple tsconfig file which should be used for build
|
|
|
|
const tsconfigPath = join(__dirname, '../../src/tsconfig-build.json');
|
|
|
|
|
|
|
|
// Read the tsconfi file
|
|
|
|
const tsConfigString = await fsReadFileAsync(tsconfigPath, { encoding: 'utf8'}) as string;
|
|
|
|
const tsConfig = JSON.parse(tsConfigString);
|
|
|
|
|
|
|
|
// Set absolute include paths
|
|
|
|
const newIncludeFiles = [];
|
|
|
|
for (const includeFile of tsConfig.include) {
|
|
|
|
newIncludeFiles.push(join(process.cwd(), includeFile));
|
|
|
|
}
|
|
|
|
tsConfig.include = newIncludeFiles;
|
|
|
|
|
|
|
|
// Write new custom tsconfig file
|
|
|
|
const { fd, path, cleanup } = await file();
|
|
|
|
await fsWriteAsync(fd, Buffer.from(JSON.stringify(tsConfig, null, 2), 'utf8'));
|
|
|
|
|
|
|
|
return {
|
|
|
|
path,
|
|
|
|
cleanup,
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
/**
|
|
|
|
* Builds and copies credentials and nodes
|
|
|
|
*
|
|
|
|
* @export
|
|
|
|
* @param {IBuildOptions} [options] Options to overwrite default behaviour
|
|
|
|
* @returns {Promise<string>}
|
|
|
|
*/
|
|
|
|
export async function buildFiles (options?: IBuildOptions): Promise<string> {
|
|
|
|
options = options || {};
|
|
|
|
|
|
|
|
// Get the path of the TypeScript cli of this project
|
|
|
|
const tscPath = join(__dirname, '../../node_modules/typescript/bin/tsc');
|
|
|
|
|
2019-09-19 06:22:22 -07:00
|
|
|
const tsconfigData = await createCustomTsconfig();
|
2019-06-23 03:35:23 -07:00
|
|
|
|
|
|
|
const outputDirectory = options.destinationFolder || UserSettings.getUserN8nFolderCustomExtensionPath();
|
|
|
|
|
2019-09-19 06:38:08 -07:00
|
|
|
// Supply a node base path so that it finds n8n-core and n8n-workflow
|
|
|
|
const nodeModulesPath = join(__dirname, '../../node_modules/');
|
|
|
|
let buildCommand = `${tscPath} --p ${tsconfigData.path} --outDir ${outputDirectory} --rootDir ${process.cwd()} --baseUrl ${nodeModulesPath}`;
|
2019-06-23 03:35:23 -07:00
|
|
|
if (options.watch === true) {
|
|
|
|
buildCommand += ' --watch';
|
|
|
|
}
|
|
|
|
|
|
|
|
let buildProcess: ChildProcess;
|
|
|
|
try {
|
|
|
|
buildProcess = spawn('node', buildCommand.split(' '), { windowsVerbatimArguments: true, cwd: process.cwd() });
|
|
|
|
|
|
|
|
// Forward the output of the child process to the main one
|
|
|
|
// that the user can see what is happening
|
|
|
|
buildProcess.stdout.pipe(process.stdout);
|
|
|
|
buildProcess.stderr.pipe(process.stderr);
|
|
|
|
|
|
|
|
// Make sure that the child process gets also always terminated
|
|
|
|
// when the main process does
|
|
|
|
process.on('exit', () => {
|
|
|
|
buildProcess.kill();
|
|
|
|
});
|
|
|
|
} catch (error) {
|
|
|
|
let errorMessage = error.message;
|
|
|
|
|
|
|
|
if (error.stdout !== undefined) {
|
|
|
|
errorMessage = `${errorMessage}\nGot following output:\n${error.stdout}`;
|
|
|
|
}
|
|
|
|
|
2019-09-19 06:22:22 -07:00
|
|
|
// Remove the tmp tsconfig file
|
|
|
|
tsconfigData.cleanup();
|
|
|
|
|
2019-06-23 03:35:23 -07:00
|
|
|
throw new Error(errorMessage);
|
|
|
|
}
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
buildProcess.on('exit', code => {
|
2019-09-19 06:22:22 -07:00
|
|
|
// Remove the tmp tsconfig file
|
|
|
|
tsconfigData.cleanup();
|
2019-09-22 02:33:53 -07:00
|
|
|
copyfiles([join(process.cwd(), './*.png'), outputDirectory], { up: true }, () => resolve(outputDirectory));
|
2019-06-23 03:35:23 -07:00
|
|
|
});
|
|
|
|
});
|
|
|
|
}
|