1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-01-08 13:06:15 +02:00
joplin/packages/tools/gulp/utils.js

229 lines
6.1 KiB
JavaScript
Raw Normal View History

const fs = require('fs-extra');
2020-11-05 18:58:23 +02:00
const execa = require('execa');
2023-07-25 16:31:44 +02:00
const glob = require('glob');
const utils = {};
utils.isLinux = () => {
return process && process.platform === 'linux';
};
utils.isWindows = () => {
return process && process.platform === 'win32';
};
utils.isMac = () => {
return process && process.platform === 'darwin';
};
2020-11-05 18:58:23 +02:00
utils.execCommandVerbose = function(commandName, args = []) {
console.info(`> ${commandName}`, args && args.length ? args : '');
const promise = execa(commandName, args);
promise.stdout.pipe(process.stdout);
promise.stderr.pipe(process.stderr);
2020-11-05 18:58:23 +02:00
return promise;
};
utils.execCommand = function(command) {
const exec = require('child_process').exec;
return new Promise((resolve, reject) => {
exec(command, { maxBuffer: 1024 * 1024 }, (error, stdout) => {
if (error) {
2020-06-24 17:33:35 +02:00
// Special case for robocopy, which will return non-zero error codes
// when sucessful. Doc is very imprecise but <= 7 seems more or less
// fine and >= 8 seems more errorish. https://ss64.com/nt/robocopy-exit.html
if (command.indexOf('robocopy') === 0 && error.code <= 7) {
resolve(stdout.trim());
return;
}
2022-07-23 11:33:12 +02:00
if (error.signal === 'SIGTERM') {
resolve('Process was killed');
} else {
2020-06-24 17:33:35 +02:00
const newError = new Error(`Code: ${error.code}: ${error.message}: ${stdout.trim()}`);
reject(newError);
}
} else {
resolve(stdout.trim());
}
});
});
};
utils.dirname = function(path) {
if (!path) throw new Error('Path is empty');
const s = path.split(/\/|\\/);
s.pop();
return s.join('/');
};
utils.basename = function(path) {
if (!path) throw new Error('Path is empty');
const s = path.split(/\/|\\/);
return s[s.length - 1];
};
utils.filename = function(path, includeDir = false) {
if (!path) throw new Error('Path is empty');
let output = includeDir ? path : utils.basename(path);
if (output.indexOf('.') < 0) return output;
output = output.split('.');
output.pop();
return output.join('.');
};
utils.toSystemSlashes = function(path) {
if (utils.isWindows()) return path.replace(/\//g, '\\');
return path.replace(/\\/g, '/');
};
utils.fileExtension = function(path) {
if (!path) throw new Error('Path is empty');
const output = path.split('.');
if (output.length <= 1) return '';
return output[output.length - 1];
};
utils.replaceFileText = async function(filePath, regex, toInsert) {
const content = await fs.readFile(filePath, 'utf8');
const newContent = content.replace(regex, toInsert);
if (newContent === content) return Promise.resolve();
await fs.writeFile(filePath, newContent);
};
utils.copyDir = async function(src, dest, options) {
options = { excluded: [],
delete: true, ...options };
src = utils.toSystemSlashes(src);
dest = utils.toSystemSlashes(dest);
await utils.mkdir(dest);
if (utils.isWindows()) {
let cmd = ['robocopy'];
cmd.push(`"${src}"`);
cmd.push(`"${dest}"`);
cmd.push('/e');
cmd.push('/r:0');
if (options.delete) cmd.push('/purge');
if (options.excluded.length) {
cmd.push('/xd');
cmd = cmd.concat(options.excluded.map(p => `"${utils.toSystemSlashes(p)}"`).join(' '));
}
await utils.execCommand(cmd.join(' '));
} else {
let excludedFlag = '';
if (options.excluded.length) {
excludedFlag = options.excluded.map(f => {
return `--exclude "${f}"`;
}).join(' ');
}
Desktop: Resolves #176: Added experimental WYSIWYG editor (#2556) * Trying to get TuiEditor to work * Tests with TinyMCE * Fixed build * Improved asset loading * Added support for Joplin source blocks * Added support for Joplin source blocks * Better integration * Make sure noteDidUpdate event is always dispatched at the right time * Minor tweaks * Fixed tests * Add support for checkboxes * Minor refactoring * Added support for file attachments * Add support for fenced code blocks * Fix new line issue on code block * Added support for Fountain scripts * Refactoring * Better handling of saving and loading notes * Fix saving and loading ntoes * Handle multi-note selection and fixed new note creation issue * Fixed newline issue in test * Fixed newline issue in test * Improve saving and loading * Improve saving and loading note * Removed undeeded prop * Fixed issue when new note being saved is incorrectly reloaded * Refactoring and improve saving of note when unmounting component * Fixed TypeScript error * Small changes * Improved further handling of saving and loading notes * Handle provisional notes and fixed various saving and loading bugs * Adding back support for HTML notes * Added support for HTML notes * Better handling of editable nodes * Preserve image HTML tag when the size is set * Handle switching between editor when the note has note finished saving * Handle templates * Handle templates * Handle loading note that is being saved * Handle note being reloaded via sync * Clean up * Clean up and improved logging * Fixed TS error * Fixed a few issues * Fixed test * Logging * Various improvements * Add blockquote support * Moved CWD operation to shim * Removed deleted files * Added support for Joplin commands
2020-03-10 01:24:57 +02:00
let deleteFlag = '';
if (options.delete) deleteFlag = '--delete';
await utils.execCommand(`rsync -a ${deleteFlag} ${excludedFlag} "${src}/" "${dest}/"`);
}
};
// Occasionally, fs.mkdirp throws a "EEXIST" error if the directory already
// exists, while it should actually ignore the error. So we have this wrapper
// that actually handle the error. It means in general this method should be
// preferred to avoid random failures on CI or when building the app.
//
// https://github.com/laurent22/joplin/issues/6935#issuecomment-1274404470
utils.mkdir = async function(dir) {
if (utils.isWindows()) {
return utils.execCommand(`if not exist "${utils.toSystemSlashes(dir)}" mkdir "${utils.toSystemSlashes(dir)}"`);
} else {
try {
// Can't return right away, or the exception won't be caught
const result = await fs.mkdirp(dir);
return result;
} catch (error) {
// Shouldn't happen but sometimes does. So we ignore the error in
// this case.
if (error.code === 'EEXIST') return;
throw error;
}
}
};
utils.mkdirp = async function(dir) {
return utils.mkdir(dir);
};
utils.copyFile = async function(src, dest) {
await fs.copy(src, dest);
};
utils.rootDir = function() {
2020-11-05 18:58:23 +02:00
return utils.dirname(utils.dirname(utils.dirname(__dirname)));
};
utils.registerGulpTasks = function(gulp, tasks) {
for (const taskName in tasks) {
gulp.task(taskName, tasks[taskName].fn);
}
};
utils.setPackagePrivateField = async function(filePath, value) {
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');
};
utils.insertContentIntoFile = async (filePath, marker, contentToInsert, createIfNotExist = false) => {
const fs = require('fs-extra');
const fileExists = await fs.pathExists(filePath);
if (!fileExists) {
if (!createIfNotExist) throw new Error(`File not found: ${filePath}`);
await fs.writeFile(filePath, `${marker}\n${contentToInsert}\n${marker}`);
} else {
let content = await fs.readFile(filePath, 'utf-8');
// [^]* matches any character including new lines
const regex = new RegExp(`${marker}[^]*?${marker}`);
content = content.replace(regex, `${marker}\n${contentToInsert}\n${marker}`);
await fs.writeFile(filePath, content);
}
};
2021-09-04 16:15:25 +02:00
utils.getFilename = (path) => {
const lastPart = path.split('/').pop();
if (lastPart.indexOf('.') < 0) return lastPart;
const splitted = lastPart.split('.');
splitted.pop();
return splitted.join('.');
};
utils.msleep = (ms) => {
return new Promise((resolve) => {
setTimeout(() => {
resolve();
}, ms);
});
};
2023-07-25 16:31:44 +02:00
utils.globSync = (pattern, options = {}) => {
let output = glob.sync(pattern, options);
output = output.map(f => f.replace(/\\/g, '/'));
output.sort();
return output;
};
module.exports = utils;