UNPKG

@graphistry/js-upload-api

Version:

Graphistry upload client for reuse by node and browser clients

502 lines 28.6 kB
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 }; } }; import { FileType, EdgeFile, NodeFile } from './File.js'; import { Privacy } from './Privacy.js'; /** * # Dataset examples * * Datasets are how to combine files into a single visualizable graph. * Powerfully, you can also specify visualization settings and * data-driven visual encodings as part of your dataset's bindings. * * For the many options, see the [JSON documentation](https://hub.graphistry.com/docs/api/). * * <br> * * --- * * <br> * * @example **Create a dataset from edges and upload using async/await** * ```javascript * import { Dataset } from '@graphistry/node-api'; * const dataset = new Dataset( * { * node_encodings: { bindings: { } }, * edge_encodings: { bindings: { source: 's', destination: 'd' } }, * metadata: {}, * name: 'testdata', * }, * edgesFile * ); * await dataset.upload(client); * console.log(`Dataset ${dataset.datasetID} uploaded to ${dataset.datasetURL}`); * ``` * * <br> * * @example **Create a dataset from nodes + edges and upload using promises** * ```javascript * import { Dataset } from '@graphistry/node-api'; * const dataset = new Dataset( * { * node_encodings: { bindings: { 'node': 'n' } }, * edge_encodings: { bindings: { 'source': 's', 'destination': 'd' } }, * metadata: {}, * name: 'testdata', * }, * edgesFile, * nodesFile * ); * dataset.upload(client).then( * () => console.log(`Dataset ${dataset.datasetID} uploaded to ${dataset.datasetURL}`) * ).catch(err => console.error('oops', err)); * ``` * * <br> * * @example **Create a dataset using Arrow** * ```javascript * import { tableFromArrays, tableToIPC, Table } from 'apache-arrow'; * import { EdgeFile } from '@graphistry/node-api'; * * //columnar data is fastest; column per attribute; reuse across datasets * const edgesJSON = {'s': ['a1', 'b2'], 'd': ['b2', 'c3']}; * const edgesTable: Table = tableFromArrays(edgesJSON); * const edgesUint8: Uint8Array = tableToIPC(edgesArr, 'file'); * const edgesFile = new EdgeFile(edgesUint8, 'arrow'); * ``` * * <br> * * @example **Add files after the Dataset is instantiated but before it has been uploaded** * ```javascript * import { Dataset } from '@graphistry/node-api'; * const dataset = new Dataset( * { * node_encodings: { bindings: { 'node': 'n' } }, * edge_encodings: { bindings: { 'source': 's', 'destination': 'd' } }, * metadata: {}, * name: 'testdata', * } * ); * dataset.addFile(nodesFile); * dataset.addFile(edgesFile); * await dataset.upload(client); * console.log(`Dataset ${dataset.datasetID} uploaded to ${dataset.datasetURL}`); * ``` * * <br> * * @example **Set privacy on uploaded dataset** * ```javascript * import { Dataset, Privacy } from '@graphistry/node-api'; * const dataset = new Dataset( * { * node_encodings: { bindings: { } }, * edge_encodings: { bindings: { 'source': 's', 'destination': 'd' } }, * metadata: {}, * name: 'testdata', * } * ); * dataset.addFile(edgesFile); * await dataset.upload(client); * await dataset.setPrivacy(client); // see additional options below * ``` * * <br> * * @example **Set simple data-driven bindings for titles, colors, icons, and labels** * ```javascript * import { Dataset } from '@graphistry/node-api'; * const dataset = new Dataset( * { * * // See also simple title, color, ... and complex color, size, icon, badge, ... * // https://hub.graphistry.com/docs/api/2/rest/upload/colors * // https://hub.graphistry.com/docs/api/2/rest/upload/complex/ * node_encodings: { * * bindings: { * 'node': 'n', //id * 'node_title': 'some_title_column' //optional * }, * * complex: { * default: { * pointSizeEncoding: { * graphType: 'point', * encodingType: 'size', * attribute: 'payment', * variation: 'categorical', * mapping: { * fixed: { * big: 500, * normal: 200, * tiny: 10 * }, * other: 100 * } * } * } * } * }, * * // See also simple title, color, ... and complex color, size, icon, badge, ... * // https://hub.graphistry.com/docs/api/2/rest/upload/colors * // https://hub.graphistry.com/docs/api/2/rest/upload/complex/ * edge_encodings: { * * bindings: { * 'source': 's', 'destination': 'd' * }, * * complex: { * default: { * edgeColorEncoding: { * graphType: 'point', * encodingType: 'color', * attribute: 'time', * variation: 'continuous', * colors: ['blue', 'yellow', 'red'] * } * } * } * }, * * // Set brand & theme: Background, foreground, logo, page metadata * // https://hub.graphistry.com/docs/api/2/rest/upload/metadata/ * metadata: { * bg: { * color: 'silver' * }, * "logo": { * url: "http://a.com/logo.png", * } * }, * name: 'testdata', * }, * * nodesFile, * edgesFile * * // Visual and layout settings * // https://hub.graphistry.com/docs/api/1/rest/url/#urloptions * { * strongGravity: true, * edgeCurvature: 0.5 * } * * ); * await dataset.upload(); * ``` */ var Dataset = /** @class */ (function () { //////////////////////////////////////////////////////////////////////////////// /** * * See examples at top of file * * Dataset definitions including required node_encodings, edge_encodings, metadata and name. * Optional definitions include edge_hypergraph_transform, and description, and various subfields. * URL settings may also be specified for additional styling. * This method autopopulates definitions edge_files, and if provided, node_files. * * Node files are optional: Nodes will be synthesized based on edges if not provided. * * If files have not been uploaded yet, this method will upload them for you. * * For more information about bindings, see https://hub.graphistry.com/docs/api/2/rest/upload/ * * For more information about URL style settings, see https://hub.graphistry.com/docs/api/1/rest/url/#urloptions * * For more information about theming, see https://hub.graphistry.com/docs/api/2/rest/upload/metadata/ * * For more information on simple encodings, see https://hub.graphistry.com/docs/api/2/rest/upload/colors * * For more information on complex encodings, see https://hub.graphistry.com/docs/api/2/rest/upload/complex/ * * @param bindings JSON dictionary of bindings * @param nodeFiles File object(s) * @param edgeFiles File object(s) * @param urlOpts JSON dictionary of URL options */ function Dataset(bindings, edgeFiles, nodeFiles, urlOpts) { if (bindings === void 0) { bindings = {}; } if (edgeFiles === void 0) { edgeFiles = []; } if (nodeFiles === void 0) { nodeFiles = []; } if (urlOpts === void 0) { urlOpts = {}; } this.bindings = bindings; this.edgeFiles = edgeFiles instanceof EdgeFile ? [edgeFiles] : edgeFiles; this.nodeFiles = nodeFiles instanceof NodeFile ? [nodeFiles] : nodeFiles; this.urlOpts = urlOpts; console.debug('Dataset constructor', this, { bindings: bindings, edgeFiles: edgeFiles, nodeFiles: nodeFiles, urlOpts: urlOpts }); for (var _i = 0, _a = this.edgeFiles; _i < _a.length; _i++) { var edgeFile = _a[_i]; if (!edgeFile.fileFormat) { throw new Error('Edge file must have a fileType'); } } for (var _b = 0, _c = this.nodeFiles; _b < _c.length; _b++) { var nodeFile = _c[_b]; if (!nodeFile.fileFormat) { throw new Error('Node file must have a fileType'); } } } Object.defineProperty(Dataset.prototype, "datasetID", { get: function () { return this._datasetID; }, enumerable: false, configurable: true }); Dataset.prototype.getCreateDatasetResponse = function () { return this._createDatasetResponse; }; Object.defineProperty(Dataset.prototype, "datasetURL", { get: function () { var _this = this; if (!this._datasetID) { throw new Error('No dataset ID yet'); } if (!this._usedClientProtocolHostname) { throw new Error('No client protocol hostname yet'); } var xtra = Object.keys(this.urlOpts).length ? "&".concat(Object.keys(this.urlOpts).map(function (k) { return "".concat(k, "=").concat(_this.urlOpts[k]); }).join('&')) : ''; return "".concat(this._usedClientProtocolHostname, "/graph/graph.html?dataset=").concat(this._datasetID).concat(xtra); }, enumerable: false, configurable: true }); //////////////////////////////////////////////////////////////////////////////// /** * * See examples at top of file * * Upload the dataset to the Graphistry server. * * If files have not been uploaded yet, this method will upload them for you. * * Upon completion, attributes datasetID and datasetURL will be set, as well as * createDatasetResponse and uploadResponse. * * @param client Client object * @returns Promise that resolves when the dataset is uploaded */ Dataset.prototype.upload = function (client) { return __awaiter(this, void 0, void 0, function () { var fileBindings, bindings; var _this = this; return __generator(this, function (_a) { switch (_a.label) { case 0: if (!client) { throw new Error('No client provided'); } if (!client.authTokenValid() && !client.isServerConfigured()) { throw new Error('Client is not configured, set token or creds'); } console.debug('Uploading dataset', { nodeFiles: this.nodeFiles, edgeFiles: this.edgeFiles }); //create+upload files as needed return [4 /*yield*/, Promise.all(this.nodeFiles.concat(this.edgeFiles).map(function (file) { return __awaiter(_this, void 0, void 0, function () { var response, ok; return __generator(this, function (_a) { switch (_a.label) { case 0: console.debug('Uploading file', file); return [4 /*yield*/, file.createFile(client)]; case 1: response = _a.sent(); console.debug("Uploaded ".concat(file.fileFormat, " file")); if (!response) { throw new Error('File creation failed 1'); } return [4 /*yield*/, file.uploadData(client)]; case 2: ok = _a.sent(); if (!ok) { throw new Error('File upload failed 2'); } return [2 /*return*/, file]; } }); }); }))]; case 1: //create+upload files as needed _a.sent(); fileBindings = { node_files: this.nodeFiles.map(function (file) { return file.fileID; }), edge_files: this.edgeFiles.map(function (file) { return file.fileID; }), }; bindings = __assign(__assign(__assign({}, (client.org ? { org_name: client.org } : {})), fileBindings), this.bindings); return [4 /*yield*/, this.createDataset(client, bindings)]; case 2: _a.sent(); return [2 /*return*/, this]; } }); }); }; //////////////////////////////////////////////////////////////////////////////// Dataset.prototype.createDataset = function (client, bindings) { return __awaiter(this, void 0, void 0, function () { var dataJsonResults, datasetID; return __generator(this, function (_a) { switch (_a.label) { case 0: this.fillMetadata(bindings, client); return [4 /*yield*/, client.post('api/v2/upload/datasets/', bindings)]; case 1: dataJsonResults = _a.sent(); this._createDatasetResponse = dataJsonResults; datasetID = dataJsonResults.data.dataset_id; this._datasetID = datasetID; this._usedClientProtocolHostname = client.clientProtocolHostname; if (!datasetID) { throw new Error('Unexpected dataset response, check dataset._createDatasetResponse'); } return [2 /*return*/, datasetID]; } }); }); }; /** * Add one or more bindings to the existing ones. In case of conflicts, override the existing ones. * * For more information about each, see https://hub.graphistry.com/docs/api/2/rest/upload/ * * @param bindings JSON dictionary of bindings to be added to the existing ones */ Dataset.prototype.updateBindings = function (bindings) { for (var _i = 0, _a = Object.entries(bindings); _i < _a.length; _i++) { var _b = _a[_i], key = _b[0], value = _b[1]; this.bindings[key] = value; } }; /** * * See examples at top of file * * Add an additional node or edge file to the existing ones. * * @param file File object. Does not need to be uploaded yet. * **/ Dataset.prototype.addFile = function (file) { console.debug('Adding file', file); if (file.type === FileType.Node) { this.nodeFiles.push(file); } else if (file.type === FileType.Edge) { this.edgeFiles.push(file); } else { throw new Error('Invalid File Type'); } }; /////////////////////////////////////////////////////////////////////////////// Dataset.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['metadata']) { data['metadata'] = {}; } var metadata = data['metadata']; if (!metadata['agent']) { metadata['agent'] = client.agent; } if (!metadata['agentversion']) { metadata['agentversion'] = client.version; } if (!metadata['apiversion']) { metadata['apiversion'] = '3'; } }; ///////////////////////////////////////////////////////////////////////////// /** * * See examples at top of file * * Set the privacy mode of the dataset. All but the client are optional. * * @param client Client object * @param mode Privacy mode. One of 'private', 'public', 'organization' * @param modeAction Capability allowed when shared * @param invitedUsers List of user IDs to share with * @param notify Whether to notify users of the share * @param message Message to include in the notification * * * @returns Promise that resolves when the privacy is set * @throws Error if the dataset has not been uploaded yet * @throws Error if server call fails */ Dataset.prototype.privacy = function (client, mode, modeAction, invitedUsers, notify, message) { if (mode === void 0) { mode = 'private'; } if (invitedUsers === void 0) { invitedUsers = []; } if (notify === void 0) { notify = false; } if (message === void 0) { message = ''; } return __awaiter(this, void 0, void 0, function () { var p; return __generator(this, function (_a) { switch (_a.label) { case 0: if (!this._datasetID) { throw new Error('Dataset must be uploaded before setting privacy'); } p = new Privacy(this._datasetID, 'dataset', mode, modeAction, invitedUsers, notify, message); return [4 /*yield*/, p.upload(client)]; case 1: return [2 /*return*/, _a.sent()]; } }); }); }; return Dataset; }()); export { Dataset }; //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiRGF0YXNldC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uL3NyYy9EYXRhc2V0LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O0FBRUEsT0FBTyxFQUFRLFFBQVEsRUFBRSxRQUFRLEVBQUUsUUFBUSxFQUFFLE1BQU0sV0FBVyxDQUFDO0FBQy9ELE9BQU8sRUFBb0IsT0FBTyxFQUFFLE1BQU0sY0FBYyxDQUFDO0FBRXpEOzs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQTRMRztBQUNIO0lBa0NJLGdGQUFnRjtJQUVoRjs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7O09BMkJHO0lBQ0gsaUJBQ0ksUUFBc0MsRUFDdEMsU0FBcUMsRUFDckMsU0FBcUMsRUFDckMsT0FBcUM7UUFIckMseUJBQUEsRUFBQSxhQUFzQztRQUN0QywwQkFBQSxFQUFBLGNBQXFDO1FBQ3JDLDBCQUFBLEVBQUEsY0FBcUM7UUFDckMsd0JBQUEsRUFBQSxZQUFxQztRQUVyQyxJQUFJLENBQUMsUUFBUSxHQUFHLFFBQVEsQ0FBQztRQUN6QixJQUFJLENBQUMsU0FBUyxHQUFHLFNBQVMsWUFBWSxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQztRQUN6RSxJQUFJLENBQUMsU0FBUyxHQUFHLFNBQVMsWUFBWSxRQUFRLENBQUMsQ0FBQyxDQUFDLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLFNBQVMsQ0FBQztRQUN6RSxJQUFJLENBQUMsT0FB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