@graphistry/js-upload-api
Version:
Graphistry upload client for reuse by node and browser clients
502 lines • 28.6 kB
JavaScript
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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