2017-06-22 21:44:38 +02:00
|
|
|
const fetch = require('node-fetch');
|
2017-06-22 23:52:27 +02:00
|
|
|
const tcpPortUsed = require('tcp-port-used');
|
|
|
|
const http = require("http");
|
|
|
|
const urlParser = require("url");
|
|
|
|
const FormData = require('form-data');
|
|
|
|
const enableServerDestroy = require('server-destroy');
|
2017-06-22 21:44:38 +02:00
|
|
|
import { stringify } from 'query-string';
|
|
|
|
|
|
|
|
class OneDriveApi {
|
|
|
|
|
|
|
|
constructor(clientId, clientSecret) {
|
|
|
|
this.clientId_ = clientId;
|
|
|
|
this.clientSecret_ = clientSecret;
|
2017-06-22 23:52:27 +02:00
|
|
|
this.auth_ = null;
|
2017-06-23 20:51:02 +02:00
|
|
|
this.listeners_ = {
|
|
|
|
'authRefreshed': [],
|
|
|
|
};
|
|
|
|
}
|
|
|
|
|
|
|
|
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';
|
|
|
|
}
|
|
|
|
|
|
|
|
setAuth(auth) {
|
|
|
|
this.auth_ = auth;
|
|
|
|
}
|
|
|
|
|
|
|
|
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
|
|
|
possibleOAuthDancePorts() {
|
2017-06-22 21:44:38 +02:00
|
|
|
return [1917, 9917, 8917];
|
|
|
|
}
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
async appDirectory() {
|
|
|
|
let r = await this.execJson('GET', '/drive/special/approot');
|
|
|
|
return r.parentReference.path + '/' + r.name;
|
|
|
|
}
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
async exec(method, path, query = null, data = null, options = null) {
|
|
|
|
method = method.toUpperCase();
|
|
|
|
|
|
|
|
if (!options) options = {};
|
|
|
|
if (!options.headers) options.headers = {};
|
|
|
|
|
|
|
|
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);
|
|
|
|
}
|
|
|
|
|
|
|
|
let url = 'https://graph.microsoft.com/v1.0' + path;
|
|
|
|
|
|
|
|
if (query) url += '?' + stringify(query);
|
|
|
|
|
|
|
|
if (data) options.body = data;
|
|
|
|
|
2017-06-23 23:32:24 +02:00
|
|
|
// console.info(method + ' ' + url);
|
|
|
|
// console.info(data);
|
2017-06-22 21:44:38 +02:00
|
|
|
|
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();
|
|
|
|
|
|
|
|
let response = await fetch(url, options);
|
|
|
|
if (!response.ok) {
|
|
|
|
let error = await response.json();
|
|
|
|
|
|
|
|
if (error && error.error && error.error.code == 'InvalidAuthenticationToken') {
|
|
|
|
await this.refreshAccessToken();
|
|
|
|
continue;
|
|
|
|
} else {
|
|
|
|
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() {
|
|
|
|
if (!this.auth_) throw new Error('Cannot refresh token: authentication data is missing');
|
|
|
|
|
|
|
|
let body = new FormData();
|
|
|
|
body.append('client_id', this.clientId());
|
|
|
|
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');
|
|
|
|
|
|
|
|
let options = {
|
|
|
|
method: 'POST',
|
|
|
|
body: body,
|
|
|
|
};
|
|
|
|
|
|
|
|
this.auth_ = null;
|
|
|
|
|
|
|
|
let response = await fetch(this.tokenBaseUrl(), options);
|
|
|
|
if (!response.ok) {
|
|
|
|
let msg = await response.text();
|
|
|
|
throw new Error(msg);
|
|
|
|
}
|
|
|
|
|
|
|
|
this.auth_ = await response.json();
|
|
|
|
|
2017-06-23 20:51:02 +02:00
|
|
|
this.dispatch('authRefreshed', this.auth_);
|
2017-06-22 23:52:27 +02:00
|
|
|
}
|
|
|
|
|
2017-06-27 21:26:29 +02:00
|
|
|
async oauthDance(targetConsole = null) {
|
|
|
|
if (targetConsole === null) targetConsole = console;
|
|
|
|
|
2017-06-22 23:52:27 +02:00
|
|
|
this.auth_ = null;
|
|
|
|
|
|
|
|
let ports = this.possibleOAuthDancePorts();
|
|
|
|
let port = null;
|
|
|
|
for (let i = 0; i < ports.length; i++) {
|
|
|
|
let inUse = await tcpPortUsed.check(ports[i]);
|
|
|
|
if (!inUse) {
|
|
|
|
port = ports[i];
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!port) throw new Error('All potential ports are in use - please report the issue at https://github.com/laurent22/joplin');
|
|
|
|
|
|
|
|
let authCodeUrl = this.authCodeUrl('http://localhost:' + port);
|
|
|
|
|
|
|
|
return new Promise((resolve, reject) => {
|
|
|
|
let server = http.createServer();
|
|
|
|
let errorMessage = null;
|
|
|
|
|
|
|
|
server.on('request', (request, response) => {
|
|
|
|
const query = urlParser.parse(request.url, true).query;
|
|
|
|
|
|
|
|
function writeResponse(code, message) {
|
|
|
|
response.writeHead(code, {"Content-Type": "text/html"});
|
|
|
|
response.write(message);
|
|
|
|
response.end();
|
|
|
|
}
|
|
|
|
|
|
|
|
if (!query.code) return writeResponse(400, '"code" query parameter is missing');
|
|
|
|
|
|
|
|
let body = new FormData();
|
|
|
|
body.append('client_id', this.clientId());
|
|
|
|
body.append('client_secret', this.clientSecret());
|
|
|
|
body.append('code', query.code ? query.code : '');
|
|
|
|
body.append('redirect_uri', 'http://localhost:' + port.toString());
|
|
|
|
body.append('grant_type', 'authorization_code');
|
|
|
|
|
|
|
|
let options = {
|
|
|
|
method: 'POST',
|
|
|
|
body: body,
|
|
|
|
};
|
|
|
|
|
|
|
|
fetch(this.tokenBaseUrl(), options).then((r) => {
|
|
|
|
if (!r.ok) {
|
|
|
|
errorMessage = 'Could not retrieve auth code: ' + r.status + ': ' + r.statusText;
|
|
|
|
writeResponse(400, errorMessage);
|
|
|
|
server.destroy();
|
|
|
|
return;
|
|
|
|
}
|
|
|
|
|
|
|
|
return r.json().then((json) => {
|
|
|
|
this.auth_ = json;
|
|
|
|
writeResponse(200, 'The application has been authorised - you may now close this browser tab.');
|
|
|
|
server.destroy();
|
|
|
|
});
|
|
|
|
});
|
|
|
|
});
|
|
|
|
|
|
|
|
server.on('close', () => {
|
|
|
|
if (errorMessage) {
|
|
|
|
reject(new Error(errorMessage));
|
|
|
|
} else {
|
|
|
|
resolve(this.auth_);
|
|
|
|
}
|
|
|
|
});
|
|
|
|
|
|
|
|
server.listen(port);
|
|
|
|
|
|
|
|
enableServerDestroy(server);
|
|
|
|
|
2017-06-27 21:26:29 +02:00
|
|
|
targetConsole.log('Please open this URL in your browser to authentify the application:');
|
|
|
|
targetConsole.log('');
|
|
|
|
targetConsole.log(authCodeUrl);
|
2017-06-22 23:52:27 +02:00
|
|
|
});
|
|
|
|
}
|
|
|
|
|
2017-06-22 21:44:38 +02:00
|
|
|
}
|
|
|
|
|
|
|
|
export { OneDriveApi };
|