UNPKG

box-ui-elements

Version:
303 lines (300 loc) • 12.9 kB
function _defineProperty(e, r, t) { return (r = _toPropertyKey(r)) in e ? Object.defineProperty(e, r, { value: t, enumerable: !0, configurable: !0, writable: !0 }) : e[r] = t, e; } function _toPropertyKey(t) { var i = _toPrimitive(t, "string"); return "symbol" == typeof i ? i : i + ""; } function _toPrimitive(t, r) { if ("object" != typeof t || !t) return t; var e = t[Symbol.toPrimitive]; if (void 0 !== e) { var i = e.call(t, r || "default"); if ("object" != typeof i) return i; throw new TypeError("@@toPrimitive must return a primitive value."); } return ("string" === r ? String : Number)(t); } import cloneDeep from 'lodash/cloneDeep'; import find from 'lodash/find'; import getProp from 'lodash/get'; import includes from 'lodash/includes'; import isArray from 'lodash/isArray'; import { areFieldValuesEqual, isEmptyValue, isMultiValuesField } from './utils'; import { JSON_PATCH_OP_ADD, JSON_PATCH_OP_REMOVE, JSON_PATCH_OP_REPLACE, JSON_PATCH_OP_TEST, METADATA_FIELD_TYPE_ENUM, METADATA_FIELD_TYPE_MULTISELECT } from '../../common/constants'; import { FIELD_ITEM_NAME, FIELD_METADATA, FIELD_EXTENSION, FIELD_PERMISSIONS } from '../../constants'; import { getMimeTypeFilter, getRangeFilter, getSelectFilter, getStringFilter, mergeQueries, mergeQueryParams } from './MetadataQueryBuilder'; const SELECT_TYPES = [METADATA_FIELD_TYPE_ENUM, METADATA_FIELD_TYPE_MULTISELECT]; export default class MetadataQueryAPIHelper { constructor(api) { _defineProperty(this, "createJSONPatchOperations", (field, oldValue, newValue) => { // check if two values are the same, return empty operations if so if (areFieldValuesEqual(oldValue, newValue)) { return []; } let operation = JSON_PATCH_OP_REPLACE; if (isEmptyValue(oldValue) && !isEmptyValue(newValue)) { operation = JSON_PATCH_OP_ADD; } if (!isEmptyValue(oldValue) && isEmptyValue(newValue)) { operation = JSON_PATCH_OP_REMOVE; } const testOp = { op: JSON_PATCH_OP_TEST, path: `/${field}`, value: oldValue }; const patchOp = { op: operation, path: `/${field}`, value: newValue }; if (operation === JSON_PATCH_OP_REMOVE) { delete patchOp.value; } return operation === JSON_PATCH_OP_ADD ? [patchOp] : [testOp, patchOp]; }); _defineProperty(this, "getMetadataQueryFields", () => { /* Example metadata query: const query = { from: 'enterprise_12345.myAwesomeTemplateKey', fields: [ 'name', // base representation field for an item (name, size, etag etc.) 'metadata.enterprise_12345.myAwesomeTemplateKey.field_1', // metadata instance field 'metadata.enterprise_12345.myAwesomeTemplateKey.field_2', // metadata instance field 'metadata.enterprise_12345.myAwesomeTemplateKey.field_3' // metadata instance field ], ancestor_folder_id: 0, }; This function will return ['field_1', 'field_2', 'field_3'] */ const { fields = [], from } = this.metadataQuery; return fields.filter(field => field.includes(from)).map(field => field.split('.').pop()); }); _defineProperty(this, "flattenMetadata", metadata => { const templateFields = getProp(this.metadataTemplate, 'fields', []); const instance = getProp(metadata, `${this.templateScope}.${this.templateKey}`); if (!instance) { return {}; } const queryFields = this.getMetadataQueryFields(); const fields = queryFields.map(queryField => { const templateField = find(templateFields, ['key', queryField]); const type = getProp(templateField, 'type'); // get data type const displayName = getProp(templateField, 'displayName', queryField); // get displayName, defaults to key const field = { key: `${FIELD_METADATA}.${this.templateScope}.${this.templateKey}.${queryField}`, value: instance[queryField], type, displayName }; if (includes(SELECT_TYPES, type)) { // get "options" for enums or multiselects field.options = getProp(templateField, 'options'); } return field; }); return { enterprise: { fields, id: instance.$id } }; }); _defineProperty(this, "getDataWithTypes", templateSchemaResponse => { this.metadataTemplate = getProp(templateSchemaResponse, 'data'); const { entries: items, next_marker: nextMarker } = this.metadataQueryResponseData; return { items, nextMarker }; }); _defineProperty(this, "getTemplateSchemaInfo", data => { this.metadataQueryResponseData = data; const [scope, key] = this.metadataQuery.from.split('.'); this.templateScope = scope; this.templateKey = key; return this.api.getMetadataAPI(true).getSchemaByTemplateKey(this.templateKey); }); /** * Generate operations for all fields update in the metadata sidepanel * * @private * @return {JSONPatchOperations} */ _defineProperty(this, "generateOperations", (item, templateOldFields, templateNewFields) => { const { scope, templateKey } = this.metadataTemplate; const itemFields = item.metadata[scope][templateKey]; const operations = templateNewFields.flatMap(newField => { let newFieldValue = newField.value; const { key, type } = newField; // when retrieve value from float type field, it gives a string instead if (type === 'float' && newFieldValue !== '') { newFieldValue = Number(newFieldValue); } const oldField = templateOldFields.find(f => f.key === key); const oldFieldValue = oldField.value; /* Generate operations array based on all the fields' orignal value and the incoming updated value. Edge Case: If there are multiple items shared different value for enum or multi-select field, the form will return 'Multiple values' as the value. In this case, it needs to generate operation based on the actual item's field value. */ const shouldUseItemFieldValue = isMultiValuesField(type, oldFieldValue) && !isMultiValuesField(type, newFieldValue); return this.createJSONPatchOperations(key, shouldUseItemFieldValue ? itemFields[key] : oldFieldValue, newFieldValue); }); return operations; }); _defineProperty(this, "queryMetadata", () => { return new Promise((resolve, reject) => { this.api.getMetadataQueryAPI().queryMetadata(this.metadataQuery, resolve, reject, { forceFetch: true }); }); }); _defineProperty(this, "fetchMetadataQueryResults", (metadataQuery, successCallback, errorCallback, fields) => { this.metadataQuery = this.verifyQueryFields(metadataQuery, fields); return this.queryMetadata().then(this.getTemplateSchemaInfo).then(this.getDataWithTypes).then(collection => { return successCallback(collection, this.metadataTemplate); }).catch(errorCallback); }); _defineProperty(this, "updateMetadata", (file, field, oldValue, newValue, successCallback, errorCallback) => { const operations = this.createJSONPatchOperations(field, oldValue, newValue); return this.api.getMetadataAPI(true).updateMetadata(file, this.metadataTemplate, operations, successCallback, errorCallback); }); _defineProperty(this, "updateMetadataWithOperations", (item, operations, successCallback, errorCallback) => { return this.api.getMetadataAPI(true).updateMetadata(item, this.metadataTemplate, operations, successCallback, errorCallback); }); _defineProperty(this, "bulkUpdateMetadata", (items, templateOldFields, templateNewFields, successCallback, errorCallback) => { const operations = []; items.forEach(item => { const operation = this.generateOperations(item, templateOldFields, templateNewFields); operations.push(operation); }); return this.api.getMetadataAPI(true).bulkUpdateMetadata(items, this.metadataTemplate, operations, successCallback, errorCallback); }); _defineProperty(this, "buildMetadataQueryParams", filters => { let argIndex = 0; let queries = []; let queryParams = {}; if (filters) { Object.keys(filters).forEach(key => { const filter = filters[key]; if (!filter) { return; } const { fieldType, value } = filter; switch (fieldType) { case 'date': case 'float': { if (typeof value === 'object' && value !== null && 'range' in value) { const result = getRangeFilter(value, key, argIndex); queryParams = mergeQueryParams(queryParams, result.queryParams); queries = mergeQueries(queries, result.queries); argIndex += result.keysGenerated; break; } break; } case 'enum': case 'multiSelect': { const arrayValue = Array.isArray(value) ? value.map(v => String(v)) : [String(value)]; let result; if (key === 'mimetype-filter') { result = getMimeTypeFilter(arrayValue, key, argIndex); } else { result = getSelectFilter(arrayValue, key, argIndex); } queryParams = mergeQueryParams(queryParams, result.queryParams); queries = mergeQueries(queries, result.queries); argIndex += result.keysGenerated; break; } case 'string': { if (value && value[0]) { const result = getStringFilter(value[0], key, argIndex); queryParams = mergeQueryParams(queryParams, result.queryParams); queries = mergeQueries(queries, result.queries); argIndex += result.keysGenerated; } break; } default: break; } }); } const query = queries.reduce((acc, curr, index) => { if (index > 0) { acc += ` AND ${curr}`; } else { acc = curr; } return acc; }, ''); return { queryParams, query }; }); _defineProperty(this, "mergeQuery", (customQuery, filterQuery) => { if (!customQuery) { return filterQuery; } if (!filterQuery) { return customQuery; } // Merge queries with AND operator return `${customQuery} AND ${filterQuery}`; }); /** * Verify that the metadata query has required fields and update it if necessary * For a file item, default fields included in the response are "type", "id", "etag" * * @param {MetadataQueryType} metadataQuery metadata query object * @param {ExternalFilterValues} [fields] optional filter values to apply to the metadata query * @return {MetadataQueryType} updated metadata query object with required fields */ _defineProperty(this, "verifyQueryFields", (metadataQuery, fields) => { const clonedQuery = cloneDeep(metadataQuery); const clonedFields = isArray(clonedQuery.fields) ? clonedQuery.fields : []; if (fields) { const { query: filterQuery, queryParams: filteredQueryParams } = this.buildMetadataQueryParams(fields); const { query: customQuery, query_params: customQueryParams } = clonedQuery; const query = this.mergeQuery(customQuery, filterQuery); const queryParams = mergeQueryParams(filteredQueryParams, customQueryParams); if (query) { clonedQuery.query = query; clonedQuery.query_params = queryParams; } } // Make sure the query fields array has "name" field which is necessary to display info. if (!clonedFields.includes(FIELD_ITEM_NAME)) { clonedFields.push(FIELD_ITEM_NAME); } if (!clonedFields.includes(FIELD_EXTENSION)) { clonedFields.push(FIELD_EXTENSION); } // This field is necessary to check if the user has permission to update metadata if (!clonedFields.includes(FIELD_PERMISSIONS)) { clonedFields.push(FIELD_PERMISSIONS); } clonedQuery.fields = clonedFields; return clonedQuery; }); this.api = api; } } //# sourceMappingURL=MetadataQueryAPIHelper.js.map