UNPKG

@bitrix24/b24jssdk

Version:

Bitrix24 REST API JavaScript SDK

127 lines (124 loc) 4.92 kB
/** * @package @bitrix24/b24jssdk * @version 1.0.5 * @copyright (c) 2026 Bitrix24 * @license MIT * @see https://github.com/bitrix24/b24jssdk * @see https://bitrix24.github.io/b24jssdk/ */ import { AbstractAction } from '../abstract-action.mjs'; import { SdkError } from '../../sdk-error.mjs'; var __defProp = Object.defineProperty; var __name = (target, value) => __defProp(target, "name", { value, configurable: true }); class FetchListV3 extends AbstractAction { static { __name(this, "FetchListV3"); } /** * Calls a REST API list method and returns an async generator for efficient large data retrieval. * Implements the fast algorithm for iterating over large datasets without loading all data into memory at once. * * @template T - The type of items in the returned arrays (default is `unknown`). * * @param {ActionFetchListV3} options - parameters for executing the request. * - `method: string` - The name of the REST API method that returns a list of data (for example: `crm.item.list`, `tasks.task.list`) * - `params?: Omit<TypeCallParams, 'pagination'>` - Request parameters, excluding the `pagination` parameter, * since the method is designed to obtain all data in one call. * Note: Use `filter`, `order`, and `select` to control the selection. * - `idKey?: string` - The name of the field containing the unique identifier of the element. * Default is 'id'. Alternatively, it can be another field, depending on the REST API data structure. * - `customKeyForResult: string` - A custom key indicating that the response REST API will be * grouped by this field. * Example: `items` to group a list of CRM items. * - `requestId?: string` - Unique request identifier for tracking. Used for query deduplication and debugging. * - `limit?: number` - How many records to retrieve at a time. Default is `50`. Maximum is `1000`. * * @returns {AsyncGenerator<T[]>} An async generator that yields chunks of data as arrays of type `T`. * Each iteration returns the next page/batch of results until all data is fetched. * * @example * import { Text } from '@bitrix24/b24jssdk' * * interface MainEventLogItem { id: number, userId: number } * const sixMonthAgo = new Date() * sixMonthAgo.setMonth((new Date()).getMonth() - 6) * sixMonthAgo.setHours(0, 0, 0) * const generator = b24.actions.v3.fetchList.make<MainEventLogItem>({ * method: 'main.eventlog.list', * params: { * filter: [ * ['timestampX', '>=', Text.toB24Format(sixMonthAgo)] // created at least 6 months ago * ], * select: ['id', 'userId'] * }, * idKey: 'id', * customKeyForResult: 'items', * requestId: 'eventlog-123', * limit: 60 * }) * * for await (const chunk of generator) { * // Process chunk (e.g., save to database, analyze, etc.) * console.log(`Processing ${chunk.length} items`) * } */ async *make(options) { const batchSize = options?.limit ?? 50; const idKey = options?.idKey ?? "id"; const customKeyForResult = options?.customKeyForResult ?? null; const params = options?.params ?? {}; const requestParams = { ...params, order: { ...params["order"] || {}, [idKey]: "ASC" }, filter: [...params["filter"] || []], pagination: { page: 0, limit: batchSize } }; let isContinue = true; let nextId = 0; do { const sendParams = { ...requestParams }; sendParams.filter.push([idKey, ">", nextId]); const response = await this._b24.actions.v3.call.make({ method: options.method, params: requestParams, requestId: options.requestId }); if (!response.isSuccess) { this._logger.error("fetchListMethod", { method: options.method, requestId: options.requestId, messages: response.getErrorMessages() }); throw new SdkError({ code: "JSSDK_CORE_B24_FETCH_LIST_METHOD_API_V3", description: `API Error: ${response.getErrorMessages().join("; ")}`, status: 500 }); } const responseData = response.getData(); if (!responseData) { isContinue = false; break; } const resultData = responseData.result[customKeyForResult]; if (resultData.length === 0) { isContinue = false; break; } yield resultData; if (resultData.length < batchSize) { isContinue = false; break; } const lastItem = resultData[resultData.length - 1]; if (lastItem && typeof lastItem[idKey] !== "undefined") { nextId = Number.parseInt(lastItem[idKey]); } else { isContinue = false; break; } } while (isContinue); } } export { FetchListV3 }; //# sourceMappingURL=fetch-list.mjs.map