2018-03-09 22:59:12 +02:00
|
|
|
const fs = require('fs-extra');
|
2019-06-15 19:58:09 +02:00
|
|
|
const { execCommand, githubRelease, githubOauthToken, isWindows, fileExists, readline } = require('./tool-utils.js');
|
2018-03-09 22:59:12 +02:00
|
|
|
const path = require('path');
|
|
|
|
const fetch = require('node-fetch');
|
|
|
|
const uriTemplate = require('uri-template');
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:19:41 +02:00
|
|
|
const projectName = 'joplin-android';
|
2018-03-09 22:59:12 +02:00
|
|
|
const rnDir = __dirname + '/../ReactNativeClient';
|
2018-01-17 22:16:13 +02:00
|
|
|
const rootDir = path.dirname(__dirname);
|
2018-03-09 22:59:12 +02:00
|
|
|
const releaseDir = rootDir + '/_releases';
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-06-15 19:58:09 +02:00
|
|
|
function wslToWinPath(wslPath) {
|
|
|
|
const s = wslPath.split('/');
|
|
|
|
if (s.length < 3) return s.join('\\');
|
|
|
|
s.splice(0, 1);
|
|
|
|
if (s[0] !== 'mnt' || s[1].length !== 1) return s.join('\\');
|
|
|
|
s.splice(0, 1);
|
|
|
|
s[0] = s[0].toUpperCase() + ':';
|
|
|
|
while (s.length && !s[s.length - 1]) s.pop();
|
|
|
|
return s.join('\\');
|
|
|
|
}
|
|
|
|
|
2018-01-17 22:16:13 +02:00
|
|
|
function increaseGradleVersionCode(content) {
|
|
|
|
const newContent = content.replace(/versionCode\s+(\d+)/, function(a, versionCode, c) {
|
|
|
|
const n = Number(versionCode);
|
2018-03-09 22:59:12 +02:00
|
|
|
if (isNaN(n) || !n) throw new Error('Invalid version code: ' + versionCode);
|
|
|
|
return 'versionCode ' + (n + 1);
|
2018-01-17 22:16:13 +02:00
|
|
|
});
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
if (newContent === content) throw new Error('Could not update version code');
|
2018-01-17 22:16:13 +02:00
|
|
|
|
|
|
|
return newContent;
|
|
|
|
}
|
|
|
|
|
|
|
|
function increaseGradleVersionName(content) {
|
|
|
|
const newContent = content.replace(/(versionName\s+"\d+?\.\d+?\.)(\d+)"/, function(match, prefix, buildNum) {
|
|
|
|
const n = Number(buildNum);
|
2018-03-09 22:59:12 +02:00
|
|
|
if (isNaN(n) || !n) throw new Error('Invalid version code: ' + versionCode);
|
2018-01-17 22:16:13 +02:00
|
|
|
return prefix + (n + 1) + '"';
|
|
|
|
});
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
if (newContent === content) throw new Error('Could not update version name');
|
2018-01-17 22:16:13 +02:00
|
|
|
|
|
|
|
return newContent;
|
|
|
|
}
|
|
|
|
|
|
|
|
function updateGradleConfig() {
|
2018-03-09 22:59:12 +02:00
|
|
|
let content = fs.readFileSync(rnDir + '/android/app/build.gradle', 'utf8');
|
2018-01-17 22:16:13 +02:00
|
|
|
content = increaseGradleVersionCode(content);
|
|
|
|
content = increaseGradleVersionName(content);
|
2018-03-09 22:59:12 +02:00
|
|
|
fs.writeFileSync(rnDir + '/android/app/build.gradle', content);
|
2018-01-17 22:16:13 +02:00
|
|
|
return content;
|
|
|
|
}
|
|
|
|
|
|
|
|
function gradleVersionName(content) {
|
|
|
|
const matches = content.match(/versionName\s+"(\d+?\.\d+?\.\d+)"/);
|
2018-03-09 22:59:12 +02:00
|
|
|
if (!matches || matches.length < 1) throw new Error('Cannot get gradle version name');
|
2018-01-17 22:16:13 +02:00
|
|
|
return matches[1];
|
|
|
|
}
|
|
|
|
|
2019-07-28 18:19:41 +02:00
|
|
|
async function createRelease(name, tagName, version) {
|
|
|
|
const originalContents = {};
|
2019-07-28 18:29:07 +02:00
|
|
|
const suffix = version + (name === 'main' ? '' : '-' + name);
|
2018-12-29 04:12:23 +02:00
|
|
|
|
2019-07-28 18:19:41 +02:00
|
|
|
console.info('Creating release: ' + suffix);
|
|
|
|
|
|
|
|
if (name === '32bit') {
|
|
|
|
let 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';
|
2018-03-09 22:59:12 +02:00
|
|
|
const apkFilePath = releaseDir + '/' + apkFilename;
|
|
|
|
const downloadUrl = 'https://github.com/laurent22/' + projectName + '/releases/download/' + tagName + '/' + apkFilename;
|
2018-01-17 22:16:13 +02:00
|
|
|
|
|
|
|
process.chdir(rootDir);
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
console.info('Running from: ' + process.cwd());
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:19:41 +02:00
|
|
|
console.info('Building APK file v' + suffix + '...');
|
2018-10-13 00:25:11 +02:00
|
|
|
|
|
|
|
let restoreDir = null;
|
2019-06-15 19:58:09 +02:00
|
|
|
let apkBuildCmd = 'assembleRelease -PbuildDir=build';
|
2019-06-15 10:44:34 +02:00
|
|
|
if (await fileExists('/mnt/c/Windows/System32/cmd.exe')) {
|
2019-06-15 19:58:09 +02:00
|
|
|
// 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 ReactNativeClient\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) + '\\ReactNativeClient\\android" && gradlew.bat ' + apkBuildCmd + '"');
|
|
|
|
console.info('');
|
2019-06-19 16:04:46 +02:00
|
|
|
await readline('Press Enter when done:');
|
2019-06-15 19:58:09 +02:00
|
|
|
apkBuildCmd = ''; // Clear the command because we've already ran it
|
2019-06-15 10:44:34 +02:00
|
|
|
} else {
|
|
|
|
process.chdir(rnDir + '/android');
|
|
|
|
apkBuildCmd = './gradlew ' + apkBuildCmd;
|
|
|
|
restoreDir = rootDir;
|
|
|
|
}
|
|
|
|
|
2019-06-15 19:58:09 +02:00
|
|
|
if (apkBuildCmd) {
|
|
|
|
console.info(apkBuildCmd);
|
|
|
|
const output = await execCommand(apkBuildCmd);
|
|
|
|
console.info(output);
|
|
|
|
}
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2018-10-13 00:25:11 +02:00
|
|
|
if (restoreDir) process.chdir(restoreDir);
|
|
|
|
|
2018-01-17 22:16:13 +02:00
|
|
|
await fs.mkdirp(releaseDir);
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
console.info('Copying APK to ' + apkFilePath);
|
2018-10-02 19:19:27 +02:00
|
|
|
await fs.copy('ReactNativeClient/android/app/build/outputs/apk/release/app-release.apk', apkFilePath);
|
2019-07-28 18:29:07 +02:00
|
|
|
|
|
|
|
if (name === 'main') {
|
|
|
|
console.info('Copying APK to ' + releaseDir + '/joplin-latest.apk');
|
|
|
|
await fs.copy('ReactNativeClient/android/app/build/outputs/apk/release/app-release.apk', releaseDir + '/joplin-latest.apk');
|
|
|
|
}
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:19:41 +02:00
|
|
|
for (let filename in originalContents) {
|
|
|
|
const content = originalContents[filename];
|
|
|
|
await fs.writeFile(filename, content);
|
|
|
|
}
|
|
|
|
|
|
|
|
return {
|
|
|
|
downloadUrl: downloadUrl,
|
2019-07-28 18:31:31 +02:00
|
|
|
apkFilename: apkFilename,
|
2019-07-28 18:33:48 +02:00
|
|
|
apkFilePath: apkFilePath,
|
2019-07-28 18:19:41 +02:00
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
async function main() {
|
|
|
|
console.info('Updating version numbers in build.gradle...');
|
|
|
|
|
|
|
|
const newContent = updateGradleConfig();
|
|
|
|
const version = gradleVersionName(newContent);
|
|
|
|
const tagName = 'android-v' + version;
|
2019-07-28 18:29:07 +02:00
|
|
|
const releaseNames = ['main', '32bit'];
|
|
|
|
const releaseFiles = {};
|
2019-07-28 18:19:41 +02:00
|
|
|
|
2019-07-28 18:29:07 +02:00
|
|
|
for (const releaseName of releaseNames) {
|
|
|
|
releaseFiles[releaseName] = await createRelease(releaseName, tagName, version);
|
|
|
|
}
|
2019-07-28 18:19:41 +02:00
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
console.info('Updating Readme URL...');
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
let readmeContent = await fs.readFile('README.md', 'utf8');
|
2019-07-28 18:29:07 +02:00
|
|
|
readmeContent = readmeContent.replace(/(https:\/\/github.com\/laurent22\/joplin-android\/releases\/download\/.*?\.apk)/, releaseFiles['main'].downloadUrl);
|
2018-03-09 22:59:12 +02:00
|
|
|
await fs.writeFile('README.md', readmeContent);
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
console.info(await execCommand('git pull'));
|
|
|
|
console.info(await execCommand('git add -A'));
|
2018-01-19 00:43:37 +02:00
|
|
|
console.info(await execCommand('git commit -m "Android release v' + version + '"'));
|
2018-03-09 22:59:12 +02:00
|
|
|
console.info(await execCommand('git tag ' + tagName));
|
|
|
|
console.info(await execCommand('git push'));
|
|
|
|
console.info(await execCommand('git push --tags'));
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
console.info('Creating GitHub release ' + tagName + '...');
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:33:48 +02:00
|
|
|
const oauthToken = await githubOauthToken();
|
2019-07-28 18:36:36 +02:00
|
|
|
const release = await githubRelease(projectName, tagName);
|
|
|
|
const uploadUrlTemplate = uriTemplate.parse(release.upload_url);
|
2019-07-28 18:33:48 +02:00
|
|
|
|
2019-07-28 18:31:31 +02:00
|
|
|
for (const releaseFilename in releaseFiles) {
|
|
|
|
const releaseFile = releaseFiles[releaseFilename];
|
|
|
|
const uploadUrl = uploadUrlTemplate.expand({ name: releaseFile.apkFilename });
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:33:48 +02:00
|
|
|
const binaryBody = await fs.readFile(releaseFile.apkFilePath);
|
2018-02-04 19:42:33 +02:00
|
|
|
|
2019-07-28 18:31:31 +02:00
|
|
|
console.info('Uploading ' + releaseFile.apkFilename + ' to ' + uploadUrl);
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:29:07 +02:00
|
|
|
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();
|
|
|
|
console.info(uploadResponseText);
|
|
|
|
}
|
2018-01-17 22:16:13 +02:00
|
|
|
|
2019-07-28 18:29:07 +02:00
|
|
|
console.info('Download URL: ' + releaseFiles['main'].downloadUrl);
|
2018-01-17 22:16:13 +02:00
|
|
|
}
|
|
|
|
|
2018-03-09 22:59:12 +02:00
|
|
|
main().catch((error) => {
|
|
|
|
console.error('Fatal error');
|
2018-01-17 22:16:13 +02:00
|
|
|
console.error(error);
|
2018-03-23 19:32:29 +02:00
|
|
|
process.exit(1);
|
2018-03-09 22:59:12 +02:00
|
|
|
});
|