booletwa
Version:
Generate TWA projects from a Web Manifest
236 lines (211 loc) • 8.8 kB
text/typescript
/*
* Copyright 2019 Google Inc. All Rights Reserved.
*
* 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 {AndroidSdkTools, Config, GradleWrapper, JdkHelper, KeyTool, Log,
ConsoleLog, TwaManifest, JarSigner, SigningKeyInfo} from '@bubblewrap/core';
import * as fs from 'fs';
import * as path from 'path';
import {enUS as messages} from '../strings';
import {Prompt, InquirerPrompt} from '../Prompt';
import {ParsedArgs} from 'minimist';
import {createValidateString} from '../inputHelpers';
import {computeChecksum, updateProject} from './shared';
import {TWA_MANIFEST_FILE_NAME} from '../constants';
// Path to the file generated when building an app bundle file using gradle.
const APP_BUNDLE_BUILD_OUTPUT_FILE_NAME = './app/build/outputs/bundle/release/app-release.aab';
const APP_BUNDLE_SIGNED_FILE_NAME = './app-release-bundle.aab'; // Final signed App Bundle file.
// Path to the file generated when building an APK file using gradle.
const APK_BUILD_OUTPUT_FILE_NAME = './app/build/outputs/apk/release/app-release-unsigned.apk';
// Final aligned and signed APK.
const APK_SIGNED_FILE_NAME = './app-release-signed.apk';
// Output file for zipalign.
const APK_ALIGNED_FILE_NAME = './app-release-unsigned-aligned.apk';
interface SigningKeyPasswords {
keystorePassword: string;
keyPassword: string;
}
class Build {
constructor(
private args: ParsedArgs,
private androidSdkTools: AndroidSdkTools,
private keyTool: KeyTool,
private gradleWrapper: GradleWrapper,
private jarSigner: JarSigner,
private log: Log = new ConsoleLog('build'),
private prompt: Prompt = new InquirerPrompt()) {
}
/**
* Checks if the twa-manifest.json file has been changed since the last time the project was generated.
*/
async hasManifestChanged(manifestFile: string): Promise<boolean> {
const targetDirectory = this.args.directory || process.cwd();
const checksumFile = path.join(targetDirectory, 'manifest-checksum.txt');
const prevChecksum = (await fs.promises.readFile(checksumFile)).toString();
const manifestContents = await fs.promises.readFile(manifestFile);
const currChecksum = computeChecksum(manifestContents);
return currChecksum != prevChecksum;
}
/**
* Checks if the keystore password and the key password are part of the environment prompts the
* user for a password otherwise.
*
* @returns {Promise<SigningKeyPasswords} the password information collected from enviromental
* variables or user input.
*/
async getPasswords(signingKeyInfo: SigningKeyInfo): Promise<SigningKeyPasswords> {
// Check if passwords are set as environment variables.
const envKeystorePass = process.env['BUBBLEWRAP_KEYSTORE_PASSWORD'];
const envKeyPass = process.env['BUBBLEWRAP_KEY_PASSWORD'];
if (envKeyPass !== undefined && envKeystorePass !== undefined) {
this.prompt.printMessage(messages.messageUsingPasswordsFromEnv);
return {
keystorePassword: envKeystorePass,
keyPassword: envKeyPass,
};
}
// Ask user for the keystore password
this.prompt.printMessage(
messages.messageEnterPasswords(signingKeyInfo.path, signingKeyInfo.alias));
const keystorePassword =
await this.prompt.promptPassword(messages.promptKeystorePassword, createValidateString(6));
const keyPassword =
await this.prompt.promptPassword(messages.promptKeyPassword, createValidateString(6));
return {
keystorePassword: keystorePassword,
keyPassword: keyPassword,
};
}
async buildApk(): Promise<void> {
await this.gradleWrapper.assembleRelease();
await this.androidSdkTools.zipalignOnlyVerification(
APK_BUILD_OUTPUT_FILE_NAME, // input file
);
fs.copyFileSync(APK_BUILD_OUTPUT_FILE_NAME, APK_ALIGNED_FILE_NAME);
}
async signApk(signingKey: SigningKeyInfo, passwords: SigningKeyPasswords): Promise<void> {
await this.androidSdkTools.apksigner(
signingKey.path,
passwords.keystorePassword,
signingKey.alias,
passwords.keyPassword,
APK_ALIGNED_FILE_NAME, // input file path
APK_SIGNED_FILE_NAME,
);
}
async buildAppBundle(): Promise<void> {
await this.gradleWrapper.bundleRelease();
}
async signAppBundle(signingKey: SigningKeyInfo, passwords: SigningKeyPasswords): Promise<void> {
await this.jarSigner.sign(signingKey, passwords.keystorePassword, passwords.keyPassword,
APP_BUNDLE_BUILD_OUTPUT_FILE_NAME, APP_BUNDLE_SIGNED_FILE_NAME);
}
/**
* Based on the promptResponse to update the project or not, run an update or print the relevant warning message.
*
* @returns {Promise<boolean>} whether the appropriate action taken (update project or print warning) was successful
*/
async runUpdate(
promptResponse: boolean,
manifestFile: string,
noUpdateMessage: string): Promise<boolean> {
if (!promptResponse) {
this.prompt.printMessage(noUpdateMessage);
return true;
}
return await updateProject(false, null, this.prompt,
this.args.directory, manifestFile);
}
async build(): Promise<boolean> {
if (!await this.androidSdkTools.checkBuildTools()) {
this.prompt.printMessage(messages.messageInstallingBuildTools);
await this.androidSdkTools.installBuildTools();
}
const manifestFile = this.args.manifest || path.join(process.cwd(), TWA_MANIFEST_FILE_NAME);
const twaManifest = await TwaManifest.fromFile(manifestFile);
const targetDirectory = this.args.directory || process.cwd();
const checksumFile = path.join(targetDirectory, 'manifest-checksum.txt');
let updateSuccessful = true;
if (!fs.existsSync(checksumFile)) {
// If checksum file doesn't exist, prompt the user about updating their project
const applyChanges = await this.prompt.promptConfirm(
messages.messageNoChecksumFileFound,
true);
updateSuccessful = await this.runUpdate(
applyChanges,
manifestFile,
messages.messageNoChecksumNoUpdate);
} else {
const hasManifestChanged = await this.hasManifestChanged(manifestFile);
if (hasManifestChanged) {
const applyChanges = await this.prompt.promptConfirm(messages.promptUpdateProject, true);
updateSuccessful = await this.runUpdate(
applyChanges,
manifestFile,
messages.messageProjectNotUpdated);
}
}
if (!updateSuccessful) {
return false;
}
let passwords = null;
let signingKey = twaManifest.signingKey;
if (!this.args.skipSigning) {
passwords = await this.getPasswords(signingKey);
signingKey = {
...signingKey,
...(this.args.signingKeyPath ? {path: this.args.signingKeyPath} : null),
...(this.args.signingKeyAlias ? {alias: this.args.signingKeyAlias} : null),
};
}
// Builds the Android Studio Project
this.prompt.printMessage(messages.messageBuildingApp);
await this.buildApk();
if (passwords) {
await this.signApk(signingKey, passwords);
}
const apkFileName = this.args.skipSigning ?
APK_ALIGNED_FILE_NAME :
APK_SIGNED_FILE_NAME;
this.prompt.printMessage(messages.messageApkSuccess(apkFileName));
await this.buildAppBundle();
if (passwords) {
await this.signAppBundle(signingKey, passwords);
}
const appBundleFileName = this.args.skipSigning ?
APP_BUNDLE_BUILD_OUTPUT_FILE_NAME :
APP_BUNDLE_SIGNED_FILE_NAME;
this.prompt.printMessage(messages.messageAppBundleSuccess(appBundleFileName));
return true;
}
}
export async function build(config: Config, args: ParsedArgs,
log: Log = new ConsoleLog('build'), prompt: Prompt = new InquirerPrompt()): Promise<boolean> {
const jdkHelper = new JdkHelper(process, config);
const androidSdkTools =
await AndroidSdkTools.create(process, config, jdkHelper, log);
const keyTool = new KeyTool(jdkHelper, log);
const gradleWrapper = new GradleWrapper(process, androidSdkTools);
const jarSigner = new JarSigner(jdkHelper);
const build = new Build(
args,
androidSdkTools,
keyTool,
gradleWrapper,
jarSigner,
log,
prompt,
);
return build.build();
}