Source: managers/metadata.ts

  1. /**
  2. * @fileoverview Manager for the Box Metadata Resource
  3. */
  4. // -----------------------------------------------------------------------------
  5. // Requirements
  6. // -----------------------------------------------------------------------------
  7. import BoxClient from '../box-client';
  8. import urlPath from '../util/url-path';
  9. const merge = require('merge-options');
  10. // -----------------------------------------------------------------------------
  11. // Typedefs
  12. // -----------------------------------------------------------------------------
  13. /**
  14. * Valid metadata field types
  15. * @readonly
  16. * @enum {MetadataFieldType}
  17. */
  18. enum MetadataFieldType {
  19. STRING = 'string',
  20. ENUM = 'enum',
  21. NUMBER = 'float',
  22. DATE = 'date',
  23. MULTI_SELECT = 'multiSelect',
  24. }
  25. /**
  26. * Metadata enum option
  27. * @typedef {Object} MetadataEnumOption
  28. * @property {string} key The option value
  29. */
  30. type MetadataEnumOption = {
  31. key: string;
  32. };
  33. /**
  34. * Field definition for a metadata template
  35. * @typedef {Object} MetadataTemplateField
  36. * @property {MetadataFieldType} type The type of the field
  37. * @property {string} key The programmatic name of the field
  38. * @property {string} displayName The display name of the field
  39. * @property {boolean} hidden Whether this field is hidden in the UI for the user and can only be set through the API instead
  40. * @property {MetadataEnumOption[]} [options] For enum fields, the options
  41. */
  42. type MetadataTemplateField = {
  43. type: MetadataFieldType;
  44. key: string;
  45. displayName: string;
  46. hidden: boolean;
  47. options?: MetadataEnumOption[];
  48. };
  49. // -----------------------------------------------------------------------------
  50. // Private
  51. // -----------------------------------------------------------------------------
  52. const PROPERTIES_TEMPLATE = 'properties',
  53. BASE_PATH = '/metadata_templates',
  54. SCHEMA_SUBRESOURCE = 'schema',
  55. ENTERPRISE_SCOPE = 'enterprise',
  56. GLOBAL_SCOPE = 'global',
  57. CASCADE_POLICIES_PATH = '/metadata_cascade_policies',
  58. QUERY_PATH = '/metadata_queries/execute_read';
  59. // -----------------------------------------------------------------------------
  60. // Public
  61. // -----------------------------------------------------------------------------
  62. /**
  63. * Simple manager for interacting with all metadata endpoints and actions.
  64. *
  65. * @constructor
  66. * @param {BoxClient} client - The Box API Client that is responsible for making calls to the API
  67. * @returns {void}
  68. */
  69. class Metadata {
  70. client: BoxClient;
  71. templates!: Record<string, any>;
  72. scopes!: Record<string, any>;
  73. cascadeResolution!: Record<string, any>;
  74. fieldTypes!: typeof MetadataFieldType;
  75. constructor(client: BoxClient) {
  76. this.client = client;
  77. }
  78. /**
  79. * Retrieve the schema definition for a metadata template
  80. *
  81. * API Endpoint: '/metadata_templates/:scope/:template'
  82. * Method: GET
  83. *
  84. * @param {string} scope - The scope of the template, e.g. "enterprise"
  85. * @param {string} template - The template to retrieve
  86. * @param {Function} [callback] - Called with the template schema if successful
  87. * @returns {Promise<Object>} A promise resolving to the template schema
  88. */
  89. getTemplateSchema(scope: string, template: string, callback?: Function) {
  90. var apiPath = urlPath(BASE_PATH, scope, template, SCHEMA_SUBRESOURCE);
  91. return this.client.wrapWithDefaultHandler(this.client.get)(
  92. apiPath,
  93. null,
  94. callback
  95. );
  96. }
  97. /**
  98. * Retrieve the schema definition for a metadata template by ID
  99. *
  100. * API Endpoint: '/metadata_templates/:id'
  101. * Method: GET
  102. *
  103. * @param {string} templateID - The ID of the template to retrieve
  104. * @param {Function} [callback] - Called with the template schema if successful
  105. * @returns {Promise<Object>} A promise resolving to the template schema
  106. */
  107. getTemplateByID(templateID: string, callback?: Function) {
  108. var apiPath = urlPath(BASE_PATH, templateID);
  109. return this.client.wrapWithDefaultHandler(this.client.get)(
  110. apiPath,
  111. null,
  112. callback
  113. );
  114. }
  115. /**
  116. * Get all templates in a given scope
  117. *
  118. * API Endpoint: '/metadata_templates/:scope'
  119. * Method: GET
  120. *
  121. * @param {string} scope - The scope to retrieve templates for
  122. * @param {Function} [callback] - Called with an array of templates when successful
  123. * @returns {Promise<Object>} A promise resolving to the collection of templates
  124. */
  125. getTemplates(scope: string, callback?: Function) {
  126. var apiPath = urlPath(BASE_PATH, scope);
  127. return this.client.wrapWithDefaultHandler(this.client.get)(
  128. apiPath,
  129. null,
  130. callback
  131. );
  132. }
  133. /**
  134. * Create a new metadata template
  135. *
  136. * API Endpoint: '/metadata_templates/schema',
  137. * Method: POST
  138. *
  139. * @param {string} templateName - The name of the metadata template
  140. * @param {MetadataTemplateField[]} fields - A list of fields for the template
  141. * @param {Object} [options] - Optional parameters, can be left null in many cases
  142. * @param {string} [options.templateKey] - The programmatic key for the template
  143. * @param {boolean} [options.hidden] - Whether the template should be hidden in the UI
  144. * @param {string} [options.scope=enterprise] - The scope for the template, only 'enterprise' is supported for now
  145. * @param {boolean} [options.copyInstanceOnItemCopy] - Whether to include the metadata when a file or folder is copied
  146. * @param {Function} [callback] - Passed the template if successful, error otherwise
  147. * @returns {Promise<Object>} A promise resolving to the created template
  148. */
  149. createTemplate(
  150. templateName: string,
  151. fields: MetadataTemplateField[],
  152. options?: {
  153. templateKey?: string;
  154. hidden?: boolean;
  155. scope?: string;
  156. copyInstanceOnItemCopy?: boolean;
  157. },
  158. callback?: Function
  159. ) {
  160. var apiPath = urlPath(BASE_PATH, SCHEMA_SUBRESOURCE),
  161. params = {
  162. body: {
  163. scope: ENTERPRISE_SCOPE,
  164. displayName: templateName,
  165. fields,
  166. },
  167. };
  168. Object.assign(params.body, options);
  169. return this.client.wrapWithDefaultHandler(this.client.post)(
  170. apiPath,
  171. params,
  172. callback
  173. );
  174. }
  175. /**
  176. * Update a metadata template via one or more non-breaking operations. Each
  177. * operation is a an object descrbing one change to the template or its
  178. * fields.
  179. *
  180. * API Endpoint: '/metadata_templates/:scope/:template/schema'
  181. * Method: PUT
  182. *
  183. * @param {string} scope - The scope of the template to modify
  184. * @param {string} template - The template to modify
  185. * @param {Object[]} operations - The operations to perform
  186. * @param {Function} [callback] - Passed the updated template if successful, error otherwise
  187. * @returns {Promise<Object>} A promise resolving to the updated template
  188. * @see {@link https://developer.box.com/en/reference/put-metadata-templates-id-id-schema/}
  189. */
  190. updateTemplate(
  191. scope: string,
  192. template: string,
  193. operations: Record<string, any>[],
  194. callback?: Function
  195. ) {
  196. var apiPath = urlPath(BASE_PATH, scope, template, SCHEMA_SUBRESOURCE),
  197. params = {
  198. body: operations,
  199. };
  200. return this.client.wrapWithDefaultHandler(this.client.put)(
  201. apiPath,
  202. params,
  203. callback
  204. );
  205. }
  206. /**
  207. * Delete a metadata template from an enterprise.
  208. *
  209. * API Endpoint: '/metadata_templates/:scope/:template/schema'
  210. * Method: DELETE
  211. *
  212. * @param {string} scope - The scope of the template to delete
  213. * @param {string} template - The template to delete
  214. * @param {Function} [callback] - Passed empty response body if successful, err otherwise
  215. * @returns {Promise<void>} A promise resolving to nothing
  216. * @see {@link https://developer.box.com/en/reference/delete-metadata-templates-id-id-schema/}
  217. */
  218. deleteTemplate(scope: string, template: string, callback?: Function) {
  219. var apiPath = urlPath(BASE_PATH, scope, template, SCHEMA_SUBRESOURCE);
  220. return this.client.wrapWithDefaultHandler(this.client.del)(
  221. apiPath,
  222. null,
  223. callback
  224. );
  225. }
  226. /**
  227. * Get the cascade policies associated with a given folder.
  228. *
  229. * API Endpoint: '/metadata_cascade_policies'
  230. * Method: GET
  231. *
  232. * @param {string} folderID The ID of the folder to get cascade policies for
  233. * @param {Object} [options] Optional parameters
  234. * @param {string} [options.owner_enterprise_id] ID of the enterprise to get policies for
  235. * @param {Function} [callback] Passed the collection of policies if successful
  236. * @returns {Promise<Object>} Promise resolving to the collection of policies
  237. */
  238. getCascadePolicies(
  239. folderID: string,
  240. options?: {
  241. owner_enterprise_id?: string;
  242. },
  243. callback?: Function
  244. ) {
  245. var apiPath = urlPath(CASCADE_POLICIES_PATH),
  246. params = {
  247. qs: Object.assign({ folder_id: folderID }, options),
  248. };
  249. return this.client.wrapWithDefaultHandler(this.client.get)(
  250. apiPath,
  251. params,
  252. callback
  253. );
  254. }
  255. /**
  256. * Get a metadata cascade policy object by ID
  257. *
  258. * API Endpoint: '/metadata_cascade_policies/:policyID'
  259. * Method: GET
  260. *
  261. * @param {string} policyID The ID of the policy to retrieve
  262. * @param {Function} [callback] Passed the cascade policy if successful
  263. * @returns {Promise<Object>} Promise resolving to the cascade policy
  264. */
  265. getCascadePolicy(policyID: string, callback?: Function) {
  266. var apiPath = urlPath(CASCADE_POLICIES_PATH, policyID);
  267. return this.client.wrapWithDefaultHandler(this.client.get)(
  268. apiPath,
  269. null,
  270. callback
  271. );
  272. }
  273. /**
  274. * Add a new cascade policy to a folder/metadata template, causing the
  275. * metadata template to be applied to all items and subfolders inside the
  276. * folder.
  277. *
  278. * API Endpoint: '/metadata_cascade_policies'
  279. * Method: POST
  280. *
  281. * @param {string} scope Metadata template scope for the template to cascade
  282. * @param {string} templateKey Metadata template key for the template to cascade
  283. * @param {string} folderID The ID of the folder to cascade over
  284. * @param {Function} [callback] Passed the cascade policy if successful
  285. * @returns {Promise<Object>} Promise resolving to the cascade policy
  286. */
  287. createCascadePolicy(
  288. scope: string,
  289. templateKey: string,
  290. folderID: string,
  291. callback?: Function
  292. ) {
  293. var apiPath = urlPath(CASCADE_POLICIES_PATH),
  294. params = {
  295. body: {
  296. folder_id: folderID,
  297. scope,
  298. templateKey,
  299. },
  300. };
  301. return this.client.wrapWithDefaultHandler(this.client.post)(
  302. apiPath,
  303. params,
  304. callback
  305. );
  306. }
  307. /**
  308. * Delete the metadata cascade policy with the given ID
  309. *
  310. * API Endpoint: '/metadata_cascade_policies/:policyID'
  311. * Method: DELETE
  312. *
  313. * @param {string} policyID The ID of the policy to delete
  314. * @param {Function} [callback] Passed nothing if successful
  315. * @returns {Promise<void>} Promise resolving to nothing
  316. */
  317. deleteCascadePolicy(policyID: string, callback?: Function) {
  318. var apiPath = urlPath(CASCADE_POLICIES_PATH, policyID);
  319. return this.client.wrapWithDefaultHandler(this.client.del)(
  320. apiPath,
  321. null,
  322. callback
  323. );
  324. }
  325. /**
  326. * If a policy already exists on a folder, this will apply that policy to all existing files and
  327. * sub-folders within the target folder.
  328. *
  329. * API Endpoint: '/metadata_cascade_policies/:policyID/apply'
  330. * Method: POST
  331. *
  332. * @param {string} policyID The ID of the policy to delete
  333. * @param {string} resolutionMethod How to resolve conflicts, either "none" or "overwrite"
  334. * @param {Function} [callback] Passed nothing if successful
  335. * @returns {Promise<void>} Promise resolving to nothing
  336. */
  337. forceApplyCascadePolicy(
  338. policyID: string,
  339. resolutionMethod: string,
  340. callback?: Function
  341. ) {
  342. var apiPath = urlPath(CASCADE_POLICIES_PATH, policyID, 'apply'),
  343. params = {
  344. body: {
  345. conflict_resolution: resolutionMethod,
  346. },
  347. };
  348. return this.client.wrapWithDefaultHandler(this.client.post)(
  349. apiPath,
  350. params,
  351. callback
  352. );
  353. }
  354. /**
  355. * Query Box items by their metadata.
  356. *
  357. * API Endpoint: '/metadata_queries/execute_read'
  358. * Method: POST
  359. *
  360. * @param {string} from - The template used in the query. Must be in the form scope.templateKey
  361. * @param {string} ancestorFolderId - The folder_id to which to restrain the query
  362. * @param {Object} [options] - Optional parameters
  363. * @param {string} [options.query] - The logical expression of the query
  364. * @param {Object} [options.query_params] - Required if query present. The arguments for the query
  365. * @param {Object} [options.order_by] - The field_key(s) to order on and the corresponding direction(s)
  366. * @param {Array} [options.fields] - An array of fields to return
  367. * @param {int} [options.limit=100] - The number of results to return for a single request
  368. * @param {string} [options.marker] - Pagination marker
  369. * @param {Function} [callback] - Passed a collection of items and their associated metadata
  370. * @returns {Promise<void>} Promise resolving to a collection of items and their associated metadata
  371. */
  372. query(
  373. from: string,
  374. ancestorFolderId: string,
  375. options?: {
  376. query?: string;
  377. query_params?: Record<string, any>;
  378. order_by?: Record<string, any>;
  379. fields?: string[];
  380. limit?: number;
  381. marker?: string;
  382. },
  383. callback?: Function
  384. ) {
  385. var body = {
  386. from,
  387. ancestor_folder_id: ancestorFolderId,
  388. };
  389. var params = {
  390. body: merge(body, options || {}),
  391. };
  392. return this.client.wrapWithDefaultHandler(this.client.post)(
  393. QUERY_PATH,
  394. params,
  395. callback
  396. );
  397. }
  398. }
  399. Metadata.prototype.templates = {
  400. PROPERTIES: PROPERTIES_TEMPLATE,
  401. };
  402. Metadata.prototype.scopes = {
  403. ENTERPRISE: ENTERPRISE_SCOPE,
  404. GLOBAL: GLOBAL_SCOPE,
  405. };
  406. Metadata.prototype.cascadeResolution = Object.freeze({
  407. PRESERVE_EXISTING: 'none',
  408. OVERWRITE: 'overwrite',
  409. });
  410. /**
  411. * Valid metadata field types
  412. * @readonly
  413. * @enum {MetadataFieldType}
  414. */
  415. Metadata.prototype.fieldTypes = MetadataFieldType;
  416. export = Metadata;