UNPKG

browserfs

Version:

A filesystem in your browser!

281 lines 38.9 kB
"use strict"; var __extends = (this && this.__extends) || function (d, b) { for (var p in b) if (b.hasOwnProperty(p)) d[p] = b[p]; function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; var file_system = require('../core/file_system'); var api_error_1 = require('../core/api_error'); var file_flag_1 = require('../core/file_flag'); var util_1 = require('../core/util'); var preload_file = require('../generic/preload_file'); var xhr = require('../generic/xhr'); var file_index_1 = require('../generic/file_index'); function tryToString(buff, encoding, cb) { try { cb(null, buff.toString(encoding)); } catch (e) { cb(e); } } var XmlHttpRequest = (function (_super) { __extends(XmlHttpRequest, _super); function XmlHttpRequest(listingUrlOrObj, prefixUrl) { if (prefixUrl === void 0) { prefixUrl = ''; } _super.call(this); if (!listingUrlOrObj) { listingUrlOrObj = 'index.json'; } if (prefixUrl.length > 0 && prefixUrl.charAt(prefixUrl.length - 1) !== '/') { prefixUrl = prefixUrl + '/'; } this.prefixUrl = prefixUrl; var listing = null; if (typeof (listingUrlOrObj) === "string") { listing = this._requestFileSync(listingUrlOrObj, 'json'); if (!listing) { throw new Error("Unable to find listing at URL: ${listingUrlOrObj}"); } } else { listing = listingUrlOrObj; } this._index = file_index_1.FileIndex.fromListing(listing); } XmlHttpRequest.prototype.empty = function () { this._index.fileIterator(function (file) { file.file_data = null; }); }; XmlHttpRequest.prototype.getXhrPath = function (filePath) { if (filePath.charAt(0) === '/') { filePath = filePath.slice(1); } return this.prefixUrl + filePath; }; XmlHttpRequest.prototype._requestFileSizeAsync = function (path, cb) { xhr.getFileSizeAsync(this.getXhrPath(path), cb); }; XmlHttpRequest.prototype._requestFileSizeSync = function (path) { return xhr.getFileSizeSync(this.getXhrPath(path)); }; XmlHttpRequest.prototype._requestFileAsync = function (p, type, cb) { xhr.asyncDownloadFile(this.getXhrPath(p), type, cb); }; XmlHttpRequest.prototype._requestFileSync = function (p, type) { return xhr.syncDownloadFile(this.getXhrPath(p), type); }; XmlHttpRequest.prototype.getName = function () { return 'XmlHttpRequest'; }; XmlHttpRequest.isAvailable = function () { return typeof XMLHttpRequest !== "undefined" && XMLHttpRequest !== null; }; XmlHttpRequest.prototype.diskSpace = function (path, cb) { cb(0, 0); }; XmlHttpRequest.prototype.isReadOnly = function () { return true; }; XmlHttpRequest.prototype.supportsLinks = function () { return false; }; XmlHttpRequest.prototype.supportsProps = function () { return false; }; XmlHttpRequest.prototype.supportsSynch = function () { return true; }; XmlHttpRequest.prototype.preloadFile = function (path, buffer) { var inode = this._index.getInode(path); if (file_index_1.isFileInode(inode)) { if (inode === null) { throw api_error_1.ApiError.ENOENT(path); } var stats = inode.getData(); stats.size = buffer.length; stats.file_data = buffer; } else { throw api_error_1.ApiError.EISDIR(path); } }; XmlHttpRequest.prototype.stat = function (path, isLstat, cb) { var inode = this._index.getInode(path); if (inode === null) { return cb(api_error_1.ApiError.ENOENT(path)); } var stats; if (file_index_1.isFileInode(inode)) { stats = inode.getData(); if (stats.size < 0) { this._requestFileSizeAsync(path, function (e, size) { if (e) { return cb(e); } stats.size = size; cb(null, stats.clone()); }); } else { cb(null, stats.clone()); } } else if (file_index_1.isDirInode(inode)) { stats = inode.getStats(); cb(null, stats); } else { cb(api_error_1.ApiError.FileError(api_error_1.ErrorCode.EINVAL, path)); } }; XmlHttpRequest.prototype.statSync = function (path, isLstat) { var inode = this._index.getInode(path); if (inode === null) { throw api_error_1.ApiError.ENOENT(path); } var stats; if (file_index_1.isFileInode(inode)) { stats = inode.getData(); if (stats.size < 0) { stats.size = this._requestFileSizeSync(path); } } else if (file_index_1.isDirInode(inode)) { stats = inode.getStats(); } else { throw api_error_1.ApiError.FileError(api_error_1.ErrorCode.EINVAL, path); } return stats; }; XmlHttpRequest.prototype.open = function (path, flags, mode, cb) { if (flags.isWriteable()) { return cb(new api_error_1.ApiError(api_error_1.ErrorCode.EPERM, path)); } var _this = this; var inode = this._index.getInode(path); if (inode === null) { return cb(api_error_1.ApiError.ENOENT(path)); } if (file_index_1.isFileInode(inode)) { var stats = inode.getData(); switch (flags.pathExistsAction()) { case file_flag_1.ActionType.THROW_EXCEPTION: case file_flag_1.ActionType.TRUNCATE_FILE: return cb(api_error_1.ApiError.EEXIST(path)); case file_flag_1.ActionType.NOP: if (stats.file_data != null) { return cb(null, new preload_file.NoSyncFile(_this, path, flags, stats.clone(), stats.file_data)); } this._requestFileAsync(path, 'buffer', function (err, buffer) { if (err) { return cb(err); } stats.size = buffer.length; stats.file_data = buffer; return cb(null, new preload_file.NoSyncFile(_this, path, flags, stats.clone(), buffer)); }); break; default: return cb(new api_error_1.ApiError(api_error_1.ErrorCode.EINVAL, 'Invalid FileMode object.')); } } else { return cb(api_error_1.ApiError.EISDIR(path)); } }; XmlHttpRequest.prototype.openSync = function (path, flags, mode) { if (flags.isWriteable()) { throw new api_error_1.ApiError(api_error_1.ErrorCode.EPERM, path); } var inode = this._index.getInode(path); if (inode === null) { throw api_error_1.ApiError.ENOENT(path); } if (file_index_1.isFileInode(inode)) { var stats = inode.getData(); switch (flags.pathExistsAction()) { case file_flag_1.ActionType.THROW_EXCEPTION: case file_flag_1.ActionType.TRUNCATE_FILE: throw api_error_1.ApiError.EEXIST(path); case file_flag_1.ActionType.NOP: if (stats.file_data != null) { return new preload_file.NoSyncFile(this, path, flags, stats.clone(), stats.file_data); } var buffer = this._requestFileSync(path, 'buffer'); stats.size = buffer.length; stats.file_data = buffer; return new preload_file.NoSyncFile(this, path, flags, stats.clone(), buffer); default: throw new api_error_1.ApiError(api_error_1.ErrorCode.EINVAL, 'Invalid FileMode object.'); } } else { throw api_error_1.ApiError.EISDIR(path); } }; XmlHttpRequest.prototype.readdir = function (path, cb) { try { cb(null, this.readdirSync(path)); } catch (e) { cb(e); } }; XmlHttpRequest.prototype.readdirSync = function (path) { var inode = this._index.getInode(path); if (inode === null) { throw api_error_1.ApiError.ENOENT(path); } else if (file_index_1.isDirInode(inode)) { return inode.getListing(); } else { throw api_error_1.ApiError.ENOTDIR(path); } }; XmlHttpRequest.prototype.readFile = function (fname, encoding, flag, cb) { var oldCb = cb; this.open(fname, flag, 0x1a4, function (err, fd) { if (err) { return cb(err); } cb = function (err, arg) { fd.close(function (err2) { if (err == null) { err = err2; } return oldCb(err, arg); }); }; var fdCast = fd; var fdBuff = fdCast.getBuffer(); if (encoding === null) { cb(err, util_1.copyingSlice(fdBuff)); } else { tryToString(fdBuff, encoding, cb); } }); }; XmlHttpRequest.prototype.readFileSync = function (fname, encoding, flag) { var fd = this.openSync(fname, flag, 0x1a4); try { var fdCast = fd; var fdBuff = fdCast.getBuffer(); if (encoding === null) { return util_1.copyingSlice(fdBuff); } return fdBuff.toString(encoding); } finally { fd.closeSync(); } }; return XmlHttpRequest; }(file_system.BaseFileSystem)); exports.__esModule = true; exports["default"] = XmlHttpRequest; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"XmlHttpRequest.js","sourceRoot":"","sources":["../../../src/backend/XmlHttpRequest.ts"],"names":[],"mappings":"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","sourcesContent":["import file_system = require('../core/file_system');\nimport {ApiError, ErrorCode} from '../core/api_error';\nimport {FileFlag, ActionType} from '../core/file_flag';\nimport {copyingSlice} from '../core/util';\nimport file = require('../core/file');\nimport Stats from '../core/node_fs_stats';\nimport preload_file = require('../generic/preload_file');\nimport xhr = require('../generic/xhr');\nimport {FileIndex, DirInode, FileInode, Inode, isFileInode, isDirInode} from '../generic/file_index';\n\n/**\n * Try to convert the given buffer into a string, and pass it to the callback.\n * Optimization that removes the needed try/catch into a helper function, as\n * this is an uncommon case.\n */\nfunction tryToString(buff: Buffer, encoding: string, cb: (e: ApiError, rv?: string) => void) {\n  try {\n    cb(null, buff.toString(encoding));\n  } catch (e) {\n    cb(e);\n  }\n}\n\n/**\n * A simple filesystem backed by XmlHttpRequests.\n */\nexport default class XmlHttpRequest extends file_system.BaseFileSystem implements file_system.FileSystem {\n  private _index: FileIndex<{}>;\n  public prefixUrl: string;\n  /**\n   * Constructs the file system.\n   * @param listingUrlOrObj index object or the path to the JSON file index generated by\n   *   tools/XHRIndexer.coffee. This can be relative to the current webpage URL\n   *   or absolutely specified.\n   * @param prefixUrl The url prefix to use for all web-server requests.\n   */\n  constructor(listingUrlOrObj: string | Object, prefixUrl: string = '') {\n    super();\n    if (!listingUrlOrObj) {\n      listingUrlOrObj = 'index.json';\n    }\n    // prefix_url must end in a directory separator.\n    if (prefixUrl.length > 0 && prefixUrl.charAt(prefixUrl.length - 1) !== '/') {\n      prefixUrl = prefixUrl + '/';\n    }\n    this.prefixUrl = prefixUrl;\n\n    let listing: Object = null;\n    if (typeof(listingUrlOrObj) === \"string\") {\n      listing = this._requestFileSync(<string> listingUrlOrObj, 'json');\n      if (!listing) {\n        throw new Error(\"Unable to find listing at URL: ${listingUrlOrObj}\");\n      }\n    } else {\n      listing = listingUrlOrObj;\n    }\n\n    this._index = FileIndex.fromListing(listing);\n  }\n\n  public empty(): void {\n    this._index.fileIterator(function(file: Stats) {\n      file.file_data = null;\n    });\n  }\n\n  private getXhrPath(filePath: string): string {\n    if (filePath.charAt(0) === '/') {\n      filePath = filePath.slice(1);\n    }\n    return this.prefixUrl + filePath;\n  }\n\n  /**\n   * Only requests the HEAD content, for the file size.\n   */\n  public _requestFileSizeAsync(path: string, cb: (err: ApiError, size?: number) => void): void {\n    xhr.getFileSizeAsync(this.getXhrPath(path), cb);\n  }\n  public _requestFileSizeSync(path: string): number {\n    return xhr.getFileSizeSync(this.getXhrPath(path));\n  }\n\n  /**\n   * Asynchronously download the given file.\n   */\n  private _requestFileAsync(p: string, type: 'buffer', cb: (err: ApiError, data?: NodeBuffer) => void): void;\n  private _requestFileAsync(p: string, type: 'json', cb: (err: ApiError, data?: any) => void): void;\n  private _requestFileAsync(p: string, type: string, cb: (err: ApiError, data?: any) => void): void;\n  private _requestFileAsync(p: string, type: string, cb: (err: ApiError, data?: any) => void): void {\n    xhr.asyncDownloadFile(this.getXhrPath(p), type, cb);\n  }\n\n  /**\n   * Synchronously download the given file.\n   */\n  private _requestFileSync(p: string, type: 'buffer'): NodeBuffer;\n  private _requestFileSync(p: string, type: 'json'): any;\n  private _requestFileSync(p: string, type: string): any;\n  private _requestFileSync(p: string, type: string): any {\n    return xhr.syncDownloadFile(this.getXhrPath(p), type);\n  }\n\n  public getName(): string {\n    return 'XmlHttpRequest';\n  }\n\n  public static isAvailable(): boolean {\n    // @todo Older browsers use a different name for XHR, iirc.\n    return typeof XMLHttpRequest !== \"undefined\" && XMLHttpRequest !== null;\n  }\n\n  public diskSpace(path: string, cb: (total: number, free: number) => void): void {\n    // Read-only file system. We could calculate the total space, but that's not\n    // important right now.\n    cb(0, 0);\n  }\n\n  public isReadOnly(): boolean {\n    return true;\n  }\n\n  public supportsLinks(): boolean {\n    return false;\n  }\n\n  public supportsProps(): boolean {\n    return false;\n  }\n\n  public supportsSynch(): boolean {\n    return true;\n  }\n\n  /**\n   * Special XHR function: Preload the given file into the index.\n   * @param [String] path\n   * @param [BrowserFS.Buffer] buffer\n   */\n  public preloadFile(path: string, buffer: NodeBuffer): void {\n    var inode = this._index.getInode(path);\n    if (isFileInode<Stats>(inode)) {\n      if (inode === null) {\n        throw ApiError.ENOENT(path);\n      }\n      var stats = inode.getData();\n      stats.size = buffer.length;\n      stats.file_data = buffer;\n    } else {\n      throw ApiError.EISDIR(path);\n    }\n  }\n\n  public stat(path: string, isLstat: boolean, cb: (e: ApiError, stat?: Stats) => void): void {\n    var inode = this._index.getInode(path);\n    if (inode === null) {\n      return cb(ApiError.ENOENT(path));\n    }\n    var stats: Stats;\n    if (isFileInode<Stats>(inode)) {\n      stats = inode.getData();\n      // At this point, a non-opened file will still have default stats from the listing.\n      if (stats.size < 0) {\n        this._requestFileSizeAsync(path, function(e: ApiError, size?: number) {\n          if (e) {\n            return cb(e);\n          }\n          stats.size = size;\n          cb(null, stats.clone());\n        });\n      } else {\n        cb(null, stats.clone());\n      }\n    } else if (isDirInode(inode)) {\n      stats = inode.getStats();\n      cb(null, stats);\n    } else {\n      cb(ApiError.FileError(ErrorCode.EINVAL, path));\n    }\n  }\n\n  public statSync(path: string, isLstat: boolean): Stats {\n    var inode = this._index.getInode(path);\n    if (inode === null) {\n      throw ApiError.ENOENT(path);\n    }\n    var stats: Stats;\n    if (isFileInode<Stats>(inode)) {\n      stats = inode.getData();\n      // At this point, a non-opened file will still have default stats from the listing.\n      if (stats.size < 0) {\n        stats.size = this._requestFileSizeSync(path);\n      }\n    } else if (isDirInode(inode)) {\n      stats = inode.getStats();\n    } else {\n      throw ApiError.FileError(ErrorCode.EINVAL, path);\n    }\n    return stats;\n  }\n\n  public open(path: string, flags: FileFlag, mode: number, cb: (e: ApiError, file?: file.File) => void): void {\n    // INVARIANT: You can't write to files on this file system.\n    if (flags.isWriteable()) {\n      return cb(new ApiError(ErrorCode.EPERM, path));\n    }\n    var _this = this;\n    // Check if the path exists, and is a file.\n    var inode = this._index.getInode(path);\n    if (inode === null) {\n      return cb(ApiError.ENOENT(path));\n    }\n    if (isFileInode<Stats>(inode)) {\n      var stats = inode.getData();\n      switch (flags.pathExistsAction()) {\n        case ActionType.THROW_EXCEPTION:\n        case ActionType.TRUNCATE_FILE:\n          return cb(ApiError.EEXIST(path));\n        case ActionType.NOP:\n          // Use existing file contents.\n          // XXX: Uh, this maintains the previously-used flag.\n          if (stats.file_data != null) {\n            return cb(null, new preload_file.NoSyncFile(_this, path, flags, stats.clone(), stats.file_data));\n          }\n          // @todo be lazier about actually requesting the file\n          this._requestFileAsync(path, 'buffer', function(err: ApiError, buffer?: NodeBuffer) {\n            if (err) {\n              return cb(err);\n            }\n            // we don't initially have file sizes\n            stats.size = buffer.length;\n            stats.file_data = buffer;\n            return cb(null, new preload_file.NoSyncFile(_this, path, flags, stats.clone(), buffer));\n          });\n          break;\n        default:\n          return cb(new ApiError(ErrorCode.EINVAL, 'Invalid FileMode object.'));\n      }\n    } else {\n      return cb(ApiError.EISDIR(path));\n    }\n  }\n\n  public openSync(path: string, flags: FileFlag, mode: number): file.File {\n    // INVARIANT: You can't write to files on this file system.\n    if (flags.isWriteable()) {\n      throw new ApiError(ErrorCode.EPERM, path);\n    }\n    // Check if the path exists, and is a file.\n    var inode = this._index.getInode(path);\n    if (inode === null) {\n      throw ApiError.ENOENT(path);\n    }\n    if (isFileInode<Stats>(inode)) {\n      var stats = inode.getData();\n      switch (flags.pathExistsAction()) {\n        case ActionType.THROW_EXCEPTION:\n        case ActionType.TRUNCATE_FILE:\n          throw ApiError.EEXIST(path);\n        case ActionType.NOP:\n          // Use existing file contents.\n          // XXX: Uh, this maintains the previously-used flag.\n          if (stats.file_data != null) {\n            return new preload_file.NoSyncFile(this, path, flags, stats.clone(), stats.file_data);\n          }\n          // @todo be lazier about actually requesting the file\n          var buffer = this._requestFileSync(path, 'buffer');\n          // we don't initially have file sizes\n          stats.size = buffer.length;\n          stats.file_data = buffer;\n          return new preload_file.NoSyncFile(this, path, flags, stats.clone(), buffer);\n        default:\n          throw new ApiError(ErrorCode.EINVAL, 'Invalid FileMode object.');\n      }\n    } else {\n      throw ApiError.EISDIR(path);\n    }\n  }\n\n  public readdir(path: string, cb: (e: ApiError, listing?: string[]) => void): void {\n    try {\n      cb(null, this.readdirSync(path));\n    } catch (e) {\n      cb(e);\n    }\n  }\n\n  public readdirSync(path: string): string[] {\n    // Check if it exists.\n    var inode = this._index.getInode(path);\n    if (inode === null) {\n      throw ApiError.ENOENT(path);\n    } else if (isDirInode(inode)) {\n      return inode.getListing();\n    } else {\n      throw ApiError.ENOTDIR(path);\n    }\n  }\n\n  /**\n   * We have the entire file as a buffer; optimize readFile.\n   */\n  public readFile(fname: string, encoding: string, flag: FileFlag, cb: (err: ApiError, data?: any) => void): void {\n    // Wrap cb in file closing code.\n    var oldCb = cb;\n    // Get file.\n    this.open(fname, flag, 0x1a4, function(err: ApiError, fd?: file.File) {\n      if (err) {\n        return cb(err);\n      }\n      cb = function(err: ApiError, arg?: Buffer) {\n        fd.close(function(err2: any) {\n          if (err == null) {\n            err = err2;\n          }\n          return oldCb(err, arg);\n        });\n      };\n      var fdCast = <preload_file.NoSyncFile<XmlHttpRequest>> fd;\n      var fdBuff = <Buffer> fdCast.getBuffer();\n      if (encoding === null) {\n        cb(err, copyingSlice(fdBuff));\n      } else {\n        tryToString(fdBuff, encoding, cb);\n      }\n    });\n  }\n\n  /**\n   * Specially-optimized readfile.\n   */\n  public readFileSync(fname: string, encoding: string, flag: FileFlag): any {\n    // Get file.\n    var fd = this.openSync(fname, flag, 0x1a4);\n    try {\n      var fdCast = <preload_file.NoSyncFile<XmlHttpRequest>> fd;\n      var fdBuff = <Buffer> fdCast.getBuffer();\n      if (encoding === null) {\n        return copyingSlice(fdBuff);\n      }\n      return fdBuff.toString(encoding);\n    } finally {\n      fd.closeSync();\n    }\n  }\n}\n"]}