1
0
mirror of https://github.com/laurent22/joplin.git synced 2025-04-20 11:28:40 +02:00
joplin/ReactNativeClient/lib/onedrive-api.js

297 lines
9.4 KiB
JavaScript
Raw Normal View History

2018-03-09 17:49:35 +00:00
const { shim } = require("lib/shim.js");
const { stringify } = require("query-string");
const { time } = require("lib/time-utils.js");
const { Logger } = require("lib/logger.js");
2017-06-22 20:44:38 +01:00
class OneDriveApi {
// `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 20:44:38 +01:00
this.clientId_ = clientId;
this.clientSecret_ = clientSecret;
2017-06-22 22:52:27 +01:00
this.auth_ = null;
this.isPublic_ = isPublic;
this.listeners_ = {
2018-03-09 17:49:35 +00:00
authRefreshed: [],
};
this.logger_ = new Logger();
}
setLogger(l) {
this.logger_ = l;
}
logger() {
return this.logger_;
}
isPublic() {
return this.isPublic_;
}
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 20:44:38 +01:00
}
2017-06-22 22:52:27 +01:00
tokenBaseUrl() {
2018-03-09 17:49:35 +00:00
return "https://login.microsoftonline.com/common/oauth2/v2.0/token";
2017-06-22 22:52:27 +01:00
}
nativeClientRedirectUrl() {
2018-03-09 17:49:35 +00:00
return "https://login.microsoftonline.com/common/oauth2/nativeclient";
}
auth() {
return this.auth_;
}
2017-06-22 22:52:27 +01:00
setAuth(auth) {
this.auth_ = auth;
2018-03-09 17:49:35 +00:00
this.dispatch("authRefreshed", this.auth());
2017-06-22 22:52:27 +01:00
}
token() {
return this.auth_ ? this.auth_.access_token : null;
2017-06-22 20:44:38 +01:00
}
clientId() {
return this.clientId_;
}
clientSecret() {
return this.clientSecret_;
}
2017-06-22 22:52:27 +01:00
async appDirectory() {
2018-03-09 17:49:35 +00:00
let r = await this.execJson("GET", "/drive/special/approot");
return r.parentReference.path + "/" + r.name;
2017-06-22 22:52:27 +01:00
}
2017-06-22 20:44:38 +01:00
authCodeUrl(redirectUri) {
let query = {
client_id: this.clientId_,
2018-03-09 17:49:35 +00:00
scope: "files.readwrite offline_access",
response_type: "code",
2017-06-22 20:44:38 +01:00
redirect_uri: redirectUri,
};
2018-03-09 17:49:35 +00:00
return "https://login.microsoftonline.com/common/oauth2/v2.0/authorize?" + stringify(query);
2017-06-22 20:44:38 +01:00
}
async execTokenRequest(code, redirectUri) {
2017-07-06 19:29:09 +00:00
let body = new shim.FormData();
2018-03-09 17:49:35 +00:00
body.append("client_id", this.clientId());
if (!this.isPublic()) body.append("client_secret", this.clientSecret());
body.append("code", code);
body.append("redirect_uri", redirectUri);
body.append("grant_type", "authorization_code");
2017-07-06 19:29:09 +00:00
const r = await shim.fetch(this.tokenBaseUrl(), {
2018-03-09 17:49:35 +00:00
method: "POST",
2017-07-06 19:29:09 +00:00
body: body,
2018-03-09 17:49:35 +00:00
});
2017-07-06 19:29:09 +00:00
if (!r.ok) {
const text = await r.text();
2018-03-09 17:49:35 +00:00
throw new Error("Could not retrieve auth code: " + r.status + ": " + r.statusText + ": " + text);
2017-07-06 19:29:09 +00:00
}
try {
const json = await r.json();
this.setAuth(json);
} catch (error) {
this.setAuth(null);
2017-07-06 19:29:09 +00:00
const text = await r.text();
2018-03-09 17:49:35 +00:00
error.message += ": " + text;
2017-07-06 19:29:09 +00:00
throw error;
}
}
2017-06-29 18:03:16 +00:00
oneDriveErrorResponseToError(errorResponse) {
2018-03-09 17:49:35 +00:00
if (!errorResponse) return new Error("Undefined error");
2017-06-29 18:03:16 +00:00
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;
2018-03-09 17:49:35 +00:00
} else {
2017-06-29 18:03:16 +00:00
return new Error(JSON.stringify(errorResponse));
}
}
2017-06-22 20:44:38 +01:00
async exec(method, path, query = null, data = null, options = null) {
2018-03-09 17:49:35 +00:00
if (!path) throw new Error("Path is required");
2017-06-22 20:44:38 +01:00
method = method.toUpperCase();
if (!options) options = {};
if (!options.headers) options.headers = {};
2018-03-09 17:49:35 +00:00
if (!options.target) options.target = "string";
2017-06-22 20:44:38 +01:00
2018-03-09 17:49:35 +00:00
if (method != "GET") {
2017-06-22 20:44:38 +01:00
options.method = method;
}
2018-03-09 17:49:35 +00:00
if (method == "PATCH" || method == "POST") {
options.headers["Content-Type"] = "application/json";
2017-06-22 20:44:38 +01:00
if (data) data = JSON.stringify(data);
}
2017-06-29 18:03:16 +00:00
let url = path;
2017-06-22 20:44:38 +01:00
2017-06-29 18:03:16 +00: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).
2018-03-09 17:49:35 +00:00
if (url.indexOf("https://") !== 0) url = "https://graph.microsoft.com/v1.0" + path;
2017-06-29 18:03:16 +00:00
if (query) {
2018-03-09 17:49:35 +00:00
url += url.indexOf("?") < 0 ? "?" : "&";
2017-06-29 18:03:16 +00:00
url += stringify(query);
}
2017-06-22 20:44:38 +01:00
if (data) options.body = data;
2017-10-15 12:13:09 +01:00
options.timeout = 1000 * 60 * 5; // in ms
for (let i = 0; i < 5; i++) {
2018-03-09 17:49:35 +00:00
options.headers["Authorization"] = "bearer " + this.token();
2017-06-22 22:52:27 +01:00
2017-07-06 22:30:45 +01:00
let response = null;
2017-07-12 23:32:08 +01:00
try {
2018-03-09 17:49:35 +00:00
if (options.source == "file" && (method == "POST" || method == "PUT")) {
2017-08-01 23:40:14 +02:00
response = await shim.uploadBlob(url, options);
2018-03-09 17:49:35 +00:00
} else if (options.target == "string") {
2017-07-12 23:32:08 +01:00
response = await shim.fetch(url, options);
2018-03-09 17:49:35 +00:00
} else {
// file
2017-07-12 23:32:08 +01:00
response = await shim.fetchBlob(url, options);
}
} catch (error) {
2018-03-09 17:49:35 +00:00
this.logger().error("Got unhandled error:", error ? error.code : "", error ? error.message : "", error);
throw error;
2017-07-06 22:30:45 +01:00
}
2017-06-22 22:52:27 +01:00
if (!response.ok) {
2017-11-30 18:29:10 +00:00
let errorResponseText = await response.text();
let errorResponse = null;
try {
2018-03-09 17:49:35 +00:00
errorResponse = JSON.parse(errorResponseText); //await response.json();
2017-11-30 18:29:10 +00:00
} catch (error) {
2018-03-09 17:49:35 +00:00
error.message = "OneDriveApi::exec: Cannot parse JSON error: " + errorResponseText + " " + error.message;
2017-11-30 18:29:10 +00:00
throw error;
}
2017-06-29 18:03:16 +00:00
let error = this.oneDriveErrorResponseToError(errorResponse);
2017-06-22 22:52:27 +01:00
2018-03-09 17:49:35 +00:00
if (error.code == "InvalidAuthenticationToken" || error.code == "unauthenticated") {
this.logger().info("Token expired: refreshing...");
2017-06-22 22:52:27 +01:00
await this.refreshAccessToken();
continue;
2018-03-09 17:49:35 +00:00
} else if (error && ((error.error && error.error.code == "generalException") || error.code == "generalException" || error.code == "EAGAIN")) {
2017-07-06 19:48:17 +00: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-06 19:48:17 +00: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' }
2018-03-09 17:49:35 +00:00
this.logger().info("Got error below - retrying (" + i + ")...");
2017-07-30 22:22:57 +02:00
this.logger().info(error);
2017-07-13 18:09:47 +00:00
await time.sleep((i + 1) * 3);
2017-12-04 23:01:22 +00:00
continue;
2018-03-09 17:49:35 +00:00
} else if (error && (error.code === "resourceModified" || (error.error && error.error.code === "resourceModified"))) {
2017-12-04 23:01:22 +00:00
// NOTE: not tested, very hard to reproduce and non-informative error message, but can be repeated
// Error: ETag does not match current item's value
// Code: resourceModified
// Header: {"_headers":{"cache-control":["private"],"transfer-encoding":["chunked"],"content-type":["application/json"],"request-id":["d...ea47"],"client-request-id":["d99...ea47"],"x-ms-ags-diagnostic":["{\"ServerInfo\":{\"DataCenter\":\"North Europe\",\"Slice\":\"SliceA\",\"Ring\":\"2\",\"ScaleUnit\":\"000\",\"Host\":\"AGSFE_IN_13\",\"ADSiteName\":\"DUB\"}}"],"duration":["96.9464"],"date":[],"connection":["close"]}}
// Request: PATCH https://graph.microsoft.com/v1.0/drive/root:/Apps/JoplinDev/f56c5601fee94b8085524513bf3e352f.md null "{\"fileSystemInfo\":{\"lastModifiedDateTime\":\"....\"}}" {"headers":{"Content-Type":"application/json","Authorization":"bearer ...
2018-03-09 17:49:35 +00:00
this.logger().info("Got error below - retrying (" + i + ")...");
2017-12-04 23:01:22 +00:00
this.logger().info(error);
await time.sleep((i + 1) * 3);
2017-07-06 19:48:17 +00:00
continue;
2018-03-09 17:49:35 +00:00
} else if (error.code == "itemNotFound" && method == "DELETE") {
2017-07-11 00:17:03 +01:00
// Deleting a non-existing item is ok - noop
return;
2017-06-22 22:52:27 +01:00
} else {
2018-03-09 17:49:35 +00:00
error.request = method + " " + url + " " + JSON.stringify(query) + " " + JSON.stringify(data) + " " + JSON.stringify(options);
2017-10-26 22:57:49 +01:00
error.headers = await response.headers;
2017-06-22 22:52:27 +01:00
throw error;
}
}
2017-06-22 20:44:38 +01:00
2017-06-22 22:52:27 +01:00
return response;
}
2018-03-09 17:49:35 +00:00
throw new Error("Could not execute request after multiple attempts: " + method + " " + url);
2017-06-22 20:44:38 +01:00
}
async execJson(method, path, query, data) {
let response = await this.exec(method, path, query, data);
2017-11-30 18:29:10 +00:00
let errorResponseText = await response.text();
try {
let output = JSON.parse(errorResponseText); //await response.json();
return output;
} catch (error) {
2018-03-09 17:49:35 +00:00
error.message = "OneDriveApi::execJson: Cannot parse JSON: " + errorResponseText + " " + error.message;
2017-11-30 18:29:10 +00:00
throw error;
//throw new Error('Cannot parse JSON: ' + text);
}
2017-06-22 20:44:38 +01:00
}
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 22:52:27 +01:00
async refreshAccessToken() {
2017-07-26 22:07:27 +01:00
if (!this.auth_ || !this.auth_.refresh_token) {
this.setAuth(null);
2018-03-09 17:49:35 +00:00
throw new Error(_("Cannot refresh token: authentication data is missing. Starting the synchronisation again may fix the problem."));
2017-07-26 22:07:27 +01:00
}
2017-06-22 22:52:27 +01:00
let body = new shim.FormData();
2018-03-09 17:49:35 +00:00
body.append("client_id", this.clientId());
if (!this.isPublic()) body.append("client_secret", this.clientSecret());
body.append("refresh_token", this.auth_.refresh_token);
body.append("redirect_uri", "http://localhost:1917");
body.append("grant_type", "refresh_token");
2017-06-22 22:52:27 +01:00
let options = {
2018-03-09 17:49:35 +00:00
method: "POST",
2017-06-22 22:52:27 +01:00
body: body,
};
let response = await shim.fetch(this.tokenBaseUrl(), options);
2017-06-22 22:52:27 +01:00
if (!response.ok) {
this.setAuth(null);
2017-06-22 22:52:27 +01:00
let msg = await response.text();
2018-03-09 17:49:35 +00:00
throw new Error(msg + ": TOKEN: " + this.auth_);
2017-06-22 22:52:27 +01:00
}
let auth = await response.json();
this.setAuth(auth);
2017-06-22 22:52:27 +01:00
}
2017-06-22 20:44:38 +01:00
}
2018-03-09 17:49:35 +00:00
module.exports = { OneDriveApi };