UNPKG

@graphistry/js-upload-api

Version:

Graphistry upload client for reuse by node and browser clients

446 lines 25 kB
var __extends = (this && this.__extends) || (function () { var extendStatics = function (d, b) { extendStatics = Object.setPrototypeOf || ({ __proto__: [] } instanceof Array && function (d, b) { d.__proto__ = b; }) || function (d, b) { for (var p in b) if (Object.prototype.hasOwnProperty.call(b, p)) d[p] = b[p]; }; return extendStatics(d, b); }; return function (d, b) { if (typeof b !== "function" && b !== null) throw new TypeError("Class extends value " + String(b) + " is not a constructor or null"); extendStatics(d, b); function __() { this.constructor = d; } d.prototype = b === null ? Object.create(b) : (__.prototype = b.prototype, new __()); }; })(); var __assign = (this && this.__assign) || function () { __assign = Object.assign || function(t) { for (var s, i = 1, n = arguments.length; i < n; i++) { s = arguments[i]; for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p)) t[p] = s[p]; } return t; }; return __assign.apply(this, arguments); }; var __awaiter = (this && this.__awaiter) || function (thisArg, _arguments, P, generator) { function adopt(value) { return value instanceof P ? value : new P(function (resolve) { resolve(value); }); } return new (P || (P = Promise))(function (resolve, reject) { function fulfilled(value) { try { step(generator.next(value)); } catch (e) { reject(e); } } function rejected(value) { try { step(generator["throw"](value)); } catch (e) { reject(e); } } function step(result) { result.done ? resolve(result.value) : adopt(result.value).then(fulfilled, rejected); } step((generator = generator.apply(thisArg, _arguments || [])).next()); }); }; var __generator = (this && this.__generator) || function (thisArg, body) { var _ = { label: 0, sent: function() { if (t[0] & 1) throw t[1]; return t[1]; }, trys: [], ops: [] }, f, y, t, g; return g = { next: verb(0), "throw": verb(1), "return": verb(2) }, typeof Symbol === "function" && (g[Symbol.iterator] = function() { return this; }), g; function verb(n) { return function (v) { return step([n, v]); }; } function step(op) { if (f) throw new TypeError("Generator is already executing."); while (_) try { if (f = 1, y && (t = op[0] & 2 ? y["return"] : op[0] ? y["throw"] || ((t = y["return"]) && t.call(y), 0) : y.next) && !(t = t.call(y, op[1])).done) return t; if (y = 0, t) op = [op[0] & 2, t.value]; switch (op[0]) { case 0: case 1: t = op; break; case 4: _.label++; return { value: op[1], done: false }; case 5: _.label++; y = op[1]; op = [0]; continue; case 7: op = _.ops.pop(); _.trys.pop(); continue; default: if (!(t = _.trys, t = t.length > 0 && t[t.length - 1]) && (op[0] === 6 || op[0] === 2)) { _ = 0; continue; } if (op[0] === 3 && (!t || (op[1] > t[0] && op[1] < t[3]))) { _.label = op[1]; break; } if (op[0] === 6 && _.label < t[1]) { _.label = t[1]; t = op; break; } if (t && _.label < t[2]) { _.label = t[2]; _.ops.push(op); break; } if (t[2]) _.ops.pop(); _.trys.pop(); continue; } op = body.call(thisArg, _); } catch (e) { op = [6, e]; y = 0; } finally { f = t = 0; } if (op[0] & 5) throw op[1]; return { value: op[0] ? op[1] : void 0, done: true }; } }; /** * @internal * * <br> * * ## FileType * * Enumeration used by {@link File} subclasses {@link EdgeFile} and {@link NodeFile} to specify whether the file is a node or edge file. * * Primarily useful for TypeScript and runtime checks to avoid mixups when passing {@link File} objects to {@link Dataset} * */ export var FileType; (function (FileType) { FileType[FileType["Node"] = 0] = "Node"; FileType[FileType["Edge"] = 1] = "Edge"; })(FileType || (FileType = {})); //////////////////////////////////////////////////////////////////////////////// /** * # File examples * * {@link File} objects are used for uploading data and then reusing as part of {@link Dataset} graph visualizations * * Powerfully, the same file may be reused in multiple {@link Dataset}s, so many variants can be made cheaply and quickly. * * For configuring supported file formats, see https://hub.graphistry.com/docs/api/2/rest/files/ . * * <br> * * --- * * <br> * * @example **Upload an {@link EdgeFile} from a JSON object in a columnar form** * ```javascript * import { EdgeFile } from '@graphistry/node-api'; * const edgesFile = new EdgeFile( * { * 's': ['a', 'b', 'c'], * 'd': ['d', 'e', 'f'], * 'v': ['v1', 'v2', 'v3'] * } * ); * await edgesFile.upload(client); * console.log(`EdgeFile uploaded as ID ${edgesFile.fileID}`); * ``` * * <br> * * @example **Upload an {@link EdgeFile} from a JSON object in a row-oriented form** * ```javascript * import { EdgeFile } from '@graphistry/node-api'; * const edgesFile = new EdgeFile( * [ * {'s': 'a', 'd': 'd', 'v': 'v1'}, * {'s': 'b', 'd': 'e', 'v': 'v2'}, * {'s': 'c', 'd': 'f', 'v': 'v3'} * ], * 'json', * 'my nodes file', * * // JSON parsing options: * // - https://hub.graphistry.com/docs/api/2/rest/upload/data/#uploadjson2 * // - https://pandas.pydata.org/docs/reference/api/pandas.read_json.html * // * // Also: file_compression, sql_transforms, ... * // https://hub.graphistry.com/docs/api/2/rest/files/ * {parser_options: {orient: 'records'}} * * ); * await edgesFile.upload(client); * console.log(`EdgeFile uploaded as ID ${edgesFile.fileID}`); * ``` * * <br> * * @example **Upload an {@link EdgeFile} using promises** * ```javascript * import { EdgeFile } from '@graphistry/node-api'; * const edgesFile = new EdgeFile({'s': ['a', 'b', 'c'], 'd': ['d', 'e', 'f'], 'v': ['v1', 'v2', 'v3']}); * edgesFile.upload(client).then( * () => console.log(`EdgeFile uploaded as ID ${edgesFile.fileID}`) * ).catch(err => console.error('oops', err)); * ``` * * <br> * * @example **Upload a {@link NodeFile} from a JSON object** * ```javascript * import { NodeFile } from '@graphistry/node-api'; * const nodesFile = new NodeFile({'n': ['a', 'b', 'c']}); * await nodesFile.upload(client); * console.log(`NodeFile uploaded as ID ${nodesFile.fileID}`); * ``` * * <br> * * @example **Upload a {@link NodeFile} from an Apache Arrow Table** * ```javascript * import { tableFromArrays, tableToIPC } from 'apache-arrow'; * import { NodeFile } from '@graphistry/node-api'; * const json = {'n': ['a', 'b', 'c']}; * const arr = tableFromArrays(json); * const uint8Buf = tableToIPC(arr); * const nodesFile = new NodeFile(uint8Buf, 'arrow'); * await nodesFile.upload(client); * console.log(`NodeFile uploaded as ID ${nodesFile.fileID}`); * ``` * * <br> * * @example **Create a {@link File} by ID (e.g., previously uploaded) for use with {@link Dataset}s** * ```javascript * import { EdgeFile, Dataset } from '@graphistry/node-api'; * const edgesFile = new EdgeFile('my_file_id'); * await (new Dataset(bindings, edgesFile)).upload(client); * console.log(`Dataset uploaded as ID ${dataset.datasetID}`); * ``` */ var File = /** @class */ (function () { //////////////////////////////////////////////////////////////////////////////// /** * * See examples at top of file * * Create a new {@link File} object for uploading or use with {@link Dataset} * * For more information on the available options, see: * * Creation step metadata options: https://hub.graphistry.com/docs/api/2/rest/files/ * * Upload step options: https://hub.graphistry.com/docs/api/2/rest/files/#uploadfiledata * * @param type FileType.Node or FileType.Edge * @param data Payload to pass to node-fetch. Use TypedArrays such as Uint8Array for binary data such as arrow * @param fileFormat File format to use, e.g. 'json', 'csv', 'arrow', 'parquet', 'orc', 'xls' * @param name Name of the file to use, e.g. 'my-file' * @param createOpts JSON post body options to use in createFile() * @param uploadUrlOpts URL options to use in uploadData() */ function File(type, data, fileFormat, name, createOpts, uploadUrlOpts) { if (data === void 0) { data = undefined; } if (fileFormat === void 0) { fileFormat = 'json'; } if (name === void 0) { name = 'my file'; } if (createOpts === void 0) { createOpts = {}; } if (uploadUrlOpts === void 0) { uploadUrlOpts = ''; } this._fileCreated = false; this._fileUploaded = false; this._fileValidated = false; if (typeof (data) == 'string') { this._fileID = data; this._fileCreated = true; this._fileUploaded = true; } else { this._data = data; } this.createOpts = createOpts; this.uploadUrlOpts = uploadUrlOpts; this.fileFormat = fileFormat; this.name = name; this.type = type; } Object.defineProperty(File.prototype, "data", { get: function () { return this._data; }, enumerable: false, configurable: true }); Object.defineProperty(File.prototype, "fileID", { get: function () { return this._fileID; }, enumerable: false, configurable: true }); Object.defineProperty(File.prototype, "fileCreated", { get: function () { return this._fileCreated; }, enumerable: false, configurable: true }); Object.defineProperty(File.prototype, "fileCreateResponse", { get: function () { return this._fileCreateResponse; }, enumerable: false, configurable: true }); Object.defineProperty(File.prototype, "fileUploaded", { get: function () { return this._fileUploaded; }, enumerable: false, configurable: true }); Object.defineProperty(File.prototype, "fileValidated", { get: function () { return this._fileValidated; }, enumerable: false, configurable: true }); Object.defineProperty(File.prototype, "fileUploadResponse", { get: function () { return this._fileUploadResponse; }, enumerable: false, configurable: true }); //////////////////////////////////////////////////////////////////////////////// /** * * See examples at top of file * * Upload curent {@link File} object to the server * * By default, this will skip reuploading files that have already been uploaded. * * @param client {@link Client} object to use for uploading * @param force If true, will force upload even if file has already been uploaded * @returns Promise that resolves to the uploaded File object when it completes uploading */ File.prototype.upload = function (client, force) { if (force === void 0) { force = false; } return __awaiter(this, void 0, void 0, function () { return __generator(this, function (_a) { switch (_a.label) { case 0: if (!client) { throw new Error('No client provided'); } return [4 /*yield*/, this.createFile(client, force)]; case 1: _a.sent(); if (!this._fileID) { throw new Error('Unexpected file creation response, check file._fileCreateResponse'); } return [4 /*yield*/, this.uploadData(client, force)]; case 2: _a.sent(); if (!this._fileCreated) { throw new Error('Unexpected file upload response, check file._fileUploadResponse'); } if (!this._fileValidated) { throw new Error('Unexpected file validation response, check file._fileValidateResponse'); } return [2 /*return*/, this]; } }); }); }; //////////////////////////////////////////////////////////////////////////////// /** * * See examples at top of file * * Helper function to create the file on the server but not yet upload its data * * By default, this will skip recreating files that have already been created. * * @param client {@link Client} object to use for uploading * @param force If true, will force creation of a new ID even if file has already been uploaded * @returns */ File.prototype.createFile = function (client, force) { if (force === void 0) { force = false; } return __awaiter(this, void 0, void 0, function () { var fileJsonResults; return __generator(this, function (_a) { switch (_a.label) { case 0: if (!force && this._fileCreated) { console.debug('File already created, skipping'); return [2 /*return*/, this._fileCreated]; } console.debug('Creating file'); return [4 /*yield*/, client.post('api/v2/files/', __assign(__assign({ file_type: this.fileFormat }, (client.org ? { org_name: client.org } : {})), this.createOpts))]; case 1: fileJsonResults = _a.sent(); console.debug('File creation response:', fileJsonResults); this._fileCreateResponse = fileJsonResults; this._fileID = fileJsonResults.file_id; this._fileCreated = !!fileJsonResults.file_id; return [2 /*return*/, fileJsonResults]; } }); }); }; /** * * See examples at top of file * * Helper function to upload the data to the server * * By default, this will skip reuploading data that has already been uploaded. * * @param client {@link Client} object to use for uploading * @param force If true, will force upload even if file has already been uploaded * @returns */ File.prototype.uploadData = function (client, force) { if (force === void 0) { force = false; } return __awaiter(this, void 0, void 0, function () { var results; return __generator(this, function (_a) { switch (_a.label) { case 0: if (!force && this._fileUploaded) { return [2 /*return*/, this._fileUploaded]; } this.fillMetadata(this._data, client); return [4 /*yield*/, client.post("api/v2/upload/files/".concat(this._fileID).concat(this.uploadUrlOpts ? "?".concat(this.uploadUrlOpts) : ''), this._data, this.fileFormat != 'json' ? {} : undefined)]; case 1: results = _a.sent(); this._fileUploadResponse = results; this._fileUploaded = !!results.is_uploaded; this._fileValidated = !!results.is_valid; return [2 /*return*/, results]; } }); }); }; /** * * See examples at top of file * * Populate data for later uploading if it wasn't set during construction * * Cannot run this function if the file has already been uploaded * * Overwrites any existing data * * @param data Data to use for uploading */ File.prototype.setData = function (data) { if (this._fileUploaded) { throw new Error('Cannot set data after file has been successfully uploaded'); } this._data = data; }; /////////////////////////////////////////////////////////////////////////////// File.prototype.fillMetadata = function (data, client) { if (!data) { throw new Error('No data to fill metadata; call setData() first or provide to File constructor'); } if (data instanceof Uint8Array) { return; } if (!data['agent_name']) { data['agent_name'] = client.agent; } if (!data['agent_version']) { data['agent_version'] = client.version; } }; return File; }()); export { File }; //////////////////////////////////////////////////////////////////////////////// /** * Helper class for tracking intent when creating a {@link File} object for uploading * * See examples at top of file * */ var EdgeFile = /** @class */ (function (_super) { __extends(EdgeFile, _super); function EdgeFile(data, fileFormat, name, createOpts, urlOpts) { if (data === void 0) { data = undefined; } if (fileFormat === void 0) { fileFormat = 'json'; } if (name === void 0) { name = 'my file'; } if (createOpts === void 0) { createOpts = {}; } if (urlOpts === void 0) { urlOpts = ''; } return _super.call(this, FileType.Edge, data, fileFormat, name, createOpts, urlOpts) || this; } return EdgeFile; }(File)); export { EdgeFile }; /** * Helper class for tracking intent when creating a {@link File} object for uploading * * See examples at top of file */ var NodeFile = /** @class */ (function (_super) { __extends(NodeFile, _super); function NodeFile(data, fileFormat, name, createOpts, urlOpts) { if (data === void 0) { data = undefined; } if (fileFormat === void 0) { fileFormat = 'json'; } if (name === void 0) { name = 'my file'; } if (createOpts === void 0) { createOpts = {}; } if (urlOpts === void 0) { urlOpts = ''; } return _super.call(this, FileType.Node, data, fileFormat, name, createOpts, urlOpts) || this; } return NodeFile; }(File)); export { NodeFile }; //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRmlsZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9GaWxlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBSUE7Ozs7Ozs7Ozs7O0dBV0c7QUFDSCxNQUFNLENBQU4sSUFBWSxRQUdYO0FBSEQsV0FBWSxRQUFRO0lBQ2hCLHVDQUFJLENBQUE7SUFDSix1Q0FBSSxDQUFBO0FBQ1IsQ0FBQyxFQUhXLFFBQVEsS0FBUixRQUFRLFFBR25CO0FBR0QsZ0ZBQWdGO0FBR2hGOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0dBb0dHO0FBQ0g7SUFxQ0ksZ0ZBQWdGO0lBRWhGOzs7Ozs7Ozs7Ozs7Ozs7O09BZ0JHO0lBQ0gsY0FBWSxJQUFjLEVBQUUsSUFBcUIsRUFBRSxVQUFtQixFQUFFLElBQWdCLEVBQUUsVUFBZSxFQUFFLGFBQWtCO1FBQWpHLHFCQUFBLEVBQUEsZ0JBQXFCO1FBQUUsMkJBQUEsRUFBQSxtQkFBbUI7UUFBRSxxQkFBQSxFQUFBLGdCQUFnQjtRQUFFLDJCQUFBLEVBQUEsZUFBZTtRQUFFLDhCQUFBLEVBQUEsa0JBQWtCO1FBMUNySCxpQkFBWSxHQUFHLEtBQUssQ0FBQztRQU1yQixrQkFBYSxHQUFHLEtBQUssQ0FBQztRQUd0QixtQkFBYyxHQUFHLEtBQUssQ0FBQztRQWtDM0IsSUFBSSxPQUFNLENBQUMsSUFBSSxDQUFDLElBQUksUUFBUSxFQUFFO1lBQzFCLElBQUksQ0FBQyxPQUFPLEdBQUcsSUFBSSxDQUFDO1lBQ3BCLElBQUksQ0FBQyxZQUFZLEdBQUcsSUFBSSxDQUFDO1lBQ3pCLElBQUksQ0FBQyxhQUFhLEdBQUcsSUFBSSxDQUFDO1NBQzdCO2FBQU07WUFDSCxJQUFJLENB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