"use strict"; var _interopRequireDefault = require("@babel/runtime/helpers/interopRequireDefault"); Object.defineProperty(exports, "__esModule", { value: true }); exports.default = void 0; var _regenerator = _interopRequireDefault(require("@babel/runtime/regenerator")); var _asyncToGenerator2 = _interopRequireDefault(require("@babel/runtime/helpers/asyncToGenerator")); var _classCallCheck2 = _interopRequireDefault(require("@babel/runtime/helpers/classCallCheck")); var _createClass2 = _interopRequireDefault(require("@babel/runtime/helpers/createClass")); var _defineProperty2 = _interopRequireDefault(require("@babel/runtime/helpers/defineProperty")); var _CoreManager = _interopRequireDefault(require("./CoreManager")); /** * Copyright (c) 2015-present, Parse, LLC. * All rights reserved. * * This source code is licensed under the BSD-style license found in the * LICENSE file in the root directory of this source tree. An additional grant * of patent rights can be found in the PATENTS file in the same directory. * * @flow */ /* global XMLHttpRequest, Blob */ var XHR = null; if (typeof XMLHttpRequest !== 'undefined') { XHR = XMLHttpRequest; } /*:: type Base64 = { base64: string };*/ /*:: type Uri = { uri: string };*/ /*:: type FileData = Array | Base64 | Blob | Uri;*/ /*:: export type FileSource = { format: 'file'; file: Blob; type: string } | { format: 'base64'; base64: string; type: string } | { format: 'uri'; uri: string; type: string };*/ var dataUriRegexp = /^data:([a-zA-Z]+\/[-a-zA-Z0-9+.]+)(;charset=[a-zA-Z0-9\-\/]*)?;base64,/; function b64Digit(number /*: number*/ ) /*: string*/ { if (number < 26) { return String.fromCharCode(65 + number); } if (number < 52) { return String.fromCharCode(97 + (number - 26)); } if (number < 62) { return String.fromCharCode(48 + (number - 52)); } if (number === 62) { return '+'; } if (number === 63) { return '/'; } throw new TypeError('Tried to encode large digit ' + number + ' in base64.'); } /** * A Parse.File is a local representation of a file that is saved to the Parse * cloud. * @alias Parse.File */ var ParseFile = /*#__PURE__*/ function () { /** * @param name {String} The file's name. This will be prefixed by a unique * value once the file has finished saving. The file name must begin with * an alphanumeric character, and consist of alphanumeric characters, * periods, spaces, underscores, or dashes. * @param data {Array} The data for the file, as either: * 1. an Array of byte value Numbers, or * 2. an Object like { base64: "..." } with a base64-encoded String. * 3. an Object like { uri: "..." } with a uri String. * 4. a File object selected with a file upload control. (3) only works * in Firefox 3.6+, Safari 6.0.2+, Chrome 7+, and IE 10+. * For example: *
   * var fileUploadControl = $("#profilePhotoFileUpload")[0];
   * if (fileUploadControl.files.length > 0) {
   *   var file = fileUploadControl.files[0];
   *   var name = "photo.jpg";
   *   var parseFile = new Parse.File(name, file);
   *   parseFile.save().then(function() {
   *     // The file has been saved to Parse.
   *   }, function(error) {
   *     // The file either could not be read, or could not be saved to Parse.
   *   });
   * }
* @param type {String} Optional Content-Type header to use for the file. If * this is omitted, the content type will be inferred from the name's * extension. */ function ParseFile(name /*: string*/ , data /*:: ?: FileData*/ , type /*:: ?: string*/ ) { (0, _classCallCheck2.default)(this, ParseFile); (0, _defineProperty2.default)(this, "_name", void 0); (0, _defineProperty2.default)(this, "_url", void 0); (0, _defineProperty2.default)(this, "_source", void 0); (0, _defineProperty2.default)(this, "_previousSave", void 0); (0, _defineProperty2.default)(this, "_data", void 0); var specifiedType = type || ''; this._name = name; if (data !== undefined) { if (Array.isArray(data)) { this._data = ParseFile.encodeBase64(data); this._source = { format: 'base64', base64: this._data, type: specifiedType }; } else if (typeof Blob !== 'undefined' && data instanceof Blob) { this._source = { format: 'file', file: data, type: specifiedType }; } else if (data && typeof data.uri === 'string' && data.uri !== undefined) { this._source = { format: 'uri', uri: data.uri, type: specifiedType }; } else if (data && typeof data.base64 === 'string') { var base64 = data.base64; var commaIndex = base64.indexOf(','); if (commaIndex !== -1) { var matches = dataUriRegexp.exec(base64.slice(0, commaIndex + 1)); // if data URI with type and charset, there will be 4 matches. this._data = base64.slice(commaIndex + 1); this._source = { format: 'base64', base64: this._data, type: matches[1] }; } else { this._data = base64; this._source = { format: 'base64', base64: base64, type: specifiedType }; } } else { throw new TypeError('Cannot create a Parse.File with that data.'); } } } /** * Return the data for the file, downloading it if not already present. * Data is present if initialized with Byte Array, Base64 or Saved with Uri. * Data is cleared if saved with File object selected with a file upload control * * @return {Promise} Promise that is resolve with base64 data */ (0, _createClass2.default)(ParseFile, [{ key: "getData", value: function () { var _getData = (0, _asyncToGenerator2.default)( /*#__PURE__*/ _regenerator.default.mark(function _callee() { var controller, result; return _regenerator.default.wrap(function (_context) { while (1) { switch (_context.prev = _context.next) { case 0: if (!this._data) { _context.next = 2; break; } return _context.abrupt("return", this._data); case 2: if (this._url) { _context.next = 4; break; } throw new Error('Cannot retrieve data for unsaved ParseFile.'); case 4: controller = _CoreManager.default.getFileController(); _context.next = 7; return controller.download(this._url); case 7: result = _context.sent; this._data = result.base64; return _context.abrupt("return", this._data); case 10: case "end": return _context.stop(); } } }, _callee, this); })); return function () { return _getData.apply(this, arguments); }; }() /** * Gets the name of the file. Before save is called, this is the filename * given by the user. After save is called, that name gets prefixed with a * unique identifier. * @return {String} */ }, { key: "name", value: function () /*: string*/ { return this._name; } /** * Gets the url of the file. It is only available after you save the file or * after you get the file from a Parse.Object. * @param {Object} options An object to specify url options * @return {String} */ }, { key: "url", value: function (options /*:: ?: { forceSecure?: boolean }*/ ) /*: ?string*/ { options = options || {}; if (!this._url) { return; } if (options.forceSecure) { return this._url.replace(/^http:\/\//i, 'https://'); } else { return this._url; } } /** * Saves the file to the Parse cloud. * @param {Object} options * * Valid options are: * @return {Promise} Promise that is resolved when the save finishes. */ }, { key: "save", value: function (options /*:: ?: FullOptions*/ ) { var _this = this; options = options || {}; var controller = _CoreManager.default.getFileController(); if (!this._previousSave) { if (this._source.format === 'file') { this._previousSave = controller.saveFile(this._name, this._source, options).then(function (res) { _this._name = res.name; _this._url = res.url; _this._data = null; return _this; }); } else if (this._source.format === 'uri') { this._previousSave = controller.download(this._source.uri).then(function (result) { var newSource = { format: 'base64', base64: result.base64, type: result.contentType }; _this._data = result.base64; return controller.saveBase64(_this._name, newSource, options); }).then(function (res) { _this._name = res.name; _this._url = res.url; return _this; }); } else { this._previousSave = controller.saveBase64(this._name, this._source, options).then(function (res) { _this._name = res.name; _this._url = res.url; return _this; }); } } if (this._previousSave) { return this._previousSave; } } }, { key: "toJSON", value: function () /*: { name: ?string, url: ?string }*/ { return { __type: 'File', name: this._name, url: this._url }; } }, { key: "equals", value: function (other /*: mixed*/ ) /*: boolean*/ { if (this === other) { return true; } // Unsaved Files are never equal, since they will be saved to different URLs return other instanceof ParseFile && this.name() === other.name() && this.url() === other.url() && typeof this.url() !== 'undefined'; } }], [{ key: "fromJSON", value: function (obj) /*: ParseFile*/ { if (obj.__type !== 'File') { throw new TypeError('JSON object does not represent a ParseFile'); } var file = new ParseFile(obj.name); file._url = obj.url; return file; } }, { key: "encodeBase64", value: function (bytes /*: Array*/ ) /*: string*/ { var chunks = []; chunks.length = Math.ceil(bytes.length / 3); for (var i = 0; i < chunks.length; i++) { var b1 = bytes[i * 3]; var b2 = bytes[i * 3 + 1] || 0; var b3 = bytes[i * 3 + 2] || 0; var has2 = i * 3 + 1 < bytes.length; var has3 = i * 3 + 2 < bytes.length; chunks[i] = [b64Digit(b1 >> 2 & 0x3F), b64Digit(b1 << 4 & 0x30 | b2 >> 4 & 0x0F), has2 ? b64Digit(b2 << 2 & 0x3C | b3 >> 6 & 0x03) : '=', has3 ? b64Digit(b3 & 0x3F) : '='].join(''); } return chunks.join(''); } }]); return ParseFile; }(); var DefaultController = { saveFile: function (name /*: string*/ , source /*: FileSource*/ , options /*:: ?: FullOptions*/ ) { if (source.format !== 'file') { throw new Error('saveFile can only be used with File-type sources.'); } // To directly upload a File, we use a REST-style AJAX request var headers = { 'X-Parse-Application-ID': _CoreManager.default.get('APPLICATION_ID'), 'Content-Type': source.type || (source.file ? source.file.type : null) }; var jsKey = _CoreManager.default.get('JAVASCRIPT_KEY'); if (jsKey) { headers['X-Parse-JavaScript-Key'] = jsKey; } var url = _CoreManager.default.get('SERVER_URL'); if (url[url.length - 1] !== '/') { url += '/'; } url += 'files/' + name; return _CoreManager.default.getRESTController().ajax('POST', url, source.file, headers, options).then(function (res) { return res.response; }); }, saveBase64: function (name /*: string*/ , source /*: FileSource*/ , options /*:: ?: FullOptions*/ ) { if (source.format !== 'base64') { throw new Error('saveBase64 can only be used with Base64-type sources.'); } var data /*: { base64: any; _ContentType?: any }*/ = { base64: source.base64 }; if (source.type) { data._ContentType = source.type; } return _CoreManager.default.getRESTController().request('POST', 'files/' + name, data, options); }, download: function (uri) { if (XHR) { return this.downloadAjax(uri); } else { return Promise.reject('Cannot make a request: No definition of XMLHttpRequest was found.'); } }, downloadAjax: function (uri) { return new Promise(function (resolve, reject) { var xhr = new XHR(); xhr.open('GET', uri, true); xhr.responseType = 'arraybuffer'; xhr.onerror = function (e) { reject(e); }; xhr.onreadystatechange = function () { if (xhr.readyState !== 4) { return; } var bytes = new Uint8Array(this.response); resolve({ base64: ParseFile.encodeBase64(bytes), contentType: xhr.getResponseHeader('content-type') }); }; xhr.send(); }); }, _setXHR: function (xhr /*: any*/ ) { XHR = xhr; } }; _CoreManager.default.setFileController(DefaultController); var _default = ParseFile; exports.default = _default;