import * as fs from 'fs-extra'; const fetch = require('node-fetch'); const execa = require('execa'); const { splitCommandString } = require('@joplin/lib/string-utils'); const moment = require('moment'); function quotePath(path: string) { if (!path) return ''; if (path.indexOf('"') < 0 && path.indexOf(' ') < 0) return path; path = path.replace(/"/, '\\"'); return `"${path}"`; } function commandToString(commandName: string, args: string[] = []) { const output = [quotePath(commandName)]; for (const arg of args) { output.push(quotePath(arg)); } return output.join(' '); } async function insertChangelog(tag: string, changelogPath: string, changelog: string, isPrerelease: boolean) { const currentText = await fs.readFile(changelogPath, 'UTF-8'); const lines = currentText.split('\n'); const beforeLines = []; const afterLines = []; for (const line of lines) { if (afterLines.length) { afterLines.push(line); continue; } if (line.indexOf('##') === 0) { afterLines.push(line); continue; } beforeLines.push(line); } const header = [ '##', `[${tag}](https://github.com/laurent22/joplin/releases/tag/${tag})`, ]; if (isPrerelease) header.push('(Pre-release)'); header.push('-'); // eslint-disable-next-line no-useless-escape header.push(`${moment.utc().format('YYYY-MM-DD\THH:mm:ss')}Z`); let newLines = []; newLines.push(header.join(' ')); newLines.push(''); newLines = newLines.concat(changelog.split('\n')); newLines.push(''); const output = beforeLines.concat(newLines).concat(afterLines); return output.join('\n'); } export async function completeReleaseWithChangelog(changelogPath: string, newVersion: string, newTag: string, appName: string, isPreRelease: boolean) { const changelog = (await execCommand2(`node ${rootDir}/packages/tools/git-changelog ${newTag}`, { })).trim(); const newChangelog = await insertChangelog(newTag, changelogPath, changelog, isPreRelease); await fs.writeFile(changelogPath, newChangelog); const finalCmds = [ 'git pull', 'git add -A', `git commit -m "${appName} ${newVersion}"`, `git tag "${newTag}"`, 'git push', 'git push --tags', ]; console.info(''); console.info('Verify that the changelog is correct:'); console.info(''); console.info(`${process.env.EDITOR} "${changelogPath}"`); console.info(''); console.info('Then run these commands:'); console.info(''); console.info(finalCmds.join(' && ')); } async function loadGitHubUsernameCache() { const path = `${__dirname}/github_username_cache.json`; if (await fs.pathExists(path)) { const jsonString = await fs.readFile(path, 'utf8'); return JSON.parse(jsonString); } return {}; } async function saveGitHubUsernameCache(cache: any) { const path = `${__dirname}/github_username_cache.json`; await fs.writeFile(path, JSON.stringify(cache)); } // Returns the project root dir export const rootDir = require('path').dirname(require('path').dirname(__dirname)); export function execCommand(command: string, options: any = null) { options = options || {}; const exec = require('child_process').exec; return new Promise((resolve, reject) => { exec(command, options, (error: any, stdout: any, stderr: any) => { if (error) { if (error.signal == 'SIGTERM') { resolve('Process was killed'); } else { reject(error); } } else { resolve([stdout.trim(), stderr.trim()].join('\n')); } }); }); } export function resolveRelativePathWithinDir(baseDir: string, ...relativePath: string[]) { const path = require('path'); const resolvedBaseDir = path.resolve(baseDir); const resolvedPath = path.resolve(baseDir, ...relativePath); if (resolvedPath.indexOf(resolvedBaseDir) !== 0) throw new Error(`Resolved path for relative path "${JSON.stringify(relativePath)}" is not within base directory "${baseDir}" (Was resolved to ${resolvedPath})`); return resolvedPath; } export function execCommandVerbose(commandName: string, args: string[] = []) { console.info(`> ${commandToString(commandName, args)}`); const promise = execa(commandName, args); promise.stdout.pipe(process.stdout); return promise; } interface ExecCommandOptions { showInput?: boolean; showOutput?: boolean; quiet?: boolean; } // There's lot of execCommandXXX functions, but eventually all scripts should // use the one below, which supports: // // - Printing the command being executed // - Printing the output in real time (piping to stdout) // - Returning the command result as string export async function execCommand2(command: string | string[], options: ExecCommandOptions = null): Promise { options = { showInput: true, showOutput: true, quiet: false, ...options, }; if (options.quiet) { options.showInput = false; options.showOutput = false; } if (options.showInput) { if (typeof command === 'string') { console.info(`> ${command}`); } else { console.info(`> ${commandToString(command[0], command.slice(1))}`); } } const args: string[] = typeof command === 'string' ? splitCommandString(command) : command as string[]; const executableName = args[0]; args.splice(0, 1); const promise = execa(executableName, args); if (options.showOutput) promise.stdout.pipe(process.stdout); const result = await promise; return result.stdout.trim(); } export function execCommandWithPipes(executable: string, args: string[]) { const spawn = require('child_process').spawn; return new Promise((resolve, reject) => { const child = spawn(executable, args, { stdio: 'inherit' }); child.on('error', (error: any) => { reject(error); }); child.on('close', (code: any) => { if (code !== 0) { reject(`Ended with code ${code}`); } else { resolve(null); } }); }); } export function toSystemSlashes(path: string) { const os = process.platform; if (os === 'win32') return path.replace(/\//g, '\\'); return path.replace(/\\/g, '/'); } export async function setPackagePrivateField(filePath: string, value: any) { const text = await fs.readFile(filePath, 'utf8'); const obj = JSON.parse(text); if (!value) { delete obj.private; } else { obj.private = true; } await fs.writeFile(filePath, JSON.stringify(obj, null, 2), 'utf8'); } export async function credentialDir() { const username = require('os').userInfo().username; const toTry = [ `c:/Users/${username}/joplin-credentials`, `/mnt/c/Users/${username}/joplin-credentials`, `/home/${username}/joplin-credentials`, `/Users/${username}/joplin-credentials`, ]; for (const dirPath of toTry) { if (await fs.pathExists(dirPath)) return dirPath; } throw new Error(`Could not find credential directory in any of these paths: ${JSON.stringify(toTry)}`); } export async function credentialFile(filename: string) { const rootDir = await credentialDir(); const output = `${rootDir}/${filename}`; if (!(await fs.pathExists(output))) throw new Error(`No such file: ${output}`); return output; } export async function readCredentialFile(filename: string) { const filePath = await credentialFile(filename); const r = await fs.readFile(filePath); return r.toString(); } export async function downloadFile(url: string, targetPath: string) { const https = require('https'); const fs = require('fs'); return new Promise((resolve, reject) => { const file = fs.createWriteStream(targetPath); https.get(url, function(response: any) { if (response.statusCode !== 200) reject(new Error(`HTTP error ${response.statusCode}`)); response.pipe(file); file.on('finish', function() { // file.close(); resolve(null); }); }).on('error', (error: any) => { reject(error); }); }); } export function fileSha256(filePath: string) { return new Promise((resolve, reject) => { const crypto = require('crypto'); const fs = require('fs'); const algo = 'sha256'; const shasum = crypto.createHash(algo); const s = fs.ReadStream(filePath); s.on('data', function(d: any) { shasum.update(d); }); s.on('end', function() { const d = shasum.digest('hex'); resolve(d); }); s.on('error', function(error: any) { reject(error); }); }); } export async function unlinkForce(filePath: string) { const fs = require('fs-extra'); try { await fs.unlink(filePath); } catch (error) { if (error.code === 'ENOENT') return; throw error; } } export function fileExists(filePath: string) { const fs = require('fs-extra'); return new Promise((resolve, reject) => { fs.stat(filePath, function(err: any) { if (err == null) { resolve(true); } else if (err.code == 'ENOENT') { resolve(false); } else { reject(err); } }); }); } export async function gitRepoClean(): Promise { const output = await execCommand2('git status --porcelain', { quiet: true }); return !output.trim(); } export async function gitRepoCleanTry() { if (!(await gitRepoClean())) throw new Error(`There are pending changes in the repository: ${process.cwd()}`); } export async function gitPullTry(ignoreIfNotBranch = true) { try { await execCommand('git pull'); } catch (error) { if (ignoreIfNotBranch && error.message.includes('no tracking information for the current branch')) { console.info('Skipping git pull because no tracking information on current branch'); } else { throw error; } } } export async function githubUsername(email: string, name: string) { const cache = await loadGitHubUsernameCache(); const cacheKey = `${email}:${name}`; if (cacheKey in cache) return cache[cacheKey]; let output = null; const oauthToken = await githubOauthToken(); const urlsToTry = [ `https://api.github.com/search/users?q=${encodeURI(email)}+in:email`, `https://api.github.com/search/users?q=user:${encodeURI(name)}`, ]; for (const url of urlsToTry) { const response = await fetch(url, { method: 'GET', headers: { 'Content-Type': 'application/json', 'Authorization': `token ${oauthToken}`, }, }); const responseText = await response.text(); if (!response.ok) continue; const responseJson = JSON.parse(responseText); if (!responseJson || !responseJson.items || responseJson.items.length !== 1) continue; output = responseJson.items[0].login; break; } cache[cacheKey] = output; await saveGitHubUsernameCache(cache); return output; } export function patreonOauthToken() { return readCredentialFile('patreon_oauth_token.txt'); } export function githubOauthToken() { return readCredentialFile('github_oauth_token.txt'); } export async function githubRelease(project: string, tagName: string, options: any = null) { options = Object.assign({}, { isDraft: false, isPreRelease: false, }, options); const oauthToken = await githubOauthToken(); const response = await fetch(`https://api.github.com/repos/laurent22/${project}/releases`, { method: 'POST', body: JSON.stringify({ tag_name: tagName, name: tagName, draft: options.isDraft, prerelease: options.isPreRelease, }), headers: { 'Content-Type': 'application/json', 'Authorization': `token ${oauthToken}`, }, }); const responseText = await response.text(); if (!response.ok) throw new Error(`Cannot create GitHub release: ${responseText}`); const responseJson = JSON.parse(responseText); if (!responseJson.url) throw new Error(`No URL for release: ${responseText}`); return responseJson; } export function readline(question: string) { return new Promise((resolve) => { const readline = require('readline'); const rl = readline.createInterface({ input: process.stdin, output: process.stdout, }); rl.question(`${question} `, (answer: string) => { resolve(answer); rl.close(); }); }); } export function isLinux() { return process && process.platform === 'linux'; } export function isWindows() { return process && process.platform === 'win32'; } export function isMac() { return process && process.platform === 'darwin'; } export async function insertContentIntoFile(filePath: string, markerOpen: string, markerClose: string, contentToInsert: string) { const fs = require('fs-extra'); let content = await fs.readFile(filePath, 'utf-8'); // [^]* matches any character including new lines const regex = new RegExp(`${markerOpen}[^]*?${markerClose}`); content = content.replace(regex, markerOpen + contentToInsert + markerClose); await fs.writeFile(filePath, content); } export function dirname(path: string) { if (!path) throw new Error('Path is empty'); const s = path.split(/\/|\\/); s.pop(); return s.join('/'); } export function basename(path: string) { if (!path) throw new Error('Path is empty'); const s = path.split(/\/|\\/); return s[s.length - 1]; } export function filename(path: string, includeDir = false) { if (!path) throw new Error('Path is empty'); const output = includeDir ? path : basename(path); if (output.indexOf('.') < 0) return output; const splitted = output.split('.'); splitted.pop(); return splitted.join('.'); } export function fileExtension(path: string) { if (!path) throw new Error('Path is empty'); const output = path.split('.'); if (output.length <= 1) return ''; return output[output.length - 1]; }