| 
									
										
										
										
											2017-07-06 18:58:01 +00:00
										 |  |  | import { shim } from 'lib/shim.js'; | 
					
						
							| 
									
										
										
										
											2017-06-22 20:44:38 +01:00
										 |  |  | import { stringify } from 'query-string'; | 
					
						
							| 
									
										
										
										
											2017-07-06 19:48:17 +00:00
										 |  |  | import { time } from 'lib/time-utils.js'; | 
					
						
							| 
									
										
										
										
											2017-07-09 16:47:05 +01:00
										 |  |  | import { Logger } from 'lib/logger.js' | 
					
						
							| 
									
										
										
										
											2017-06-22 20:44:38 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | class OneDriveApi { | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01: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 20:44:38 +01:00
										 |  |  | 		this.clientId_ = clientId; | 
					
						
							|  |  |  | 		this.clientSecret_ = clientSecret; | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 		this.auth_ = null; | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01:00
										 |  |  | 		this.isPublic_ = isPublic; | 
					
						
							| 
									
										
										
										
											2017-06-23 18:51:02 +00:00
										 |  |  | 		this.listeners_ = { | 
					
						
							|  |  |  | 			'authRefreshed': [], | 
					
						
							|  |  |  | 		}; | 
					
						
							| 
									
										
										
										
											2017-07-09 16:47:05 +01:00
										 |  |  | 		this.logger_ = new Logger(); | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	setLogger(l) { | 
					
						
							|  |  |  | 		this.logger_ = l; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 	logger() { | 
					
						
							|  |  |  | 		return this.logger_; | 
					
						
							| 
									
										
										
										
											2017-06-23 18:51:02 +00:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01:00
										 |  |  | 	isPublic() { | 
					
						
							|  |  |  | 		return this.isPublic_; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 18:51:02 +00: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 20:44:38 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 	tokenBaseUrl() { | 
					
						
							|  |  |  | 		return 'https://login.microsoftonline.com/common/oauth2/v2.0/token'; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 18:58:01 +00:00
										 |  |  | 	auth() { | 
					
						
							|  |  |  | 		return this.auth_; | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 	setAuth(auth) { | 
					
						
							|  |  |  | 		this.auth_ = auth; | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01: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() { | 
					
						
							| 
									
										
										
										
											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 22:52:27 +01:00
										 |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 20:44:38 +01: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-06 23:15:31 +01:00
										 |  |  | 	async execTokenRequest(code, redirectUri) { | 
					
						
							| 
									
										
										
										
											2017-07-06 19:29:09 +00:00
										 |  |  | 		let body = new shim.FormData(); | 
					
						
							|  |  |  | 		body.append('client_id', this.clientId()); | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01:00
										 |  |  | 		if (!this.isPublic()) body.append('client_secret', this.clientSecret()); | 
					
						
							| 
									
										
										
										
											2017-07-06 19:29:09 +00: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-06 23:15:31 +01:00
										 |  |  | 			this.setAuth(null); | 
					
						
							| 
									
										
										
										
											2017-07-06 19:29:09 +00:00
										 |  |  | 			const text = await r.text(); | 
					
						
							|  |  |  | 			error.message += ': ' + text; | 
					
						
							|  |  |  | 			throw error; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 	} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-29 18:03:16 +00: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 20:44:38 +01:00
										 |  |  | 	async exec(method, path, query = null, data = null, options = null) { | 
					
						
							| 
									
										
										
										
											2017-07-24 18:01:40 +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 = {}; | 
					
						
							| 
									
										
										
										
											2017-07-06 22:30:45 +01:00
										 |  |  | 		if (!options.target) options.target = 'string'; | 
					
						
							| 
									
										
										
										
											2017-06-22 20:44:38 +01:00
										 |  |  | 
 | 
					
						
							|  |  |  | 		if (method != 'GET') { | 
					
						
							|  |  |  | 			options.method = method; | 
					
						
							|  |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 22:32:24 +01:00
										 |  |  | 		if (method == 'PATCH' || method == 'POST') { | 
					
						
							| 
									
										
										
										
											2017-06-22 20:44:38 +01:00
										 |  |  | 			options.headers['Content-Type'] = 'application/json'; | 
					
						
							|  |  |  | 			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).
 | 
					
						
							|  |  |  | 		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 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
 | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-23 18:51:02 +00:00
										 |  |  | 		for (let i = 0; i < 5; i++) { | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 			options.headers['Authorization'] = 'bearer ' + this.token(); | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 22:30:45 +01:00
										 |  |  | 			let response = null; | 
					
						
							| 
									
										
										
										
											2017-07-12 23:32:08 +01: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-12 23:32:08 +01:00
										 |  |  | 					response = await shim.fetch(url, options); | 
					
						
							|  |  |  | 				} else { // file
 | 
					
						
							|  |  |  | 					response = await shim.fetchBlob(url, options); | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 			} catch (error) { | 
					
						
							| 
									
										
										
										
											2017-10-15 12:13:09 +01:00
										 |  |  | 				let canRetry = true; | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-13 18:09:47 +00: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-10-15 12:13:09 +01:00
										 |  |  | 				} else if (error.code == 'ECONNRESET') { | 
					
						
							|  |  |  | 					// request to https://public-ch3302....1fab24cb1bd5f.md failed, reason: socket hang up"
 | 
					
						
							| 
									
										
										
										
											2017-10-18 23:13:53 +01:00
										 |  |  | 				} else if (error.code == 'ENOTFOUND') { | 
					
						
							|  |  |  | 					// OneDrive (or Node?) sometimes sends back a "not found" error for resources
 | 
					
						
							|  |  |  | 					// that definitely exist and in this case repeating the request works.
 | 
					
						
							|  |  |  | 					// Error is:
 | 
					
						
							|  |  |  | 					// request to https://graph.microsoft.com/v1.0/drive/special/approot failed, reason: getaddrinfo ENOTFOUND graph.microsoft.com graph.microsoft.com:443
 | 
					
						
							| 
									
										
										
										
											2017-10-15 12:13:09 +01:00
										 |  |  | 				} else if (error.message.indexOf('network timeout') === 0) { | 
					
						
							|  |  |  | 					// network timeout at: https://public-ch3302...859f9b0e3ab.md
 | 
					
						
							|  |  |  | 				} else { | 
					
						
							|  |  |  | 					canRetry = false; | 
					
						
							|  |  |  | 				} | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | 				if (canRetry) { | 
					
						
							|  |  |  | 					this.logger().info('Got error code ' + error.code + ': ' + error.message + ' - retrying (' + i + ')...'); | 
					
						
							| 
									
										
										
										
											2017-07-13 18:09:47 +00:00
										 |  |  | 					await time.sleep((i + 1) * 3); | 
					
						
							|  |  |  | 					continue; | 
					
						
							|  |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2017-10-15 12:13:09 +01:00
										 |  |  | 					this.logger().error('Got unhandled error:', error ? error.code : '', error ? error.message : '', error); | 
					
						
							| 
									
										
										
										
											2017-07-13 18:09:47 +00:00
										 |  |  | 					throw error; | 
					
						
							|  |  |  | 				} | 
					
						
							| 
									
										
										
										
											2017-07-06 22:30:45 +01:00
										 |  |  | 			} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 			if (!response.ok) { | 
					
						
							| 
									
										
										
										
											2017-06-29 18:03:16 +00:00
										 |  |  | 				let errorResponse = await response.json(); | 
					
						
							|  |  |  | 				let error = this.oneDriveErrorResponseToError(errorResponse); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-07 23:25:03 +01:00
										 |  |  | 				if (error.code == 'InvalidAuthenticationToken' || error.code == 'unauthenticated') { | 
					
						
							| 
									
										
										
										
											2017-07-09 16:47:05 +01:00
										 |  |  | 					this.logger().info('Token expired: refreshing...'); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 					await this.refreshAccessToken(); | 
					
						
							|  |  |  | 					continue; | 
					
						
							| 
									
										
										
										
											2017-07-10 00:20:38 +01: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-10 00:20:38 +01:00
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											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' }
 | 
					
						
							| 
									
										
										
										
											2017-07-30 22:22:57 +02:00
										 |  |  | 					this.logger().info('Got error below - retrying (' + i + ')...'); | 
					
						
							|  |  |  | 					this.logger().info(error); | 
					
						
							| 
									
										
										
										
											2017-07-13 18:09:47 +00:00
										 |  |  | 					await time.sleep((i + 1) * 3); | 
					
						
							| 
									
										
										
										
											2017-07-06 19:48:17 +00:00
										 |  |  | 					continue; | 
					
						
							| 
									
										
										
										
											2017-07-11 00:17:03 +01:00
										 |  |  | 				} else if (error.code == 'itemNotFound' && method == 'DELETE') { | 
					
						
							|  |  |  | 					// Deleting a non-existing item is ok - noop
 | 
					
						
							|  |  |  | 					return; | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 				} else { | 
					
						
							| 
									
										
										
										
											2017-06-29 18:03:16 +00:00
										 |  |  | 					error.request = method + ' ' + url + ' ' + JSON.stringify(query) + ' ' + JSON.stringify(data) + ' ' + JSON.stringify(options); | 
					
						
							| 
									
										
										
										
											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; | 
					
						
							|  |  |  | 		} | 
					
						
							| 
									
										
										
										
											2017-06-23 18:51:02 +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); | 
					
						
							|  |  |  | 		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 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); | 
					
						
							| 
									
										
										
										
											2017-07-28 18:13:07 +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
										 |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 18:58:01 +00:00
										 |  |  | 		let body = new shim.FormData(); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 		body.append('client_id', this.clientId()); | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01:00
										 |  |  | 		if (!this.isPublic()) body.append('client_secret', this.clientSecret()); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01: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 18:58:01 +00:00
										 |  |  | 		let response = await shim.fetch(this.tokenBaseUrl(), options); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 		if (!response.ok) { | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +01:00
										 |  |  | 			this.setAuth(null); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 			let msg = await response.text(); | 
					
						
							| 
									
										
										
										
											2017-07-15 17:14:15 +01:00
										 |  |  | 			throw new Error(msg + ': TOKEN: ' + this.auth_); | 
					
						
							| 
									
										
										
										
											2017-06-22 22:52:27 +01:00
										 |  |  | 		} | 
					
						
							|  |  |  | 
 | 
					
						
							| 
									
										
										
										
											2017-07-06 23:15:31 +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
										 |  |  | } | 
					
						
							|  |  |  | 
 | 
					
						
							|  |  |  | export { OneDriveApi }; |