2016-03-28 20:33:20 +00:00
|
|
|
'use strict';
|
|
|
|
|
2016-04-08 08:01:32 +00:00
|
|
|
const cookies = require('js-cookie');
|
2016-03-28 20:33:20 +00:00
|
|
|
const request = require('superagent');
|
|
|
|
const config = require('./config.js');
|
2016-04-07 17:03:49 +00:00
|
|
|
const events = require('./events.js');
|
2017-01-08 21:24:46 +00:00
|
|
|
const progress = require('./util/progress.js');
|
2017-01-20 20:51:04 +00:00
|
|
|
const uri = require('./util/uri.js');
|
2016-03-28 20:33:20 +00:00
|
|
|
|
2017-01-08 22:52:20 +00:00
|
|
|
let fileTokens = {};
|
|
|
|
|
2016-06-14 08:31:48 +00:00
|
|
|
class Api extends events.EventTarget {
|
2016-03-30 18:45:37 +00:00
|
|
|
constructor() {
|
2016-06-14 08:31:48 +00:00
|
|
|
super();
|
2016-03-30 20:05:57 +00:00
|
|
|
this.user = null;
|
2016-03-30 18:45:37 +00:00
|
|
|
this.userName = null;
|
|
|
|
this.userPassword = null;
|
2016-04-11 16:45:58 +00:00
|
|
|
this.cache = {};
|
2016-05-08 14:59:25 +00:00
|
|
|
this.allRanks = [
|
|
|
|
'anonymous',
|
|
|
|
'restricted',
|
|
|
|
'regular',
|
|
|
|
'power',
|
|
|
|
'moderator',
|
|
|
|
'administrator',
|
|
|
|
'nobody',
|
|
|
|
];
|
2016-06-19 17:16:40 +00:00
|
|
|
this.rankNames = new Map([
|
|
|
|
['anonymous', 'Anonymous'],
|
|
|
|
['restricted', 'Restricted user'],
|
|
|
|
['regular', 'Regular user'],
|
|
|
|
['power', 'Power user'],
|
|
|
|
['moderator', 'Moderator'],
|
|
|
|
['administrator', 'Administrator'],
|
|
|
|
['nobody', 'Nobody'],
|
|
|
|
]);
|
2016-03-30 18:45:37 +00:00
|
|
|
}
|
|
|
|
|
2016-05-22 09:16:25 +00:00
|
|
|
get(url, options) {
|
2016-04-11 16:45:58 +00:00
|
|
|
if (url in this.cache) {
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
resolve(this.cache[url]);
|
|
|
|
});
|
|
|
|
}
|
2017-01-07 13:28:36 +00:00
|
|
|
return this._wrappedRequest(url, request.get, {}, {}, options)
|
2016-05-22 09:16:25 +00:00
|
|
|
.then(response => {
|
|
|
|
this.cache[url] = response;
|
|
|
|
return Promise.resolve(response);
|
|
|
|
});
|
2016-03-28 20:33:20 +00:00
|
|
|
}
|
|
|
|
|
2016-05-22 09:16:25 +00:00
|
|
|
post(url, data, files, options) {
|
2016-04-11 16:45:58 +00:00
|
|
|
this.cache = {};
|
2017-01-07 13:28:36 +00:00
|
|
|
return this._wrappedRequest(url, request.post, data, files, options);
|
2016-03-28 20:33:20 +00:00
|
|
|
}
|
|
|
|
|
2016-05-22 09:16:25 +00:00
|
|
|
put(url, data, files, options) {
|
2016-04-11 16:45:58 +00:00
|
|
|
this.cache = {};
|
2017-01-07 13:28:36 +00:00
|
|
|
return this._wrappedRequest(url, request.put, data, files, options);
|
2016-04-07 20:54:45 +00:00
|
|
|
}
|
|
|
|
|
2016-08-06 20:44:04 +00:00
|
|
|
delete(url, data, options) {
|
2016-04-11 16:45:58 +00:00
|
|
|
this.cache = {};
|
2017-01-07 13:28:36 +00:00
|
|
|
return this._wrappedRequest(url, request.delete, data, {}, options);
|
2016-03-28 20:33:20 +00:00
|
|
|
}
|
|
|
|
|
2016-03-30 20:05:57 +00:00
|
|
|
hasPrivilege(lookup) {
|
|
|
|
let minViableRank = null;
|
|
|
|
for (let privilege of Object.keys(config.privileges)) {
|
|
|
|
if (!privilege.startsWith(lookup)) {
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
const rankName = config.privileges[privilege];
|
2016-05-08 14:59:25 +00:00
|
|
|
const rankIndex = this.allRanks.indexOf(rankName);
|
2016-03-30 20:05:57 +00:00
|
|
|
if (minViableRank === null || rankIndex < minViableRank) {
|
|
|
|
minViableRank = rankIndex;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
if (minViableRank === null) {
|
2016-06-11 07:30:22 +00:00
|
|
|
throw `Bad privilege name: ${lookup}`;
|
2016-03-30 20:05:57 +00:00
|
|
|
}
|
|
|
|
let myRank = this.user !== null ?
|
2016-05-08 14:59:25 +00:00
|
|
|
this.allRanks.indexOf(this.user.rank) :
|
2016-03-30 20:05:57 +00:00
|
|
|
0;
|
|
|
|
return myRank >= minViableRank;
|
2016-03-30 18:45:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-08 08:01:32 +00:00
|
|
|
loginFromCookies() {
|
2017-01-08 01:12:38 +00:00
|
|
|
const auth = cookies.getJSON('auth');
|
|
|
|
return auth && auth.user && auth.password ?
|
|
|
|
this.login(auth.user, auth.password, true) :
|
|
|
|
Promise.resolve();
|
2016-04-08 08:01:32 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
login(userName, userPassword, doRemember) {
|
2016-06-19 19:37:44 +00:00
|
|
|
this.cache = {};
|
2016-03-30 19:01:18 +00:00
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
this.userName = userName;
|
|
|
|
this.userPassword = userPassword;
|
2016-04-03 13:12:15 +00:00
|
|
|
this.get('/user/' + userName + '?bump-login=true')
|
2016-03-30 20:05:57 +00:00
|
|
|
.then(response => {
|
2016-04-08 08:01:32 +00:00
|
|
|
const options = {};
|
|
|
|
if (doRemember) {
|
|
|
|
options.expires = 365;
|
|
|
|
}
|
|
|
|
cookies.set(
|
|
|
|
'auth',
|
|
|
|
{'user': userName, 'password': userPassword},
|
|
|
|
options);
|
2016-05-30 20:20:42 +00:00
|
|
|
this.user = response;
|
2016-03-30 20:05:57 +00:00
|
|
|
resolve();
|
2016-06-14 08:31:48 +00:00
|
|
|
this.dispatchEvent(new CustomEvent('login'));
|
2017-01-08 01:12:38 +00:00
|
|
|
}, error => {
|
|
|
|
reject(error);
|
2016-03-30 19:01:18 +00:00
|
|
|
this.logout();
|
|
|
|
});
|
|
|
|
});
|
2016-03-30 18:45:37 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
logout() {
|
2016-04-01 11:09:07 +00:00
|
|
|
this.user = null;
|
2016-03-30 18:45:37 +00:00
|
|
|
this.userName = null;
|
|
|
|
this.userPassword = null;
|
2016-06-14 08:31:48 +00:00
|
|
|
this.dispatchEvent(new CustomEvent('logout'));
|
2016-03-30 18:45:37 +00:00
|
|
|
}
|
|
|
|
|
2016-04-09 17:53:53 +00:00
|
|
|
forget() {
|
|
|
|
cookies.remove('auth');
|
|
|
|
}
|
|
|
|
|
2016-04-16 13:07:33 +00:00
|
|
|
isLoggedIn(user) {
|
|
|
|
if (user) {
|
|
|
|
return this.userName !== null &&
|
|
|
|
this.userName.toLowerCase() === user.name.toLowerCase();
|
|
|
|
} else {
|
|
|
|
return this.userName !== null;
|
|
|
|
}
|
2016-03-30 18:45:37 +00:00
|
|
|
}
|
|
|
|
|
2016-05-20 19:35:12 +00:00
|
|
|
_getFullUrl(url) {
|
2016-09-03 23:25:19 +00:00
|
|
|
const fullUrl =
|
|
|
|
(config.apiUrl + '/' + url).replace(/([^:])\/+/g, '$1/');
|
|
|
|
const matches = fullUrl.match(/^([^?]*)\??(.*)$/);
|
|
|
|
const baseUrl = matches[1];
|
|
|
|
const request = matches[2];
|
|
|
|
return [baseUrl, request];
|
2016-03-28 20:33:20 +00:00
|
|
|
}
|
2017-01-07 13:28:36 +00:00
|
|
|
|
2017-01-08 22:52:20 +00:00
|
|
|
_getFileId(file) {
|
|
|
|
if (file.constructor === String) {
|
|
|
|
return file;
|
|
|
|
}
|
|
|
|
return file.name + file.size;
|
|
|
|
}
|
|
|
|
|
2017-01-07 13:28:36 +00:00
|
|
|
_wrappedRequest(url, requestFactory, data, files, options) {
|
|
|
|
// transform the request: upload each file, then make the request use
|
|
|
|
// its tokens.
|
|
|
|
data = Object.assign({}, data);
|
|
|
|
let abortFunction = () => {};
|
2017-01-07 15:24:56 +00:00
|
|
|
let promise = Promise.resolve();
|
2017-01-07 13:28:36 +00:00
|
|
|
if (files) {
|
|
|
|
for (let key of Object.keys(files)) {
|
2017-01-08 22:52:20 +00:00
|
|
|
const file = files[key];
|
|
|
|
const fileId = this._getFileId(file);
|
|
|
|
if (fileTokens[fileId]) {
|
|
|
|
data[key + 'Token'] = fileTokens[fileId];
|
2017-01-07 13:28:36 +00:00
|
|
|
} else {
|
|
|
|
promise = promise
|
|
|
|
.then(() => {
|
2017-01-07 15:24:56 +00:00
|
|
|
let uploadPromise = this._upload(file);
|
|
|
|
abortFunction = () => uploadPromise.abort();
|
|
|
|
return uploadPromise;
|
2017-01-07 13:28:36 +00:00
|
|
|
})
|
|
|
|
.then(token => {
|
|
|
|
abortFunction = () => {};
|
2017-01-08 22:52:20 +00:00
|
|
|
fileTokens[fileId] = token;
|
2017-01-07 13:28:36 +00:00
|
|
|
data[key + 'Token'] = token;
|
|
|
|
return Promise.resolve();
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
2017-01-07 15:24:56 +00:00
|
|
|
promise = promise.then(
|
|
|
|
() => {
|
|
|
|
let requestPromise = this._rawRequest(
|
|
|
|
url, requestFactory, data, {}, options);
|
|
|
|
abortFunction = () => requestPromise.abort();
|
|
|
|
return requestPromise;
|
2017-01-08 10:04:21 +00:00
|
|
|
})
|
|
|
|
.catch(error => {
|
2017-01-08 19:56:48 +00:00
|
|
|
if (error.response && error.response.name ===
|
2017-01-08 10:04:21 +00:00
|
|
|
'MissingOrExpiredRequiredFileError') {
|
|
|
|
for (let key of Object.keys(files)) {
|
2017-01-08 22:52:20 +00:00
|
|
|
const file = files[key];
|
|
|
|
const fileId = this._getFileId(file);
|
|
|
|
fileTokens[fileId] = null;
|
2017-01-08 10:04:21 +00:00
|
|
|
}
|
|
|
|
error.message =
|
|
|
|
'The uploaded file has expired; ' +
|
|
|
|
'please resend the form to reupload.';
|
|
|
|
}
|
2017-01-07 15:24:56 +00:00
|
|
|
return Promise.reject(error);
|
|
|
|
});
|
2017-01-07 13:28:36 +00:00
|
|
|
promise.abort = () => abortFunction();
|
|
|
|
return promise;
|
|
|
|
}
|
|
|
|
|
|
|
|
_upload(file, options) {
|
|
|
|
let abortFunction = () => {};
|
|
|
|
let returnedPromise = new Promise((resolve, reject) => {
|
2017-01-07 15:24:56 +00:00
|
|
|
let uploadPromise = this._rawRequest(
|
|
|
|
'/uploads', request.post, {}, {content: file}, options);
|
|
|
|
abortFunction = () => uploadPromise.abort();
|
|
|
|
return uploadPromise.then(
|
2017-01-07 13:28:36 +00:00
|
|
|
response => {
|
|
|
|
abortFunction = () => {};
|
2017-01-07 15:24:56 +00:00
|
|
|
return resolve(response.token);
|
|
|
|
}, reject);
|
2017-01-07 13:28:36 +00:00
|
|
|
});
|
|
|
|
returnedPromise.abort = () => abortFunction();
|
|
|
|
return returnedPromise;
|
|
|
|
}
|
|
|
|
|
|
|
|
_rawRequest(url, requestFactory, data, files, options) {
|
|
|
|
options = options || {};
|
|
|
|
data = Object.assign({}, data);
|
|
|
|
const [fullUrl, query] = this._getFullUrl(url);
|
|
|
|
|
2017-01-07 15:24:56 +00:00
|
|
|
let abortFunction = () => {};
|
|
|
|
let returnedPromise = new Promise((resolve, reject) => {
|
2017-01-07 13:28:36 +00:00
|
|
|
let req = requestFactory(fullUrl);
|
|
|
|
|
|
|
|
req.set('Accept', 'application/json');
|
|
|
|
|
|
|
|
if (query) {
|
|
|
|
req.query(query);
|
|
|
|
}
|
|
|
|
|
|
|
|
if (files) {
|
|
|
|
for (let key of Object.keys(files)) {
|
|
|
|
const value = files[key];
|
|
|
|
if (value.constructor === String) {
|
|
|
|
data[key + 'Url'] = value;
|
|
|
|
} else {
|
|
|
|
req.attach(key, value || new Blob());
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (data) {
|
|
|
|
if (files && Object.keys(files).length) {
|
|
|
|
req.attach('metadata', new Blob([JSON.stringify(data)]));
|
|
|
|
} else {
|
|
|
|
req.set('Content-Type', 'application/json');
|
|
|
|
req.send(data);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
if (this.userName && this.userPassword) {
|
|
|
|
req.auth(
|
|
|
|
this.userName,
|
|
|
|
encodeURIComponent(this.userPassword)
|
|
|
|
.replace(/%([0-9A-F]{2})/g, (match, p1) => {
|
|
|
|
return String.fromCharCode('0x' + p1);
|
|
|
|
}));
|
|
|
|
}
|
|
|
|
} catch (e) {
|
2017-01-08 01:12:38 +00:00
|
|
|
reject(
|
|
|
|
new Error('Authentication error (malformed credentials)'));
|
2017-01-07 13:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if (!options.noProgress) {
|
2017-01-08 21:24:46 +00:00
|
|
|
progress.start();
|
2017-01-07 13:28:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
abortFunction = () => {
|
|
|
|
req.abort(); // does *NOT* call the callback passed in .end()
|
2017-01-08 21:24:46 +00:00
|
|
|
progress.done();
|
2017-01-08 01:12:38 +00:00
|
|
|
reject(
|
|
|
|
new Error('The request was aborted due to user cancel.'));
|
2017-01-07 13:28:36 +00:00
|
|
|
};
|
|
|
|
|
|
|
|
req.end((error, response) => {
|
2017-01-08 21:24:46 +00:00
|
|
|
progress.done();
|
2017-01-07 15:24:56 +00:00
|
|
|
abortFunction = () => {};
|
2017-01-07 13:28:36 +00:00
|
|
|
if (error) {
|
2017-01-08 01:12:38 +00:00
|
|
|
if (response && response.body) {
|
|
|
|
error = new Error(
|
|
|
|
response.body.description || 'Unknown error');
|
|
|
|
error.response = response.body;
|
|
|
|
}
|
|
|
|
reject(error);
|
2017-01-07 13:28:36 +00:00
|
|
|
} else {
|
|
|
|
resolve(response.body);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
});
|
2017-01-07 15:24:56 +00:00
|
|
|
returnedPromise.abort = () => abortFunction();
|
|
|
|
return returnedPromise;
|
2017-01-07 13:28:36 +00:00
|
|
|
}
|
2016-03-28 20:33:20 +00:00
|
|
|
}
|
|
|
|
|
2016-03-31 22:20:34 +00:00
|
|
|
module.exports = new Api();
|