|
| 1 | +import { |
| 2 | + BlockObjectResponse, |
| 3 | + DatabaseObjectResponse, |
| 4 | + PageObjectResponse, |
| 5 | + PartialBlockObjectResponse, |
| 6 | + PartialDatabaseObjectResponse, |
| 7 | + PartialPageObjectResponse, |
| 8 | + PartialUserObjectResponse, |
| 9 | + UserObjectResponse, |
| 10 | +} from "./api-endpoints" |
| 11 | + |
| 12 | +interface PaginatedArgs { |
| 13 | + start_cursor?: string |
| 14 | +} |
| 15 | + |
| 16 | +interface PaginatedList<T> { |
| 17 | + object: "list" |
| 18 | + results: T[] |
| 19 | + next_cursor: string | null |
| 20 | + has_more: boolean |
| 21 | +} |
| 22 | + |
| 23 | +/** |
| 24 | + * Returns an async iterator over the results of any paginated Notion API. |
| 25 | + * |
| 26 | + * Example (given a notion Client called `notion`): |
| 27 | + * |
| 28 | + * ``` |
| 29 | + * for await (const block of iteratePaginatedAPI(notion.blocks.children.list, { |
| 30 | + * block_id: parentBlockId, |
| 31 | + * })) { |
| 32 | + * // Do something with block. |
| 33 | + * } |
| 34 | + * ``` |
| 35 | + * |
| 36 | + * @param listFn A bound function on the Notion client that represents a conforming paginated |
| 37 | + * API. Example: `notion.blocks.children.list`. |
| 38 | + * @param firstPageArgs Arguments that should be passed to the API on the first and subsequent |
| 39 | + * calls to the API. Any necessary `next_cursor` will be automatically populated by |
| 40 | + * this function. Example: `{ block_id: "<my block id>" }` |
| 41 | + */ |
| 42 | +export async function* iteratePaginatedAPI<Args extends PaginatedArgs, Item>( |
| 43 | + listFn: (args: Args) => Promise<PaginatedList<Item>>, |
| 44 | + firstPageArgs: Args |
| 45 | +): AsyncIterableIterator<Item> { |
| 46 | + let nextCursor: string | null | undefined = firstPageArgs.start_cursor |
| 47 | + do { |
| 48 | + const response: PaginatedList<Item> = await listFn({ |
| 49 | + ...firstPageArgs, |
| 50 | + start_cursor: nextCursor, |
| 51 | + }) |
| 52 | + yield* response.results |
| 53 | + nextCursor = response.next_cursor |
| 54 | + } while (nextCursor) |
| 55 | +} |
| 56 | + |
1 | 57 | /**
|
2 |
| - * Utility for enforcing exhaustiveness checks in the type system. |
| 58 | + * Collect all of the results of paginating an API into an in-memory array. |
3 | 59 | *
|
4 |
| - * @see https://basarat.gitbook.io/typescript/type-system/discriminated-unions#throw-in-exhaustive-checks |
| 60 | + * Example (given a notion Client called `notion`): |
5 | 61 | *
|
6 |
| - * @param value The variable with no remaining values |
| 62 | + * ``` |
| 63 | + * const blocks = collectPaginatedAPI(notion.blocks.children.list, { |
| 64 | + * block_id: parentBlockId, |
| 65 | + * }) |
| 66 | + * // Do something with blocks. |
| 67 | + * ``` |
| 68 | + * |
| 69 | + * @param listFn A bound function on the Notion client that represents a conforming paginated |
| 70 | + * API. Example: `notion.blocks.children.list`. |
| 71 | + * @param firstPageArgs Arguments that should be passed to the API on the first and subsequent |
| 72 | + * calls to the API. Any necessary `next_cursor` will be automatically populated by |
| 73 | + * this function. Example: `{ block_id: "<my block id>" }` |
| 74 | + */ |
| 75 | +export async function collectPaginatedAPI<Args extends PaginatedArgs, Item>( |
| 76 | + listFn: (args: Args) => Promise<PaginatedList<Item>>, |
| 77 | + firstPageArgs: Args |
| 78 | +): Promise<Item[]> { |
| 79 | + const results: Item[] = [] |
| 80 | + for await (const item of iteratePaginatedAPI(listFn, firstPageArgs)) { |
| 81 | + results.push(item) |
| 82 | + } |
| 83 | + return results |
| 84 | +} |
| 85 | + |
| 86 | +/** |
| 87 | + * @returns `true` if `response` is a full `BlockObjectResponse`. |
7 | 88 | */
|
8 |
| -export function assertNever(value: never): never { |
9 |
| - throw new Error(`Unexpected value should never occur: ${value}`) |
| 89 | +export function isFullBlock( |
| 90 | + response: BlockObjectResponse | PartialBlockObjectResponse |
| 91 | +): response is BlockObjectResponse { |
| 92 | + return "type" in response |
10 | 93 | }
|
11 | 94 |
|
12 |
| -type AllKeys<T> = T extends unknown ? keyof T : never |
| 95 | +/** |
| 96 | + * @returns `true` if `response` is a full `PageObjectResponse`. |
| 97 | + */ |
| 98 | +export function isFullPage( |
| 99 | + response: PageObjectResponse | PartialPageObjectResponse |
| 100 | +): response is PageObjectResponse { |
| 101 | + return "url" in response |
| 102 | +} |
13 | 103 |
|
14 |
| -export function pick<O extends unknown, K extends AllKeys<O>>( |
15 |
| - base: O, |
16 |
| - keys: readonly K[] |
17 |
| -): Pick<O, K> { |
18 |
| - const entries = keys.map(key => [key, base?.[key]]) |
19 |
| - return Object.fromEntries(entries) |
| 104 | +/** |
| 105 | + * @returns `true` if `response` is a full `DatabaseObjectResponse`. |
| 106 | + */ |
| 107 | +export function isFullDatabase( |
| 108 | + response: DatabaseObjectResponse | PartialDatabaseObjectResponse |
| 109 | +): response is DatabaseObjectResponse { |
| 110 | + return "title" in response |
20 | 111 | }
|
21 | 112 |
|
22 |
| -export function isObject(o: unknown): o is Record<PropertyKey, unknown> { |
23 |
| - return typeof o === "object" && o !== null |
| 113 | +/** |
| 114 | + * @returns `true` if `response` is a full `UserObjectResponse`. |
| 115 | + */ |
| 116 | +export function isFullUser( |
| 117 | + response: UserObjectResponse | PartialUserObjectResponse |
| 118 | +): response is UserObjectResponse { |
| 119 | + return "type" in response |
24 | 120 | }
|
0 commit comments