mirror of
https://github.com/laurent22/joplin.git
synced 2025-01-05 12:50:29 +02:00
270 lines
6.8 KiB
TypeScript
270 lines
6.8 KiB
TypeScript
import { PaginationOrderDir } from '../../models/utils/types';
|
|
import { ErrorMethodNotAllowed, ErrorForbidden, ErrorBadRequest, ErrorNotFound } from './utils/errors';
|
|
|
|
import route_folders from './routes/folders';
|
|
import route_notes from './routes/notes';
|
|
import route_resources from './routes/resources';
|
|
import route_tags from './routes/tags';
|
|
import route_master_keys from './routes/master_keys';
|
|
import route_search from './routes/search';
|
|
import route_ping from './routes/ping';
|
|
import route_auth from './routes/auth';
|
|
|
|
const { ltrimSlashes } = require('../../path-utils');
|
|
const md5 = require('md5');
|
|
|
|
export enum RequestMethod {
|
|
GET = 'GET',
|
|
POST = 'POST',
|
|
PUT = 'PUT',
|
|
DELETE = 'DELETE',
|
|
}
|
|
|
|
export interface RequestFile {
|
|
path: string;
|
|
}
|
|
|
|
interface RequestQuery {
|
|
fields?: string[] | string;
|
|
token?: string;
|
|
nounce?: string;
|
|
page?: number;
|
|
|
|
// Search engine query
|
|
query?: string;
|
|
type?: string; // Model type as a string (eg. "note", "folder")
|
|
|
|
as_tree?: number;
|
|
|
|
// Pagination
|
|
limit?: number;
|
|
order_dir?: PaginationOrderDir;
|
|
order_by?: string;
|
|
|
|
// Auth token
|
|
auth_token?: string;
|
|
}
|
|
|
|
export interface Request {
|
|
method: RequestMethod;
|
|
path: string;
|
|
query: RequestQuery;
|
|
body: any;
|
|
bodyJson_: any;
|
|
bodyJson: any;
|
|
files: RequestFile[];
|
|
params: any[];
|
|
action?: any;
|
|
}
|
|
|
|
export enum AuthTokenStatus {
|
|
Waiting = 'waiting',
|
|
Accepted = 'accepted',
|
|
Rejected = 'rejected',
|
|
}
|
|
|
|
interface AuthToken {
|
|
value: string;
|
|
status: AuthTokenStatus;
|
|
}
|
|
|
|
export interface RequestContext {
|
|
dispatch: Function;
|
|
authToken: AuthToken;
|
|
token: string;
|
|
}
|
|
|
|
type RouteFunction = (request: Request, id: string, link: string, context: RequestContext)=> Promise<any | void>;
|
|
|
|
interface ResourceNameToRoute {
|
|
[key: string]: RouteFunction;
|
|
}
|
|
|
|
export default class Api {
|
|
|
|
private token_: string | Function;
|
|
private authToken_: AuthToken = null;
|
|
private knownNounces_: any = {};
|
|
private actionApi_: any;
|
|
private resourceNameToRoute_: ResourceNameToRoute = {};
|
|
private dispatch_: Function;
|
|
|
|
public constructor(token: string | Function = null, dispatch: Function = null, actionApi: any = null) {
|
|
this.token_ = token;
|
|
this.actionApi_ = actionApi;
|
|
this.dispatch_ = dispatch;
|
|
|
|
this.resourceNameToRoute_ = {
|
|
ping: route_ping,
|
|
notes: route_notes,
|
|
folders: route_folders,
|
|
tags: route_tags,
|
|
resources: route_resources,
|
|
master_keys: route_master_keys,
|
|
search: route_search,
|
|
services: this.action_services.bind(this),
|
|
auth: route_auth,
|
|
};
|
|
|
|
this.dispatch = this.dispatch.bind(this);
|
|
}
|
|
|
|
public get token(): string {
|
|
return typeof this.token_ === 'function' ? this.token_() : this.token_;
|
|
}
|
|
|
|
private dispatch(action: any) {
|
|
if (action.type === 'API_AUTH_TOKEN_SET') {
|
|
this.authToken_ = {
|
|
value: action.value,
|
|
status: AuthTokenStatus.Waiting,
|
|
};
|
|
|
|
this.dispatch_({
|
|
type: 'API_NEED_AUTH_SET',
|
|
value: true,
|
|
});
|
|
|
|
return;
|
|
}
|
|
|
|
return this.dispatch_(action);
|
|
}
|
|
|
|
public acceptAuthToken(accept: boolean) {
|
|
if (!this.authToken_) throw new Error('Auth token is not set');
|
|
|
|
this.authToken_.status = accept ? AuthTokenStatus.Accepted : AuthTokenStatus.Rejected;
|
|
|
|
this.dispatch_({
|
|
type: 'API_NEED_AUTH_SET',
|
|
value: false,
|
|
});
|
|
}
|
|
|
|
private parsePath(path: string) {
|
|
path = ltrimSlashes(path);
|
|
if (!path) return { fn: null, params: [] };
|
|
|
|
const pathParts = path.split('/');
|
|
const callSuffix = pathParts.splice(0, 1)[0];
|
|
const fn = this.resourceNameToRoute_[callSuffix];
|
|
|
|
return {
|
|
fn: fn,
|
|
params: pathParts,
|
|
};
|
|
}
|
|
|
|
// Response can be any valid JSON object, so a string, and array or an object (key/value pairs).
|
|
public async route(method: RequestMethod, path: string, query: RequestQuery = null, body: any = null, files: RequestFile[] = null): Promise<any> {
|
|
if (!files) files = [];
|
|
if (!query) query = {};
|
|
|
|
const parsedPath = this.parsePath(path);
|
|
if (!parsedPath.fn) throw new ErrorNotFound(); // Nothing at the root yet
|
|
|
|
if (query && query.nounce) {
|
|
const requestMd5 = md5(JSON.stringify([method, path, body, query, files.length]));
|
|
if (this.knownNounces_[query.nounce] === requestMd5) {
|
|
throw new ErrorBadRequest('Duplicate Nounce');
|
|
}
|
|
this.knownNounces_[query.nounce] = requestMd5;
|
|
}
|
|
|
|
let id = null;
|
|
let link = null;
|
|
const params = parsedPath.params;
|
|
|
|
if (params.length >= 1) {
|
|
id = params[0];
|
|
params.splice(0, 1);
|
|
if (params.length >= 1) {
|
|
link = params[0];
|
|
params.splice(0, 1);
|
|
}
|
|
}
|
|
|
|
const request: Request = {
|
|
method,
|
|
path: ltrimSlashes(path),
|
|
query: query ? query : {},
|
|
body,
|
|
bodyJson_: null,
|
|
bodyJson: function(disallowedProperties: string[] = null) {
|
|
if (!this.bodyJson_) this.bodyJson_ = JSON.parse(this.body);
|
|
|
|
if (disallowedProperties) {
|
|
const filteredBody = Object.assign({}, this.bodyJson_);
|
|
for (let i = 0; i < disallowedProperties.length; i++) {
|
|
const n = disallowedProperties[i];
|
|
delete filteredBody[n];
|
|
}
|
|
return filteredBody;
|
|
}
|
|
|
|
return this.bodyJson_;
|
|
},
|
|
files,
|
|
params,
|
|
};
|
|
|
|
this.checkToken_(request);
|
|
|
|
const context: RequestContext = {
|
|
dispatch: this.dispatch,
|
|
token: this.token,
|
|
authToken: this.authToken_,
|
|
};
|
|
|
|
try {
|
|
return await parsedPath.fn(request, id, link, context);
|
|
} catch (error) {
|
|
if (!error.httpCode) error.httpCode = 500;
|
|
throw error;
|
|
}
|
|
}
|
|
|
|
private checkToken_(request: Request) {
|
|
// For now, whitelist some calls to allow the web clipper to work
|
|
// without an extra auth step
|
|
// const whiteList = [['GET', 'ping'], ['GET', 'tags'], ['GET', 'folders'], ['POST', 'notes']];
|
|
|
|
const whiteList = [
|
|
['GET', 'ping'],
|
|
['GET', 'auth'],
|
|
['POST', 'auth'],
|
|
['GET', 'auth/check'],
|
|
];
|
|
|
|
for (let i = 0; i < whiteList.length; i++) {
|
|
if (whiteList[i][0] === request.method && whiteList[i][1] === request.path) return;
|
|
}
|
|
|
|
// If the API has been initialized without a token, it means no auth is
|
|
// needed. This is for example when it is used as the plugin data API.
|
|
if (!this.token) return;
|
|
|
|
if (!request.query || !request.query.token) throw new ErrorForbidden('Missing "token" parameter');
|
|
if (request.query.token !== this.token) throw new ErrorForbidden('Invalid "token" parameter');
|
|
}
|
|
|
|
private async execServiceActionFromRequest_(externalApi: any, request: Request) {
|
|
const action = externalApi[request.action];
|
|
if (!action) throw new ErrorNotFound(`Invalid action: ${request.action}`);
|
|
const args = Object.assign({}, request);
|
|
delete args.action;
|
|
return action(args);
|
|
}
|
|
|
|
private async action_services(request: Request, serviceName: string) {
|
|
if (request.method !== RequestMethod.POST) throw new ErrorMethodNotAllowed();
|
|
if (!this.actionApi_) throw new ErrorNotFound('No action API has been setup!');
|
|
if (!this.actionApi_[serviceName]) throw new ErrorNotFound(`No such service: ${serviceName}`);
|
|
|
|
const externalApi = this.actionApi_[serviceName]();
|
|
return this.execServiceActionFromRequest_(externalApi, JSON.parse(request.body));
|
|
}
|
|
|
|
}
|