UNPKG

@harishreddym/baqend

Version:

Baqend JavaScript SDK

559 lines (489 loc) 14.1 kB
'use strict'; const CommunicationError = require('../error/CommunicationError'); /** * Checks whether the user uses a browser which does support revalidation. */ const REVALIDATION_SUPPORTED = typeof navigator === 'undefined' || (typeof chrome !== 'undefined' && /google/i.test(navigator.vendor)) || (/cros i686/i.test(navigator.platform)); /** * @typedef {'json'|'text'|'blob'|'buffer'|'arraybuffer'|'data-url'|'form'} EntityType */ const RESPONSE_TYPE = Symbol('ResponseType'); /** * @alias connector.Message */ class Message { /** * Creates a new message class with the given message specification * @param {Object} specification * @return {Class<Message>} */ static create(specification) { const parts = specification.path.split('?'); const path = parts[0].split(/[:*]\w*/); const query = []; if (parts[1]) { parts[1].split('&').forEach((arg) => { const part = arg.split('='); query.push(part[0]); }); } specification.dynamic = specification.path.indexOf('*') !== -1; specification.path = path; specification.query = query; return class extends Message { get spec() { return specification; } }; } /** * Creates a new message class with the given message specification and a full path * @param {Object} specification * @param {Object} members additional members applied to the created message * @return {Class<Message>} */ static createExternal(specification, members) { specification.path = [specification.path]; /** * @ignore */ const cls = class extends Message { get spec() { return specification; } }; Object.assign(cls.prototype, members); return cls; } get isBinary() { return this.request.type in Message.BINARY || this[RESPONSE_TYPE] in Message.BINARY; } /** * @param {string} arguments... The path arguments */ constructor() { /** @type boolean */ this.withCredentials = false; /** @type util.TokenStorage */ this.tokenStorage = null; /** @type connector.Message~progressCallback */ this.progressCallback = null; const args = arguments; let index = 0; let path = this.spec.path; if (Object(path) instanceof Array) { path = this.spec.path[0]; const len = this.spec.path.length; for (let i = 1; i < len; i += 1) { if (this.spec.dynamic && len - 1 === i) { path += args[index].split('/').map(encodeURIComponent).join('/'); } else { path += encodeURIComponent(args[index]) + this.spec.path[i]; } index += 1; } } let query = ''; for (let i = 0; i < this.spec.query.length; i += 1) { const arg = args[index]; index += 1; if (arg !== undefined && arg !== null) { query += (query || path.indexOf('?') !== -1) ? '&' : '?'; query += this.spec.query[i] + '=' + encodeURIComponent(arg); } } this.request = { method: this.spec.method, path: path + query, entity: null, headers: {}, }; if (args[index]) { this.entity(args[index], 'json'); } this.responseType('json'); } /** * Gets the value of a the specified request header * @param {string} name The header name * @return {string} The header value * @name header * @memberOf connector.Message.prototype * @method */ /** * Sets the value of a the specified request header * @param {string} name The header name * @param {string} value The header value if omitted the value will be returned * @return {this} This message object */ header(name, value) { if (value !== undefined) { this.request.headers[name] = value; return this; } return this.request.headers[name]; } /** * Sets the entity type * @param {*} data The data to send * @param {EntityType} [type="json"] the type of the data one of 'json'|'text'|'blob'|'arraybuffer' defaults to 'json' * @return {this} This message object */ entity(data, type) { let requestType = type; if (!requestType) { if (typeof data === 'string') { if (/^data:(.+?)(;base64)?,.*$/.test(data)) { requestType = 'data-url'; } else { requestType = 'text'; } } else if (typeof Blob !== 'undefined' && data instanceof Blob) { requestType = 'blob'; } else if (typeof Buffer !== 'undefined' && data instanceof Buffer) { requestType = 'buffer'; } else if (typeof ArrayBuffer !== 'undefined' && data instanceof ArrayBuffer) { requestType = 'arraybuffer'; } else if (typeof FormData !== 'undefined' && data instanceof FormData) { requestType = 'form'; } else { requestType = 'json'; } } this.request.type = requestType; this.request.entity = data; return this; } /** * Get the mimeType * @return {string} This message object * @name mimeType * @memberOf connector.Message.prototype * @method */ /** * Sets the mimeType * @param {string} mimeType the mimeType of the data * @return {this} This message object */ mimeType(mimeType) { return this.header('content-type', mimeType); } /** * Gets the contentLength * @return {number} * @name contentLength * @memberOf connector.Message.prototype * @method */ /** * Sets the contentLength * @param {number} contentLength the content length of the data * @return {this} This message object */ contentLength(contentLength) { return this.header('content-length', contentLength); } /** * Gets the request conditional If-Match header * @return {string} This message object * @name ifMatch * @memberOf connector.Message.prototype * @method */ /** * Sets the request conditional If-Match header * @param {string} eTag the If-Match ETag value * @return {this} This message object */ ifMatch(eTag) { return this.header('If-Match', this.formatETag(eTag)); } /** * Gets the request a ETag based conditional header * @return {string} * @name ifNoneMatch * @memberOf connector.Message.prototype * @method */ /** * Sets the request a ETag based conditional header * @param {string} eTag The ETag value * @return {this} This message object */ ifNoneMatch(eTag) { return this.header('If-None-Match', this.formatETag(eTag)); } /** * Gets the request date based conditional header * @return {string} This message object * @name ifUnmodifiedSince * @memberOf connector.Message.prototype * @method */ /** * Sets the request date based conditional header * @param {Date} date The date value * @return {this} This message object */ ifUnmodifiedSince(date) { // IE 10 returns UTC strings and not an RFC-1123 GMT date string return this.header('if-unmodified-since', date && date.toUTCString().replace('UTC', 'GMT')); } /** * Indicates that the request should not be served by a local cache * @return {this} */ noCache() { if (!REVALIDATION_SUPPORTED) { this.ifMatch('') // is needed for firefox or safari (but forbidden for chrome) .ifNoneMatch('-'); // is needed for edge and ie (but forbidden for chrome) } return this.cacheControl('max-age=0, no-cache'); } /** * Gets the cache control header * @return {string} * @name cacheControl * @memberOf connector.Message.prototype * @method */ /** * Sets the cache control header * @param {string} value The cache control flags * @return {this} This message object */ cacheControl(value) { return this.header('cache-control', value); } /** * Gets the ACL of a file into the Baqend-Acl header * @return {string} This message object * @name acl * @memberOf connector.Message.prototype * @method */ /** * Sets and encodes the ACL of a file into the Baqend-Acl header * @param {Acl} acl the file ACLs * @return {this} This message object */ acl(acl) { return this.header('baqend-acl', acl && JSON.stringify(acl)); } /** * Gets and encodes the custom headers of a file into the Baqend-Custom-Headers header * @return {string} This message object * @name customHeaders * @memberOf connector.Message.prototype * @method */ /** * Sets and encodes the custom headers of a file into the Baqend-Custom-Headers header * @param {*} customHeaders the file custom headers * @return {this} This message object */ customHeaders(customHeaders) { return this.header('baqend-custom-headers', customHeaders && JSON.stringify(customHeaders)); } /** * Gets the request accept header * @return {string} This message object * @name accept * @memberOf connector.Message.prototype * @method */ /** * Sets the request accept header * @param {string} accept the accept header value * @return {this} This message object */ accept(accept) { return this.header('accept', accept); } /** * Gets the response type which should be returned * @return {string} This message object * @name responseType * @memberOf connector.Message.prototype * @method */ /** * Sets the response type which should be returned * @param {string} type The response type one of 'json'|'text'|'blob'|'arraybuffer' defaults to 'json' * @return {this} This message object */ responseType(type) { if (type !== undefined) { this[RESPONSE_TYPE] = type; return this; } return this[RESPONSE_TYPE]; } /** * Gets the progress callback * @return {connector.Message~progressCallback} The callback set * @name progress * @memberOf connector.Message.prototype * @method */ /** * Sets the progress callback * @param {connector.Message~progressCallback} callback * @return {this} This message object */ progress(callback) { if (callback !== undefined) { this.progressCallback = callback; return this; } return this.progressCallback; } /** * Adds the given string to the request path * * If the parameter is an object, it will be serialized as a query string. * * @param {string|Object<string,string>} query which will added to the request path * @return {this} */ addQueryString(query) { if (Object(query) instanceof String) { this.request.path += query; return this; } if (query) { let sep = this.request.path.indexOf('?') >= 0 ? '&' : '?'; Object.keys(query).forEach((key) => { this.request.path += sep + key + '=' + encodeURIComponent(query[key]); sep = '&'; }); } return this; } formatETag(eTag) { let tag = eTag; if (tag && tag !== '*') { tag = '' + tag; if (tag.indexOf('"') === -1) { tag = '"' + tag + '"'; } } return tag; } /** * Handle the receive * @param {Object} response The received response headers and data * @return {void} */ doReceive(response) { if (this.spec.status.indexOf(response.status) === -1) { throw new CommunicationError(this, response); } } } /** * The message specification * @name spec * @memberOf connector.Message.prototype * @type {Object} */ Object.assign(Message, { /** * @alias connector.Message.StatusCode * @enum {number} */ StatusCode: { NOT_MODIFIED: 304, BAD_CREDENTIALS: 460, BUCKET_NOT_FOUND: 461, INVALID_PERMISSION_MODIFICATION: 462, INVALID_TYPE_VALUE: 463, OBJECT_NOT_FOUND: 404, OBJECT_OUT_OF_DATE: 412, PERMISSION_DENIED: 466, QUERY_DISPOSED: 467, QUERY_NOT_SUPPORTED: 468, SCHEMA_NOT_COMPATIBLE: 469, SCHEMA_STILL_EXISTS: 470, SYNTAX_ERROR: 471, TRANSACTION_INACTIVE: 472, TYPE_ALREADY_EXISTS: 473, TYPE_STILL_REFERENCED: 474, SCRIPT_ABORTION: 475, }, BINARY: { blob: true, buffer: true, stream: true, arraybuffer: true, 'data-url': true, base64: true, }, GoogleOAuth: Message.createExternal({ method: 'OAUTH', path: 'https://accounts.google.com/o/oauth2/auth?response_type=code&access_type=online', query: ['client_id', 'scope', 'state'], status: [200], }, { addRedirectOrigin(baseUri) { this.addQueryString({ redirect_uri: baseUri + '/db/User/OAuth/google', }); }, }), FacebookOAuth: Message.createExternal({ method: 'OAUTH', path: 'https://www.facebook.com/dialog/oauth?response_type=code', query: ['client_id', 'scope', 'state'], status: [200], }, { addRedirectOrigin(baseUri) { this.addQueryString({ redirect_uri: baseUri + '/db/User/OAuth/facebook', }); }, }), GitHubOAuth: Message.createExternal({ method: 'OAUTH', path: 'https://github.com/login/oauth/authorize?response_type=code&access_type=online', query: ['client_id', 'scope', 'state'], status: [200], }, { addRedirectOrigin(baseUri) { this.addQueryString({ redirect_uri: baseUri + '/db/User/OAuth/github', }); }, }), LinkedInOAuth: Message.createExternal({ method: 'OAUTH', path: 'https://www.linkedin.com/uas/oauth2/authorization?response_type=code&access_type=online', query: ['client_id', 'scope', 'state'], status: [200], }, { addRedirectOrigin(baseUri) { this.addQueryString({ redirect_uri: baseUri + '/db/User/OAuth/linkedin', }); }, }), TwitterOAuth: Message.createExternal({ method: 'OAUTH', path: '', query: [], status: [200], }, { addRedirectOrigin(baseUri) { this.request.path = baseUri + '/db/User/OAuth1/twitter'; }, }), }); module.exports = Message; /** * The progress callback is called, when you send a message to the server and a progress is noticed * @callback connector.Message~progressCallback * @param {ProgressEvent} event The Progress Event * @return {*} unused */