diff --git a/src/index.ts b/src/index.ts index 96b8525dd..10de28bf2 100644 --- a/src/index.ts +++ b/src/index.ts @@ -1,9 +1,9 @@ /// import 'tslib' -import { forEach, clone, uuid, concat, dropEle, hasMorePages, pagination, eventToRE } from './utils' +import { forEach, clone, uuid, concat, dropEle, hasMorePages, pagination, eventToRE, normBulkUpdate } from './utils' -export { hasMorePages, pagination } +export { hasMorePages, pagination, normBulkUpdate } export const Utils = { forEach, clone, uuid, concat, dropEle } export { eventParser } from './sockets/EventParser' diff --git a/src/utils/httpclient.ts b/src/utils/httpclient.ts new file mode 100644 index 000000000..f3c8be9e8 --- /dev/null +++ b/src/utils/httpclient.ts @@ -0,0 +1,56 @@ +import { Omit } from './internalTypes' + +/** + * 从类型 T(如 `{ _taskId: TaskId[] }`)上获得给定字段 K + * (如 `_taskId`)对应的数组的元素类型(如 `TaskId`)。如果 + * `T[K]` 不是数组类型,则放弃类型推断,返回 any。 + */ +export type ArrayPropertyElement = T[K] extends (infer U)[] + ? U + : any + +/** + * 生成的结果类型,替换了类型 T(如 `{ taskIds: TaskId[], isArchived: boolean }`) + * 上的字段 K(如 `taskIds`)为字段 S(如 `_id`),而字段 S + * 对应的值类型则是字段 K 对应的数组值的元素类型(如 `TaskId`)。 + * 如果 `T[K]` 不是数组类型,则字段 S(如 `_id`)的类型将是 any。 + */ +export type NormBulkUpdateResult = Array< + Record> & Omit +> + +/** + * 将批量 PUT 的返回结果转变为可以直接被缓存层消费的数据。 + * 用法如:有 response$ 的元素形状为 + * `{ taskIds: TaskId[], isArchived: boolean, updated: string }` + * 则调用 `normBulkUpdate(response$, 'taskIds', '_id')` 将推出元素形状为 + * `{ _id: TaskId, isArchived: boolean, updated: string }[]` + * 的数据。 + */ +export function normBulkUpdate< + T extends {}, + K extends keyof T, + U extends string +>( + response: T, + responseIdsField: K, + entityIdField: U +): NormBulkUpdateResult { + const { [responseIdsField]: ids, ...rest } = response as any + + return ids + ? (ids as Array>).map((id) => { + const existingValue = rest[entityIdField] + if (existingValue != null && existingValue !== id) { + throw new Error( + `normBulkUpdate: specified key-value pair(${entityIdField}-${id})` + + ` is conflicting to an existing one(${entityIdField}-${existingValue}).)` + ) + } + return { + ...rest, + [entityIdField]: id + } + }) + : [] +} diff --git a/src/utils/index.ts b/src/utils/index.ts index dfe59e77e..44003f7dc 100644 --- a/src/utils/index.ts +++ b/src/utils/index.ts @@ -2,3 +2,4 @@ export * from './helper' export * from './internalTypes' export { createProxy } from './proxy' export { eventToRegexp as eventToRE } from './eventToRegexp' +export * from './httpclient' diff --git a/src/utils/internalTypes.ts b/src/utils/internalTypes.ts index a5da2075c..27d4894bd 100644 --- a/src/utils/internalTypes.ts +++ b/src/utils/internalTypes.ts @@ -18,6 +18,8 @@ export type Dict = { [key: string]: T } +export type Omit = Pick> + export type TableInfo = { tabName: string, pkName: string diff --git a/test/app.ts b/test/app.ts index 0cf16af35..6fe98e8ee 100644 --- a/test/app.ts +++ b/test/app.ts @@ -13,3 +13,4 @@ import './mock' import './apis' import './sockets' import './net' +import './utils/index' diff --git a/test/utils/httpclient.ts b/test/utils/httpclient.ts new file mode 100644 index 000000000..f50b39bb3 --- /dev/null +++ b/test/utils/httpclient.ts @@ -0,0 +1,74 @@ +import { expect } from 'chai' +import { it, describe } from 'tman' +import { TaskId } from 'teambition-types' +import { normBulkUpdate } from '../../src/utils' + +describe('httpclient utils spec', () => { + it(`${normBulkUpdate.name} should produce [] when responseIdsField is undefined`, () => { + expect(normBulkUpdate( + { + taskIds: ['123', '456'], + isArchived: true, + updated: '2018-08-21T05:43:10.000Z' + } as any, + '_taskIds', + '_id' + )).to.deep.equal([]) + }) + + it(`${normBulkUpdate.name} should produce [] when responseIdsField is empty`, () => { + expect(normBulkUpdate( + { + taskIds: [], + isArchived: true, + updated: '2018-08-21T05:43:10.000Z' + }, + 'taskIds', + '_id' + )).to.deep.equal([]) + }) + + it(`${normBulkUpdate.name} should work when responseIdsField and entityIdField are the same`, () => { + expect(normBulkUpdate( + { + _id: ['123' as TaskId, '456' as TaskId], + isArchived: true, + updated: '2018-08-21T05:43:10.000Z' + }, + '_id', + '_id' + )).to.deep.equal([ + { _id: '123', isArchived: true, updated: '2018-08-21T05:43:10.000Z' }, + { _id: '456', isArchived: true, updated: '2018-08-21T05:43:10.000Z' } + ]) + }) + + it(`${normBulkUpdate.name} should work when responseIdsField and entityIdField are different`, () => { + expect(normBulkUpdate( + { + taskIds: ['123' as TaskId, '456' as TaskId], + isArchived: true, + updated: '2018-08-21T05:43:10.000Z' + }, + 'taskIds', + '_id' + )).to.deep.equal([ + { _id: '123', isArchived: true, updated: '2018-08-21T05:43:10.000Z' }, + { _id: '456', isArchived: true, updated: '2018-08-21T05:43:10.000Z' } + ]) + }) + + it(`${normBulkUpdate.name} should throw when property conflict happends`, () => { + expect(() => normBulkUpdate( + { + taskIds: ['123', '456'], + cid: '789', // 语义为 child id + isArchived: true, + updated: '2018-08-21T05:43:10.000Z' + }, + 'taskIds', + 'cid' // 语义为 backbone 的 client id + )).to.throw() + }) + +}) diff --git a/test/utils/index.ts b/test/utils/index.ts new file mode 100644 index 000000000..00442d7b4 --- /dev/null +++ b/test/utils/index.ts @@ -0,0 +1 @@ +import './httpclient'