const fs = require('fs-extra'); const { execCommandVerbose, execCommandWithPipes, githubRelease, githubOauthToken, fileExists } = require('./tool-utils.js'); const path = require('path'); const fetch = require('node-fetch'); const uriTemplate = require('uri-template'); const projectName = 'joplin-android'; const rootDir = path.dirname(path.dirname(__dirname)); const rnDir = `${rootDir}/packages/app-mobile`; const releaseDir = `${rnDir}/dist`; function increaseGradleVersionCode(content) { const newContent = content.replace(/versionCode\s+(\d+)/, function(a, versionCode) { const n = Number(versionCode); if (isNaN(n) || !n) throw new Error(`Invalid version code: ${versionCode}`); return `versionCode ${n + 1}`; }); if (newContent === content) throw new Error('Could not update version code'); return newContent; } function increaseGradleVersionName(content) { const newContent = content.replace(/(versionName\s+"\d+?\.\d+?\.)(\d+)"/, function(match, prefix, buildNum) { const n = Number(buildNum); if (isNaN(n)) throw new Error(`Invalid version code: ${buildNum}`); return `${prefix + (n + 1)}"`; }); if (newContent === content) throw new Error('Could not update version name'); return newContent; } function updateGradleConfig() { let content = fs.readFileSync(`${rnDir}/android/app/build.gradle`, 'utf8'); content = increaseGradleVersionCode(content); content = increaseGradleVersionName(content); fs.writeFileSync(`${rnDir}/android/app/build.gradle`, content); return content; } function gradleVersionName(content) { const matches = content.match(/versionName\s+"(\d+?\.\d+?\.\d+)"/); if (!matches || matches.length < 1) throw new Error('Cannot get gradle version name'); return matches[1]; } async function createRelease(name, tagName, version) { const originalContents = {}; const suffix = version + (name === 'main' ? '' : `-${name}`); console.info(`Creating release: ${suffix}`); if (name === '32bit') { const filename = `${rnDir}/android/app/build.gradle`; let content = await fs.readFile(filename, 'utf8'); originalContents[filename] = content; content = content.replace(/abiFilters "armeabi-v7a", "x86", "arm64-v8a", "x86_64"/, 'abiFilters "armeabi-v7a", "x86"'); content = content.replace(/include "armeabi-v7a", "x86", "arm64-v8a", "x86_64"/, 'include "armeabi-v7a", "x86"'); await fs.writeFile(filename, content); } const apkFilename = `joplin-v${suffix}.apk`; const apkFilePath = `${releaseDir}/${apkFilename}`; const downloadUrl = `https://github.com/laurent22/${projectName}/releases/download/${tagName}/${apkFilename}`; process.chdir(rootDir); console.info(`Running from: ${process.cwd()}`); console.info(`Building APK file v${suffix}...`); let restoreDir = null; let apkBuildCmd = ''; const apkBuildCmdArgs = ['assembleRelease', '-PbuildDir=build']; if (await fileExists('/mnt/c/Windows/System32/cmd.exe')) { // In recent versions (of Gradle? React Native?), running gradlew.bat from WSL throws the following error: // Error: Command failed: /mnt/c/Windows/System32/cmd.exe /c "cd packages\app-mobile\android && gradlew.bat assembleRelease -PbuildDir=build" // FAILURE: Build failed with an exception. // * What went wrong: // Could not determine if Stdout is a console: could not get handle file information (errno 1) // So we need to manually run the command from DOS, and then coming back here to finish the process once it's done. // console.info('Run this command from DOS:'); // console.info(''); // console.info(`cd "${wslToWinPath(rootDir)}\\packages\\app-mobile\\android" && gradlew.bat ${apkBuildCmd}"`); // console.info(''); // await readline('Press Enter when done:'); // apkBuildCmd = ''; // Clear the command because we've already ran it // process.chdir(`${rnDir}/android`); // apkBuildCmd = `/mnt/c/Windows/System32/cmd.exe /c "cd packages\\app-mobile\\android && gradlew.bat ${apkBuildCmd}"`; // restoreDir = rootDir; // apkBuildCmd = `/mnt/c/Windows/System32/cmd.exe /c "cd packages\\app-mobile\\android && gradlew.bat ${apkBuildCmd}"`; await execCommandWithPipes('/mnt/c/Windows/System32/cmd.exe', ['/c', `cd packages\\app-mobile\\android && gradlew.bat ${apkBuildCmd}`]); apkBuildCmd = ''; } else { process.chdir(`${rnDir}/android`); apkBuildCmd = './gradlew'; restoreDir = rootDir; } if (apkBuildCmd) { await execCommandVerbose(apkBuildCmd, apkBuildCmdArgs); } if (restoreDir) process.chdir(restoreDir); await fs.mkdirp(releaseDir); console.info(`Copying APK to ${apkFilePath}`); await fs.copy(`${rnDir}/android/app/build/outputs/apk/release/app-release.apk`, apkFilePath); if (name === 'main') { console.info(`Copying APK to ${releaseDir}/joplin-latest.apk`); await fs.copy(`${rnDir}/android/app/build/outputs/apk/release/app-release.apk`, `${releaseDir}/joplin-latest.apk`); } for (const filename in originalContents) { const content = originalContents[filename]; await fs.writeFile(filename, content); } return { downloadUrl: downloadUrl, apkFilename: apkFilename, apkFilePath: apkFilePath, }; } async function main() { const argv = require('yargs').argv; if (!['release', 'prerelease'].includes(argv.type)) throw new Error('Must specify release type. Either --type=release or --type=prerelease'); const isPreRelease = argv.type === 'prerelease'; if (isPreRelease) console.info('Creating pre-release'); console.info('Updating version numbers in build.gradle...'); const newContent = updateGradleConfig(); const version = gradleVersionName(newContent); const tagName = `android-v${version}`; const releaseNames = ['main', '32bit']; const releaseFiles = {}; for (const releaseName of releaseNames) { releaseFiles[releaseName] = await createRelease(releaseName, tagName, version); } if (!isPreRelease) { console.info('Updating Readme URL...'); let readmeContent = await fs.readFile(`${rootDir}/README.md`, 'utf8'); readmeContent = readmeContent.replace(/(https:\/\/github.com\/laurent22\/joplin-android\/releases\/download\/android-v\d+\.\d+\.\d+\/joplin-v\d+\.\d+\.\d+\.apk)/, releaseFiles['main'].downloadUrl); readmeContent = readmeContent.replace(/(https:\/\/github.com\/laurent22\/joplin-android\/releases\/download\/android-v\d+\.\d+\.\d+\/joplin-v\d+\.\d+\.\d+-32bit\.apk)/, releaseFiles['32bit'].downloadUrl); await fs.writeFile(`${rootDir}/README.md`, readmeContent); } await execCommandVerbose('git', ['pull']); await execCommandVerbose('git', ['add', '-A']); await execCommandVerbose('git', ['commit', '-m', `Android release v${version}`]); await execCommandVerbose('git', ['tag', tagName]); await execCommandVerbose('git', ['push']); await execCommandVerbose('git', ['push', '--tags']); console.info(`Creating GitHub release ${tagName}...`); const releaseOptions = { isPreRelease: isPreRelease }; const oauthToken = await githubOauthToken(); const release = await githubRelease(projectName, tagName, releaseOptions); const uploadUrlTemplate = uriTemplate.parse(release.upload_url); for (const releaseFilename in releaseFiles) { const releaseFile = releaseFiles[releaseFilename]; const uploadUrl = uploadUrlTemplate.expand({ name: releaseFile.apkFilename }); const binaryBody = await fs.readFile(releaseFile.apkFilePath); console.info(`Uploading ${releaseFile.apkFilename} to ${uploadUrl}`); const uploadResponse = await fetch(uploadUrl, { method: 'POST', body: binaryBody, headers: { 'Content-Type': 'application/vnd.android.package-archive', 'Authorization': `token ${oauthToken}`, 'Content-Length': binaryBody.length, }, }); const uploadResponseText = await uploadResponse.text(); const uploadResponseObject = JSON.parse(uploadResponseText); if (!uploadResponseObject || !uploadResponseObject.browser_download_url) throw new Error('Could not upload file to GitHub'); } console.info(`Main download URL: ${releaseFiles['main'].downloadUrl}`); } main().catch((error) => { console.error('Fatal error'); console.error(error); process.exit(1); });