/* Copyright 2016-2018 Balena Ltd. Licensed under the Apache License, Version 2.0 (the "License"); you may not use this file except in compliance with the License. You may obtain a copy of the License at http://www.apache.org/licenses/LICENSE-2.0 Unless required by applicable law or agreed to in writing, software distributed under the License is distributed on an "AS IS" BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. See the License for the specific language governing permissions and limitations under the License. */ import * as JSONStream from 'JSONStream'; import * as request from 'request'; import { BalenaSDK } from 'balena-sdk'; import * as Stream from 'stream'; import { Pack } from 'tar-stream'; import { TypedError } from 'typed-error'; import { RegistrySecrets } from 'resin-multibuild'; import { tarDirectory } from './compose'; const DEBUG_MODE = !!process.env.DEBUG; const CURSOR_METADATA_REGEX = /([a-z]+)([0-9]+)?/; const TRIM_REGEX = /\n+$/; export interface BuildOpts { emulated: boolean; nocache: boolean; registrySecrets: RegistrySecrets; } export interface RemoteBuild { app: string; owner: string; source: string; auth: string; baseUrl: string; opts: BuildOpts; sdk: BalenaSDK; // For internal use releaseId?: number; hadError?: boolean; } interface BuilderMessage { message: string; type?: string; replace?: boolean; isError?: boolean; // These will be set when the type === 'metadata' resource?: string; value?: string; } export class RemoteBuildFailedError extends TypedError { public constructor(message = 'Remote build failed') { super(message); } } async function getBuilderEndpoint( baseUrl: string, owner: string, app: string, opts: BuildOpts, ): Promise { const querystring = await import('querystring'); const args = querystring.stringify({ owner, app, emulated: opts.emulated, nocache: opts.nocache, }); // Note that using https (rather than http) is a requirement when using the // --registry-secrets feature, as the secrets are not otherwise encrypted. return `https://builder.${baseUrl}/v3/build?${args}`; } export async function startRemoteBuild(build: RemoteBuild): Promise { const Bluebird = await import('bluebird'); const stream = await getRequestStream(build); // Special windows handling (win64 also reports win32) if (process.platform === 'win32') { const readline = (await import('readline')).createInterface({ input: process.stdin, output: process.stdout, }); readline.on('SIGINT', () => process.emit('SIGINT')); } return new Bluebird((resolve, reject) => { // Setup interrupt handlers so we can cancel the build if the user presses // ctrl+c // This is necessary because the `exit-hook` module is used by several // dependencies, and will exit without calling the following handler. // Once https://github.com/balena-io/balena-cli/issues/867 has been solved, // we are free to (and definitely should) remove the below line process.removeAllListeners('SIGINT'); process.on('SIGINT', () => { process.stderr.write('Received SIGINT, cleaning up. Please wait.\n'); cancelBuildIfNecessary(build).then(() => { stream.end(); process.exit(130); }); }); stream.on('data', getBuilderMessageHandler(build)); stream.on('end', resolve); stream.on('error', reject); }).then(() => { if (build.hadError) { throw new RemoteBuildFailedError(); } }); } async function handleBuilderMetadata(obj: BuilderMessage, build: RemoteBuild) { const { stripIndent } = await import('common-tags'); switch (obj.resource) { case 'cursor': const readline = await import('readline'); if (obj.value == null) { return; } const match = obj.value.match(CURSOR_METADATA_REGEX); if (!match) { // FIXME: Make this error nicer. console.log( stripIndent` Warning: ignoring unknown builder command. You may experience odd build output. Maybe you need to update balena-cli?`, ); return; } const value = match[1]; const amount = match[2] || 1; switch (value) { case 'erase': readline.clearLine(process.stdout, 0); process.stdout.write('\r'); break; case 'up': readline.moveCursor(process.stdout, 0, -amount); break; case 'down': readline.moveCursor(process.stdout, 0, amount); break; } break; case 'buildLogId': // The name of this resource is slightly dated, but this is the release // id from the API. We need to save this so that if the user ctrl+c's the // build we can cancel it on the API. build.releaseId = parseInt(obj.value!, 10); break; } } function getBuilderMessageHandler( build: RemoteBuild, ): (obj: BuilderMessage) => Promise { return async (obj: BuilderMessage) => { if (DEBUG_MODE) { console.log(`[debug] handling message: ${JSON.stringify(obj)}`); } if (obj.type != null && obj.type === 'metadata') { return handleBuilderMetadata(obj, build); } if (obj.message) { const readline = await import('readline'); readline.clearLine(process.stdout, 0); const message = obj.message.replace(TRIM_REGEX, ''); if (obj.replace) { process.stdout.write(`\r${message}`); } else { process.stdout.write(`\r${message}\n`); } } if (obj.isError) { build.hadError = true; } }; } async function cancelBuildIfNecessary(build: RemoteBuild): Promise { if (build.releaseId != null) { await build.sdk.pine.patch({ resource: 'release', id: build.releaseId, body: { status: 'cancelled', end_timestamp: Date.now(), }, }); } } /** * Return a callback function that takes a tar-stream Pack object as argument * and uses it to add the '.balena/registry-secrets.json' metadata file that * contains usernames and passwords to private docker registries. The builder * will remove the file from the tar stream and use the secrets to pull base * images from users' private registries. * @param registrySecrets JS object containing registry usernames and passwords * @returns A callback function, or undefined if registrySecrets is empty */ function getTarStreamCallbackForRegistrySecrets( registrySecrets: RegistrySecrets, ): ((pack: Pack) => void) | undefined { if (Object.keys(registrySecrets).length > 0) { return (pack: Pack) => { pack.entry( { name: '.balena/registry-secrets.json' }, JSON.stringify(registrySecrets), ); }; } } async function getRequestStream(build: RemoteBuild): Promise { const path = await import('path'); const visuals = await import('resin-cli-visuals'); const zlib = await import('zlib'); const tarSpinner = new visuals.Spinner('Packaging the project source...'); tarSpinner.start(); // Tar the directory so that we can send it to the builder const tarStream = await tarDirectory( path.resolve(build.source), getTarStreamCallbackForRegistrySecrets(build.opts.registrySecrets), ); tarSpinner.stop(); const url = await getBuilderEndpoint( build.baseUrl, build.owner, build.app, build.opts, ); if (DEBUG_MODE) { console.log(`[debug] Connecting to builder at ${url}`); } const post = request.post({ url, auth: { bearer: build.auth, }, headers: { 'Content-Encoding': 'gzip', }, body: tarStream.pipe( zlib.createGzip({ level: 6, }), ), }); const uploadSpinner = new visuals.Spinner( 'Uploading source package to balena cloud', ); uploadSpinner.start(); const parseStream = post.pipe(JSONStream.parse('*')); parseStream.on('data', () => uploadSpinner.stop()); return parseStream as Stream.Duplex; }