mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-11 18:24:43 +02:00
165 lines
4.8 KiB
TypeScript
165 lines
4.8 KiB
TypeScript
const { Logger } = require('lib/logger.js');
|
|
const { shim } = require('lib/shim.js');
|
|
const JoplinError = require('lib/JoplinError');
|
|
const { rtrimSlashes } = require('lib/path-utils.js');
|
|
const base64 = require('base-64');
|
|
const { _ } = require('lib/locale');
|
|
|
|
interface JoplinServerApiOptions {
|
|
username: Function,
|
|
password: Function,
|
|
baseUrl: Function,
|
|
}
|
|
|
|
export default class JoplinServerApi {
|
|
|
|
logger_:any;
|
|
options_:JoplinServerApiOptions;
|
|
kvStore_:any;
|
|
|
|
constructor(options:JoplinServerApiOptions) {
|
|
this.logger_ = new Logger();
|
|
this.options_ = options;
|
|
this.kvStore_ = null;
|
|
}
|
|
|
|
setLogger(l:any) {
|
|
this.logger_ = l;
|
|
}
|
|
|
|
logger():any {
|
|
return this.logger_;
|
|
}
|
|
|
|
setKvStore(v:any) {
|
|
this.kvStore_ = v;
|
|
}
|
|
|
|
kvStore() {
|
|
if (!this.kvStore_) throw new Error('JoplinServerApi.kvStore_ is not set!!');
|
|
return this.kvStore_;
|
|
}
|
|
|
|
authToken():string {
|
|
if (!this.options_.username() || !this.options_.password()) return null;
|
|
try {
|
|
// Note: Non-ASCII passwords will throw an error about Latin1 characters - https://github.com/laurent22/joplin/issues/246
|
|
// Tried various things like the below, but it didn't work on React Native:
|
|
// return base64.encode(utf8.encode(this.options_.username() + ':' + this.options_.password()));
|
|
return base64.encode(`${this.options_.username()}:${this.options_.password()}`);
|
|
} catch (error) {
|
|
error.message = `Cannot encode username/password: ${error.message}`;
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
baseUrl():string {
|
|
return rtrimSlashes(this.options_.baseUrl());
|
|
}
|
|
|
|
static baseUrlFromNextcloudWebDavUrl(webDavUrl:string) {
|
|
// http://nextcloud.local/remote.php/webdav/Joplin
|
|
// http://nextcloud.local/index.php/apps/joplin/api
|
|
const splitted = webDavUrl.split('/remote.php/webdav');
|
|
if (splitted.length !== 2) throw new Error(`Unsupported WebDAV URL format: ${webDavUrl}`);
|
|
return `${splitted[0]}/index.php/apps/joplin/api`;
|
|
}
|
|
|
|
syncTargetId(settings:any) {
|
|
const s = settings['sync.5.syncTargets'][settings['sync.5.path']];
|
|
if (!s) throw new Error(`Joplin Nextcloud app not configured for URL: ${this.baseUrl()}`);
|
|
return s.uuid;
|
|
}
|
|
|
|
static connectionErrorMessage(error:any) {
|
|
const msg = error && error.message ? error.message : 'Unknown error';
|
|
return _('Could not connect to the Joplin Nextcloud app. Please check the configuration in the Synchronisation config screen. Full error was:\n\n%s', msg);
|
|
}
|
|
|
|
async setupSyncTarget(webDavUrl:string) {
|
|
return this.exec('POST', 'sync_targets', {
|
|
webDavUrl: webDavUrl,
|
|
});
|
|
}
|
|
|
|
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 ' + '\''}${options.body}'`);
|
|
output.push(url);
|
|
|
|
return output.join(' ');
|
|
}
|
|
|
|
async exec(method:string, path:string = '', body:any = null, headers:any = null, options:any = null):Promise<any> {
|
|
if (headers === null) headers = {};
|
|
if (options === null) options = {};
|
|
|
|
const authToken = this.authToken();
|
|
|
|
if (authToken) headers['Authorization'] = `Basic ${authToken}`;
|
|
|
|
headers['Content-Type'] = 'application/json';
|
|
|
|
if (typeof body === 'object' && body !== null) body = JSON.stringify(body);
|
|
|
|
const fetchOptions:any = {};
|
|
fetchOptions.headers = headers;
|
|
fetchOptions.method = method;
|
|
if (options.path) fetchOptions.path = options.path;
|
|
if (body) fetchOptions.body = body;
|
|
|
|
const url = `${this.baseUrl()}/${path}`;
|
|
|
|
let response = null;
|
|
|
|
// console.info('WebDAV Call', method + ' ' + url, headers, options);
|
|
console.info(this.requestToCurl_(url, fetchOptions));
|
|
|
|
if (typeof body === 'string') fetchOptions.headers['Content-Length'] = `${shim.stringByteLength(body)}`;
|
|
response = await shim.fetch(url, fetchOptions);
|
|
|
|
const responseText = await response.text();
|
|
|
|
const responseJson_:any = null;
|
|
const loadResponseJson = async () => {
|
|
if (!responseText) return null;
|
|
if (responseJson_) return responseJson_;
|
|
try {
|
|
return JSON.parse(responseText);
|
|
} catch (error) {
|
|
throw new Error(`Cannot parse JSON: ${responseText.substr(0, 8192)}`);
|
|
}
|
|
};
|
|
|
|
const newError = (message:string, code:number = 0) => {
|
|
return new JoplinError(`${method} ${path}: ${message} (${code})`, code);
|
|
};
|
|
|
|
if (!response.ok) {
|
|
let json = null;
|
|
try {
|
|
json = await loadResponseJson();
|
|
} catch (error) {
|
|
throw newError(`Unknown error: ${responseText.substr(0, 8192)}`, response.status);
|
|
}
|
|
|
|
const trace = json.stacktrace ? `\n${json.stacktrace}` : '';
|
|
let message = json.error;
|
|
if (!message) message = responseText.substr(0, 8192);
|
|
throw newError(message + trace, response.status);
|
|
}
|
|
|
|
const output = await loadResponseJson();
|
|
return output;
|
|
}
|
|
}
|