1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-09 08:45:55 +02:00
joplin/CliClient/app/main.js
2017-07-02 11:34:07 +01:00

833 lines
23 KiB
JavaScript

#!/usr/bin/env node
require('source-map-support').install();
require('babel-plugin-transform-runtime');
import { FileApi } from 'lib/file-api.js';
import { FileApiDriverOneDrive } from 'lib/file-api-driver-onedrive.js';
import { FileApiDriverMemory } from 'lib/file-api-driver-memory.js';
import { FileApiDriverLocal } from 'lib/file-api-driver-local.js';
import { Database } from 'lib/database.js';
import { DatabaseDriverNode } from 'lib/database-driver-node.js';
import { BaseModel } from 'lib/base-model.js';
import { Folder } from 'lib/models/folder.js';
import { BaseItem } from 'lib/models/base-item.js';
import { Note } from 'lib/models/note.js';
import { Setting } from 'lib/models/setting.js';
import { Synchronizer } from 'lib/synchronizer.js';
import { Logger } from 'lib/logger.js';
import { uuid } from 'lib/uuid.js';
import { sprintf } from 'sprintf-js';
import { importEnex } from 'import-enex';
import { filename, basename } from 'lib/path-utils.js';
import { _ } from 'lib/locale.js';
import os from 'os';
import fs from 'fs-extra';
process.on('unhandledRejection', (reason, p) => {
console.error('Unhandled promise rejection', p, 'reason:', reason);
});
const packageJson = require('./package.json');
let initArgs = {
profileDir: null,
}
Setting.setConstant('appId', 'net.cozic.joplin-cli');
let currentFolder = null;
let commands = [];
let database_ = null;
let synchronizers_ = {};
let logger = new Logger();
let dbLogger = new Logger();
let syncLogger = new Logger();
commands.push({
usage: 'version',
description: 'Displays version information',
action: function(args, end) {
this.log(packageJson.name + ' ' + packageJson.version);
end();
},
});
commands.push({
usage: 'mkbook <notebook>',
aliases: ['mkdir'],
description: 'Creates a new notebook',
action: function(args, end) {
Folder.save({ title: args['notebook'] }).then((folder) => {
switchCurrentFolder(folder);
}).catch((error) => {
this.log(error);
}).then(() => {
end();
});
},
});
commands.push({
usage: 'mknote <note-title>',
aliases: ['touch'],
description: 'Creates a new note',
action: function(args, end) {
if (!currentFolder) {
this.log('Notes can only be created within a notebook.');
end();
return;
}
let note = {
title: args['note-title'],
parent_id: currentFolder.id,
};
Note.save(note).catch((error) => {
this.log(error);
}).then((note) => {
end();
});
},
});
commands.push({
usage: 'use <notebook>',
aliases: ['cd'],
description: 'Switches to [notebook] - all further operations will happen within this notebook.',
action: async function(args, end) {
let folderTitle = args['notebook'];
let folder = await Folder.loadByField('title', folderTitle);
if (!folder) return cmdError(this, _('Invalid folder title: %s', folderTitle), end);
switchCurrentFolder(folder);
end();
},
autocomplete: autocompleteFolders,
});
commands.push({
usage: 'set <title> <name> [value]',
description: 'Sets the property <name> of the given item to the given [value].',
action: async function(args, end) {
try {
let promise = null;
let title = args['title'];
let propName = args['name'];
let propValue = args['value'];
if (!propValue) propValue = '';
let item = null;
if (!currentFolder) {
item = await Folder.loadByField('title', title);
} else {
item = await Note.loadFolderNoteByField(currentFolder.id, 'title', title);
}
if (!item) {
item = await BaseItem.loadItemById(title);
}
if (!item) {
this.log(_('No item with title "%s" found.', title));
end();
return;
}
let newItem = {
id: item.id,
type_: item.type_,
};
newItem[propName] = propValue;
let ItemClass = BaseItem.itemClass(newItem);
await ItemClass.save(newItem);
} catch(error) {
this.log(error);
}
end();
},
autocomplete: autocompleteItems,
});
commands.push({
usage: 'cat <title>',
description: 'Displays the given item data.',
action: function(args, end) {
let title = args['title'];
let promise = null;
if (!currentFolder) {
promise = Folder.loadByField('title', title);
} else {
promise = Note.loadFolderNoteByField(currentFolder.id, 'title', title);
}
promise.then((item) => {
if (!item) {
this.log(_('No item with title "%s" found.', title));
end();
return;
}
if (!currentFolder) {
this.log(Folder.serialize(item));
} else {
this.log(Note.serialize(item));
}
}).catch((error) => {
this.log(error);
}).then(() => {
end();
});
},
autocomplete: autocompleteItems,
});
commands.push({
usage: 'rm <pattern>',
description: 'Deletes the given item. For a notebook, all the notes within that notebook will be deleted. Use `rm ../<notebook>` to delete a notebook.',
options: [
['-f, --force', 'Deletes the items without asking for confirmation.'],
],
action: async function(args, end) {
try {
let pattern = args['pattern'];
let itemType = null;
let force = args.options && args.options.force === true;
if (pattern.indexOf('*') < 0) { // Handle it as a simple title
if (pattern.substr(0, 3) == '../') {
itemType = BaseModel.MODEL_TYPE_FOLDER;
pattern = pattern.substr(3);
} else {
itemType = BaseModel.MODEL_TYPE_NOTE;
}
let item = await BaseItem.loadItemByField(itemType, 'title', pattern);
if (!item) throw new Error(_('No item with title "%s" found.', pattern));
let ok = force ? true : await cmdPromptConfirm(this, _('Delete item?'));
if (ok) {
await BaseItem.deleteItem(itemType, item.id);
if (currentFolder && currentFolder.id == item.id) {
let f = await Folder.defaultFolder();
switchCurrentFolder(f);
}
}
} else { // Handle it as a glob pattern
if (currentFolder) {
let notes = await Note.previews(currentFolder.id, { titlePattern: pattern });
if (!notes.length) throw new Error(_('No note matches this pattern: "%s"', pattern));
let ok = force ? true : await cmdPromptConfirm(this, _('%d notes match this pattern. Delete them?', notes.length));
if (ok) {
for (let i = 0; i < notes.length; i++) {
await Note.delete(notes[i].id);
}
}
}
}
} catch (error) {
this.log(error);
}
end();
},
autocomplete: autocompleteItems,
});
commands.push({
usage: 'mv <pattern> <notebook>',
description: 'Moves the notes matching <pattern> to <notebook>.',
action: async function(args, end) {
try {
if (!currentFolder) throw new Error(_('Please select a notebook first.'));
let pattern = args['pattern'];
let folder = await Folder.loadByField('title', args['notebook']);
if (!folder) throw new Error(_('No folder with title "%s"', args['notebook']));
let notes = await Note.previews(currentFolder.id, { titlePattern: pattern });
if (!notes.length) throw new Error(_('No note matches this pattern: "%s"', pattern));
for (let i = 0; i < notes.length; i++) {
await Note.save({ id: notes[i].id, parent_id: folder.id });
}
} catch (error) {
this.log(error);
}
end();
},
autocomplete: autocompleteItems,
});
commands.push({
usage: 'dump',
description: 'Dumps the complete database as JSON.',
action: async function(args, end) {
try {
let items = [];
let folders = await Folder.all();
for (let i = 0; i < folders.length; i++) {
let folder = folders[i];
let notes = await Note.previews(folder.id);
items.push(folder);
items = items.concat(notes);
}
this.log(JSON.stringify(items));
} catch (error) {
this.log(error);
}
end();
},
autocomplete: autocompleteFolders,
});
commands.push({
usage: 'ls [pattern]',
description: 'Displays the notes in [notebook]. Use `ls ..` to display the list of notebooks.',
options: [
['-n, --lines <num>', 'Displays only the first top <num> lines.'],
['-s, --sort <field>', 'Sorts the item by <field> (eg. title, updated_time, created_time).'],
['-r, --reverse', 'Reverses the sorting order.'],
['-t, --type <type>', 'Displays only the items of the specific type(s). Can be `n` for notes, `t` for todos, or `nt` for notes and todos (eg. `-tt` would display only the todos, while `-ttd` would display notes and todos.'],
['-f, --format <format>', 'Either "text" or "json"'],
],
action: async function(args, end) {
try {
let pattern = args['pattern'];
let suffix = '';
let items = [];
let options = args.options;
let queryOptions = {};
if (options.lines) queryOptions.limit = options.lines;
if (options.sort) {
queryOptions.orderBy = options.sort;
queryOptions.orderByDir = 'ASC';
}
if (options.reverse === true) queryOptions.orderByDir = queryOptions.orderByDir == 'ASC' ? 'DESC' : 'ASC';
queryOptions.caseInsensitive = true;
if (options.type) {
queryOptions.itemTypes = [];
if (options.type.indexOf('n') >= 0) queryOptions.itemTypes.push('note');
if (options.type.indexOf('t') >= 0) queryOptions.itemTypes.push('todo');
}
if (pattern) queryOptions.titlePattern = pattern;
if (pattern == '..' || !currentFolder) {
items = await Folder.all(queryOptions);
suffix = '/';
} else {
if (!currentFolder) throw new Error(_('Please select a notebook first.'));
items = await Note.previews(currentFolder.id, queryOptions);
}
if (options.format && options.format == 'json') {
this.log(JSON.stringify(items));
} else {
for (let i = 0; i < items.length; i++) {
let item = items[i];
let line = '';
if (!!item.is_todo) {
line += sprintf('[%s] ', !!item.todo_completed ? 'X' : ' ');
}
line += item.title + suffix;
this.log(line);
}
}
} catch (error) {
this.log(error);
}
end();
},
autocomplete: autocompleteFolders,
});
commands.push({
usage: 'config [name] [value]',
description: 'Gets or sets a config value. If [value] is not provided, it will show the value of [name]. If neither [name] nor [value] is provided, it will list the current configuration.',
action: async function(args, end) {
try {
if (!args.name && !args.value) {
let keys = Setting.publicKeys();
for (let i = 0; i < keys.length; i++) {
this.log(keys[i] + ' = ' + Setting.value(keys[i]));
}
} else if (args.name && !args.value) {
this.log(args.name + ' = ' + Setting.value(args.name));
} else {
Setting.setValue(args.name, args.value);
await Setting.saveAll();
}
} catch(error) {
this.log(error);
}
end();
},
});
commands.push({
usage: 'sync',
description: 'Synchronizes with remote storage.',
options: [
['--random-failures', 'For debugging purposes. Do not use.'],
],
action: function(args, end) {
let redrawnCalled = false;
let options = {
onProgress: (report) => {
let line = [];
if (report.remotesToUpdate) line.push(_('Items to upload: %d/%d.', report.createRemote + report.updateRemote, report.remotesToUpdate));
if (report.remotesToDelete) line.push(_('Remote items to delete: %d/%d.', report.deleteRemote, report.remotesToDelete));
if (report.localsToUdpate) line.push(_('Items to download: %d/%d.', report.createLocal + report.updateLocal, report.localsToUdpate));
if (report.localsToDelete) line.push(_('Local items to delete: %d/%d.', report.deleteLocal, report.localsToDelete));
//redrawnCalled = true;
//vorpal.ui.redraw(line.join(' '));
},
onMessage: (msg) => {
if (redrawnCalled) vorpal.ui.redraw.done();
this.log(msg);
},
randomFailures: args.options['random-failures'] === true,
};
this.log(_('Synchronization target: %s', Setting.value('sync.target')));
synchronizer(Setting.value('sync.target')).then((s) => {
this.log(_('Starting synchronization...'));
return s.start(options);
}).catch((error) => {
this.log(error);
}).then(() => {
if (redrawnCalled) vorpal.ui.redraw.done();
this.log(_('Done.'));
end();
});
},
});
commands.push({
usage: 'import-enex <file> [notebook]',
description: _('Imports an Evernote notebook file (.enex file).'),
options: [
['--fuzzy-matching', 'For debugging purposes. Do not use.'],
],
action: async function(args, end) {
let redrawnCalled = false;
try {
let filePath = args.file;
let folder = null;
let folderTitle = args['notebook'];
if (folderTitle) {
folder = await Folder.loadByField('title', folderTitle);
if (!folder) {
let ok = await cmdPromptConfirm(this, _('Folder does not exists: "%s". Create it?', folderTitle))
if (!ok) {
end();
return;
}
folder = await Folder.save({ title: folderTitle });
}
} else {
folderTitle = filename(filePath);
folderTitle = _('Imported - %s', folderTitle);
let inc = 0;
while (true) {
let t = folderTitle + (inc ? ' (' + inc + ')' : '');
let f = await Folder.loadByField('title', t);
if (!f) {
folderTitle = t;
break;
}
inc++;
}
}
let ok = await cmdPromptConfirm(this, _('File "%s" will be imported into notebook "%s". Continue?', basename(filePath), folderTitle))
if (!ok) {
end();
return;
}
let options = {
fuzzyMatching: args.options['fuzzy-matching'] === true,
onProgress: (progressState) => {
let line = [];
line.push(_('Found: %d.', progressState.loaded));
line.push(_('Created: %d.', progressState.created));
if (progressState.updated) line.push(_('Updated: %d.', progressState.updated));
if (progressState.skipped) line.push(_('Skipped: %d.', progressState.skipped));
if (progressState.resourcesCreated) line.push(_('Resources: %d.', progressState.resourcesCreated));
redrawnCalled = true;
vorpal.ui.redraw(line.join(' '));
},
onError: (error) => {
if (redrawnCalled) vorpal.ui.redraw.done();
let s = error.trace ? error.trace : error.toString();
this.log(s);
},
}
folder = !folder ? await Folder.save({ title: folderTitle }) : folder;
this.log(_('Importing notes...'));
await importEnex(folder.id, filePath, options);
this.log(_('Done.'));
} catch (error) {
this.log(error);
}
if (redrawnCalled) vorpal.ui.redraw.done();
end();
},
});
function commandByName(name) {
for (let i = 0; i < commands.length; i++) {
let c = commands[i];
let n = c.usage.split(' ');
n = n[0].trim();
if (n == name) return c;
if (c.aliases && c.aliases.indexOf(name) >= 0) return c;
}
return null;
}
function execCommand(name, args) {
return new Promise((resolve, reject) => {
let cmd = commandByName(name);
if (!cmd) {
reject(new Error('Unknown command: ' + name));
} else {
cmd.action(args, function() {
resolve();
});
}
});
}
// async function execCommand(args) {
// var parseArgs = require('minimist');
// let results = parseArgs(args);
// //var results = vorpal.parse(args, { use: 'minimist' });
// if (!results['_'].length) throw new Error(_('Invalid command: %s', args));
// console.info(results);
// let commandName = results['_'].splice(0, 1);
// let cmd = commandByName(commandName);
// if (!cmd) throw new Error(_('Unknown command: %s', args));
// let usage = cmd.usage.split(' ');
// let commandArgs = [];
// usage.splice(0, 1);
// for (let i = 0; i < usage.length; i++) {
// let u = usage[i].trim();
// if (u == '') continue;
// let required = false;
// if (u.length >= 3 && u[0] == '<' && u[u.length - 1] == '>') {
// required = true;
// u = u.substr(1, u.length - 2);
// }
// if (u.length >= 3 && u[0] == '[' && u[u.length - 1] == ']') {
// u = u.substr(1, u.length - 2);
// }
// if (required && !results['_'].length) throw new Error(_('Missing argument: %s', args));
// if (!results['_'].length) break;
// console.info(u);
// commandArgs[u] = results['_'].splice(0, 1);
// }
// console.info(commandArgs);
// // usage: 'import-enex <file> [notebook]',
// // description: _('Imports en Evernote notebook file (.enex file).'),
// // options: [
// // ['--fuzzy-matching', 'For debugging purposes. Do not use.'],
// // ],
// }
async function synchronizer(syncTarget) {
if (synchronizers_[syncTarget]) return synchronizers_[syncTarget];
let fileApi = null;
if (syncTarget == 'onedrive') {
const CLIENT_ID = 'e09fc0de-c958-424f-83a2-e56a721d331b';
const CLIENT_SECRET = 'JA3cwsqSGHFtjMwd5XoF5L5';
let driver = new FileApiDriverOneDrive(CLIENT_ID, CLIENT_SECRET);
let auth = Setting.value('sync.onedrive.auth');
if (auth) {
auth = JSON.parse(auth);
} else {
auth = await driver.api().oauthDance(vorpal);
Setting.setValue('sync.onedrive.auth', JSON.stringify(auth));
}
driver.api().setAuth(auth);
driver.api().on('authRefreshed', (a) => {
Setting.setValue('sync.onedrive.auth', JSON.stringify(a));
});
let appDir = await driver.api().appDirectory();
logger.info('App dir: ' + appDir);
fileApi = new FileApi(appDir, driver);
fileApi.setLogger(logger);
} else if (syncTarget == 'memory') {
fileApi = new FileApi('joplin', new FileApiDriverMemory());
fileApi.setLogger(logger);
} else if (syncTarget == 'local') {
let syncDir = Setting.value('sync.local.path');
if (!syncDir) syncDir = Setting.value('profileDir') + '/sync';
vorpal.log(_('Synchronizing with directory "%s"', syncDir));
await fs.mkdirp(syncDir, 0o755);
fileApi = new FileApi(syncDir, new FileApiDriverLocal());
fileApi.setLogger(logger);
} else {
throw new Error('Unknown backend: ' + syncTarget);
}
synchronizers_[syncTarget] = new Synchronizer(database_, fileApi);
synchronizers_[syncTarget].setLogger(syncLogger);
return synchronizers_[syncTarget];
}
function switchCurrentFolder(folder) {
currentFolder = folder;
Setting.setValue('activeFolderId', folder ? folder.id : '');
updatePrompt();
}
function promptString() {
let path = '~';
if (currentFolder) {
path += '/' + currentFolder.title;
}
return 'joplin:' + path + '$ ';
}
function updatePrompt() {
vorpal.delimiter(promptString());
}
// For now, to go around this issue: https://github.com/dthree/vorpal/issues/114
function quotePromptArg(s) {
if (s.indexOf(' ') >= 0) {
return '"' + s + '"';
}
return s;
}
function autocompleteFolders() {
return Folder.all().then((folders) => {
let output = [];
for (let i = 0; i < folders.length; i++) {
output.push(quotePromptArg(folders[i].title));
}
output.push('..');
output.push('.');
return output;
});
}
function autocompleteItems() {
let promise = null;
if (!currentFolder) {
promise = Folder.all();
} else {
promise = Note.previews(currentFolder.id);
}
return promise.then((items) => {
let output = [];
for (let i = 0; i < items.length; i++) {
output.push(quotePromptArg(items[i].title));
}
return output;
});
}
function cmdError(commandInstance, msg, end) {
commandInstance.log(msg);
end();
}
function cmdPromptConfirm(commandInstance, message) {
return new Promise((resolve, reject) => {
let options = {
type: 'confirm',
name: 'ok',
default: false, // This needs to be false so that, when pressing Ctrl+C, the prompt returns false
message: message,
};
commandInstance.prompt(options, (result) => {
if (result.ok) {
resolve(true);
} else {
resolve(false);
}
});
});
}
// Handles the initial flags passed to main script and
// returns the remaining args.
async function handleStartFlags(argv) {
argv = argv.slice(0);
argv.splice(0, 2); // First arguments are the node executable, and the node JS file
while (argv.length) {
let arg = argv[0];
let nextArg = argv.length >= 2 ? argv[1] : null;
if (arg == '--profile') {
if (!nextArg) {
throw new Error(_('Usage: --profile <dir-path>'));
}
initArgs.profileDir = nextArg;
argv.splice(0, 2);
continue;
}
if (arg.length && arg[0] == '-') {
throw new Error(_('Unknown flag: %s', arg));
} else {
break;
}
}
return argv;
}
function escapeShellArg(arg) {
if (arg.indexOf('"') >= 0 && arg.indexOf("'") >= 0) throw new Error(_('Command line argument "%s" contains both quotes and double-quotes - aborting.', arg)); // Hopeless case
let quote = '"';
if (arg.indexOf('"') >= 0) quote = "'";
if (arg.indexOf(' ') >= 0 || arg.indexOf("\t") >= 0) return quote + arg + quote;
return arg;
}
function shellArgsToString(args) {
let output = [];
for (let i = 0; i < args.length; i++) {
output.push(escapeShellArg(args[i]));
}
return output.join(' ');
}
process.stdin.on('keypress', (_, key) => {
if (key && key.name === 'return') {
updatePrompt();
}
if (key.name === 'tab') {
vorpal.ui.imprint();
vorpal.log(vorpal.ui.input());
}
});
const vorpal = require('vorpal')();
async function main() {
for (let commandIndex = 0; commandIndex < commands.length; commandIndex++) {
let c = commands[commandIndex];
let o = vorpal.command(c.usage, c.description);
if (c.options) {
for (let i = 0; i < c.options.length; i++) {
let options = c.options[i];
if (options.length == 2) o.option(options[0], options[1]);
if (options.length == 3) o.option(options[0], options[1], options[2]);
}
}
if (c.aliases) {
for (let i = 0; i < c.aliases.length; i++) {
o.alias(c.aliases[i]);
}
}
if (c.autocomplete) {
o.autocomplete({
data: c.autocomplete,
});
}
o.action(c.action);
}
let argv = process.argv;
argv = await handleStartFlags(argv);
const profileDir = initArgs.profileDir ? initArgs.profileDir : os.homedir() + '/.config/' + Setting.value('appName');
const resourceDir = profileDir + '/resources';
Setting.setConstant('profileDir', profileDir);
Setting.setConstant('resourceDir', resourceDir);
await fs.mkdirp(profileDir, 0o755);
await fs.mkdirp(resourceDir, 0o755);
logger.addTarget('file', { path: profileDir + '/log.txt' });
logger.setLevel(Logger.LEVEL_DEBUG);
dbLogger.addTarget('file', { path: profileDir + '/log-database.txt' });
dbLogger.setLevel(Logger.LEVEL_DEBUG);
syncLogger.addTarget('file', { path: profileDir + '/log-sync.txt' });
syncLogger.setLevel(Logger.LEVEL_DEBUG);
logger.info(sprintf('Starting %s %s...', packageJson.name, packageJson.version));
logger.info('Profile directory: ' + profileDir);
database_ = new Database(new DatabaseDriverNode());
database_.setLogger(dbLogger);
await database_.open({ name: profileDir + '/database.sqlite' });
BaseModel.db_ = database_;
await Setting.load();
let activeFolderId = Setting.value('activeFolderId');
let activeFolder = null;
if (activeFolderId) activeFolder = await Folder.load(activeFolderId);
if (!activeFolder) activeFolder = await Folder.defaultFolder();
Setting.setValue('activeFolderId', activeFolder ? activeFolder.id : '');
if (activeFolder) await execCommand('cd', { 'notebook': activeFolder.title }); // Use execCommand() so that no history entry is created
// If we still have arguments, pass it to Vorpal and exit
if (argv.length) {
let cmd = shellArgsToString(argv);
await vorpal.exec(cmd);
await vorpal.exec('exit');
return;
} else {
vorpal.history(Setting.value('appId')); // Enables persistent history
vorpal.delimiter(promptString()).show();
if (!activeFolder) {
vorpal.log(_('No notebook is defined. Create one with `mkbook <notebook>`.'));
}
}
}
main().catch((error) => {
vorpal.log('Fatal error:');
vorpal.log(error);
});