1
0
mirror of https://github.com/laurent22/joplin.git synced 2024-12-27 10:32:58 +02:00
joplin/CliClient/app/app.js

357 lines
9.8 KiB
JavaScript
Raw Normal View History

const { BaseApplication } = require('lib/BaseApplication');
const { createStore, applyMiddleware } = require('redux');
const { reducer, defaultState } = require('lib/reducer.js');
const { JoplinDatabase } = require('lib/joplin-database.js');
const { Database } = require('lib/database.js');
const { FoldersScreenUtils } = require('lib/folders-screen-utils.js');
const { DatabaseDriverNode } = require('lib/database-driver-node.js');
const { BaseModel } = require('lib/base-model.js');
const { Folder } = require('lib/models/folder.js');
const { BaseItem } = require('lib/models/base-item.js');
const { Note } = require('lib/models/note.js');
const { Tag } = require('lib/models/tag.js');
const { Setting } = require('lib/models/setting.js');
const { Logger } = require('lib/logger.js');
const { sprintf } = require('sprintf-js');
const { reg } = require('lib/registry.js');
const { fileExtension } = require('lib/path-utils.js');
const { shim } = require('lib/shim.js');
const { _, setLocale, defaultLocale, closestSupportedLocale } = require('lib/locale.js');
const os = require('os');
const fs = require('fs-extra');
const { cliUtils } = require('./cli-utils.js');
2017-10-06 19:38:17 +02:00
const EventEmitter = require('events');
2017-07-10 22:03:46 +02:00
class Application extends BaseApplication {
2017-07-10 22:03:46 +02:00
constructor() {
super();
2017-07-10 22:03:46 +02:00
this.showPromptString_ = true;
2017-08-04 19:51:01 +02:00
this.commands_ = {};
this.commandMetadata_ = null;
2017-08-04 18:50:12 +02:00
this.activeCommand_ = null;
2017-08-04 19:11:10 +02:00
this.allCommandsLoaded_ = false;
2017-08-21 20:32:43 +02:00
this.showStackTraces_ = false;
2017-10-05 19:17:56 +02:00
this.gui_ = null;
2017-07-10 22:03:46 +02:00
}
2017-10-09 00:34:01 +02:00
gui() {
return this.gui_;
}
commandStdoutMaxWidth() {
2017-10-24 22:22:57 +02:00
return this.gui().stdoutMaxWidth();
2017-10-09 00:34:01 +02:00
}
2017-07-17 21:19:01 +02:00
async guessTypeAndLoadItem(pattern, options = null) {
let type = BaseModel.TYPE_NOTE;
if (pattern.indexOf('/') === 0) {
type = BaseModel.TYPE_FOLDER;
pattern = pattern.substr(1);
}
return this.loadItem(type, pattern, options);
}
2017-07-15 17:35:40 +02:00
async loadItem(type, pattern, options = null) {
let output = await this.loadItems(type, pattern, options);
if (output.length > 1) {
2017-10-27 00:22:36 +02:00
// output.sort((a, b) => { return a.user_updated_time < b.user_updated_time ? +1 : -1; });
2017-09-24 16:48:23 +02:00
2017-10-27 00:22:36 +02:00
// let answers = { 0: _('[Cancel]') };
// for (let i = 0; i < output.length; i++) {
// answers[i + 1] = output[i].title;
// }
2017-09-24 16:48:23 +02:00
2017-10-08 19:50:43 +02:00
// Not really useful with new UI?
throw new Error(_('More than one item match "%s". Please narrow down your query.', pattern));
// let msg = _('More than one item match "%s". Please select one:', pattern);
// const response = await cliUtils.promptMcq(msg, answers);
// if (!response) return null;
return output[response - 1];
} else {
return output.length ? output[0] : null;
}
2017-07-10 22:03:46 +02:00
}
2017-07-11 20:17:23 +02:00
async loadItems(type, pattern, options = null) {
2017-10-27 00:22:36 +02:00
if (type === 'folderOrNote') {
const folders = await this.loadItems(BaseModel.TYPE_FOLDER, pattern, options);
if (folders.length) return folders;
return await this.loadItems(BaseModel.TYPE_NOTE, pattern, options);
}
pattern = pattern ? pattern.toString() : '';
2017-07-15 17:35:40 +02:00
if (type == BaseModel.TYPE_FOLDER && (pattern == Folder.conflictFolderTitle() || pattern == Folder.conflictFolderId())) return [Folder.conflictFolder()];
2017-07-11 20:17:23 +02:00
if (!options) options = {};
2017-07-13 23:26:45 +02:00
2017-07-11 20:17:23 +02:00
const parent = options.parent ? options.parent : app().currentFolder();
const ItemClass = BaseItem.itemClass(type);
if (type == BaseModel.TYPE_NOTE && pattern.indexOf('*') >= 0) { // Handle it as pattern
if (!parent) throw new Error(_('No notebook selected.'));
return await Note.previews(parent.id, { titlePattern: pattern });
} else { // Single item
let item = null;
if (type == BaseModel.TYPE_NOTE) {
if (!parent) throw new Error(_('No notebook has been specified.'));
item = await ItemClass.loadFolderNoteByField(parent.id, 'title', pattern);
} else {
item = await ItemClass.loadByTitle(pattern);
}
if (item) return [item];
item = await ItemClass.load(pattern); // Load by id
if (item) return [item];
if (pattern.length >= 2) {
return await ItemClass.loadByPartialId(pattern);
2017-07-11 20:17:23 +02:00
}
}
return [];
2017-07-10 22:03:46 +02:00
}
stdout(text) {
return this.gui().stdout(text);
}
2017-10-07 19:07:38 +02:00
setupCommand(cmd) {
cmd.setStdout((text) => {
return this.stdout(text);
2017-10-07 19:07:38 +02:00
});
2017-10-14 23:44:50 +02:00
cmd.setDispatcher((action) => {
if (this.store()) {
return this.store().dispatch(action);
} else {
return (action) => {};
}
2017-10-14 20:03:23 +02:00
});
2017-10-07 19:07:38 +02:00
cmd.setPrompt(async (message, options) => {
2017-10-28 19:55:45 +02:00
if (!options) options = {};
if (!options.type) options.type = 'boolean';
if (!options.booleanAnswerDefault) options.booleanAnswerDefault = 'y';
if (!options.answers) options.answers = options.booleanAnswerDefault === 'y' ? [_('Y'), _('n')] : [_('N'), _('y')];
2017-10-07 19:07:38 +02:00
if (options.type == 'boolean') {
message += ' (' + options.answers.join('/') + ')';
}
2017-10-28 19:55:45 +02:00
let answer = await this.gui().prompt('', message + ' ');
2017-10-15 18:57:09 +02:00
if (options.type === 'boolean') {
2017-10-28 19:55:45 +02:00
if (answer === null) return false; // Pressed ESCAPE
if (!answer) answer = options.answers[0];
let positiveIndex = options.booleanAnswerDefault == 'y' ? 0 : 1;
return answer.toLowerCase() === options.answers[positiveIndex].toLowerCase();
2017-10-07 19:07:38 +02:00
}
});
return cmd;
}
2017-10-09 22:29:49 +02:00
async exit(code = 0) {
const doExit = async () => {
this.gui().exit();
await super.exit(code);
};
// Give it a few seconds to cancel otherwise exit anyway
setTimeout(async () => {
await doExit();
}, 5000);
if (await reg.syncTarget().syncStarted()) {
this.stdout(_('Cancelling background synchronisation... Please wait.'));
const sync = await reg.syncTarget().synchronizer();
await sync.cancel();
}
await doExit();
2017-10-09 22:29:49 +02:00
}
2017-08-04 19:11:10 +02:00
commands() {
if (this.allCommandsLoaded_) return this.commands_;
2017-07-10 22:03:46 +02:00
fs.readdirSync(__dirname).forEach((path) => {
if (path.indexOf('command-') !== 0) return;
const ext = fileExtension(path)
if (ext != 'js') return;
let CommandClass = require('./' + path);
let cmd = new CommandClass();
2017-07-19 00:14:20 +02:00
if (!cmd.enabled()) return;
2017-10-07 19:07:38 +02:00
cmd = this.setupCommand(cmd);
2017-08-04 19:51:01 +02:00
this.commands_[cmd.name()] = cmd;
2017-07-10 22:03:46 +02:00
});
2017-08-04 19:11:10 +02:00
this.allCommandsLoaded_ = true;
return this.commands_;
}
async commandNames() {
const metadata = await this.commandMetadata();
let output = [];
for (let n in metadata) {
if (!metadata.hasOwnProperty(n)) continue;
output.push(n);
}
return output;
2017-07-10 22:03:46 +02:00
}
2017-08-04 19:51:01 +02:00
async commandMetadata() {
if (this.commandMetadata_) return this.commandMetadata_;
2017-07-24 22:36:49 +02:00
const osTmpdir = require('os-tmpdir');
2017-08-04 19:51:01 +02:00
const storage = require('node-persist');
await storage.init({ dir: osTmpdir() + '/commandMetadata', ttl: 1000 * 60 * 60 * 24 });
2017-07-24 22:36:49 +02:00
2017-08-04 19:51:01 +02:00
let output = await storage.getItem('metadata');
if (Setting.value('env') != 'dev' && output) {
this.commandMetadata_ = output;
return Object.assign({}, this.commandMetadata_);
}
2017-08-04 19:11:10 +02:00
const commands = this.commands();
2017-07-24 22:36:49 +02:00
2017-08-04 19:51:01 +02:00
output = {};
2017-08-04 19:11:10 +02:00
for (let n in commands) {
if (!commands.hasOwnProperty(n)) continue;
const cmd = commands[n];
2017-08-04 19:51:01 +02:00
output[n] = cmd.metadata();
2017-07-24 22:36:49 +02:00
}
2017-08-04 19:51:01 +02:00
await storage.setItem('metadata', output);
this.commandMetadata_ = output;
return Object.assign({}, this.commandMetadata_);
2017-07-24 22:36:49 +02:00
}
hasGui() {
return this.gui() && !this.gui().isDummy();
}
2017-08-03 19:48:14 +02:00
findCommandByName(name) {
2017-08-04 19:11:10 +02:00
if (this.commands_[name]) return this.commands_[name];
2017-08-03 19:48:14 +02:00
let CommandClass = null;
try {
CommandClass = require(__dirname + '/command-' + name + '.js');
2017-08-03 19:48:14 +02:00
} catch (error) {
let e = new Error('No such command: ' + name);
e.type = 'notFound';
throw e;
}
2017-10-06 19:38:17 +02:00
2017-08-03 19:48:14 +02:00
let cmd = new CommandClass();
2017-10-07 19:07:38 +02:00
cmd = this.setupCommand(cmd);
2017-08-04 19:11:10 +02:00
this.commands_[name] = cmd;
return this.commands_[name];
2017-08-03 19:48:14 +02:00
}
dummyGui() {
return {
isDummy: () => { return true; },
prompt: (initialText = '', promptString = '') => { return cliUtils.prompt(initialText, promptString); },
showConsole: () => {},
maximizeConsole: () => {},
stdout: (text) => { console.info(text); },
fullScreen: (b=true) => {},
exit: () => {},
2017-10-24 21:52:26 +02:00
showModalOverlay: (text) => {},
hideModalOverlay: () => {},
2017-10-24 22:22:57 +02:00
stdoutMaxWidth: () => { return 78; }
};
}
2017-08-03 19:48:14 +02:00
async execCommand(argv) {
if (!argv.length) return this.execCommand(['help']);
2017-10-07 18:30:27 +02:00
reg.logger().info('execCommand()', argv);
2017-08-03 19:48:14 +02:00
const commandName = argv[0];
2017-08-04 18:50:12 +02:00
this.activeCommand_ = this.findCommandByName(commandName);
let outException = null;
try {
if (this.gui().isDummy() && !this.activeCommand_.supportsUi('cli')) throw new Error(_('The command "%s" is only available in GUI mode', this.activeCommand_.name()));
const cmdArgs = cliUtils.makeCommandArgs(this.activeCommand_, argv);
await this.activeCommand_.action(cmdArgs);
} catch (error) {
outException = error;
}
2017-10-14 20:03:23 +02:00
this.activeCommand_ = null;
if (outException) throw outException;
2017-08-04 18:50:12 +02:00
}
2017-08-20 16:29:18 +02:00
currentCommand() {
return this.activeCommand_;
2017-08-03 19:48:14 +02:00
}
async start(argv) {
argv = await super.start(argv);
2017-10-25 19:41:36 +02:00
cliUtils.setStdout((object) => {
return this.stdout(object);
});
// If we have some arguments left at this point, it's a command
// so execute it.
if (argv.length) {
this.gui_ = this.dummyGui();
try {
await this.execCommand(argv);
} catch (error) {
if (this.showStackTraces_) {
console.info(error);
} else {
console.info(error.message);
}
}
} else { // Otherwise open the GUI
this.initRedux();
const AppGui = require('./app-gui.js');
this.gui_ = new AppGui(this, this.store());
this.gui_.setLogger(this.logger_);
await this.gui_.start();
2017-10-21 18:53:43 +02:00
// Since the settings need to be loaded before the store is created, it will never
2017-11-08 23:22:24 +02:00
// receive the SETTING_UPDATE_ALL even, which mean state.settings will not be
2017-10-21 18:53:43 +02:00
// initialised. So we manually call dispatchUpdateAll() to force an update.
Setting.dispatchUpdateAll();
await FoldersScreenUtils.refreshFolders();
2017-10-22 19:12:16 +02:00
const tags = await Tag.allWithNotes();
this.dispatch({
2017-11-08 23:22:24 +02:00
type: 'TAG_UPDATE_ALL',
2017-10-22 19:12:16 +02:00
tags: tags,
});
this.store().dispatch({
2017-11-08 23:22:24 +02:00
type: 'FOLDER_SELECT',
id: Setting.value('activeFolderId'),
});
}
2017-07-10 22:03:46 +02:00
}
}
let application_ = null;
function app() {
if (application_) return application_;
application_ = new Application();
return application_;
}
2017-11-03 02:13:17 +02:00
module.exports = { app };