X-Git-Url: https://git.arvados.org/arvados.git/blobdiff_plain/90df34ba0e84b5735c48382362284b5f0382dd0e..273119605b7f33940a5ef9b1422eb1ff152d6764:/src/common/webdav.ts diff --git a/src/common/webdav.ts b/src/common/webdav.ts index 27e1f22de5..bb8a68bdd2 100644 --- a/src/common/webdav.ts +++ b/src/common/webdav.ts @@ -2,19 +2,33 @@ // // SPDX-License-Identifier: AGPL-3.0 +import { customEncodeURI } from "./url"; + export class WebDAV { - defaults: WebDAVDefaults = { + private defaults: WebDAVDefaults = { baseURL: '', - headers: {}, + headers: { + 'Cache-Control': 'must-revalidate' + }, }; constructor(config?: Partial, private createRequest = () => new XMLHttpRequest()) { if (config) { - this.defaults = { ...this.defaults, ...config }; + this.defaults = { + ...this.defaults, + ...config, + headers: { + ...this.defaults.headers, + ...config.headers + }, + }; } } + getBaseUrl = (): string => this.defaults.baseURL; + setAuthorization = (token?) => this.defaults.headers.Authorization = token; + propfind = (url: string, config: WebDAVRequestConfig = {}) => this.request({ ...config, url, @@ -25,21 +39,47 @@ export class WebDAV { this.request({ ...config, url, method: 'PUT', - data, + data }) + get = (url: string, config: WebDAVRequestConfig = {}) => + this.request({ + ...config, url, + method: 'GET' + }) + + upload = (url: string, files: File[], config: WebDAVRequestConfig = {}) => { + return Promise.all( + files.map(file => this.request({ + ...config, url, + method: 'PUT', + data: file + })) + ); + } + copy = (url: string, destination: string, config: WebDAVRequestConfig = {}) => this.request({ ...config, url, method: 'COPY', - headers: { ...config.headers, Destination: this.defaults.baseURL + destination } + headers: { + ...config.headers, + Destination: this.defaults.baseURL + ? this.defaults.baseURL.replace(/\/+$/, '') + '/' + destination.replace(/^\/+/, '') + : destination + } }) move = (url: string, destination: string, config: WebDAVRequestConfig = {}) => this.request({ ...config, url, method: 'MOVE', - headers: { ...config.headers, Destination: this.defaults.baseURL + destination } + headers: { + ...config.headers, + Destination: this.defaults.baseURL + ? this.defaults.baseURL.replace(/\/+$/, '') + '/' + destination.replace(/^\/+/, '') + : destination + } }) delete = (url: string, config: WebDAVRequestConfig = {}) => @@ -51,24 +91,53 @@ export class WebDAV { private request = (config: RequestConfig) => { return new Promise((resolve, reject) => { const r = this.createRequest(); - r.open(config.method, this.defaults.baseURL + config.url); + this.defaults.baseURL = this.defaults.baseURL.replace(/\/+$/, ''); + r.open(config.method, + `${this.defaults.baseURL + ? this.defaults.baseURL+'/' + : ''}${customEncodeURI(config.url)}`); const headers = { ...this.defaults.headers, ...config.headers }; Object .keys(headers) .forEach(key => r.setRequestHeader(key, headers[key])); + if (!(window as any).cancelTokens) { + Object.assign(window, { cancelTokens: {} }); + } + + (window as any).cancelTokens[config.url] = () => { + resolve(r); + r.abort(); + } + if (config.onUploadProgress) { r.upload.addEventListener('progress', config.onUploadProgress); } - r.addEventListener('load', () => resolve(r)); - r.addEventListener('error', () => reject(r)); + // This event gets triggered on *any* server response + r.addEventListener('load', () => { + if (r.status >= 400) { + return reject(r); + } else { + return resolve(r); + } + }); + + // This event gets triggered on network errors + r.addEventListener('error', () => { + return reject(r); + }); + + r.upload.addEventListener('error', () => { + return reject(r); + }); r.send(config.data); }); } } + export interface WebDAVRequestConfig { headers?: { [key: string]: string;