2017-07-06 20:58:01 +02:00
|
|
|
import { shim } from 'lib/shim.js';
|
2017-06-22 21:44:38 +02:00
|
|
|
import { stringify } from 'query-string';
|
2017-07-06 21:48:17 +02:00
|
|
|
import { time } from 'lib/time-utils.js';
|
2017-07-09 17:47:05 +02:00
|
|
|
import { Logger } from 'lib/logger.js'
|
2017-06-22 21:44:38 +02:00
|
|
|
|
|
|
|
class OneDriveApi {
|
|
|
|
|
2017-07-07 00:15:31 +02:00
|
|
|
// `isPublic` is to tell OneDrive whether the application is a "public" one (Mobile and desktop
|
|
|
|
// apps are considered "public"), in which case the secret should not be sent to the API.
|
|
|
|
// In practice the React Native app is public, and the Node one is not because we
|
|
|
|
// use a local server for the OAuth dance.
|
|
|
|
constructor(clientId, clientSecret, isPublic) {
|
2017-06-22 21:44:38 +02:00
|
|
|
this.clientId_ = clientId;
|
|
|
|
this.clientSecret_ = clientSecret;
|
2017-06-22 23:52:27 +02:00
|
|
|
this.auth_ = null;
|
2017-07-07 00:15:31 +02:00
|
|
|
this.isPublic_ = isPublic;
|
2017-06-23 20:51:02 +02:00
|
|
|
this.listeners_ = {
|
|
|
|
'authRefreshed': [],
|
|
|
|
};
|
2017-07-09 17:47:05 +02:00
|
|
|
this.logger_ = new Logger();
|
|
|
|
}
|
|
|
|
|
|
|
|
setLogger(l) {
|
|
|
|
this.logger_ = l;
|
|
|
|
}
|
|
|
|
|
|
|
|
logger() {
|
|
|
|
return this.logger_;
|
2017-06-23 20:51:02 +02:00
|
|
|
}
|
|
|
|
|
2017-07-07 00:15:31 +02:00
|
|
|
isPublic() {
|
|
|
|
return this.isPublic_;
|
|
|
|
}
|
|
|
|
|
2017-06-23 20:51:02 +02:00
|
|
|
dispatch(eventName, param) {
|
|
|
|
let ls = this.listeners_[eventName];
|
|
|
|
for (let i = 0; i < ls.length; i++) {
|
|
|
|
ls[i](param);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
on(eventName, callback) {
|
|
|
|
this.listeners_[eventName].push(callback);
|
2017-06-22 21:44:38 +02:00
|
|
|
}
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
tokenBaseUrl() {
|
|
|
|
return 'https://login.microsoftonline.com/common/oauth2/v2.0/token';
|
|
|
|
}
|
|
|
|
|
2017-07-06 20:58:01 +02:00
|
|
|
auth() {
|
|
|
|
return this.auth_;
|
|
|
|
}
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
setAuth(auth) {
|
|
|
|
this.auth_ = auth;
|
2017-07-07 00:15:31 +02:00
|
|
|
this.dispatch('authRefreshed', this.auth());
|
2017-06-22 23:52:27 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
token() {
|
|
|
|
return this.auth_ ? this.auth_.access_token : null;
|
2017-06-22 21:44:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
clientId() {
|
|
|
|
return this.clientId_;
|
|
|
|
}
|
|
|
|
|
|
|
|
clientSecret() {
|
|
|
|
return this.clientSecret_;
|
|
|
|
}
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
async appDirectory() {
|
2017-07-30 22:22:57 +02:00
|
|
|
let r = await this.execJson('GET', '/drive/special/approot');
|
|
|
|
return r.parentReference.path + '/' + r.name;
|
2017-06-22 23:52:27 +02:00
|
|
|
}
|
|
|
|
|
2017-06-22 21:44:38 +02:00
|
|
|
authCodeUrl(redirectUri) {
|
|
|
|
let query = {
|
|
|
|
client_id: this.clientId_,
|
|
|
|
scope: 'files.readwrite offline_access',
|
|
|
|
response_type: 'code',
|
|
|
|
redirect_uri: redirectUri,
|
|
|
|
};
|
|
|
|
return 'https://login.microsoftonline.com/common/oauth2/v2.0/authorize?' + stringify(query);
|
|
|
|
}
|
|
|
|
|
2017-07-07 00:15:31 +02:00
|
|
|
async execTokenRequest(code, redirectUri) {
|
2017-07-06 21:29:09 +02:00
|
|
|
let body = new shim.FormData();
|
|
|
|
body.append('client_id', this.clientId());
|
2017-07-07 00:15:31 +02:00
|
|
|
if (!this.isPublic()) body.append('client_secret', this.clientSecret());
|
2017-07-06 21:29:09 +02:00
|
|
|
body.append('code', code);
|
|
|
|
body.append('redirect_uri', redirectUri);
|
|
|
|
body.append('grant_type', 'authorization_code');
|
|
|
|
|
|
|
|
const r = await shim.fetch(this.tokenBaseUrl(), {
|
|
|
|
method: 'POST',
|
|
|
|
body: body,
|
|
|
|
})
|
|
|
|
|
|
|
|
if (!r.ok) {
|
|
|
|
const text = await r.text();
|
|
|
|
throw new Error('Could not retrieve auth code: ' + r.status + ': ' + r.statusText + ': ' + text);
|
|
|
|
}
|
|
|
|
|
|
|
|
try {
|
|
|
|
const json = await r.json();
|
|
|
|
this.setAuth(json);
|
|
|
|
} catch (error) {
|
2017-07-07 00:15:31 +02:00
|
|
|
this.setAuth(null);
|
2017-07-06 21:29:09 +02:00
|
|
|
const text = await r.text();
|
|
|
|
error.message += ': ' + text;
|
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-29 20:03:16 +02:00
|
|
|
oneDriveErrorResponseToError(errorResponse) {
|
|
|
|
if (!errorResponse) return new Error('Undefined error');
|
|
|
|
|
|
|
|
if (errorResponse.error) {
|
|
|
|
let e = errorResponse.error;
|
|
|
|
let output = new Error(e.message);
|
|
|
|
if (e.code) output.code = e.code;
|
|
|
|
if (e.innerError) output.innerError = e.innerError;
|
|
|
|
return output;
|
|
|
|
} else {
|
|
|
|
return new Error(JSON.stringify(errorResponse));
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2017-06-22 21:44:38 +02:00
|
|
|
async exec(method, path, query = null, data = null, options = null) {
|
2017-07-24 20:01:40 +02:00
|
|
|
if (!path) throw new Error('Path is required');
|
|
|
|
|
2017-06-22 21:44:38 +02:00
|
|
|
method = method.toUpperCase();
|
|
|
|
|
|
|
|
if (!options) options = {};
|
|
|
|
if (!options.headers) options.headers = {};
|
2017-07-06 23:30:45 +02:00
|
|
|
if (!options.target) options.target = 'string';
|
2017-06-22 21:44:38 +02:00
|
|
|
|
|
|
|
if (method != 'GET') {
|
|
|
|
options.method = method;
|
|
|
|
}
|
|
|
|
|
2017-06-23 23:32:24 +02:00
|
|
|
if (method == 'PATCH' || method == 'POST') {
|
2017-06-22 21:44:38 +02:00
|
|
|
options.headers['Content-Type'] = 'application/json';
|
|
|
|
if (data) data = JSON.stringify(data);
|
|
|
|
}
|
|
|
|
|
2017-06-29 20:03:16 +02:00
|
|
|
let url = path;
|
2017-06-22 21:44:38 +02:00
|
|
|
|
2017-06-29 20:03:16 +02:00
|
|
|
// In general, `path` contains a path relative to the base URL, but in some
|
|
|
|
// cases the full URL is provided (for example, when it's a URL that was
|
|
|
|
// retrieved from the API).
|
|
|
|
if (url.indexOf('https://') !== 0) url = 'https://graph.microsoft.com/v1.0' + path;
|
|
|
|
|
|
|
|
if (query) {
|
|
|
|
url += url.indexOf('?') < 0 ? '?' : '&';
|
|
|
|
url += stringify(query);
|
|
|
|
}
|
2017-06-22 21:44:38 +02:00
|
|
|
|
|
|
|
if (data) options.body = data;
|
|
|
|
|
2017-06-23 20:51:02 +02:00
|
|
|
for (let i = 0; i < 5; i++) {
|
2017-06-22 23:52:27 +02:00
|
|
|
options.headers['Authorization'] = 'bearer ' + this.token();
|
|
|
|
|
2017-07-06 23:30:45 +02:00
|
|
|
let response = null;
|
2017-07-13 00:32:08 +02:00
|
|
|
try {
|
2017-08-01 23:40:14 +02:00
|
|
|
if (options.source == 'file' && (method == 'POST' || method == 'PUT')) {
|
|
|
|
response = await shim.uploadBlob(url, options);
|
|
|
|
} else if (options.target == 'string') {
|
2017-07-13 00:32:08 +02:00
|
|
|
response = await shim.fetch(url, options);
|
|
|
|
} else { // file
|
|
|
|
response = await shim.fetchBlob(url, options);
|
|
|
|
}
|
|
|
|
} catch (error) {
|
2017-07-13 20:09:47 +02:00
|
|
|
if (error.message == 'Network request failed') {
|
|
|
|
// Unfortunately the error 'Network request failed' doesn't have a type
|
|
|
|
// or error code, so hopefully that message won't change and is not localized
|
2017-07-30 22:22:57 +02:00
|
|
|
this.logger().info('Got error "Network request failed" - retrying (' + i + ')...');
|
2017-07-13 20:09:47 +02:00
|
|
|
await time.sleep((i + 1) * 3);
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
throw error;
|
|
|
|
}
|
2017-07-06 23:30:45 +02:00
|
|
|
}
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
if (!response.ok) {
|
2017-06-29 20:03:16 +02:00
|
|
|
let errorResponse = await response.json();
|
|
|
|
let error = this.oneDriveErrorResponseToError(errorResponse);
|
2017-06-22 23:52:27 +02:00
|
|
|
|
2017-07-08 00:25:03 +02:00
|
|
|
if (error.code == 'InvalidAuthenticationToken' || error.code == 'unauthenticated') {
|
2017-07-09 17:47:05 +02:00
|
|
|
this.logger().info('Token expired: refreshing...');
|
2017-06-22 23:52:27 +02:00
|
|
|
await this.refreshAccessToken();
|
|
|
|
continue;
|
2017-07-10 01:20:38 +02:00
|
|
|
} else if (error && ((error.error && error.error.code == 'generalException') || error.code == 'generalException' || error.code == 'EAGAIN')) {
|
2017-07-06 21:48:17 +02:00
|
|
|
// Rare error (one Google hit) - I guess the request can be repeated
|
|
|
|
// { error:
|
|
|
|
// { code: 'generalException',
|
|
|
|
// message: 'An error occurred in the data store.',
|
|
|
|
// innerError:
|
|
|
|
// { 'request-id': 'b4310552-c18a-45b1-bde1-68e2c2345eef',
|
|
|
|
// date: '2017-06-29T00:15:50' } } }
|
2017-07-10 01:20:38 +02:00
|
|
|
|
2017-07-06 21:48:17 +02:00
|
|
|
// { FetchError: request to https://graph.microsoft.com/v1.0/drive/root:/Apps/Joplin/.sync/7ee5dc04afcb414aa7c684bfc1edba8b.md_1499352102856 failed, reason: connect EAGAIN 65.52.64.250:443 - Local (0.0.0.0:54374)
|
|
|
|
// name: 'FetchError',
|
|
|
|
// message: 'request to https://graph.microsoft.com/v1.0/drive/root:/Apps/Joplin/.sync/7ee5dc04afcb414aa7c684bfc1edba8b.md_1499352102856 failed, reason: connect EAGAIN 65.52.64.250:443 - Local (0.0.0.0:54374)',
|
|
|
|
// type: 'system',
|
|
|
|
// errno: 'EAGAIN',
|
|
|
|
// code: 'EAGAIN' }
|
2017-07-30 22:22:57 +02:00
|
|
|
this.logger().info('Got error below - retrying (' + i + ')...');
|
|
|
|
this.logger().info(error);
|
2017-07-13 20:09:47 +02:00
|
|
|
await time.sleep((i + 1) * 3);
|
2017-07-06 21:48:17 +02:00
|
|
|
continue;
|
2017-07-11 01:17:03 +02:00
|
|
|
} else if (error.code == 'itemNotFound' && method == 'DELETE') {
|
|
|
|
// Deleting a non-existing item is ok - noop
|
|
|
|
return;
|
2017-06-22 23:52:27 +02:00
|
|
|
} else {
|
2017-06-29 20:03:16 +02:00
|
|
|
error.request = method + ' ' + url + ' ' + JSON.stringify(query) + ' ' + JSON.stringify(data) + ' ' + JSON.stringify(options);
|
2017-06-22 23:52:27 +02:00
|
|
|
throw error;
|
|
|
|
}
|
|
|
|
}
|
2017-06-22 21:44:38 +02:00
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
return response;
|
|
|
|
}
|
2017-06-23 20:51:02 +02:00
|
|
|
|
|
|
|
throw new Error('Could not execute request after multiple attempts: ' + method + ' ' + url);
|
2017-06-22 21:44:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
async execJson(method, path, query, data) {
|
|
|
|
let response = await this.exec(method, path, query, data);
|
|
|
|
let output = await response.json();
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
|
|
|
async execText(method, path, query, data) {
|
|
|
|
let response = await this.exec(method, path, query, data);
|
|
|
|
let output = await response.text();
|
|
|
|
return output;
|
|
|
|
}
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
async refreshAccessToken() {
|
2017-07-26 23:07:27 +02:00
|
|
|
if (!this.auth_ || !this.auth_.refresh_token) {
|
|
|
|
this.setAuth(null);
|
2017-07-28 20:13:07 +02:00
|
|
|
throw new Error(_('Cannot refresh token: authentication data is missing. Starting the synchronisation again may fix the problem.'));
|
2017-07-26 23:07:27 +02:00
|
|
|
}
|
2017-06-22 23:52:27 +02:00
|
|
|
|
2017-07-06 20:58:01 +02:00
|
|
|
let body = new shim.FormData();
|
2017-06-22 23:52:27 +02:00
|
|
|
body.append('client_id', this.clientId());
|
2017-07-07 00:15:31 +02:00
|
|
|
if (!this.isPublic()) body.append('client_secret', this.clientSecret());
|
2017-06-22 23:52:27 +02:00
|
|
|
body.append('refresh_token', this.auth_.refresh_token);
|
|
|
|
body.append('redirect_uri', 'http://localhost:1917');
|
|
|
|
body.append('grant_type', 'refresh_token');
|
|
|
|
|
|
|
|
let options = {
|
|
|
|
method: 'POST',
|
|
|
|
body: body,
|
|
|
|
};
|
|
|
|
|
2017-07-06 20:58:01 +02:00
|
|
|
let response = await shim.fetch(this.tokenBaseUrl(), options);
|
2017-06-22 23:52:27 +02:00
|
|
|
if (!response.ok) {
|
2017-07-07 00:15:31 +02:00
|
|
|
this.setAuth(null);
|
2017-06-22 23:52:27 +02:00
|
|
|
let msg = await response.text();
|
2017-07-15 18:14:15 +02:00
|
|
|
throw new Error(msg + ': TOKEN: ' + this.auth_);
|
2017-06-22 23:52:27 +02:00
|
|
|
}
|
|
|
|
|
2017-07-07 00:15:31 +02:00
|
|
|
let auth = await response.json();
|
|
|
|
this.setAuth(auth);
|
2017-06-22 23:52:27 +02:00
|
|
|
}
|
|
|
|
|
2017-06-22 21:44:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export { OneDriveApi };
|