1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-11-27 08:21:03 +02:00

Generator: Fixed issue that could result in an invalid JPL file

This commit is contained in:
Laurent Cozic 2021-01-13 12:16:36 +00:00
parent 6cdb16c8ab
commit 3abf21d4ae
4 changed files with 83 additions and 22 deletions

View File

@ -2,7 +2,7 @@
"name": "<%= packageName %>",
"version": "1.0.0",
"scripts": {
"dist": "webpack",
"dist": "webpack --joplin-plugin-config buildMain && webpack --joplin-plugin-config buildExtraScripts && webpack --joplin-plugin-config createArchive",
"prepare": "npm run dist",
"update": "npm install -g generator-joplin && yo joplin --update"
},
@ -21,6 +21,7 @@
"ts-loader": "^7.0.5",
"typescript": "^3.9.3",
"webpack": "^4.43.0",
"webpack-cli": "^3.3.11"
"webpack-cli": "^3.3.11",
"yargs": "^16.2.0"
}
}

View File

@ -33,10 +33,6 @@ const manifest = readManifest(manifestPath);
const pluginArchiveFilePath = path.resolve(publishDir, `${manifest.id}.jpl`);
const pluginInfoFilePath = path.resolve(publishDir, `${manifest.id}.json`);
fs.removeSync(distDir);
fs.removeSync(publishDir);
fs.mkdirpSync(publishDir);
function validatePackageJson() {
const content = JSON.parse(fs.readFileSync(packageJsonPath, 'utf8'));
if (!content.name || content.name.indexOf('joplin-plugin-') !== 0) {
@ -109,6 +105,7 @@ function createPluginInfo(manifestPath, destPath, jplFilePath) {
function onBuildCompleted() {
try {
fs.removeSync(path.resolve(publishDir, 'index.js'));
createPluginArchive(distDir, pluginArchiveFilePath);
createPluginInfo(manifestPath, pluginInfoFilePath, pluginArchiveFilePath);
validatePackageJson();
@ -174,6 +171,16 @@ const extraScriptConfig = Object.assign({}, baseConfig, {
},
});
const createArchiveConfig = {
stats: 'errors-only',
entry: './dist/index.js',
output: {
filename: 'index.js',
path: publishDir,
},
plugins: [new WebpackOnBuildPlugin(onBuildCompleted)],
};
function resolveExtraScriptPath(name) {
const relativePath = `./src/${name}`;
@ -196,8 +203,8 @@ function resolveExtraScriptPath(name) {
};
}
function addExtraScriptConfigs(baseConfig, userConfig) {
if (!userConfig.extraScripts.length) return baseConfig;
function buildExtraScriptConfigs(userConfig) {
if (!userConfig.extraScripts.length) return [];
const output = [];
@ -209,22 +216,55 @@ function addExtraScriptConfigs(baseConfig, userConfig) {
}));
}
return baseConfig.concat(output);
return output;
}
function addLastConfigStep(config) {
const lastConfig = config[config.length - 1];
if (!lastConfig.plugins) lastConfig.plugins = [];
lastConfig.plugins.push(new WebpackOnBuildPlugin(onBuildCompleted));
config[config.length - 1] = lastConfig;
return config;
function main(processArgv) {
const yargs = require('yargs/yargs');
const argv = yargs(processArgv).argv;
const configName = argv['joplin-plugin-config'];
if (!configName) throw new Error('A config file must be specified via the --joplin-plugin-config flag');
// Webpack configurations run in parallel, while we need them to run in
// sequence, and to do that it seems the only way is to run webpack multiple
// times, with different config each time.
const configs = {
// Builds the main src/index.ts and copy the extra content from /src to
// /dist including scripts, CSS and any other asset.
buildMain: pluginConfig,
// Builds the extra scripts as defined in plugin.config.json. When doing
// so, some JavaScript files that were copied in the previous might be
// overwritten here by the compiled version. This is by design. The
// result is that JS files that don't need compilation, are simply
// copied to /dist, while those that do need it are correctly compiled.
buildExtraScripts: buildExtraScriptConfigs(userConfig),
// Ths config is for creating the .jpl, which is done via the plugin, so
// it doesn't actually need an entry and output, however webpack won't
// run without this. So we give it an entry that we know is going to
// exist and output in the publish dir. Then the plugin will delete this
// temporary file before packaging the plugin.
createArchive: createArchiveConfig,
};
// If we are running the first config step, we clean up and create the build
// directories.
if (configName === 'buildMain') {
fs.removeSync(distDir);
fs.removeSync(publishDir);
fs.mkdirpSync(publishDir);
}
return configs[configName];
}
let exportedConfigs = [pluginConfig];
let exportedConfigs = [];
try {
exportedConfigs = addExtraScriptConfigs(exportedConfigs, userConfig);
exportedConfigs = addLastConfigStep(exportedConfigs);
exportedConfigs = main(process.argv);
} catch (error) {
console.error(chalk.red(error.message));
process.exit(1);

View File

@ -26,6 +26,11 @@ function mergePackageKey(parentKey, source, dest) {
} else if (typeof source[k] === 'object' && !Array.isArray(source[k]) && source[k] !== null) {
// If it's an object, recursively process it
output[k] = mergePackageKey(k, source[k], output[k]);
} else if (parentKey === 'scripts' && ['dist', 'prepare', 'update'].includes(k)) {
// When merging scripts, we need to take certain keys from source
// because if they aren't right, the plugin might not build
// correctly.
output[k] = source[k];
} else {
// Otherwise, the default is to preserve the destination key
output[k] = dest[k];
@ -36,12 +41,12 @@ function mergePackageKey(parentKey, source, dest) {
}
function mergeIgnoreFile(source, dest) {
const output = source.split('\n').concat(dest.split('\n'));
const output = dest.trim().split('\n').concat(source.trim().split('\n'));
return output.filter(function(item, pos) {
return `${output.filter(function(item, pos) {
if (!item) return true; // Leave blank lines
return output.indexOf(item) === pos;
}).join('\n');
}).join('\n').trim()}\n`;
}
function packageNameFromPluginName(pluginName) {

View File

@ -1,4 +1,4 @@
const { packageNameFromPluginName } = require('./utils');
const { packageNameFromPluginName, mergeIgnoreFile } = require('./utils');
describe('utils', () => {
@ -29,4 +29,19 @@ describe('utils', () => {
expect(hasThrown).toBe(true);
});
test('mergeIgnoreFile', () => {
const testCases = [
['line1\nline2\n', 'newline\n', 'line1\nline2\nnewline\n'],
['line1\nline2\n', 'line1\nnewline\n', 'line1\nline2\nnewline\n'],
];
for (const t of testCases) {
const userVersion = t[0];
const frameworkVersion = t[1];
const expected = t[2];
const actual = mergeIgnoreFile(frameworkVersion, userVersion);
expect(actual).toBe(expected);
}
});
});