mirror of
https://github.com/laurent22/joplin.git
synced 2024-12-24 10:27:10 +02:00
All: Refactored REST API to make it testable and to allow further extension
This commit is contained in:
parent
bc09d2c640
commit
8a619e4b8b
1
CliClient/.gitignore
vendored
1
CliClient/.gitignore
vendored
@ -13,6 +13,7 @@ tests/fuzzing.*
|
|||||||
tests/fuzzing -*
|
tests/fuzzing -*
|
||||||
tests/logs/*
|
tests/logs/*
|
||||||
tests/cli-integration/
|
tests/cli-integration/
|
||||||
|
tests/tmp/
|
||||||
*.mo
|
*.mo
|
||||||
*.*~
|
*.*~
|
||||||
tests/sync
|
tests/sync
|
||||||
|
874
CliClient/package-lock.json
generated
874
CliClient/package-lock.json
generated
File diff suppressed because it is too large
Load Diff
@ -37,6 +37,7 @@
|
|||||||
"fs-extra": "^5.0.0",
|
"fs-extra": "^5.0.0",
|
||||||
"html-entities": "^1.2.1",
|
"html-entities": "^1.2.1",
|
||||||
"html-minifier": "^3.5.15",
|
"html-minifier": "^3.5.15",
|
||||||
|
"image-data-uri": "^2.0.0",
|
||||||
"image-type": "^3.0.0",
|
"image-type": "^3.0.0",
|
||||||
"joplin-turndown": "^4.0.8",
|
"joplin-turndown": "^4.0.8",
|
||||||
"joplin-turndown-plugin-gfm": "^1.0.7",
|
"joplin-turndown-plugin-gfm": "^1.0.7",
|
||||||
@ -57,7 +58,7 @@
|
|||||||
"redux": "^3.7.2",
|
"redux": "^3.7.2",
|
||||||
"sax": "^1.2.2",
|
"sax": "^1.2.2",
|
||||||
"server-destroy": "^1.0.1",
|
"server-destroy": "^1.0.1",
|
||||||
"sharp": "^0.18.4",
|
"sharp": "^0.20.8",
|
||||||
"sprintf-js": "^1.1.1",
|
"sprintf-js": "^1.1.1",
|
||||||
"sqlite3": "^4.0.1",
|
"sqlite3": "^4.0.1",
|
||||||
"string-padding": "^1.0.2",
|
"string-padding": "^1.0.2",
|
||||||
|
118
CliClient/tests/services_rest_Api.js
Normal file
118
CliClient/tests/services_rest_Api.js
Normal file
@ -0,0 +1,118 @@
|
|||||||
|
require('app-module-path').addPath(__dirname);
|
||||||
|
|
||||||
|
const { time } = require('lib/time-utils.js');
|
||||||
|
const { fileContentEqual, setupDatabase, setupDatabaseAndSynchronizer, db, synchronizer, fileApi, sleep, clearDatabase, switchClient, syncTargetId, objectsEqual, checkThrowAsync } = require('test-utils.js');
|
||||||
|
const markdownUtils = require('lib/markdownUtils.js');
|
||||||
|
const Api = require('lib/services/rest/Api');
|
||||||
|
const Folder = require('lib/models/Folder');
|
||||||
|
const Resource = require('lib/models/Resource');
|
||||||
|
|
||||||
|
process.on('unhandledRejection', (reason, p) => {
|
||||||
|
console.log('Unhandled Rejection at: Promise', p, 'reason:', reason);
|
||||||
|
});
|
||||||
|
|
||||||
|
let api = null;
|
||||||
|
|
||||||
|
describe('services_rest_Api', function() {
|
||||||
|
|
||||||
|
beforeEach(async (done) => {
|
||||||
|
api = new Api();
|
||||||
|
await setupDatabaseAndSynchronizer(1);
|
||||||
|
await switchClient(1);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should ping', async (done) => {
|
||||||
|
const response = await api.route('GET', 'ping');
|
||||||
|
expect(response).toBe('JoplinClipperServer');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should handle Not Found errors', async (done) => {
|
||||||
|
const hasThrown = await checkThrowAsync(async () => await api.route('GET', 'pong'));
|
||||||
|
expect(hasThrown).toBe(true);
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should get folders', async (done) => {
|
||||||
|
let f1 = await Folder.save({ title: "mon carnet" });
|
||||||
|
const response = await api.route('GET', 'folders');
|
||||||
|
expect(response.length).toBe(1);
|
||||||
|
expect(response[0].title).toBe('mon carnet');
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create notes', async (done) => {
|
||||||
|
let response = null;
|
||||||
|
const f = await Folder.save({ title: "mon carnet" });
|
||||||
|
|
||||||
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
||||||
|
title: 'testing',
|
||||||
|
parent_id: f.id,
|
||||||
|
}));
|
||||||
|
expect(response.title).toBe('testing');
|
||||||
|
expect(!!response.id).toBe(true);
|
||||||
|
|
||||||
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
||||||
|
title: 'testing',
|
||||||
|
parent_id: f.id,
|
||||||
|
}));
|
||||||
|
expect(response.title).toBe('testing');
|
||||||
|
expect(!!response.id).toBe(true);
|
||||||
|
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create notes with images', async (done) => {
|
||||||
|
let response = null;
|
||||||
|
const f = await Folder.save({ title: "mon carnet" });
|
||||||
|
|
||||||
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
||||||
|
title: 'testing image',
|
||||||
|
parent_id: f.id,
|
||||||
|
image_data_url: ""
|
||||||
|
}));
|
||||||
|
|
||||||
|
const resources = await Resource.all();
|
||||||
|
expect(resources.length).toBe(1);
|
||||||
|
|
||||||
|
const resource = resources[0];
|
||||||
|
expect(response.body.indexOf(resource.id) >= 0).toBe(true);
|
||||||
|
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should create notes from HTML', async (done) => {
|
||||||
|
let response = null;
|
||||||
|
const f = await Folder.save({ title: "mon carnet" });
|
||||||
|
|
||||||
|
response = await api.route('POST', 'notes', null, JSON.stringify({
|
||||||
|
title: 'testing HTML',
|
||||||
|
parent_id: f.id,
|
||||||
|
body_html: '<b>Bold text</b>',
|
||||||
|
}));
|
||||||
|
|
||||||
|
expect(response.body).toBe('**Bold text**');
|
||||||
|
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
it('should filter fields', async (done) => {
|
||||||
|
let f = api.fields_({ query: { fields: 'one,two' } }, []);
|
||||||
|
expect(f.length).toBe(2);
|
||||||
|
expect(f[0]).toBe('one');
|
||||||
|
expect(f[1]).toBe('two');
|
||||||
|
|
||||||
|
f = api.fields_({ query: { fields: 'one ,, two ' } }, []);
|
||||||
|
expect(f.length).toBe(2);
|
||||||
|
expect(f[0]).toBe('one');
|
||||||
|
expect(f[1]).toBe('two');
|
||||||
|
|
||||||
|
f = api.fields_({ query: { fields: ' ' } }, ['def']);
|
||||||
|
expect(f.length).toBe(1);
|
||||||
|
expect(f[0]).toBe('def');
|
||||||
|
|
||||||
|
done();
|
||||||
|
});
|
||||||
|
|
||||||
|
});
|
@ -48,7 +48,9 @@ EncryptionService.fsDriver_ = fsDriver;
|
|||||||
FileApiDriverLocal.fsDriver_ = fsDriver;
|
FileApiDriverLocal.fsDriver_ = fsDriver;
|
||||||
|
|
||||||
const logDir = __dirname + '/../tests/logs';
|
const logDir = __dirname + '/../tests/logs';
|
||||||
|
const tempDir = __dirname + '/../tests/tmp';
|
||||||
fs.mkdirpSync(logDir, 0o755);
|
fs.mkdirpSync(logDir, 0o755);
|
||||||
|
fs.mkdirpSync(tempDir, 0o755);
|
||||||
|
|
||||||
SyncTargetRegistry.addClass(SyncTargetMemory);
|
SyncTargetRegistry.addClass(SyncTargetMemory);
|
||||||
SyncTargetRegistry.addClass(SyncTargetFilesystem);
|
SyncTargetRegistry.addClass(SyncTargetFilesystem);
|
||||||
@ -80,6 +82,7 @@ BaseItem.loadClass('MasterKey', MasterKey);
|
|||||||
|
|
||||||
Setting.setConstant('appId', 'net.cozic.joplin-cli');
|
Setting.setConstant('appId', 'net.cozic.joplin-cli');
|
||||||
Setting.setConstant('appType', 'cli');
|
Setting.setConstant('appType', 'cli');
|
||||||
|
Setting.setConstant('tempDir', tempDir);
|
||||||
|
|
||||||
BaseService.logger_ = logger;
|
BaseService.logger_ = logger;
|
||||||
|
|
||||||
|
@ -1,19 +1,10 @@
|
|||||||
const { netUtils } = require('lib/net-utils');
|
const { netUtils } = require('lib/net-utils');
|
||||||
const urlParser = require("url");
|
const urlParser = require("url");
|
||||||
const Note = require('lib/models/Note');
|
|
||||||
const Folder = require('lib/models/Folder');
|
|
||||||
const Resource = require('lib/models/Resource');
|
|
||||||
const Tag = require('lib/models/Tag');
|
|
||||||
const Setting = require('lib/models/Setting');
|
const Setting = require('lib/models/Setting');
|
||||||
const { shim } = require('lib/shim');
|
|
||||||
const md5 = require('md5');
|
|
||||||
const { fileExtension, safeFileExtension, safeFilename, filename } = require('lib/path-utils');
|
|
||||||
const HtmlToMd = require('lib/HtmlToMd');
|
|
||||||
const { Logger } = require('lib/logger.js');
|
const { Logger } = require('lib/logger.js');
|
||||||
const markdownUtils = require('lib/markdownUtils');
|
|
||||||
const mimeUtils = require('lib/mime-utils.js').mime;
|
|
||||||
const randomClipperPort = require('lib/randomClipperPort');
|
const randomClipperPort = require('lib/randomClipperPort');
|
||||||
const enableServerDestroy = require('server-destroy');
|
const enableServerDestroy = require('server-destroy');
|
||||||
|
const Api = require('lib/services/rest/Api');
|
||||||
|
|
||||||
class ClipperServer {
|
class ClipperServer {
|
||||||
|
|
||||||
@ -22,6 +13,7 @@ class ClipperServer {
|
|||||||
this.startState_ = 'idle';
|
this.startState_ = 'idle';
|
||||||
this.server_ = null;
|
this.server_ = null;
|
||||||
this.port_ = null;
|
this.port_ = null;
|
||||||
|
this.api_ = new Api();
|
||||||
}
|
}
|
||||||
|
|
||||||
static instance() {
|
static instance() {
|
||||||
@ -32,6 +24,7 @@ class ClipperServer {
|
|||||||
|
|
||||||
setLogger(l) {
|
setLogger(l) {
|
||||||
this.logger_ = l;
|
this.logger_ = l;
|
||||||
|
this.api_.setLogger(l);
|
||||||
}
|
}
|
||||||
|
|
||||||
logger() {
|
logger() {
|
||||||
@ -65,140 +58,6 @@ class ClipperServer {
|
|||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
|
||||||
htmlToMdParser() {
|
|
||||||
if (this.htmlToMdParser_) return this.htmlToMdParser_;
|
|
||||||
this.htmlToMdParser_ = new HtmlToMd();
|
|
||||||
return this.htmlToMdParser_;
|
|
||||||
}
|
|
||||||
|
|
||||||
async requestNoteToNote(requestNote) {
|
|
||||||
const output = {
|
|
||||||
title: requestNote.title ? requestNote.title : '',
|
|
||||||
body: requestNote.body ? requestNote.body : '',
|
|
||||||
};
|
|
||||||
|
|
||||||
if (requestNote.body_html) {
|
|
||||||
// Parsing will not work if the HTML is not wrapped in a top level tag, which is not guaranteed
|
|
||||||
// when getting the content from elsewhere. So here wrap it - it won't change anything to the final
|
|
||||||
// rendering but it makes sure everything will be parsed.
|
|
||||||
output.body = await this.htmlToMdParser().parse('<div>' + requestNote.body_html + '</div>', {
|
|
||||||
baseUrl: requestNote.base_url ? requestNote.base_url : '',
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
if (requestNote.parent_id) {
|
|
||||||
output.parent_id = requestNote.parent_id;
|
|
||||||
} else {
|
|
||||||
const folder = await Folder.defaultFolder();
|
|
||||||
if (!folder) throw new Error('Cannot find folder for note');
|
|
||||||
output.parent_id = folder.id;
|
|
||||||
}
|
|
||||||
|
|
||||||
if (requestNote.source_url) output.source_url = requestNote.source_url;
|
|
||||||
if (requestNote.author) output.author = requestNote.author;
|
|
||||||
|
|
||||||
return output;
|
|
||||||
}
|
|
||||||
|
|
||||||
// Note must have been saved first
|
|
||||||
async attachImageFromDataUrl_(note, imageDataUrl, cropRect) {
|
|
||||||
const tempDir = Setting.value('tempDir');
|
|
||||||
const mime = mimeUtils.fromDataUrl(imageDataUrl);
|
|
||||||
let ext = mimeUtils.toFileExtension(mime) || '';
|
|
||||||
if (ext) ext = '.' + ext;
|
|
||||||
const tempFilePath = tempDir + '/' + md5(Math.random() + '_' + Date.now()) + ext;
|
|
||||||
const imageConvOptions = {};
|
|
||||||
if (cropRect) imageConvOptions.cropRect = cropRect;
|
|
||||||
await shim.imageFromDataUrl(imageDataUrl, tempFilePath, imageConvOptions);
|
|
||||||
return await shim.attachFileToNote(note, tempFilePath);
|
|
||||||
}
|
|
||||||
|
|
||||||
async downloadImage_(url) {
|
|
||||||
const tempDir = Setting.value('tempDir');
|
|
||||||
|
|
||||||
const isDataUrl = url && url.toLowerCase().indexOf('data:') === 0;
|
|
||||||
|
|
||||||
const name = isDataUrl ? md5(Math.random() + '_' + Date.now()) : filename(url);
|
|
||||||
let fileExt = isDataUrl ? mimeUtils.toFileExtension(mimeUtils.fromDataUrl(url)) : safeFileExtension(fileExtension(url).toLowerCase());
|
|
||||||
if (fileExt) fileExt = '.' + fileExt;
|
|
||||||
let imagePath = tempDir + '/' + safeFilename(name) + fileExt;
|
|
||||||
if (await shim.fsDriver().exists(imagePath)) imagePath = tempDir + '/' + safeFilename(name) + '_' + md5(Math.random() + '_' + Date.now()).substr(0,10) + fileExt;
|
|
||||||
|
|
||||||
try {
|
|
||||||
if (isDataUrl) {
|
|
||||||
await shim.imageFromDataUrl(url, imagePath);
|
|
||||||
} else {
|
|
||||||
await shim.fetchBlob(url, { path: imagePath });
|
|
||||||
}
|
|
||||||
return imagePath;
|
|
||||||
} catch (error) {
|
|
||||||
this.logger().warn('Cannot download image at ' + url, error);
|
|
||||||
return '';
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
async downloadImages_(urls) {
|
|
||||||
const PromisePool = require('es6-promise-pool')
|
|
||||||
|
|
||||||
const output = {};
|
|
||||||
|
|
||||||
let urlIndex = 0;
|
|
||||||
const promiseProducer = () => {
|
|
||||||
if (urlIndex >= urls.length) return null;
|
|
||||||
|
|
||||||
const url = urls[urlIndex++];
|
|
||||||
|
|
||||||
return new Promise(async (resolve, reject) => {
|
|
||||||
const imagePath = await this.downloadImage_(url);
|
|
||||||
if (imagePath) output[url] = { path: imagePath };
|
|
||||||
resolve();
|
|
||||||
});
|
|
||||||
}
|
|
||||||
|
|
||||||
const concurrency = 3
|
|
||||||
const pool = new PromisePool(promiseProducer, concurrency)
|
|
||||||
await pool.start()
|
|
||||||
|
|
||||||
return output;
|
|
||||||
}
|
|
||||||
|
|
||||||
async createResourcesFromPaths_(urls) {
|
|
||||||
for (let url in urls) {
|
|
||||||
if (!urls.hasOwnProperty(url)) continue;
|
|
||||||
const urlInfo = urls[url];
|
|
||||||
try {
|
|
||||||
const resource = await shim.createResourceFromPath(urlInfo.path);
|
|
||||||
urlInfo.resource = resource;
|
|
||||||
} catch (error) {
|
|
||||||
this.logger().warn('Cannot create resource for ' + url, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
return urls;
|
|
||||||
}
|
|
||||||
|
|
||||||
async removeTempFiles_(urls) {
|
|
||||||
for (let url in urls) {
|
|
||||||
if (!urls.hasOwnProperty(url)) continue;
|
|
||||||
const urlInfo = urls[url];
|
|
||||||
try {
|
|
||||||
await shim.fsDriver().remove(urlInfo.path);
|
|
||||||
} catch (error) {
|
|
||||||
this.logger().warn('Cannot remove ' + urlInfo.path, error);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
replaceImageUrlsByResources_(md, urls) {
|
|
||||||
let output = md.replace(/(!\[.*?\]\()([^\s\)]+)(.*?\))/g, (match, before, imageUrl, after) => {
|
|
||||||
const urlInfo = urls[imageUrl];
|
|
||||||
if (!urlInfo || !urlInfo.resource) return before + imageUrl + after;
|
|
||||||
const resourceUrl = Resource.internalUrl(urlInfo.resource);
|
|
||||||
return before + resourceUrl + after;
|
|
||||||
});
|
|
||||||
|
|
||||||
return output;
|
|
||||||
}
|
|
||||||
|
|
||||||
async findAvailablePort() {
|
async findAvailablePort() {
|
||||||
const tcpPortUsed = require('tcp-port-used');
|
const tcpPortUsed = require('tcp-port-used');
|
||||||
|
|
||||||
@ -251,26 +110,33 @@ class ClipperServer {
|
|||||||
response.end();
|
response.end();
|
||||||
}
|
}
|
||||||
|
|
||||||
const requestId = Date.now();
|
const writeResponse = (code, response) => {
|
||||||
this.logger().info('Request (' + requestId + '): ' + request.method + ' ' + request.url);
|
if (typeof response === 'string') {
|
||||||
|
writeResponseText(code, response);
|
||||||
|
} else {
|
||||||
|
writeResponseJson(code, response);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger().info('Request: ' + request.method + ' ' + request.url);
|
||||||
|
|
||||||
const url = urlParser.parse(request.url, true);
|
const url = urlParser.parse(request.url, true);
|
||||||
|
|
||||||
if (request.method === 'GET') {
|
const execRequest = async (request, body = '') => {
|
||||||
if (url.pathname === '/ping') {
|
try {
|
||||||
return writeResponseText(200, 'JoplinClipperServer');
|
const response = await this.api_.route(request.method, url.pathname, url.query, body);
|
||||||
|
writeResponse(200, response);
|
||||||
|
} catch (error) {
|
||||||
|
console.error(error);
|
||||||
|
writeResponse(error.httpCode ? error.httpCode : 500, error.message);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if (url.pathname === '/folders') {
|
if (request.method === 'OPTIONS') {
|
||||||
const structure = await Folder.allAsTree({ fields: ['id', 'parent_id', 'title'] });
|
writeCorsHeaders(200);
|
||||||
return writeResponseJson(200, structure);
|
response.end();
|
||||||
}
|
} else {
|
||||||
|
if (request.method === 'POST') {
|
||||||
if (url.pathname === '/tags') {
|
|
||||||
return writeResponseJson(200, await Tag.all({ fields: ['id', 'title'] }));
|
|
||||||
}
|
|
||||||
} else if (request.method === 'POST') {
|
|
||||||
if (url.pathname === '/notes') {
|
|
||||||
let body = '';
|
let body = '';
|
||||||
|
|
||||||
request.on('data', (data) => {
|
request.on('data', (data) => {
|
||||||
@ -278,57 +144,14 @@ class ClipperServer {
|
|||||||
});
|
});
|
||||||
|
|
||||||
request.on('end', async () => {
|
request.on('end', async () => {
|
||||||
try {
|
execRequest(request, body);
|
||||||
const requestNote = JSON.parse(body);
|
|
||||||
let note = await this.requestNoteToNote(requestNote);
|
|
||||||
|
|
||||||
const imageUrls = markdownUtils.extractImageUrls(note.body);
|
|
||||||
|
|
||||||
this.logger().info('Request (' + requestId + '): Downloading images: ' + imageUrls.length);
|
|
||||||
|
|
||||||
let result = await this.downloadImages_(imageUrls);
|
|
||||||
|
|
||||||
this.logger().info('Request (' + requestId + '): Creating resources from paths: ' + Object.getOwnPropertyNames(result).length);
|
|
||||||
|
|
||||||
result = await this.createResourcesFromPaths_(result);
|
|
||||||
await this.removeTempFiles_(result);
|
|
||||||
note.body = this.replaceImageUrlsByResources_(note.body, result);
|
|
||||||
|
|
||||||
this.logger().info('Request (' + requestId + '): Saving note...');
|
|
||||||
|
|
||||||
note = await Note.save(note);
|
|
||||||
|
|
||||||
if (requestNote.tags) {
|
|
||||||
const tagTitles = requestNote.tags.split(',');
|
|
||||||
await Tag.setNoteTagsByTitles(note.id, tagTitles);
|
|
||||||
}
|
|
||||||
|
|
||||||
if (requestNote.image_data_url) {
|
|
||||||
await this.attachImageFromDataUrl_(note, requestNote.image_data_url, requestNote.crop_rect);
|
|
||||||
}
|
|
||||||
|
|
||||||
this.logger().info('Request (' + requestId + '): Created note ' + note.id);
|
|
||||||
return writeResponseJson(200, note);
|
|
||||||
} catch (error) {
|
|
||||||
this.logger().error(error);
|
|
||||||
return writeResponseJson(400, { errorCode: 'exception', errorMessage: error.message });
|
|
||||||
}
|
|
||||||
});
|
});
|
||||||
} else {
|
} else {
|
||||||
return writeResponseJson(404, { errorCode: 'not_found' });
|
execRequest(request);
|
||||||
}
|
}
|
||||||
} else if (request.method === 'OPTIONS') {
|
|
||||||
writeCorsHeaders(200);
|
|
||||||
response.end();
|
|
||||||
} else {
|
|
||||||
return writeResponseJson(405, { errorCode: 'method_not_allowed' });
|
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
this.server_.on('close', () => {
|
|
||||||
|
|
||||||
});
|
|
||||||
|
|
||||||
enableServerDestroy(this.server_);
|
enableServerDestroy(this.server_);
|
||||||
|
|
||||||
this.logger().info('Starting Clipper server on port ' + this.port_);
|
this.logger().info('Starting Clipper server on port ' + this.port_);
|
||||||
|
288
ReactNativeClient/lib/services/rest/Api.js
Normal file
288
ReactNativeClient/lib/services/rest/Api.js
Normal file
@ -0,0 +1,288 @@
|
|||||||
|
const { ltrimSlashes } = require('lib/path-utils.js');
|
||||||
|
const Folder = require('lib/models/Folder');
|
||||||
|
const Note = require('lib/models/Note');
|
||||||
|
const Tag = require('lib/models/Tag');
|
||||||
|
const Setting = require('lib/models/Setting');
|
||||||
|
const markdownUtils = require('lib/markdownUtils');
|
||||||
|
const mimeUtils = require('lib/mime-utils.js').mime;
|
||||||
|
const { Logger } = require('lib/logger.js');
|
||||||
|
const md5 = require('md5');
|
||||||
|
const { shim } = require('lib/shim');
|
||||||
|
const HtmlToMd = require('lib/HtmlToMd');
|
||||||
|
const { fileExtension, safeFileExtension, safeFilename, filename } = require('lib/path-utils');
|
||||||
|
|
||||||
|
class ApiError extends Error {
|
||||||
|
|
||||||
|
constructor(message, httpCode = 400) {
|
||||||
|
super(message);
|
||||||
|
this.httpCode_ = httpCode;
|
||||||
|
}
|
||||||
|
|
||||||
|
get httpCode() {
|
||||||
|
return this.httpCode_;
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class MethodNotAllowedError extends ApiError {
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super('Method Not Allowed', 405);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class NotFoundError extends ApiError {
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
super('Not Found', 404);
|
||||||
|
}
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
class Api {
|
||||||
|
|
||||||
|
constructor() {
|
||||||
|
this.logger_ = new Logger();
|
||||||
|
}
|
||||||
|
|
||||||
|
async route(method, path, query = null, body = null) {
|
||||||
|
path = ltrimSlashes(path);
|
||||||
|
const callName = 'action_' + path;
|
||||||
|
if (!this[callName]) throw new NotFoundError();
|
||||||
|
|
||||||
|
try {
|
||||||
|
return this[callName]({
|
||||||
|
method: method,
|
||||||
|
query: query ? query : {},
|
||||||
|
body: body,
|
||||||
|
});
|
||||||
|
} catch (error) {
|
||||||
|
if (!error.httpCode) error.httpCode = 500;
|
||||||
|
throw error;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
setLogger(l) {
|
||||||
|
this.logger_ = l;
|
||||||
|
}
|
||||||
|
|
||||||
|
logger() {
|
||||||
|
return this.logger_;
|
||||||
|
}
|
||||||
|
|
||||||
|
fields_(request, defaultFields) {
|
||||||
|
const query = request.query;
|
||||||
|
if (!query || !query.fields) return defaultFields;
|
||||||
|
const fields = query.fields.split(',').map(f => f.trim()).filter(f => !!f);
|
||||||
|
return fields.length ? fields : defaultFields;
|
||||||
|
}
|
||||||
|
|
||||||
|
async action_ping(request) {
|
||||||
|
if (request.method === 'GET') {
|
||||||
|
return 'JoplinClipperServer';
|
||||||
|
}
|
||||||
|
throw new MethodNotAllowedError();
|
||||||
|
}
|
||||||
|
|
||||||
|
async action_folders(request) {
|
||||||
|
if (request.method === 'GET') {
|
||||||
|
return await Folder.allAsTree({ fields: this.fields_(request, ['id', 'parent_id', 'title']) });
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new MethodNotAllowedError();
|
||||||
|
}
|
||||||
|
|
||||||
|
async action_tags(request) {
|
||||||
|
if (request.method === 'GET') {
|
||||||
|
return await Tag.all({ fields: this.fields_(request, ['id', 'title']) })
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new MethodNotAllowedError();
|
||||||
|
}
|
||||||
|
|
||||||
|
async action_notes(request) {
|
||||||
|
if (request.method === 'POST') {
|
||||||
|
const requestId = Date.now();
|
||||||
|
const requestNote = JSON.parse(request.body);
|
||||||
|
let note = await this.requestNoteToNote(requestNote);
|
||||||
|
|
||||||
|
const imageUrls = markdownUtils.extractImageUrls(note.body);
|
||||||
|
|
||||||
|
this.logger().info('Request (' + requestId + '): Downloading images: ' + imageUrls.length);
|
||||||
|
|
||||||
|
let result = await this.downloadImages_(imageUrls);
|
||||||
|
|
||||||
|
this.logger().info('Request (' + requestId + '): Creating resources from paths: ' + Object.getOwnPropertyNames(result).length);
|
||||||
|
|
||||||
|
result = await this.createResourcesFromPaths_(result);
|
||||||
|
await this.removeTempFiles_(result);
|
||||||
|
note.body = this.replaceImageUrlsByResources_(note.body, result);
|
||||||
|
|
||||||
|
this.logger().info('Request (' + requestId + '): Saving note...');
|
||||||
|
|
||||||
|
note = await Note.save(note);
|
||||||
|
|
||||||
|
if (requestNote.tags) {
|
||||||
|
const tagTitles = requestNote.tags.split(',');
|
||||||
|
await Tag.setNoteTagsByTitles(note.id, tagTitles);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (requestNote.image_data_url) {
|
||||||
|
note = await this.attachImageFromDataUrl_(note, requestNote.image_data_url, requestNote.crop_rect);
|
||||||
|
}
|
||||||
|
|
||||||
|
this.logger().info('Request (' + requestId + '): Created note ' + note.id);
|
||||||
|
|
||||||
|
return note;
|
||||||
|
}
|
||||||
|
|
||||||
|
throw new MethodNotAllowedError();
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
// ========================================================================================================================
|
||||||
|
// UTILIY FUNCTIONS
|
||||||
|
// ========================================================================================================================
|
||||||
|
|
||||||
|
htmlToMdParser() {
|
||||||
|
if (this.htmlToMdParser_) return this.htmlToMdParser_;
|
||||||
|
this.htmlToMdParser_ = new HtmlToMd();
|
||||||
|
return this.htmlToMdParser_;
|
||||||
|
}
|
||||||
|
|
||||||
|
async requestNoteToNote(requestNote) {
|
||||||
|
const output = {
|
||||||
|
title: requestNote.title ? requestNote.title : '',
|
||||||
|
body: requestNote.body ? requestNote.body : '',
|
||||||
|
};
|
||||||
|
|
||||||
|
if (requestNote.body_html) {
|
||||||
|
// Parsing will not work if the HTML is not wrapped in a top level tag, which is not guaranteed
|
||||||
|
// when getting the content from elsewhere. So here wrap it - it won't change anything to the final
|
||||||
|
// rendering but it makes sure everything will be parsed.
|
||||||
|
output.body = await this.htmlToMdParser().parse('<div>' + requestNote.body_html + '</div>', {
|
||||||
|
baseUrl: requestNote.base_url ? requestNote.base_url : '',
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (requestNote.parent_id) {
|
||||||
|
output.parent_id = requestNote.parent_id;
|
||||||
|
} else {
|
||||||
|
const folder = await Folder.defaultFolder();
|
||||||
|
if (!folder) throw new Error('Cannot find folder for note');
|
||||||
|
output.parent_id = folder.id;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (requestNote.source_url) output.source_url = requestNote.source_url;
|
||||||
|
if (requestNote.author) output.author = requestNote.author;
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
// Note must have been saved first
|
||||||
|
async attachImageFromDataUrl_(note, imageDataUrl, cropRect) {
|
||||||
|
const tempDir = Setting.value('tempDir');
|
||||||
|
const mime = mimeUtils.fromDataUrl(imageDataUrl);
|
||||||
|
let ext = mimeUtils.toFileExtension(mime) || '';
|
||||||
|
if (ext) ext = '.' + ext;
|
||||||
|
const tempFilePath = tempDir + '/' + md5(Math.random() + '_' + Date.now()) + ext;
|
||||||
|
const imageConvOptions = {};
|
||||||
|
if (cropRect) imageConvOptions.cropRect = cropRect;
|
||||||
|
await shim.imageFromDataUrl(imageDataUrl, tempFilePath, imageConvOptions);
|
||||||
|
return await shim.attachFileToNote(note, tempFilePath);
|
||||||
|
}
|
||||||
|
|
||||||
|
async downloadImage_(url) {
|
||||||
|
const tempDir = Setting.value('tempDir');
|
||||||
|
|
||||||
|
const isDataUrl = url && url.toLowerCase().indexOf('data:') === 0;
|
||||||
|
|
||||||
|
const name = isDataUrl ? md5(Math.random() + '_' + Date.now()) : filename(url);
|
||||||
|
let fileExt = isDataUrl ? mimeUtils.toFileExtension(mimeUtils.fromDataUrl(url)) : safeFileExtension(fileExtension(url).toLowerCase());
|
||||||
|
if (fileExt) fileExt = '.' + fileExt;
|
||||||
|
let imagePath = tempDir + '/' + safeFilename(name) + fileExt;
|
||||||
|
if (await shim.fsDriver().exists(imagePath)) imagePath = tempDir + '/' + safeFilename(name) + '_' + md5(Math.random() + '_' + Date.now()).substr(0,10) + fileExt;
|
||||||
|
|
||||||
|
try {
|
||||||
|
if (isDataUrl) {
|
||||||
|
await shim.imageFromDataUrl(url, imagePath);
|
||||||
|
} else {
|
||||||
|
await shim.fetchBlob(url, { path: imagePath });
|
||||||
|
}
|
||||||
|
return imagePath;
|
||||||
|
} catch (error) {
|
||||||
|
this.logger().warn('Cannot download image at ' + url, error);
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async downloadImages_(urls) {
|
||||||
|
const PromisePool = require('es6-promise-pool')
|
||||||
|
|
||||||
|
const output = {};
|
||||||
|
|
||||||
|
let urlIndex = 0;
|
||||||
|
const promiseProducer = () => {
|
||||||
|
if (urlIndex >= urls.length) return null;
|
||||||
|
|
||||||
|
const url = urls[urlIndex++];
|
||||||
|
|
||||||
|
return new Promise(async (resolve, reject) => {
|
||||||
|
const imagePath = await this.downloadImage_(url);
|
||||||
|
if (imagePath) output[url] = { path: imagePath };
|
||||||
|
resolve();
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
const concurrency = 3
|
||||||
|
const pool = new PromisePool(promiseProducer, concurrency)
|
||||||
|
await pool.start()
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
async createResourcesFromPaths_(urls) {
|
||||||
|
for (let url in urls) {
|
||||||
|
if (!urls.hasOwnProperty(url)) continue;
|
||||||
|
const urlInfo = urls[url];
|
||||||
|
try {
|
||||||
|
const resource = await shim.createResourceFromPath(urlInfo.path);
|
||||||
|
urlInfo.resource = resource;
|
||||||
|
} catch (error) {
|
||||||
|
this.logger().warn('Cannot create resource for ' + url, error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return urls;
|
||||||
|
}
|
||||||
|
|
||||||
|
async removeTempFiles_(urls) {
|
||||||
|
for (let url in urls) {
|
||||||
|
if (!urls.hasOwnProperty(url)) continue;
|
||||||
|
const urlInfo = urls[url];
|
||||||
|
try {
|
||||||
|
await shim.fsDriver().remove(urlInfo.path);
|
||||||
|
} catch (error) {
|
||||||
|
this.logger().warn('Cannot remove ' + urlInfo.path, error);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
replaceImageUrlsByResources_(md, urls) {
|
||||||
|
let output = md.replace(/(!\[.*?\]\()([^\s\)]+)(.*?\))/g, (match, before, imageUrl, after) => {
|
||||||
|
const urlInfo = urls[imageUrl];
|
||||||
|
if (!urlInfo || !urlInfo.resource) return before + imageUrl + after;
|
||||||
|
const resourceUrl = Resource.internalUrl(urlInfo.resource);
|
||||||
|
return before + resourceUrl + after;
|
||||||
|
});
|
||||||
|
|
||||||
|
return output;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
}
|
||||||
|
|
||||||
|
module.exports = Api;
|
@ -186,7 +186,11 @@ function shimInit() {
|
|||||||
const mime = mimeUtils.fromDataUrl(imageDataUrl);
|
const mime = mimeUtils.fromDataUrl(imageDataUrl);
|
||||||
await shim.writeImageToFile(image, mime, filePath);
|
await shim.writeImageToFile(image, mime, filePath);
|
||||||
} else {
|
} else {
|
||||||
throw new Error('Node support not implemented');
|
if (options.cropRect) throw new Error('Crop rect not supported in Node');
|
||||||
|
|
||||||
|
const imageDataURI = require('image-data-uri');
|
||||||
|
const result = imageDataURI.decode(imageDataUrl);
|
||||||
|
await shim.fsDriver().writeFile(filePath, result.dataBuffer, 'buffer');
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user