2024-01-20 16:29:21 +02:00
|
|
|
"use strict";
|
|
|
|
var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) {
|
|
|
|
function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); }
|
|
|
|
return new (P || (P = Promise))(function (resolve, reject) {
|
|
|
|
function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } }
|
|
|
|
function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } }
|
|
|
|
function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); }
|
|
|
|
step((generator = generator.apply(thisArg, _arguments || [])).next());
|
|
|
|
});
|
|
|
|
};
|
|
|
|
Object.defineProperty(exports, "__esModule", { value: true });
|
|
|
|
const BaseApplication_1 = require("@joplin/lib/BaseApplication");
|
|
|
|
const folders_screen_utils_js_1 = require("@joplin/lib/folders-screen-utils.js");
|
|
|
|
const ResourceService_1 = require("@joplin/lib/services/ResourceService");
|
|
|
|
const BaseModel_1 = require("@joplin/lib/BaseModel");
|
|
|
|
const Folder_1 = require("@joplin/lib/models/Folder");
|
|
|
|
const BaseItem_1 = require("@joplin/lib/models/BaseItem");
|
|
|
|
const Note_1 = require("@joplin/lib/models/Note");
|
|
|
|
const Tag_1 = require("@joplin/lib/models/Tag");
|
|
|
|
const Setting_1 = require("@joplin/lib/models/Setting");
|
|
|
|
const registry_js_1 = require("@joplin/lib/registry.js");
|
|
|
|
const path_utils_1 = require("@joplin/lib/path-utils");
|
|
|
|
const utils_1 = require("@joplin/utils");
|
|
|
|
const locale_1 = require("@joplin/lib/locale");
|
|
|
|
const fs_extra_1 = require("fs-extra");
|
|
|
|
const RevisionService_1 = require("@joplin/lib/services/RevisionService");
|
|
|
|
const shim_1 = require("@joplin/lib/shim");
|
|
|
|
const setupCommand_1 = require("./setupCommand");
|
2018-03-09 22:59:12 +02:00
|
|
|
const { cliUtils } = require('./cli-utils.js');
|
2020-11-07 17:59:37 +02:00
|
|
|
const Cache = require('@joplin/lib/Cache');
|
2024-01-20 16:29:21 +02:00
|
|
|
const { splitCommandBatch } = require('@joplin/lib/string-utils');
|
|
|
|
class Application extends BaseApplication_1.default {
|
|
|
|
constructor() {
|
|
|
|
super(...arguments);
|
|
|
|
this.commands_ = {};
|
|
|
|
this.commandMetadata_ = null;
|
|
|
|
this.activeCommand_ = null;
|
|
|
|
this.allCommandsLoaded_ = false;
|
|
|
|
this.gui_ = null;
|
|
|
|
this.cache_ = new Cache();
|
|
|
|
}
|
|
|
|
gui() {
|
|
|
|
return this.gui_;
|
|
|
|
}
|
|
|
|
commandStdoutMaxWidth() {
|
|
|
|
return this.gui().stdoutMaxWidth();
|
|
|
|
}
|
|
|
|
guessTypeAndLoadItem(pattern, options = null) {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
let type = BaseModel_1.default.TYPE_NOTE;
|
|
|
|
if (pattern.indexOf('/') === 0) {
|
|
|
|
type = BaseModel_1.default.TYPE_FOLDER;
|
|
|
|
pattern = pattern.substr(1);
|
|
|
|
}
|
|
|
|
return this.loadItem(type, pattern, options);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
loadItem(type, pattern, options = null) {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
const output = yield this.loadItems(type, pattern, options);
|
|
|
|
if (output.length > 1) {
|
|
|
|
// output.sort((a, b) => { return a.user_updated_time < b.user_updated_time ? +1 : -1; });
|
|
|
|
// let answers = { 0: _('[Cancel]') };
|
|
|
|
// for (let i = 0; i < output.length; i++) {
|
|
|
|
// answers[i + 1] = output[i].title;
|
|
|
|
// }
|
|
|
|
// Not really useful with new UI?
|
|
|
|
throw new Error((0, locale_1._)('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;
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
loadItems(type, pattern, options = null) {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
if (type === 'folderOrNote') {
|
|
|
|
const folders = yield this.loadItems(BaseModel_1.default.TYPE_FOLDER, pattern, options);
|
|
|
|
if (folders.length)
|
|
|
|
return folders;
|
|
|
|
return yield this.loadItems(BaseModel_1.default.TYPE_NOTE, pattern, options);
|
|
|
|
}
|
|
|
|
pattern = pattern ? pattern.toString() : '';
|
|
|
|
if (type === BaseModel_1.default.TYPE_FOLDER && (pattern === Folder_1.default.conflictFolderTitle() || pattern === Folder_1.default.conflictFolderId()))
|
|
|
|
return [Folder_1.default.conflictFolder()];
|
|
|
|
if (!options)
|
|
|
|
options = {};
|
|
|
|
const parent = options.parent ? options.parent : app().currentFolder();
|
|
|
|
const ItemClass = BaseItem_1.default.itemClass(type);
|
|
|
|
if (type === BaseModel_1.default.TYPE_NOTE && pattern.indexOf('*') >= 0) {
|
|
|
|
// Handle it as pattern
|
|
|
|
if (!parent)
|
|
|
|
throw new Error((0, locale_1._)('No notebook selected.'));
|
|
|
|
return yield Note_1.default.previews(parent.id, { titlePattern: pattern });
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Single item
|
|
|
|
let item = null;
|
|
|
|
if (type === BaseModel_1.default.TYPE_NOTE) {
|
|
|
|
if (!parent)
|
|
|
|
throw new Error((0, locale_1._)('No notebook has been specified.'));
|
|
|
|
item = yield ItemClass.loadFolderNoteByField(parent.id, 'title', pattern);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
item = yield ItemClass.loadByTitle(pattern);
|
|
|
|
}
|
|
|
|
if (item)
|
|
|
|
return [item];
|
|
|
|
item = yield ItemClass.load(pattern); // Load by id
|
|
|
|
if (item)
|
|
|
|
return [item];
|
|
|
|
if (pattern.length >= 2) {
|
|
|
|
return yield ItemClass.loadByPartialId(pattern);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return [];
|
|
|
|
});
|
|
|
|
}
|
|
|
|
setupCommand(cmd) {
|
|
|
|
return (0, setupCommand_1.default)(cmd, (t) => this.stdout(t), () => this.store(), () => this.gui());
|
|
|
|
}
|
|
|
|
stdout(text) {
|
|
|
|
return this.gui().stdout(text);
|
|
|
|
}
|
|
|
|
exit(code = 0) {
|
|
|
|
const _super = Object.create(null, {
|
|
|
|
exit: { get: () => super.exit }
|
|
|
|
});
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
const doExit = () => __awaiter(this, void 0, void 0, function* () {
|
|
|
|
this.gui().exit();
|
|
|
|
yield _super.exit.call(this, code);
|
|
|
|
});
|
|
|
|
// Give it a few seconds to cancel otherwise exit anyway
|
|
|
|
shim_1.default.setTimeout(() => __awaiter(this, void 0, void 0, function* () {
|
|
|
|
yield doExit();
|
|
|
|
}), 5000);
|
|
|
|
if (yield registry_js_1.reg.syncTarget().syncStarted()) {
|
|
|
|
this.stdout((0, locale_1._)('Cancelling background synchronisation... Please wait.'));
|
|
|
|
const sync = yield registry_js_1.reg.syncTarget().synchronizer();
|
|
|
|
yield sync.cancel();
|
|
|
|
}
|
|
|
|
yield doExit();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
commands(uiType = null) {
|
|
|
|
if (!this.allCommandsLoaded_) {
|
|
|
|
// eslint-disable-next-line github/array-foreach -- Old code before rule was applied
|
|
|
|
(0, fs_extra_1.readdirSync)(__dirname).forEach(path => {
|
|
|
|
if (path.indexOf('command-') !== 0)
|
|
|
|
return;
|
|
|
|
if (path.endsWith('.test.js'))
|
|
|
|
return;
|
|
|
|
const ext = (0, path_utils_1.fileExtension)(path);
|
|
|
|
if (ext !== 'js')
|
|
|
|
return;
|
|
|
|
const CommandClass = require(`./${path}`);
|
|
|
|
let cmd = new CommandClass();
|
|
|
|
if (!cmd.enabled())
|
|
|
|
return;
|
|
|
|
cmd = this.setupCommand(cmd);
|
|
|
|
this.commands_[cmd.name()] = cmd;
|
|
|
|
});
|
|
|
|
this.allCommandsLoaded_ = true;
|
|
|
|
}
|
|
|
|
if (uiType !== null) {
|
|
|
|
const temp = {};
|
|
|
|
for (const n in this.commands_) {
|
|
|
|
if (!this.commands_.hasOwnProperty(n))
|
|
|
|
continue;
|
|
|
|
const c = this.commands_[n];
|
|
|
|
if (!c.supportsUi(uiType))
|
|
|
|
continue;
|
|
|
|
temp[n] = c;
|
|
|
|
}
|
|
|
|
return temp;
|
|
|
|
}
|
|
|
|
return this.commands_;
|
|
|
|
}
|
|
|
|
commandNames() {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
const metadata = yield this.commandMetadata();
|
|
|
|
const output = [];
|
|
|
|
for (const n in metadata) {
|
|
|
|
if (!metadata.hasOwnProperty(n))
|
|
|
|
continue;
|
|
|
|
output.push(n);
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
commandMetadata() {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
if (this.commandMetadata_)
|
|
|
|
return this.commandMetadata_;
|
|
|
|
let output = yield this.cache_.getItem('metadata');
|
|
|
|
if (output) {
|
|
|
|
this.commandMetadata_ = output;
|
|
|
|
return Object.assign({}, this.commandMetadata_);
|
|
|
|
}
|
|
|
|
const commands = this.commands();
|
|
|
|
output = {};
|
|
|
|
for (const n in commands) {
|
|
|
|
if (!commands.hasOwnProperty(n))
|
|
|
|
continue;
|
|
|
|
const cmd = commands[n];
|
|
|
|
output[n] = cmd.metadata();
|
|
|
|
}
|
|
|
|
yield this.cache_.setItem('metadata', output, 1000 * 60 * 60 * 24);
|
|
|
|
this.commandMetadata_ = output;
|
|
|
|
return Object.assign({}, this.commandMetadata_);
|
|
|
|
});
|
|
|
|
}
|
|
|
|
hasGui() {
|
|
|
|
return this.gui() && !this.gui().isDummy();
|
|
|
|
}
|
|
|
|
findCommandByName(name) {
|
|
|
|
if (this.commands_[name])
|
|
|
|
return this.commands_[name];
|
|
|
|
let CommandClass = null;
|
|
|
|
try {
|
|
|
|
CommandClass = require(`${__dirname}/command-${name}.js`);
|
|
|
|
}
|
|
|
|
catch (error) {
|
|
|
|
if (error.message && error.message.indexOf('Cannot find module') >= 0) {
|
|
|
|
const e = new Error((0, locale_1._)('No such command: %s', name));
|
|
|
|
e.type = 'notFound';
|
|
|
|
throw e;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
let cmd = new CommandClass();
|
|
|
|
cmd = this.setupCommand(cmd);
|
|
|
|
this.commands_[name] = cmd;
|
|
|
|
return this.commands_[name];
|
|
|
|
}
|
|
|
|
dummyGui() {
|
|
|
|
return {
|
|
|
|
isDummy: () => {
|
|
|
|
return true;
|
|
|
|
},
|
|
|
|
prompt: (initialText = '', promptString = '', options = null) => {
|
|
|
|
return cliUtils.prompt(initialText, promptString, options);
|
|
|
|
},
|
|
|
|
showConsole: () => { },
|
|
|
|
maximizeConsole: () => { },
|
|
|
|
stdout: (text) => {
|
|
|
|
// eslint-disable-next-line no-console
|
|
|
|
console.info(text);
|
|
|
|
},
|
|
|
|
fullScreen: () => { },
|
|
|
|
exit: () => { },
|
|
|
|
showModalOverlay: () => { },
|
|
|
|
hideModalOverlay: () => { },
|
|
|
|
stdoutMaxWidth: () => {
|
|
|
|
return 100;
|
|
|
|
},
|
|
|
|
forceRender: () => { },
|
|
|
|
termSaveState: () => { },
|
|
|
|
termRestoreState: () => { },
|
|
|
|
};
|
|
|
|
}
|
|
|
|
execCommand(argv) {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
if (!argv.length)
|
|
|
|
return this.execCommand(['help']);
|
|
|
|
// reg.logger().debug('execCommand()', argv);
|
|
|
|
const commandName = argv[0];
|
|
|
|
this.activeCommand_ = this.findCommandByName(commandName);
|
|
|
|
let outException = null;
|
|
|
|
try {
|
|
|
|
if (this.gui().isDummy() && !this.activeCommand_.supportsUi('cli'))
|
|
|
|
throw new Error((0, locale_1._)('The command "%s" is only available in GUI mode', this.activeCommand_.name()));
|
|
|
|
const cmdArgs = cliUtils.makeCommandArgs(this.activeCommand_, argv);
|
|
|
|
yield this.activeCommand_.action(cmdArgs);
|
|
|
|
}
|
|
|
|
catch (error) {
|
|
|
|
outException = error;
|
|
|
|
}
|
|
|
|
this.activeCommand_ = null;
|
|
|
|
if (outException)
|
|
|
|
throw outException;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
currentCommand() {
|
|
|
|
return this.activeCommand_;
|
|
|
|
}
|
|
|
|
loadKeymaps() {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
const defaultKeyMap = [
|
|
|
|
{ keys: [':'], type: 'function', command: 'enter_command_line_mode' },
|
|
|
|
{ keys: ['TAB'], type: 'function', command: 'focus_next' },
|
|
|
|
{ keys: ['SHIFT_TAB'], type: 'function', command: 'focus_previous' },
|
|
|
|
{ keys: ['UP'], type: 'function', command: 'move_up' },
|
|
|
|
{ keys: ['DOWN'], type: 'function', command: 'move_down' },
|
|
|
|
{ keys: ['PAGE_UP'], type: 'function', command: 'page_up' },
|
|
|
|
{ keys: ['PAGE_DOWN'], type: 'function', command: 'page_down' },
|
|
|
|
{ keys: ['ENTER'], type: 'function', command: 'activate' },
|
|
|
|
{ keys: ['DELETE', 'BACKSPACE'], type: 'function', command: 'delete' },
|
|
|
|
{ keys: ['n'], type: 'function', command: 'next_link' },
|
|
|
|
{ keys: ['b'], type: 'function', command: 'previous_link' },
|
|
|
|
{ keys: ['o'], type: 'function', command: 'open_link' },
|
|
|
|
{ keys: [' '], type: 'prompt', command: 'todo toggle $n' },
|
|
|
|
{ keys: ['tc'], type: 'function', command: 'toggle_console' },
|
|
|
|
{ keys: ['tm'], type: 'function', command: 'toggle_metadata' },
|
|
|
|
{ keys: ['ti'], type: 'function', command: 'toggle_ids' },
|
|
|
|
{ keys: ['/'], type: 'prompt', command: 'search ""', cursorPosition: -2 },
|
|
|
|
{ keys: ['mn'], type: 'prompt', command: 'mknote ""', cursorPosition: -2 },
|
|
|
|
{ keys: ['mt'], type: 'prompt', command: 'mktodo ""', cursorPosition: -2 },
|
|
|
|
{ keys: ['mb'], type: 'prompt', command: 'mkbook ""', cursorPosition: -2 },
|
|
|
|
{ keys: ['yn'], type: 'prompt', command: 'cp $n ""', cursorPosition: -2 },
|
|
|
|
{ keys: ['dn'], type: 'prompt', command: 'mv $n ""', cursorPosition: -2 },
|
|
|
|
];
|
|
|
|
// Filter the keymap item by command so that items in keymap.json can override
|
|
|
|
// the default ones.
|
|
|
|
const itemsByCommand = {};
|
|
|
|
for (let i = 0; i < defaultKeyMap.length; i++) {
|
|
|
|
itemsByCommand[defaultKeyMap[i].command] = defaultKeyMap[i];
|
|
|
|
}
|
|
|
|
const filePath = `${Setting_1.default.value('profileDir')}/keymap.json`;
|
|
|
|
if (yield (0, fs_extra_1.pathExists)(filePath)) {
|
|
|
|
try {
|
|
|
|
let configString = yield (0, fs_extra_1.readFile)(filePath, 'utf-8');
|
|
|
|
configString = configString.replace(/^\s*\/\/.*/, ''); // Strip off comments
|
|
|
|
const keymap = JSON.parse(configString);
|
|
|
|
for (let keymapIndex = 0; keymapIndex < keymap.length; keymapIndex++) {
|
|
|
|
const item = keymap[keymapIndex];
|
|
|
|
itemsByCommand[item.command] = item;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (error) {
|
|
|
|
let msg = error.message ? error.message : '';
|
|
|
|
msg = `Could not load keymap ${filePath}\n${msg}`;
|
|
|
|
error.message = msg;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
const output = [];
|
|
|
|
for (const n in itemsByCommand) {
|
|
|
|
if (!itemsByCommand.hasOwnProperty(n))
|
|
|
|
continue;
|
|
|
|
output.push(itemsByCommand[n]);
|
|
|
|
}
|
|
|
|
// Map reserved shortcuts to their equivalent key
|
|
|
|
// https://github.com/cronvel/terminal-kit/issues/101
|
|
|
|
for (let i = 0; i < output.length; i++) {
|
|
|
|
const newKeys = output[i].keys.map(k => {
|
|
|
|
k = k.replace(/CTRL_H/g, 'BACKSPACE');
|
|
|
|
k = k.replace(/CTRL_I/g, 'TAB');
|
|
|
|
k = k.replace(/CTRL_M/g, 'ENTER');
|
|
|
|
return k;
|
|
|
|
});
|
|
|
|
output[i].keys = newKeys;
|
|
|
|
}
|
|
|
|
return output;
|
|
|
|
});
|
|
|
|
}
|
|
|
|
commandList(argv) {
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
if (argv.length && argv[0] === 'batch') {
|
|
|
|
const commands = [];
|
|
|
|
const commandLines = splitCommandBatch(yield (0, fs_extra_1.readFile)(argv[1], 'utf-8'));
|
|
|
|
for (const commandLine of commandLines) {
|
|
|
|
if (!commandLine.trim())
|
|
|
|
continue;
|
|
|
|
const splitted = (0, utils_1.splitCommandString)(commandLine.trim());
|
|
|
|
commands.push(splitted);
|
|
|
|
}
|
|
|
|
return commands;
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
return [argv];
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
|
|
|
// We need this special case here because by the time the `version` command
|
|
|
|
// runs, the keychain has already been setup.
|
|
|
|
checkIfKeychainEnabled(argv) {
|
|
|
|
return argv.indexOf('version') < 0;
|
|
|
|
}
|
|
|
|
start(argv) {
|
|
|
|
const _super = Object.create(null, {
|
|
|
|
start: { get: () => super.start }
|
|
|
|
});
|
|
|
|
return __awaiter(this, void 0, void 0, function* () {
|
|
|
|
const keychainEnabled = this.checkIfKeychainEnabled(argv);
|
|
|
|
argv = yield _super.start.call(this, argv, { keychainEnabled });
|
|
|
|
cliUtils.setStdout((object) => {
|
|
|
|
return this.stdout(object);
|
|
|
|
});
|
|
|
|
this.initRedux();
|
|
|
|
// If we have some arguments left at this point, it's a command
|
|
|
|
// so execute it.
|
|
|
|
if (argv.length) {
|
|
|
|
this.gui_ = this.dummyGui();
|
|
|
|
this.currentFolder_ = yield Folder_1.default.load(Setting_1.default.value('activeFolderId'));
|
|
|
|
yield this.applySettingsSideEffects();
|
|
|
|
try {
|
|
|
|
const commands = yield this.commandList(argv);
|
|
|
|
for (const command of commands) {
|
|
|
|
yield this.execCommand(command);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
catch (error) {
|
|
|
|
if (this.showStackTraces_) {
|
|
|
|
console.error(error);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// eslint-disable-next-line no-console
|
|
|
|
console.info(error.message);
|
|
|
|
}
|
|
|
|
process.exit(1);
|
|
|
|
}
|
|
|
|
yield Setting_1.default.saveAll();
|
|
|
|
// Need to call exit() explicitly, otherwise Node wait for any timeout to complete
|
|
|
|
// https://stackoverflow.com/questions/18050095
|
|
|
|
process.exit(0);
|
|
|
|
}
|
|
|
|
else {
|
|
|
|
// Otherwise open the GUI
|
|
|
|
const keymap = yield this.loadKeymaps();
|
|
|
|
const AppGui = require('./app-gui.js');
|
|
|
|
this.gui_ = new AppGui(this, this.store(), keymap);
|
|
|
|
this.gui_.setLogger(this.logger());
|
|
|
|
yield this.gui_.start();
|
|
|
|
// Since the settings need to be loaded before the store is created, it will never
|
|
|
|
// receive the SETTING_UPDATE_ALL even, which mean state.settings will not be
|
|
|
|
// initialised. So we manually call dispatchUpdateAll() to force an update.
|
|
|
|
Setting_1.default.dispatchUpdateAll();
|
|
|
|
yield (0, folders_screen_utils_js_1.refreshFolders)((action) => this.store().dispatch(action));
|
|
|
|
const tags = yield Tag_1.default.allWithNotes();
|
|
|
|
ResourceService_1.default.runInBackground();
|
|
|
|
RevisionService_1.default.instance().runInBackground();
|
|
|
|
this.dispatch({
|
|
|
|
type: 'TAG_UPDATE_ALL',
|
|
|
|
items: tags,
|
|
|
|
});
|
|
|
|
this.store().dispatch({
|
|
|
|
type: 'FOLDER_SELECT',
|
|
|
|
id: Setting_1.default.value('activeFolderId'),
|
|
|
|
});
|
|
|
|
this.startRotatingLogMaintenance(Setting_1.default.value('profileDir'));
|
|
|
|
}
|
|
|
|
});
|
|
|
|
}
|
2017-07-10 22:03:46 +02:00
|
|
|
}
|
|
|
|
let application_ = null;
|
|
|
|
function app() {
|
2024-01-20 16:29:21 +02:00
|
|
|
if (application_)
|
|
|
|
return application_;
|
|
|
|
application_ = new Application();
|
|
|
|
return application_;
|
2017-07-10 22:03:46 +02:00
|
|
|
}
|
2024-01-20 16:29:21 +02:00
|
|
|
exports.default = app;
|
|
|
|
//# sourceMappingURL=app.js.map
|