2020-11-05 16:58:23 +00:00
|
|
|
import shim from './shim';
|
|
|
|
import { _ } from './locale';
|
2021-01-29 18:45:11 +00:00
|
|
|
const { rtrimSlashes } = require('./path-utils.js');
|
2020-11-05 16:58:23 +00:00
|
|
|
const JoplinError = require('./JoplinError');
|
2021-01-29 18:45:11 +00:00
|
|
|
const { stringify } = require('query-string');
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
interface Options {
|
|
|
|
baseUrl(): string;
|
|
|
|
username(): string;
|
|
|
|
password(): string;
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
enum ExecOptionsResponseFormat {
|
|
|
|
Json = 'json',
|
|
|
|
Text = 'text',
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
enum ExecOptionsTarget {
|
|
|
|
String = 'string',
|
|
|
|
File = 'file',
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
interface ExecOptions {
|
|
|
|
responseFormat?: ExecOptionsResponseFormat;
|
|
|
|
target?: ExecOptionsTarget;
|
|
|
|
path?: string;
|
|
|
|
source?: string;
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
export default class JoplinServerApi {
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
private options_: Options;
|
|
|
|
private session_: any;
|
2021-02-01 10:48:37 +00:00
|
|
|
private debugRequests_: boolean = false;
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
public constructor(options: Options) {
|
|
|
|
this.options_ = options;
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
private baseUrl() {
|
2019-12-13 01:16:34 +00:00
|
|
|
return rtrimSlashes(this.options_.baseUrl());
|
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
private async session() {
|
|
|
|
// TODO: handle invalid session
|
|
|
|
if (this.session_) return this.session_;
|
|
|
|
|
|
|
|
this.session_ = await this.exec('POST', 'api/sessions', null, {
|
|
|
|
email: this.options_.username(),
|
|
|
|
password: this.options_.password(),
|
|
|
|
});
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
return this.session_;
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
private async sessionId() {
|
|
|
|
const session = await this.session();
|
|
|
|
return session ? session.id : '';
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
public async shareFile(pathOrId: string) {
|
|
|
|
return this.exec('POST', 'api/shares', null, {
|
|
|
|
file_id: pathOrId,
|
|
|
|
type: 1, // ShareType.Link
|
2019-12-13 01:16:34 +00:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
public static connectionErrorMessage(error: any) {
|
|
|
|
const msg = error && error.message ? error.message : 'Unknown error';
|
|
|
|
return _('Could not connect to Joplin Server. Please check the Synchronisation options in the config screen. Full error was:\n\n%s', msg);
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
public shareUrl(share: any): string {
|
|
|
|
return `${this.baseUrl()}/shares/${share.id}`;
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-02-01 10:48:37 +00:00
|
|
|
private requestToCurl_(url: string, options: any) {
|
|
|
|
const output = [];
|
|
|
|
output.push('curl');
|
|
|
|
output.push('-v');
|
|
|
|
if (options.method) output.push(`-X ${options.method}`);
|
|
|
|
if (options.headers) {
|
|
|
|
for (const n in options.headers) {
|
|
|
|
if (!options.headers.hasOwnProperty(n)) continue;
|
|
|
|
output.push(`${'-H ' + '"'}${n}: ${options.headers[n]}"`);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (options.body) output.push(`${'--data ' + '\''}${JSON.stringify(options.body)}'`);
|
|
|
|
output.push(url);
|
|
|
|
|
|
|
|
return output.join(' ');
|
|
|
|
}
|
2021-01-29 18:45:11 +00:00
|
|
|
|
|
|
|
public async exec(method: string, path: string = '', query: Record<string, any> = null, body: any = null, headers: any = null, options: ExecOptions = null) {
|
2019-12-13 01:16:34 +00:00
|
|
|
if (headers === null) headers = {};
|
|
|
|
if (options === null) options = {};
|
2021-01-29 18:45:11 +00:00
|
|
|
if (!options.responseFormat) options.responseFormat = ExecOptionsResponseFormat.Json;
|
|
|
|
if (!options.target) options.target = ExecOptionsTarget.String;
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
let sessionId = '';
|
|
|
|
if (path !== 'api/sessions' && !sessionId) {
|
|
|
|
sessionId = await this.sessionId();
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
if (sessionId) headers['X-API-AUTH'] = sessionId;
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2020-11-12 19:13:28 +00:00
|
|
|
const fetchOptions: any = {};
|
2019-12-13 01:16:34 +00:00
|
|
|
fetchOptions.headers = headers;
|
|
|
|
fetchOptions.method = method;
|
|
|
|
if (options.path) fetchOptions.path = options.path;
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
if (body) {
|
|
|
|
if (typeof body === 'object') {
|
|
|
|
fetchOptions.body = JSON.stringify(body);
|
|
|
|
fetchOptions.headers['Content-Type'] = 'application/json';
|
|
|
|
} else {
|
|
|
|
fetchOptions.body = body;
|
|
|
|
}
|
|
|
|
|
|
|
|
fetchOptions.headers['Content-Length'] = `${shim.stringByteLength(fetchOptions.body)}`;
|
|
|
|
}
|
|
|
|
|
|
|
|
let url = `${this.baseUrl()}/${path}`;
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
if (query) {
|
|
|
|
url += url.indexOf('?') < 0 ? '?' : '&';
|
|
|
|
url += stringify(query);
|
|
|
|
}
|
|
|
|
|
|
|
|
let response: any = null;
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-02-01 10:48:37 +00:00
|
|
|
if (this.debugRequests_) {
|
|
|
|
console.info('Joplin API Call', `${method} ${url}`, headers, options);
|
|
|
|
console.info(this.requestToCurl_(url, fetchOptions));
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
if (options.source == 'file' && (method == 'POST' || method == 'PUT')) {
|
|
|
|
if (fetchOptions.path) {
|
|
|
|
const fileStat = await shim.fsDriver().stat(fetchOptions.path);
|
|
|
|
if (fileStat) fetchOptions.headers['Content-Length'] = `${fileStat.size}`;
|
|
|
|
}
|
|
|
|
response = await shim.uploadBlob(url, fetchOptions);
|
|
|
|
} else if (options.target == 'string') {
|
|
|
|
if (typeof body === 'string') fetchOptions.headers['Content-Length'] = `${shim.stringByteLength(body)}`;
|
|
|
|
response = await shim.fetch(url, fetchOptions);
|
|
|
|
} else {
|
|
|
|
// file
|
|
|
|
response = await shim.fetchBlob(url, fetchOptions);
|
|
|
|
}
|
2019-12-13 01:16:34 +00:00
|
|
|
|
|
|
|
const responseText = await response.text();
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
// console.info('Joplin API Response', responseText);
|
|
|
|
|
|
|
|
// Creates an error object with as much data as possible as it will appear in the log, which will make debugging easier
|
|
|
|
const newError = (message: string, code: number = 0) => {
|
|
|
|
// Gives a shorter response for error messages. Useful for cases where a full HTML page is accidentally loaded instead of
|
|
|
|
// JSON. That way the error message will still show there's a problem but without filling up the log or screen.
|
|
|
|
const shortResponseText = (`${responseText}`).substr(0, 1024);
|
|
|
|
return new JoplinError(`${method} ${path}: ${message} (${code}): ${shortResponseText}`, code);
|
|
|
|
};
|
|
|
|
|
|
|
|
let responseJson_: any = null;
|
2019-12-13 01:16:34 +00:00
|
|
|
const loadResponseJson = async () => {
|
|
|
|
if (!responseText) return null;
|
|
|
|
if (responseJson_) return responseJson_;
|
2021-01-29 18:45:11 +00:00
|
|
|
responseJson_ = JSON.parse(responseText);
|
|
|
|
if (!responseJson_) throw newError('Cannot parse JSON response', response.status);
|
|
|
|
return responseJson_;
|
2019-12-13 01:16:34 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
if (!response.ok) {
|
2021-01-29 18:45:11 +00:00
|
|
|
if (options.target === 'file') throw newError('fetchBlob error', response.status);
|
|
|
|
|
2019-12-13 01:16:34 +00:00
|
|
|
let json = null;
|
|
|
|
try {
|
|
|
|
json = await loadResponseJson();
|
|
|
|
} catch (error) {
|
2021-01-29 18:45:11 +00:00
|
|
|
// Just send back the plain text in newErro()
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
if (json && json.error) {
|
|
|
|
throw newError(`${json.error}`, json.code ? json.code : response.status);
|
|
|
|
}
|
|
|
|
|
|
|
|
throw newError('Unknown error', response.status);
|
2019-12-13 01:16:34 +00:00
|
|
|
}
|
|
|
|
|
2021-01-29 18:45:11 +00:00
|
|
|
if (options.responseFormat === 'text') return responseText;
|
|
|
|
|
2019-12-13 01:16:34 +00:00
|
|
|
const output = await loadResponseJson();
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
}
|