fireberry-api-client 1.0.2-beta.2.2 → 1.0.2-beta.2.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
@@ -1 +1 @@
1
- {"version":3,"sources":["../../src/constants/objectIds.ts","../../src/utils/queryBuilder.ts","../../src/utils/fieldMapping.ts","../../src/constants/objectNames.ts","../../src/constants/fieldTypes.ts","../../src/utils/relatedFieldMapping.ts","../../src/constants/excludedFields.ts","../../src/utils/fieldTypes.ts","../../src/sdk/index.ts"],"names":[],"mappings":"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","file":"index.js","sourcesContent":["/**\r\n * Object Type ID to Primary Key Field Mapping\r\n * Maps Fireberry object type IDs to their primary key field names\r\n * Generated from actual API responses\r\n */\r\nexport const OBJECT_ID_MAP: Record<number, string> = {\r\n 1: 'accountid', // Account\r\n 2: 'contactid', // Contact\r\n 3: 'leadid', // Lead\r\n 4: 'opportunityid', // Opportunity\r\n 5: 'casesid', // Cases\r\n 6: 'activityid', // Activity\r\n 7: 'noteid', // Note\r\n 8: 'competitorid', // Competitor\r\n 9: 'crmuserid', // CrmUser\r\n 10: 'taskid', // Task\r\n 12: 'quoteid', // Quote\r\n 13: 'crmorderid', // CrmOrder\r\n 14: 'productid', // Product\r\n 17: 'crmorderitemid', // CrmOrderItem\r\n 20: 'emailtemplateid', // EmailTemplate\r\n 23: 'businessunitid', // BusinessUnit\r\n 25: 'orgid', // Org\r\n 27: 'printtemplateid', // PrintTemplate\r\n 28: 'contractid', // Contract\r\n 33: 'accountproductid', // AccountProduct\r\n 46: 'projectid', // Project\r\n 55: 'wfruleid', // WFRule\r\n 58: 'mdobjectid', // MDObject\r\n 64: 'roleid', // Role\r\n 67: 'campaignid', // Campaign\r\n 70: 'crmuserloginid', // CrmUserLogin\r\n 73: 'systemfieldid', // SystemField\r\n 76: 'articleid', // Article\r\n 77: 'linkid', // Link\r\n 78: 'invoiceid', // Invoice\r\n 80: 'invoicereceiptitemid', // InvoiceReceiptItem\r\n 81: 'invoiceid', // InvoiceNo\r\n 82: 'invoiceid', // InvoiceDraft\r\n 83: 'invoiceid', // InvoiceReceipt\r\n 84: 'invoiceid', // InvoiceReno\r\n 85: 'invoiceid', // InvoiceCredit\r\n 86: 'invoiceid', // InvoiceDelivery\r\n 89: 'iprestrictionid', // IpRestriction\r\n 90: 'transactionitemid', // TransactionItem\r\n 93: 'chargeid', // Charge\r\n 100: 'calllogid', // calllog\r\n 101: 'attendanceclockid', // AttendanceClock\r\n 102: 'activitylogid', // ActivityLog\r\n 104: 'conversationid', // Conversation\r\n 105: 'teaminboxid', // TeamInbox\r\n 106: 'texttemplateid', // TextTemplate\r\n 107: 'facebookconnectionid', // FacebookConnection\r\n 109: 'auditlogid', // AuditLog\r\n 110: 'smstemplateid', // SMSTemplate\r\n 111: 'providerverificationid', // ProviderVerification\r\n 114: 'calendarresourceid', // CalendarResource\r\n 115: 'journeyid', // Journey\r\n 116: 'profileid', // Profile\r\n 117: 'landingpageid', // LandingPage\r\n};\r\n\r\n/**\r\n * Gets the primary key field name for a given object type\r\n *\r\n * @param objectTypeId - The numeric object type ID\r\n * @returns The correct ID field name for the object type\r\n */\r\nexport function getObjectIdFieldName(objectTypeId: string | number): string {\r\n const objectTypeNum =\r\n typeof objectTypeId === 'string' ? parseInt(objectTypeId, 10) : objectTypeId;\r\n\r\n // Check if it's a mapped base object\r\n if (OBJECT_ID_MAP[objectTypeNum]) {\r\n return OBJECT_ID_MAP[objectTypeNum];\r\n }\r\n\r\n // For custom objects (1000 and up), use the pattern customobjectXid\r\n if (objectTypeNum >= 1000) {\r\n return `customobject${objectTypeNum}id`;\r\n }\r\n\r\n // Fallback to generic 'id' for unmapped objects\r\n return 'id';\r\n}\r\n","import type { QueryOperator, QueryMetadata, QueryResultWithMetadata, QueryExplainResult } from '../types/query';\r\nimport { getObjectIdFieldName } from '../constants/objectIds';\r\n\r\n/**\r\n * Regular expression to match pure date format (YYYY-MM-DD).\r\n * Does not match datetime formats like YYYY-MM-DDTHH:MM:SS or YYYY-MM-DD HH:MM:SS.\r\n */\r\nconst PURE_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}$/;\r\n\r\n/**\r\n * Checks if a value is a pure date string (YYYY-MM-DD format).\r\n * Returns false for datetime formats that include time components.\r\n *\r\n * @param value - The value to check\r\n * @returns True if the value is a pure date string\r\n *\r\n * @example\r\n * isPureDate('2024-01-15') // true\r\n * isPureDate('2024-01-15T10:30:00') // false\r\n * isPureDate('2024-01-15 10:30:00') // false\r\n * isPureDate('123') // false\r\n */\r\nexport function isPureDate(value: string): boolean {\r\n return PURE_DATE_REGEX.test(value);\r\n}\r\n\r\n/**\r\n * Adds a specified number of days to a date string.\r\n * Works with both pure dates (YYYY-MM-DD) and datetime formats.\r\n *\r\n * @param dateStr - The date string to modify\r\n * @param days - Number of days to add (can be negative)\r\n * @returns The modified date in YYYY-MM-DD format\r\n *\r\n * @example\r\n * addDays('2024-01-15', 1) // '2024-01-16'\r\n * addDays('2024-01-31', 1) // '2024-02-01'\r\n * addDays('2024-03-01', -1) // '2024-02-29' (leap year)\r\n */\r\nexport function addDays(dateStr: string, days: number): string {\r\n // Extract just the date part if datetime format\r\n const datePart = dateStr.split(/[T\\s]/)[0];\r\n const date = new Date(datePart + 'T00:00:00');\r\n date.setDate(date.getDate() + days);\r\n\r\n const year = date.getFullYear();\r\n const month = String(date.getMonth() + 1).padStart(2, '0');\r\n const day = String(date.getDate()).padStart(2, '0');\r\n\r\n return `${year}-${month}-${day}`;\r\n}\r\n\r\n/**\r\n * Escapes special characters in query values to prevent query injection.\r\n * This is a security measure to ensure user-provided values cannot modify\r\n * the query structure or inject additional query logic.\r\n *\r\n * @param value - The value to escape\r\n * @returns Escaped value safe for use in Fireberry queries\r\n */\r\nexport function escapeQueryValue(value: string): string {\r\n if (!value) {\r\n return '';\r\n }\r\n // Escape backslashes first to avoid double-escaping\r\n let escaped = value.replace(/\\\\/g, '\\\\\\\\');\r\n // Escape parentheses which could break out of conditions\r\n escaped = escaped.replace(/\\(/g, '\\\\(');\r\n escaped = escaped.replace(/\\)/g, '\\\\)');\r\n // Escape logical operators that could inject additional conditions\r\n // Using word boundaries to only match standalone operators\r\n escaped = escaped.replace(/\\bor\\b/gi, '\\\\or');\r\n escaped = escaped.replace(/\\band\\b/gi, '\\\\and');\r\n return escaped;\r\n}\r\n\r\n/**\r\n * Sanitizes a query string to ensure proper syntax for the Fireberry API\r\n * Handles common syntax issues and removes extraneous elements\r\n *\r\n * @param query - Query string to sanitize\r\n * @returns Sanitized query string\r\n */\r\nexport function sanitizeQuery(query: string): string {\r\n if (!query) {\r\n return '';\r\n }\r\n\r\n // First, protect special operators from being modified\r\n // Temporarily mark is-null and is-not-null operators\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+)\\s+(is-null|is-not-null)\\s*\\)/g,\r\n '($1 __SPECIAL_OPERATOR__$2)',\r\n );\r\n\r\n // Also protect text operators like start-with, not-start-with\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+)\\s+(start-with|not-start-with)\\s+([^)]+)\\s*\\)/g,\r\n '($1 __TEXT_OPERATOR__$2 $3)',\r\n );\r\n\r\n // Fix missing operators: (field value) -> (field = value)\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+(?:field|Field|system|System)[0-9]*)\\s+(?!__SPECIAL_OPERATOR__|__TEXT_OPERATOR__)([^()<>=!]+)\\s*\\)/g,\r\n '($1 = $2)',\r\n );\r\n\r\n // Fix with a more general pattern for any field-value pair without operator\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+)\\s+(?!__SPECIAL_OPERATOR__|__TEXT_OPERATOR__|<=|>=|!=|<|>|=\\s)([^()<>]+)\\s*\\)/g,\r\n '($1 = $2)',\r\n );\r\n\r\n // Restore special operators\r\n query = query.replace(/__SPECIAL_OPERATOR__/g, '');\r\n query = query.replace(/__TEXT_OPERATOR__/g, '');\r\n\r\n // Remove parentheses containing only a comparison operator\r\n query = query.replace(/\\(\\s*(?:<=|>=|!=|<|>|=)\\s*\\)/g, '');\r\n // Remove parentheses containing only text operators\r\n query = query.replace(/\\(\\s*(?:start-with|not-start-with)\\s*\\)/gi, '');\r\n // Remove parentheses containing only logical operators (AND/OR)\r\n query = query.replace(/\\(\\s*(?:and|or)\\s*\\)/gi, '');\r\n // Remove empty parentheses\r\n query = query.replace(/\\(\\s*\\)/g, '');\r\n // Remove logical operators without operands at start/end\r\n query = query.replace(/^\\s*(and|or)\\s*/gi, '');\r\n query = query.replace(/\\s*(and|or)\\s*$/gi, '');\r\n // Remove redundant nested parentheses: ((x)) -> (x)\r\n const nestedPattern = /\\(\\s*\\(([^()]+)\\)\\s*\\)/g;\r\n while (nestedPattern.test(query)) {\r\n query = query.replace(nestedPattern, '($1)');\r\n }\r\n // Collapse multiple spaces\r\n query = query.replace(/\\s+/g, ' ');\r\n return query.trim();\r\n}\r\n\r\n/**\r\n * Gets today's date in YYYY-MM-DD format\r\n */\r\nexport function getToday(): string {\r\n const now = new Date();\r\n const year = now.getFullYear();\r\n const month = String(now.getMonth() + 1).padStart(2, '0');\r\n const day = String(now.getDate()).padStart(2, '0');\r\n return `${year}-${month}-${day}`;\r\n}\r\n\r\n/**\r\n * Gets the start of current week (Monday) in YYYY-MM-DD format\r\n */\r\nexport function getStartOfWeek(): string {\r\n const now = new Date();\r\n const day = now.getDay();\r\n // Adjust so Monday is first day (day 0 = Sunday, so Monday = 1)\r\n const diff = now.getDate() - day + (day === 0 ? -6 : 1);\r\n const monday = new Date(now);\r\n monday.setDate(diff);\r\n const year = monday.getFullYear();\r\n const month = String(monday.getMonth() + 1).padStart(2, '0');\r\n const dayStr = String(monday.getDate()).padStart(2, '0');\r\n return `${year}-${month}-${dayStr}`;\r\n}\r\n\r\n/**\r\n * Gets the start of current month in YYYY-MM-DD format\r\n */\r\nexport function getStartOfMonth(): string {\r\n const now = new Date();\r\n const year = now.getFullYear();\r\n const month = String(now.getMonth() + 1).padStart(2, '0');\r\n return `${year}-${month}-01`;\r\n}\r\n\r\n/**\r\n * Date condition builder for fluent date query construction\r\n */\r\nexport interface DateConditionBuilder {\r\n /** Records from today */\r\n today(): QueryBuilder;\r\n /** Records from this week (Monday to now) */\r\n thisWeek(): QueryBuilder;\r\n /** Records from this month */\r\n thisMonth(): QueryBuilder;\r\n /** Records between two dates (inclusive) */\r\n between(startDate: string, endDate: string): QueryBuilder;\r\n /** Records from the last N days (including today) */\r\n daysAgo(days: number): QueryBuilder;\r\n /** Records before date */\r\n before(date: string): QueryBuilder;\r\n /** Records after date */\r\n after(date: string): QueryBuilder;\r\n /** Records on or before date */\r\n onOrBefore(date: string): QueryBuilder;\r\n /** Records on or after date */\r\n onOrAfter(date: string): QueryBuilder;\r\n}\r\n\r\n/**\r\n * Condition builder for fluent query construction\r\n */\r\nexport interface ConditionBuilder {\r\n /** Equals comparison (=) */\r\n equals(value: string | number): QueryBuilder;\r\n /** Not equals comparison (!=) */\r\n notEquals(value: string | number): QueryBuilder;\r\n /** IN comparison - matches any of the provided values (joined with OR) */\r\n in(values: (string | number)[]): QueryBuilder;\r\n /** Less than comparison (<) - works with numbers and dates */\r\n lessThan(value: string | number): QueryBuilder;\r\n /** Greater than comparison (>) - works with numbers and dates */\r\n greaterThan(value: string | number): QueryBuilder;\r\n /** Less than or equal (<=) - auto-converts pure dates (YYYY-MM-DD) to < nextDay */\r\n lessThanOrEqual(value: string | number): QueryBuilder;\r\n /** Greater than or equal (>=) - works with numbers and dates */\r\n greaterThanOrEqual(value: string | number): QueryBuilder;\r\n /** Contains value (translates to start-with %value) */\r\n contains(value: string): QueryBuilder;\r\n /** Does not contain value (translates to not-start-with %value) */\r\n notContains(value: string): QueryBuilder;\r\n /** Starts with value (start-with) */\r\n startsWith(value: string): QueryBuilder;\r\n /** Does not start with value (not-start-with) */\r\n notStartsWith(value: string): QueryBuilder;\r\n /** Field is null (is-null) */\r\n isNull(): QueryBuilder;\r\n /** Field is not null (is-not-null) */\r\n isNotNull(): QueryBuilder;\r\n}\r\n\r\n/**\r\n * Internal representation of a query condition\r\n */\r\ninterface QueryCondition {\r\n field: string;\r\n operator: QueryOperator;\r\n value?: string;\r\n}\r\n\r\n/**\r\n * Fluent query builder for constructing Fireberry queries\r\n *\r\n * @example\r\n * ```typescript\r\n * // Build a query string\r\n * const query = new QueryBuilder()\r\n * .where('statuscode').equals('1')\r\n * .and()\r\n * .where('emailaddress1').contains('@example.com')\r\n * .build();\r\n * // Output: \"(statuscode = 1) and (emailaddress1 start-with %@example.com)\"\r\n *\r\n * // With select and execute (requires client)\r\n * const result = await new QueryBuilder(client)\r\n * .objectType('1')\r\n * .select('accountid', 'name', 'emailaddress1')\r\n * .where('statuscode').equals('1')\r\n * .limit(100)\r\n * .execute();\r\n * ```\r\n */\r\n/**\r\n * Client interface for query execution\r\n */\r\ninterface QueryClient {\r\n query(options: {\r\n objectType: string;\r\n fields: string[];\r\n query: string;\r\n showRealValue: boolean;\r\n sortBy?: string;\r\n sortType?: 'asc' | 'desc';\r\n limit?: number;\r\n page?: number;\r\n signal?: AbortSignal;\r\n }): Promise<QueryResult>;\r\n}\r\n\r\n/**\r\n * Query result type\r\n */\r\ninterface QueryResult {\r\n records: Record<string, unknown>[];\r\n total: number;\r\n success: boolean;\r\n}\r\n\r\nexport class QueryBuilder {\r\n private conditions: QueryCondition[] = [];\r\n private joinOperators: ('and' | 'or')[] = [];\r\n private currentField: string | null = null;\r\n private selectedFields: string[] = [];\r\n private objectTypeId: string | null = null;\r\n private sortByField: string | null = null;\r\n private sortDirection: 'asc' | 'desc' = 'desc';\r\n private limitValue: number | null = null;\r\n private pageNumber: number = 1;\r\n private showRealValueFlag: boolean = true;\r\n private client: QueryClient | null = null;\r\n\r\n /**\r\n * Creates a new QueryBuilder\r\n * @param client - Optional FireberryClient for executing queries\r\n */\r\n constructor(client?: QueryClient) {\r\n this.client = client ?? null;\r\n }\r\n\r\n /**\r\n * Sets the object type for the query\r\n * @param objectType - Object type ID (e.g., '1' for Account)\r\n */\r\n objectType(objectType: string | number): this {\r\n this.objectTypeId = String(objectType);\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds fields to select\r\n * @param fields - Field names to select\r\n */\r\n select(...fields: string[]): this {\r\n this.selectedFields.push(...fields);\r\n return this;\r\n }\r\n\r\n /**\r\n * Starts a new WHERE condition\r\n * @param field - Field name to filter on\r\n */\r\n where(field: string): ConditionBuilder {\r\n this.currentField = field;\r\n return this.createConditionBuilder();\r\n }\r\n\r\n /**\r\n * Starts a new WHERE condition for a date field with date-specific helpers\r\n * @param field - Date field name to filter on\r\n *\r\n * @example\r\n * ```typescript\r\n * // Records created today\r\n * qb.whereDate('createdon').today()\r\n *\r\n * // Records from this week\r\n * qb.whereDate('createdon').thisWeek()\r\n *\r\n * // Records from the last 30 days\r\n * qb.whereDate('createdon').daysAgo(30)\r\n *\r\n * // Records between two dates\r\n * qb.whereDate('createdon').between('2024-01-01', '2024-12-31')\r\n * ```\r\n */\r\n whereDate(field: string): DateConditionBuilder {\r\n return this.createDateConditionBuilder(field);\r\n }\r\n\r\n /**\r\n * Adds a WHERE condition for the primary ID field, automatically mapped based on object type\r\n * @param value - The ID value to match\r\n * @throws Error if objectType is not set\r\n *\r\n * @example\r\n * ```typescript\r\n * // Instead of knowing the exact field name:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereId('abc123') // Automatically uses 'accountid' for object type 1\r\n * .execute();\r\n *\r\n * // Equivalent to:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .where('accountid').equals('abc123')\r\n * .execute();\r\n * ```\r\n */\r\n whereId(value: string | number): this {\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type must be set before using whereId(). Call .objectType() first.');\r\n }\r\n const idField = getObjectIdFieldName(this.objectTypeId);\r\n this.addCondition(idField, '=', String(value));\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds WHERE conditions for multiple primary ID values, joined with OR\r\n * @param values - Array of ID values to match\r\n * @throws Error if objectType is not set\r\n * @throws Error if values array is empty\r\n *\r\n * @example\r\n * ```typescript\r\n * // Query multiple accounts by ID:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereIds(['id1', 'id2', 'id3'])\r\n * .execute();\r\n *\r\n * // Generates: (accountid = id1) or (accountid = id2) or (accountid = id3)\r\n *\r\n * // Can be combined with other conditions:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereIds(['id1', 'id2'])\r\n * .and()\r\n * .where('statuscode').equals('1')\r\n * .execute();\r\n * ```\r\n */\r\n whereIds(values: (string | number)[]): this {\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type must be set before using whereIds(). Call .objectType() first.');\r\n }\r\n if (!values || values.length === 0) {\r\n throw new Error('whereIds() requires at least one ID value.');\r\n }\r\n const idField = getObjectIdFieldName(this.objectTypeId);\r\n\r\n // Add first condition\r\n this.addCondition(idField, '=', String(values[0]));\r\n\r\n // Add remaining conditions with OR\r\n for (let i = 1; i < values.length; i++) {\r\n this.joinOperators.push('or');\r\n this.addCondition(idField, '=', String(values[i]));\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a WHERE IN condition for a field with multiple values, joined with OR\r\n * @param field - Field name to filter on\r\n * @param values - Array of values to match\r\n * @throws Error if values array is empty\r\n *\r\n * @example\r\n * ```typescript\r\n * // Query accounts with specific status codes:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereIn('statuscode', [1, 2, 3])\r\n * .execute();\r\n *\r\n * // Generates: (statuscode = 1) or (statuscode = 2) or (statuscode = 3)\r\n * ```\r\n */\r\n whereIn(field: string, values: (string | number)[]): this {\r\n if (!values || values.length === 0) {\r\n throw new Error('whereIn() requires at least one value.');\r\n }\r\n // Add first condition\r\n this.addCondition(field, '=', String(values[0]));\r\n // Add remaining conditions with OR\r\n for (let i = 1; i < values.length; i++) {\r\n this.joinOperators.push('or');\r\n this.addCondition(field, '=', String(values[i]));\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds an AND logical operator\r\n */\r\n and(): this {\r\n if (this.conditions.length > 0) {\r\n this.joinOperators.push('and');\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds an OR logical operator\r\n */\r\n or(): this {\r\n if (this.conditions.length > 0) {\r\n this.joinOperators.push('or');\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the sort field and direction\r\n * @param field - Field to sort by\r\n * @param direction - Sort direction ('asc' or 'desc')\r\n */\r\n sortBy(field: string, direction: 'asc' | 'desc' = 'desc'): this {\r\n this.sortByField = field;\r\n this.sortDirection = direction;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the maximum number of records to return\r\n * @param count - Maximum record count\r\n */\r\n limit(count: number): this {\r\n this.limitValue = count;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the page number for pagination\r\n * @param page - Page number (1-based)\r\n */\r\n page(page: number): this {\r\n this.pageNumber = page;\r\n return this;\r\n }\r\n\r\n /**\r\n * Controls whether to show real values (labels) for dropdown fields\r\n * @param show - Whether to show real values (default: true)\r\n */\r\n showRealValue(show: boolean): this {\r\n this.showRealValueFlag = show;\r\n return this;\r\n }\r\n\r\n /**\r\n * Builds the query string from conditions\r\n * @returns The built query string\r\n */\r\n build(): string {\r\n if (this.conditions.length === 0) {\r\n return '';\r\n }\r\n\r\n const parts: string[] = [];\r\n\r\n for (let i = 0; i < this.conditions.length; i++) {\r\n const condition = this.conditions[i];\r\n let conditionStr: string;\r\n\r\n if (condition.operator === 'is-null' || condition.operator === 'is-not-null') {\r\n conditionStr = `(${condition.field} ${condition.operator})`;\r\n } else {\r\n const escapedValue = escapeQueryValue(condition.value || '');\r\n conditionStr = `(${condition.field} ${condition.operator} ${escapedValue})`;\r\n }\r\n\r\n parts.push(conditionStr);\r\n\r\n // Add join operator if there's a next condition\r\n if (i < this.joinOperators.length) {\r\n parts.push(this.joinOperators[i]);\r\n }\r\n }\r\n\r\n return parts.join(' ');\r\n }\r\n\r\n /**\r\n * Returns the selected fields array\r\n * Useful for inspecting the query configuration\r\n */\r\n getFields(): string[] {\r\n return [...this.selectedFields];\r\n }\r\n\r\n /**\r\n * Converts the query builder state to a payload compatible with @fireberry/sdk\r\n *\r\n * @returns Object with `fields` (comma-separated string) and `query` (filter string)\r\n *\r\n * @example\r\n * ```typescript\r\n * import FireberryClientSDK from '@fireberry/sdk/client';\r\n * import { QueryBuilder } from 'fireberry-api-client';\r\n *\r\n * const sdk = new FireberryClientSDK();\r\n * await sdk.initializeContext();\r\n *\r\n * const payload = new QueryBuilder()\r\n * .select('accountid', 'accountname')\r\n * .where('statuscode').equals('1')\r\n * .toSDKPayload();\r\n *\r\n * // Use with SDK\r\n * const results = await sdk.api.query(1, payload);\r\n * ```\r\n */\r\n toSDKPayload(): { fields: string; query: string; page_size?: number; page_number?: number } {\r\n const payload: { fields: string; query: string; page_size?: number; page_number?: number } = {\r\n fields: this.selectedFields.length > 0 ? this.selectedFields.join(',') : '*',\r\n query: this.build(),\r\n };\r\n\r\n if (this.limitValue !== null) {\r\n payload.page_size = this.limitValue;\r\n }\r\n\r\n if (this.pageNumber > 1) {\r\n payload.page_number = this.pageNumber;\r\n }\r\n\r\n return payload;\r\n }\r\n\r\n /**\r\n * Executes the query and returns the count of matching records\r\n * Uses minimal field selection (ID only) for efficiency\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns Number of matching records\r\n *\r\n * @example\r\n * ```typescript\r\n * const activeCount = await client.queryBuilder()\r\n * .objectType(1)\r\n * .where('statuscode').equals('1')\r\n * .count();\r\n *\r\n * console.log(`Found ${activeCount} active accounts`);\r\n * ```\r\n */\r\n async count(signal?: AbortSignal): Promise<number> {\r\n if (!this.client) {\r\n throw new Error('QueryBuilder requires a client to execute queries. Pass a FireberryClient to the constructor.');\r\n }\r\n\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type is required. Use .objectType() before executing.');\r\n }\r\n\r\n // Use minimal fields for efficiency - just get the ID field\r\n const idField = getObjectIdFieldName(this.objectTypeId);\r\n\r\n const result = await this.client.query({\r\n objectType: this.objectTypeId,\r\n fields: [idField],\r\n query: this.build(),\r\n showRealValue: false, // No need for labels\r\n signal,\r\n });\r\n\r\n return result.total;\r\n }\r\n\r\n /**\r\n * Executes the query and returns the first record or null\r\n * Automatically sets limit to 1 for efficiency\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns First record or null if no records found\r\n *\r\n * @example\r\n * ```typescript\r\n * const account = await client.queryBuilder()\r\n * .objectType(1)\r\n * .where('accountname').equals('Acme Corp')\r\n * .first();\r\n *\r\n * if (account) {\r\n * console.log(account.accountid);\r\n * }\r\n * ```\r\n */\r\n async first(signal?: AbortSignal): Promise<Record<string, unknown> | null> {\r\n // Temporarily set limit to 1 for efficiency\r\n const originalLimit = this.limitValue;\r\n this.limitValue = 1;\r\n\r\n try {\r\n const result = await this.execute(signal);\r\n return result.records[0] ?? null;\r\n } finally {\r\n // Restore original limit\r\n this.limitValue = originalLimit;\r\n }\r\n }\r\n\r\n /**\r\n * Executes the query (requires client to be set)\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns Query results\r\n */\r\n async execute(signal?: AbortSignal): Promise<QueryResult> {\r\n if (!this.client) {\r\n throw new Error('QueryBuilder requires a client to execute queries. Pass a FireberryClient to the constructor.');\r\n }\r\n\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type is required. Use .objectType() before executing.');\r\n }\r\n\r\n const queryOptions: Parameters<QueryClient['query']>[0] = {\r\n objectType: this.objectTypeId,\r\n fields: this.selectedFields.length > 0 ? this.selectedFields : ['*'],\r\n query: this.build(),\r\n showRealValue: this.showRealValueFlag,\r\n };\r\n\r\n if (this.sortByField) {\r\n queryOptions.sortBy = this.sortByField;\r\n queryOptions.sortType = this.sortDirection;\r\n }\r\n\r\n if (this.limitValue !== null) {\r\n queryOptions.limit = this.limitValue;\r\n }\r\n\r\n if (this.pageNumber > 1) {\r\n queryOptions.page = this.pageNumber;\r\n }\r\n\r\n if (signal) {\r\n queryOptions.signal = signal;\r\n }\r\n\r\n return this.client.query(queryOptions);\r\n }\r\n\r\n /**\r\n * Executes the query and returns results with debugging metadata\r\n * Includes query string, fields, pagination info, and execution time\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns Query results with metadata\r\n *\r\n * @example\r\n * ```typescript\r\n * const result = await client.queryBuilder()\r\n * .objectType(1)\r\n * .select('accountid', 'accountname')\r\n * .where('statuscode').equals('1')\r\n * .executeWithDebug();\r\n *\r\n * console.log('Query:', result.metadata.queryString);\r\n * console.log('Fields:', result.metadata.fields);\r\n * console.log('Execution time:', result.metadata.executionTimeMs, 'ms');\r\n * console.log('Records:', result.records.length);\r\n * ```\r\n */\r\n async executeWithDebug(signal?: AbortSignal): Promise<QueryResultWithMetadata> {\r\n if (!this.client) {\r\n throw new Error('QueryBuilder requires a client to execute queries. Pass a FireberryClient to the constructor.');\r\n }\r\n\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type is required. Use .objectType() before executing.');\r\n }\r\n\r\n const startTime = Date.now();\r\n const fields = this.selectedFields.length > 0 ? this.selectedFields : ['*'];\r\n const queryString = this.build();\r\n\r\n const queryOptions: Parameters<QueryClient['query']>[0] = {\r\n objectType: this.objectTypeId,\r\n fields,\r\n query: queryString,\r\n showRealValue: this.showRealValueFlag,\r\n };\r\n\r\n if (this.sortByField) {\r\n queryOptions.sortBy = this.sortByField;\r\n queryOptions.sortType = this.sortDirection;\r\n }\r\n\r\n if (this.limitValue !== null) {\r\n queryOptions.limit = this.limitValue;\r\n }\r\n\r\n if (this.pageNumber > 1) {\r\n queryOptions.page = this.pageNumber;\r\n }\r\n\r\n if (signal) {\r\n queryOptions.signal = signal;\r\n }\r\n\r\n const result = await this.client.query(queryOptions);\r\n const executionTimeMs = Date.now() - startTime;\r\n\r\n const metadata: QueryMetadata = {\r\n objectType: this.objectTypeId,\r\n fields,\r\n queryString,\r\n pageNumber: this.pageNumber,\r\n pageSize: this.limitValue ?? 500,\r\n autoPage: true, // Default behavior\r\n executionTimeMs,\r\n };\r\n\r\n if (this.sortByField) {\r\n metadata.sortBy = this.sortByField;\r\n metadata.sortType = this.sortDirection;\r\n }\r\n\r\n if (this.limitValue !== null) {\r\n metadata.limit = this.limitValue;\r\n }\r\n\r\n return {\r\n ...result,\r\n metadata,\r\n };\r\n }\r\n\r\n /**\r\n * Analyzes the query without executing it (dry run)\r\n * Returns information about what the query will do, potential issues, and optimization suggestions\r\n *\r\n * @returns Query analysis with warnings and suggestions\r\n *\r\n * @example\r\n * ```typescript\r\n * const analysis = client.queryBuilder()\r\n * .objectType('1')\r\n * .select('*')\r\n * .where('statuscode').equals('1')\r\n * .explain();\r\n *\r\n * console.log('Query:', analysis.query);\r\n * console.log('Warnings:', analysis.warnings);\r\n * console.log('Suggestions:', analysis.suggestions);\r\n * console.log('Estimated API calls:', analysis.estimatedApiCalls);\r\n * ```\r\n */\r\n explain(): QueryExplainResult {\r\n const warnings: string[] = [];\r\n const suggestions: string[] = [];\r\n\r\n // Check object type\r\n if (!this.objectTypeId) {\r\n warnings.push('Object type is not set. Call .objectType() before executing.');\r\n }\r\n\r\n // Analyze fields\r\n const fields = this.selectedFields.length > 0 ? this.selectedFields : ['*'];\r\n const usesWildcard = fields.includes('*') || fields.length === 0;\r\n\r\n if (usesWildcard) {\r\n warnings.push('Using wildcard (*) fields may include unnecessary data and slow down queries.');\r\n suggestions.push('Consider selecting only the specific fields you need with .select()');\r\n }\r\n\r\n // Analyze query\r\n const queryString = this.build();\r\n const conditionCount = this.conditions.length;\r\n\r\n if (conditionCount === 0 && !this.limitValue) {\r\n warnings.push('No query conditions or limit set. This may return a large number of records.');\r\n suggestions.push('Add filters with .where() or set a .limit() to control result size.');\r\n }\r\n\r\n // Check for potential performance issues\r\n const hasContainsOperator = this.conditions.some(\r\n (c) => c.operator === 'start-with' && c.value?.startsWith('%'),\r\n );\r\n if (hasContainsOperator) {\r\n warnings.push('Contains queries (using % prefix) may be slower than exact matches.');\r\n }\r\n\r\n // Check for OR conditions (may affect performance)\r\n const hasOrConditions = this.joinOperators.some((op) => op === 'or');\r\n if (hasOrConditions && conditionCount > 5) {\r\n warnings.push('Multiple OR conditions may affect query performance.');\r\n suggestions.push('Consider breaking into separate queries if possible.');\r\n }\r\n\r\n // Sorting analysis\r\n const sortField = this.sortByField || 'modifiedon';\r\n const sortDirection = this.sortDirection;\r\n\r\n // Estimate API calls\r\n let estimatedApiCalls = 1;\r\n const pageSize = 500; // Default page size\r\n const willAutoPage = this.limitValue === null; // Auto-page when no limit\r\n\r\n if (willAutoPage && conditionCount === 0) {\r\n // Without conditions, could be many pages\r\n estimatedApiCalls = -1; // Unknown/many\r\n warnings.push('Without filters, query may require many API calls for pagination.');\r\n } else if (this.limitValue !== null) {\r\n estimatedApiCalls = Math.ceil(this.limitValue / pageSize);\r\n }\r\n\r\n // Check for missing index hints\r\n if (!this.sortByField && conditionCount > 0) {\r\n suggestions.push('Consider adding .sortBy() to control result ordering.');\r\n }\r\n\r\n // Check showRealValue impact\r\n if (this.showRealValueFlag) {\r\n suggestions.push('showRealValue is enabled (default). Set .showRealValue(false) if you only need IDs.');\r\n }\r\n\r\n return {\r\n objectType: this.objectTypeId || '(not set)',\r\n query: queryString || '(no conditions)',\r\n fields,\r\n usesWildcard,\r\n willAutoPage,\r\n limit: this.limitValue,\r\n pageSize,\r\n sorting: {\r\n field: sortField,\r\n direction: sortDirection,\r\n },\r\n estimatedApiCalls,\r\n warnings,\r\n suggestions,\r\n conditionCount,\r\n showRealValue: this.showRealValueFlag,\r\n };\r\n }\r\n\r\n /**\r\n * Creates a condition builder for the current field\r\n */\r\n private createConditionBuilder(): ConditionBuilder {\r\n const field = this.currentField!;\r\n\r\n return {\r\n equals: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '=', String(value));\r\n return this;\r\n },\r\n notEquals: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '!=', String(value));\r\n return this;\r\n },\r\n in: (values: (string | number)[]): QueryBuilder => {\r\n if (!values || values.length === 0) {\r\n throw new Error('in() requires at least one value.');\r\n }\r\n // Add first condition\r\n this.addCondition(field, '=', String(values[0]));\r\n // Add remaining conditions with OR\r\n for (let i = 1; i < values.length; i++) {\r\n this.joinOperators.push('or');\r\n this.addCondition(field, '=', String(values[i]));\r\n }\r\n return this;\r\n },\r\n lessThan: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '<', String(value));\r\n return this;\r\n },\r\n greaterThan: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '>', String(value));\r\n return this;\r\n },\r\n lessThanOrEqual: (value: string | number): QueryBuilder => {\r\n const strValue = String(value);\r\n // Fireberry API bug: <= with pure dates (YYYY-MM-DD) behaves like <\r\n // because it interprets the date as midnight (00:00:00).\r\n // Auto-convert to < nextDay for correct \"on or before\" behavior.\r\n if (isPureDate(strValue)) {\r\n const nextDay = addDays(strValue, 1);\r\n this.addCondition(field, '<', nextDay);\r\n } else {\r\n this.addCondition(field, '<=', strValue);\r\n }\r\n return this;\r\n },\r\n greaterThanOrEqual: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '>=', String(value));\r\n return this;\r\n },\r\n contains: (value: string): QueryBuilder => {\r\n // Contains translates to \"start-with %value\"\r\n this.addCondition(field, 'start-with', `%${value}`);\r\n return this;\r\n },\r\n notContains: (value: string): QueryBuilder => {\r\n // Not contains translates to \"not-start-with %value\"\r\n this.addCondition(field, 'not-start-with', `%${value}`);\r\n return this;\r\n },\r\n startsWith: (value: string): QueryBuilder => {\r\n this.addCondition(field, 'start-with', value);\r\n return this;\r\n },\r\n notStartsWith: (value: string): QueryBuilder => {\r\n this.addCondition(field, 'not-start-with', value);\r\n return this;\r\n },\r\n isNull: (): QueryBuilder => {\r\n this.addCondition(field, 'is-null');\r\n return this;\r\n },\r\n isNotNull: (): QueryBuilder => {\r\n this.addCondition(field, 'is-not-null');\r\n return this;\r\n },\r\n };\r\n }\r\n\r\n /**\r\n * Creates a date condition builder for the specified field\r\n */\r\n private createDateConditionBuilder(field: string): DateConditionBuilder {\r\n return {\r\n today: (): QueryBuilder => {\r\n const today = getToday();\r\n // Records from today: >= today AND < tomorrow\r\n this.addCondition(field, '>=', today);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', addDays(today, 1));\r\n return this;\r\n },\r\n thisWeek: (): QueryBuilder => {\r\n const startOfWeek = getStartOfWeek();\r\n const tomorrow = addDays(getToday(), 1);\r\n // Records from start of week to now: >= monday AND < tomorrow\r\n this.addCondition(field, '>=', startOfWeek);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', tomorrow);\r\n return this;\r\n },\r\n thisMonth: (): QueryBuilder => {\r\n const startOfMonth = getStartOfMonth();\r\n const tomorrow = addDays(getToday(), 1);\r\n // Records from start of month to now: >= first day AND < tomorrow\r\n this.addCondition(field, '>=', startOfMonth);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', tomorrow);\r\n return this;\r\n },\r\n between: (startDate: string, endDate: string): QueryBuilder => {\r\n // Records between dates (inclusive): >= start AND < day after end\r\n this.addCondition(field, '>=', startDate);\r\n this.joinOperators.push('and');\r\n // Use < next day for inclusive end date (handles API quirk)\r\n const dayAfterEnd = isPureDate(endDate) ? addDays(endDate, 1) : endDate;\r\n this.addCondition(field, '<', dayAfterEnd);\r\n return this;\r\n },\r\n daysAgo: (days: number): QueryBuilder => {\r\n const today = getToday();\r\n const startDate = addDays(today, -days);\r\n const tomorrow = addDays(today, 1);\r\n // Records from N days ago to now: >= (today - N) AND < tomorrow\r\n this.addCondition(field, '>=', startDate);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', tomorrow);\r\n return this;\r\n },\r\n before: (date: string): QueryBuilder => {\r\n this.addCondition(field, '<', date);\r\n return this;\r\n },\r\n after: (date: string): QueryBuilder => {\r\n // After a date means > end of that day\r\n // For pure dates, use > date (API treats as > midnight, so actually > end of previous day)\r\n // We need >= next day for \"after\" to work correctly\r\n const nextDay = isPureDate(date) ? addDays(date, 1) : date;\r\n this.addCondition(field, '>=', nextDay);\r\n return this;\r\n },\r\n onOrBefore: (date: string): QueryBuilder => {\r\n // Use < next day for inclusive (handles API quirk with <=)\r\n const nextDay = isPureDate(date) ? addDays(date, 1) : date;\r\n this.addCondition(field, '<', nextDay);\r\n return this;\r\n },\r\n onOrAfter: (date: string): QueryBuilder => {\r\n this.addCondition(field, '>=', date);\r\n return this;\r\n },\r\n };\r\n }\r\n\r\n /**\r\n * Adds a condition to the query\r\n */\r\n private addCondition(field: string, operator: QueryOperator, value?: string): void {\r\n this.conditions.push({ field, operator, value });\r\n this.currentField = null;\r\n }\r\n}\r\n","/**\r\n * Special field name mappings that don't follow standard patterns\r\n * Key: original field name, Value: corresponding label field\r\n */\r\nconst SPECIAL_LABEL_FIELD_MAPPINGS: Record<string, string> = {\r\n objectid: 'objecttitle',\r\n lastactionid: 'lastactiontitle',\r\n wfruleid: 'rulename', // Object 55 (Workflow Rules)\r\n noteid: 'subject', // Object 7 (Note) - noteid maps to subject\r\n};\r\n\r\n/**\r\n * Fields ending with \"id\" that should NOT have \"id\" replaced with \"name\"\r\n * These fields will use the default append \"name\" behavior instead\r\n */\r\nconst EXCLUDED_ID_FIELDS: string[] = [\r\n 'businessunitid', // → businessunitidname (not businessunitname)\r\n 'crmuserid', // → no name field exists\r\n 'languageid', // → languageidname (not languagename)\r\n];\r\n\r\n/**\r\n * Fields ending with \"code\" that should NOT have \"code\" removed\r\n * These fields will use the default append \"name\" behavior instead\r\n */\r\nconst EXCLUDED_CODE_FIELDS: string[] = [\r\n 'duplicaterecordcode', // → duplicaterecordcodename (not duplicaterecord)\r\n];\r\n\r\n/**\r\n * Fields that have NO corresponding label field at all\r\n * These fields should not have any name transformation applied\r\n * Return empty string to signal \"no label field exists\"\r\n */\r\nconst FIELDS_WITHOUT_LABEL_FIELD: string[] = [\r\n 'systemfieldid', // Object 73 - no name field exists\r\n 'fieldobjecttype', // Object 73 - base field doesn't exist, only *name exists\r\n 'invoiceid', // Objects 78, 81 - no name field exists\r\n 'calllogid', // Object 100 - no name field exists (primary key)\r\n 'attendanceclockid', // Object 101 - no name field exists (primary key)\r\n 'activitylogid', // Object 102 - no name field exists (primary key)\r\n 'conversationid', // Object 104 - no name field exists (primary key)\r\n 'texttemplateid', // Text Template - no name field exists (primary key)\r\n 'smstemplateid', // Object 110 - no name field exists (primary key)\r\n 'deletedby', // Object 7 - no name field exists\r\n 'recordid', // Object 7 - no name field exists\r\n 'objecttypecode', // Object 7 - no name field exists\r\n];\r\n\r\n/**\r\n * Object-type specific overrides for field name transformations\r\n * Key: object type ID, Value: { excludedIdFields, excludedCodeFields }\r\n */\r\nconst OBJECT_TYPE_OVERRIDES: Record<\r\n number,\r\n { excludedIdFields?: string[]; excludedCodeFields?: string[] }\r\n> = {\r\n // CRM Orders (13) - uses *idname and *codename patterns\r\n 13: {\r\n excludedIdFields: ['ownerid', 'accountid', 'printtemplateid', 'pcfaccountid'],\r\n excludedCodeFields: [\r\n 'rounddiscountcode',\r\n 'taxincludecode',\r\n 'currencycode',\r\n 'transmissioncode',\r\n 'creditrejectioncode1',\r\n 'creditrejectioncode2',\r\n 'apartmentcode1',\r\n 'apartmentcode2',\r\n 'restrictioncode1',\r\n 'restrictioncode2',\r\n 'casescode1',\r\n 'casescode2',\r\n 'returncode1',\r\n 'returncode2',\r\n 'statuscode',\r\n ],\r\n },\r\n // Products (14)\r\n 14: {\r\n excludedCodeFields: ['categorycode'], // → categoryname (not category)\r\n },\r\n // CRM Order Items (17) - uses *idname pattern\r\n 17: {\r\n excludedIdFields: ['crmorderid', 'productid', 'ownerid'],\r\n },\r\n // Email Templates (20)\r\n 20: {\r\n excludedIdFields: ['mdobjectid'],\r\n },\r\n // Print Templates (27)\r\n 27: {\r\n excludedIdFields: ['mdobjectid'],\r\n },\r\n // System Fields (73) - uses *idname pattern\r\n 73: {\r\n excludedIdFields: ['mdobjectid', 'ownerid'],\r\n },\r\n // Invoices (78) - uses *codename and *idname patterns extensively\r\n 78: {\r\n excludedIdFields: [\r\n 'crmorderid',\r\n 'invoicereceiptid',\r\n 'accountid',\r\n 'ownerid',\r\n 'invoicerenoid',\r\n ],\r\n excludedCodeFields: [\r\n 'taxincludecode',\r\n 'depositcode',\r\n 'rounddiscountcode',\r\n 'currencycode',\r\n 'statecode',\r\n 'invoicetypecode',\r\n ],\r\n },\r\n // Invoice No (81) - uses *codename and *idname patterns extensively\r\n 81: {\r\n excludedIdFields: [\r\n 'ownerid',\r\n 'invoicecreditid',\r\n 'crmorderid',\r\n 'invoicereceiptid',\r\n 'invoicedeliveryid',\r\n 'accountid',\r\n ],\r\n excludedCodeFields: ['currencycode', 'rounddiscountcode', 'statecode', 'taxincludecode'],\r\n },\r\n // Invoice Draft (82) - uses *codename and *idname patterns extensively\r\n 82: {\r\n excludedIdFields: ['accountid', 'crmorderid', 'ownerid', 'invoicerenoid'],\r\n excludedCodeFields: ['taxincludecode', 'statecode', 'rounddiscountcode', 'currencycode'],\r\n },\r\n // Invoice Receipt (83) - uses *codename and *idname patterns\r\n 83: {\r\n excludedIdFields: ['invoicenoid', 'crmorderid', 'ownerid', 'accountid'],\r\n excludedCodeFields: ['statecode', 'currencycode'],\r\n },\r\n // Invoice Tax Receipt (84) - uses *codename and *idname patterns\r\n 84: {\r\n excludedIdFields: ['crmorderid', 'ownerid', 'invoicecreditid', 'accountid'],\r\n excludedCodeFields: ['rounddiscountcode', 'statecode', 'taxincludecode', 'currencycode'],\r\n },\r\n // Invoice Credit (85) - uses *codename and *idname patterns\r\n 85: {\r\n excludedIdFields: ['ownerid', 'crmorderid', 'accountid'],\r\n excludedCodeFields: ['taxincludecode', 'rounddiscountcode', 'currencycode', 'statecode'],\r\n },\r\n // Invoice (86) - uses *codename and *idname patterns\r\n 86: {\r\n excludedIdFields: ['accountid', 'ownerid', 'crmorderid'],\r\n excludedCodeFields: ['statecode', 'rounddiscountcode', 'currencycode', 'taxincludecode'],\r\n },\r\n // Call Log (100) - uses *idname pattern\r\n 100: {\r\n excludedIdFields: ['contactid', 'leadid', 'accountid', 'ownerid'],\r\n },\r\n // Attendance Clock (101) - uses *idname pattern\r\n 101: {\r\n excludedIdFields: ['ownerid'],\r\n },\r\n // Activity Log (102) - uses *idname and *codename patterns\r\n 102: {\r\n excludedIdFields: ['contactid', 'ownerid'],\r\n excludedCodeFields: ['objecttypecode', 'typecode', 'resultcode'],\r\n },\r\n // Conversation (104) - uses *idname pattern\r\n 104: {\r\n excludedIdFields: ['leadid', 'ownerid', 'contactid', 'accountid'],\r\n },\r\n // Text Template (106) - uses *idname pattern\r\n 106: {\r\n excludedIdFields: ['ownerid'],\r\n },\r\n // SMS Template (110) - uses *idname pattern\r\n 110: {\r\n excludedIdFields: ['ownerid'],\r\n },\r\n};\r\n\r\n/**\r\n * Converts a field API name to its corresponding label field.\r\n *\r\n * Rules:\r\n * - Special mappings (e.g., objectid → objecttitle)\r\n * - Fields starting with \"pcf\" (custom fields): append \"name\" → pcf_field → pcf_fieldname\r\n * - Fields ending with \"code\" (unless excluded): remove \"code\" → statuscode → status\r\n * - Fields ending with \"id\" (unless excluded): replace \"id\" with \"name\" → accountid → accountname\r\n * - All other fields: append \"name\"\r\n *\r\n * @param fieldName - The field API name\r\n * @param objectType - The object type ID (required for object-specific overrides)\r\n * @returns The corresponding label field, or empty string if no label field exists\r\n */\r\nexport function getLabelFieldForField(fieldName: string, objectType: string | number): string {\r\n // Check special mappings first\r\n if (SPECIAL_LABEL_FIELD_MAPPINGS[fieldName]) {\r\n return SPECIAL_LABEL_FIELD_MAPPINGS[fieldName];\r\n }\r\n\r\n // Check if field has no label field at all - return empty string\r\n if (FIELDS_WITHOUT_LABEL_FIELD.includes(fieldName)) {\r\n return '';\r\n }\r\n\r\n // Check for custom object primary keys (customobject{N}id pattern) → name\r\n if (/^customobject\\d+id$/.test(fieldName)) {\r\n return 'name';\r\n }\r\n\r\n // Custom fields (pcf prefix) - just append \"name\"\r\n if (fieldName.startsWith('pcf')) {\r\n return `${fieldName}name`;\r\n }\r\n\r\n // Get object-type specific overrides\r\n const objectTypeNum =\r\n typeof objectType === 'string' ? parseInt(objectType, 10) : objectType;\r\n const overrides = OBJECT_TYPE_OVERRIDES[objectTypeNum] || null;\r\n\r\n // For custom objects (1000+), add default exclusions\r\n const isCustomObject = objectTypeNum >= 1000;\r\n const customObjectExclusions = isCustomObject ? ['ownerid'] : [];\r\n\r\n // Combine global and object-specific exclusions\r\n const excludedCodeFields = [\r\n ...EXCLUDED_CODE_FIELDS,\r\n ...(overrides?.excludedCodeFields || []),\r\n ];\r\n const excludedIdFields = [\r\n ...EXCLUDED_ID_FIELDS,\r\n ...(overrides?.excludedIdFields || []),\r\n ...customObjectExclusions,\r\n ];\r\n\r\n // Remove \"code\" suffix (unless excluded) → statuscode → status\r\n if (fieldName.endsWith('code') && !excludedCodeFields.includes(fieldName)) {\r\n return fieldName.slice(0, -4);\r\n }\r\n\r\n // Replace \"id\" suffix with \"name\" (unless excluded) → accountid → accountname\r\n if (fieldName.endsWith('id') && !excludedIdFields.includes(fieldName)) {\r\n return fieldName.slice(0, -2) + 'name';\r\n }\r\n\r\n // Default: append \"name\"\r\n return `${fieldName}name`;\r\n}\r\n","/**\r\n * Object Type ID to Name Field Mapping\r\n * Maps Fireberry object type IDs to their display name field\r\n * Generated from actual API responses\r\n */\r\nexport const OBJECT_NAME_MAP: Record<number, string> = {\r\n 1: 'accountname', // Account\r\n 2: 'fullname', // Contact\r\n 3: 'fullname', // Lead\r\n 4: 'name', // Opportunity\r\n 5: 'title', // Cases\r\n 6: 'subject', // Activity\r\n 7: 'notetext', // Note\r\n 8: 'competitorname', // Competitor\r\n 9: 'fullname', // CrmUser\r\n 10: 'subject', // Task\r\n 12: 'quotenumber', // Quote\r\n 13: 'crmordernumber', // CrmOrder\r\n 14: 'name', // Product\r\n 17: 'productname', // CrmOrderItem\r\n 20: 'title', // EmailTemplate\r\n 23: 'name', // BusinessUnit\r\n 25: 'orgname', // Org\r\n 27: 'name', // PrintTemplate\r\n 28: 'contractname', // Contract\r\n 33: 'productid', // AccountProduct (uses productid as display)\r\n 46: 'projectname', // Project\r\n 55: 'rulename', // WFRule\r\n 58: 'name', // MDObject\r\n 64: 'rolename', // Role\r\n 67: 'campaignname', // Campaign\r\n 70: 'browsername', // CrmUserLogin\r\n 73: 'label', // SystemField (using label as name field)\r\n 76: 'articlename', // Article\r\n 77: 'linkname', // Link\r\n 78: 'invoicenumber', // Invoice\r\n 80: 'documentnumber', // InvoiceReceiptItem\r\n 81: 'invoicenumber', // InvoiceNo\r\n 82: 'invoicenumber', // InvoiceDraft\r\n 83: 'invoicenumber', // InvoiceReceipt\r\n 84: 'invoicenumber', // InvoiceReno\r\n 85: 'invoicenumber', // InvoiceCredit\r\n 86: 'invoicenumber', // InvoiceDelivery\r\n 89: 'name', // IpRestriction\r\n 90: 'documentnumber', // TransactionItem\r\n 93: 'name', // Charge\r\n 100: 'callerid', // calllog\r\n 101: 'name', // AttendanceClock\r\n 102: 'activitylognumber', // ActivityLog\r\n 104: 'subject', // Conversation\r\n 105: 'name', // TeamInbox\r\n 106: 'name', // TextTemplate\r\n 107: 'name', // FacebookConnection\r\n 109: 'name', // AuditLog\r\n 110: 'name', // SMSTemplate\r\n 111: 'name', // ProviderVerification\r\n 114: 'name', // CalendarResource\r\n 115: 'name', // Journey\r\n 116: 'name', // Profile\r\n 117: 'name', // LandingPage\r\n};\r\n\r\n/**\r\n * Gets the display name field for a given object type\r\n *\r\n * @param objectTypeId - The numeric object type ID\r\n * @returns The name field for the object type\r\n */\r\nexport function getNameFieldByObjectType(objectTypeId: string | number): string {\r\n const objectTypeNum =\r\n typeof objectTypeId === 'string' ? parseInt(objectTypeId, 10) : objectTypeId;\r\n\r\n // Check if it's a mapped base object\r\n if (OBJECT_NAME_MAP[objectTypeNum]) {\r\n return OBJECT_NAME_MAP[objectTypeNum];\r\n }\r\n\r\n // For custom objects (1000 and up), use 'name'\r\n if (objectTypeNum >= 1000) {\r\n return 'name';\r\n }\r\n\r\n // Fallback to 'name' for unmapped objects\r\n return 'name';\r\n}\r\n","/**\r\n * Field Type System IDs from Fireberry CRM\r\n * These UUIDs identify different field types in the metadata API\r\n */\r\nexport const FIELD_TYPE_IDS = {\r\n DROPDOWN: 'b4919f2e-2996-48e4-a03c-ba39fb64386c',\r\n LOOKUP: 'a8fcdf65-91bc-46fd-82f6-1234758345a1',\r\n EMAIL: 'c713d2f7-8fa9-43c3-8062-f07486eaf567',\r\n TEXT: 'a1e7ed6f-5083-477b-b44c-9943a6181359',\r\n URL: 'c820d32f-44df-4c2a-9c1e-18734e864fd5',\r\n LONG_TEXT: '80108f9d-1e75-40fa-9fa9-02be4ddc1da1',\r\n DATETIME: 'ce972d02-5013-46d4-9d1d-f09df1ac346a',\r\n DATE: '83bf530c-e04c-462b-9ffc-a46f750fc072',\r\n HTML: 'ed2ad39d-32fc-4585-8f5b-2e93463f050a',\r\n TELEPHONE: '3f62f67a-1cee-403a-bec6-aa02a9804edb',\r\n NUMERIC: '6a34bfe3-fece-4da1-9136-a7b1e5ae3319',\r\n} as const;\r\n\r\n/**\r\n * Human-readable mappings for field types\r\n * Used for display purposes\r\n */\r\nexport const FIELD_TYPE_MAPPINGS: Record<string, string> = {\r\n [FIELD_TYPE_IDS.DROPDOWN]: 'Dropdown',\r\n [FIELD_TYPE_IDS.EMAIL]: 'Email',\r\n [FIELD_TYPE_IDS.TEXT]: 'Text',\r\n [FIELD_TYPE_IDS.LOOKUP]: 'Lookup',\r\n [FIELD_TYPE_IDS.URL]: 'URL',\r\n [FIELD_TYPE_IDS.LONG_TEXT]: 'Long Text',\r\n [FIELD_TYPE_IDS.DATETIME]: 'DateTime',\r\n [FIELD_TYPE_IDS.DATE]: 'Date',\r\n [FIELD_TYPE_IDS.HTML]: 'HTML',\r\n [FIELD_TYPE_IDS.TELEPHONE]: 'Telephone',\r\n [FIELD_TYPE_IDS.NUMERIC]: 'Number',\r\n};\r\n","/**\r\n * Related Field Mapping Utilities\r\n *\r\n * Functions for parsing and resolving related object field references.\r\n * Related fields use the pattern: {referenceField}_{relatedField}\r\n * Example: accountid_telephone1, contactid_fullname, accountid_statuscode\r\n */\r\n\r\nimport { OBJECT_ID_MAP } from '../constants/objectIds';\r\nimport { FIELD_TYPE_IDS } from '../constants/fieldTypes';\r\nimport { getLabelFieldForField } from './fieldMapping';\r\nimport type { FireberryField } from '../types/metadata';\r\n\r\n/**\r\n * Reverse mapping from ID field name to object type ID\r\n * Generated from OBJECT_ID_MAP\r\n */\r\nexport const ID_FIELD_TO_OBJECT_TYPE: Record<string, number> = Object.entries(OBJECT_ID_MAP).reduce(\r\n (acc, [objectTypeId, idFieldName]) => {\r\n acc[idFieldName] = parseInt(objectTypeId, 10);\r\n return acc;\r\n },\r\n {} as Record<string, number>,\r\n);\r\n\r\n/**\r\n * Information about a parsed related field\r\n */\r\nexport interface RelatedFieldInfo {\r\n /** The original field name (e.g., \"accountid_telephone1\") */\r\n originalField: string;\r\n /** The reference/lookup field (e.g., \"accountid\") */\r\n referenceField: string;\r\n /** The field on the related object (e.g., \"telephone1\") */\r\n relatedField: string;\r\n /** The object type ID of the related object (e.g., 1 for Account) */\r\n relatedObjectType: number;\r\n}\r\n\r\n/**\r\n * Result of resolving a related field with its label field(s)\r\n */\r\nexport interface RelatedFieldResolution {\r\n /** The original field name */\r\n originalField: string;\r\n /** All fields that should be selected/returned */\r\n fields: string[];\r\n /** The primary value field */\r\n valueField: string;\r\n /** The label/display field (may be same as valueField for non-code fields) */\r\n labelField: string;\r\n /** The code field if applicable (for dropdown fields) */\r\n codeField?: string;\r\n /** Whether this is a code field (ends with 'code') */\r\n isCodeField: boolean;\r\n /** Related object type ID */\r\n relatedObjectType: number;\r\n /** Field type from metadata (if available) */\r\n fieldType?: string;\r\n}\r\n\r\n/**\r\n * Field metadata map type - maps field name to field info\r\n */\r\nexport type FieldMetadataMap = Map<string, FireberryField>;\r\n\r\n/**\r\n * Gets the object type ID from a reference field name\r\n *\r\n * @param referenceField - The reference field name (e.g., \"accountid\", \"contactid\")\r\n * @returns The object type ID, or null if not a known reference field\r\n *\r\n * @example\r\n * getObjectTypeFromReferenceField('accountid') // 1\r\n * getObjectTypeFromReferenceField('contactid') // 2\r\n * getObjectTypeFromReferenceField('leadid') // 3\r\n * getObjectTypeFromReferenceField('unknown') // null\r\n */\r\nexport function getObjectTypeFromReferenceField(referenceField: string): number | null {\r\n // Check standard mappings\r\n if (ID_FIELD_TO_OBJECT_TYPE[referenceField]) {\r\n return ID_FIELD_TO_OBJECT_TYPE[referenceField];\r\n }\r\n\r\n // Check for custom object pattern: customobject{N}id\r\n const customMatch = referenceField.match(/^customobject(\\d+)id$/);\r\n if (customMatch) {\r\n return parseInt(customMatch[1], 10);\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * Parses a related field name into its components\r\n *\r\n * @param fieldName - The field name to parse (e.g., \"accountid_telephone1\")\r\n * @returns RelatedFieldInfo if it's a valid related field, null otherwise\r\n *\r\n * @example\r\n * parseRelatedField('accountid_telephone1')\r\n * // { originalField: 'accountid_telephone1', referenceField: 'accountid',\r\n * // relatedField: 'telephone1', relatedObjectType: 1 }\r\n *\r\n * parseRelatedField('accountid_statuscode')\r\n * // { originalField: 'accountid_statuscode', referenceField: 'accountid',\r\n * // relatedField: 'statuscode', relatedObjectType: 1 }\r\n *\r\n * parseRelatedField('telephone1') // null (not a related field)\r\n */\r\nexport function parseRelatedField(fieldName: string): RelatedFieldInfo | null {\r\n // Must contain underscore\r\n const underscoreIndex = fieldName.indexOf('_');\r\n if (underscoreIndex === -1) {\r\n return null;\r\n }\r\n\r\n // Try to find a valid reference field by checking progressively longer prefixes\r\n // This handles cases like pcf_accountid_status where we need to find the right split\r\n const parts = fieldName.split('_');\r\n\r\n // Start from the first part and try combinations\r\n for (let i = 0; i < parts.length - 1; i++) {\r\n const potentialRefField = parts.slice(0, i + 1).join('_');\r\n const objectType = getObjectTypeFromReferenceField(potentialRefField);\r\n\r\n if (objectType !== null) {\r\n const relatedField = parts.slice(i + 1).join('_');\r\n if (relatedField) {\r\n return {\r\n originalField: fieldName,\r\n referenceField: potentialRefField,\r\n relatedField,\r\n relatedObjectType: objectType,\r\n };\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * Determines if a field is a \"code\" field (stores internal value, has label equivalent)\r\n *\r\n * @param fieldName - The field name to check\r\n * @returns True if the field is a code field\r\n */\r\nexport function isCodeField(fieldName: string): boolean {\r\n return fieldName.endsWith('code');\r\n}\r\n\r\n/**\r\n * Gets the code field name for a label field\r\n * For example: status -> statuscode\r\n *\r\n * @param labelField - The label field name\r\n * @returns The code field name\r\n */\r\nexport function getCodeFieldFromLabel(labelField: string): string {\r\n return `${labelField}code`;\r\n}\r\n\r\n/**\r\n * Gets the label field name for a code field\r\n * For example: statuscode -> status\r\n *\r\n * @param codeField - The code field name\r\n * @returns The label field name\r\n */\r\nexport function getLabelFieldFromCode(codeField: string): string {\r\n if (codeField.endsWith('code')) {\r\n return codeField.slice(0, -4);\r\n }\r\n return codeField;\r\n}\r\n\r\n/**\r\n * Checks if a field is a dropdown type based on metadata\r\n *\r\n * @param fieldName - The field name to check\r\n * @param metadata - Field metadata map\r\n * @returns True if the field is a dropdown type\r\n */\r\nexport function isDropdownFieldByMetadata(\r\n fieldName: string,\r\n metadata: FieldMetadataMap,\r\n): boolean {\r\n const field = metadata.get(fieldName);\r\n return field?.systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n}\r\n\r\n/**\r\n * Resolver class for related fields that uses metadata to determine field types\r\n *\r\n * @example\r\n * ```typescript\r\n * // Create resolver with metadata\r\n * const fields = await client.metadata.getFields('1');\r\n * const resolver = new RelatedFieldResolver();\r\n * resolver.setMetadata(1, fields.fields);\r\n *\r\n * // Resolve a related field\r\n * const resolution = resolver.resolve('accountid_statuscode');\r\n * // Returns both statuscode and status fields for dropdown types\r\n * ```\r\n */\r\nexport class RelatedFieldResolver {\r\n private metadataByObjectType: Map<number, FieldMetadataMap> = new Map();\r\n\r\n /**\r\n * Sets field metadata for an object type\r\n *\r\n * @param objectType - The object type ID\r\n * @param fields - Array of field metadata\r\n */\r\n setMetadata(objectType: number, fields: FireberryField[]): void {\r\n const fieldMap: FieldMetadataMap = new Map();\r\n for (const field of fields) {\r\n fieldMap.set(field.fieldName, field);\r\n }\r\n this.metadataByObjectType.set(objectType, fieldMap);\r\n }\r\n\r\n /**\r\n * Gets field metadata for an object type\r\n *\r\n * @param objectType - The object type ID\r\n * @returns Field metadata map, or undefined if not loaded\r\n */\r\n getMetadata(objectType: number): FieldMetadataMap | undefined {\r\n return this.metadataByObjectType.get(objectType);\r\n }\r\n\r\n /**\r\n * Checks if metadata is loaded for an object type\r\n *\r\n * @param objectType - The object type ID\r\n * @returns True if metadata is loaded\r\n */\r\n hasMetadata(objectType: number): boolean {\r\n return this.metadataByObjectType.has(objectType);\r\n }\r\n\r\n /**\r\n * Clears all cached metadata\r\n */\r\n clearMetadata(): void {\r\n this.metadataByObjectType.clear();\r\n }\r\n\r\n /**\r\n * Resolves a related field into all necessary fields for querying and display\r\n * Uses metadata to determine if fields are dropdown types that need code/label pairs\r\n *\r\n * @param fieldName - The related field name to resolve\r\n * @param showRealValue - Whether labels are being returned (default: true).\r\n * When true, expands dropdown fields to include both code and label.\r\n * When false, returns only the requested field (no expansion needed).\r\n * @returns Resolution with all fields and metadata, or null if not a related field\r\n *\r\n * @example\r\n * // With showRealValue=true (default), dropdown fields get code/label pairs\r\n * resolver.resolve('accountid_status')\r\n * // { fields: ['accountid_status', 'accountid_statuscode'], ... }\r\n *\r\n * // With showRealValue=false, no expansion\r\n * resolver.resolve('accountid_status', false)\r\n * // { fields: ['accountid_status'], ... }\r\n *\r\n * // Regular fields stay as-is regardless of showRealValue\r\n * resolver.resolve('accountid_telephone1')\r\n * // { fields: ['accountid_telephone1'], ... }\r\n */\r\n resolve(fieldName: string, showRealValue: boolean = true): RelatedFieldResolution | null {\r\n const parsed = parseRelatedField(fieldName);\r\n if (!parsed) {\r\n return null;\r\n }\r\n\r\n const { referenceField, relatedField, relatedObjectType } = parsed;\r\n const metadata = this.metadataByObjectType.get(relatedObjectType);\r\n const fieldIsCode = isCodeField(relatedField);\r\n const fieldMeta = metadata?.get(relatedField);\r\n\r\n // When showRealValue is false, no expansion needed - just return the field as-is\r\n if (!showRealValue) {\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n isCodeField: fieldIsCode,\r\n relatedObjectType,\r\n fieldType: fieldMeta?.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // For code fields, we want to also get the label\r\n if (fieldIsCode) {\r\n const labelOnRelated = getLabelFieldForField(relatedField, relatedObjectType);\r\n const labelFieldName = labelOnRelated\r\n ? `${referenceField}_${labelOnRelated}`\r\n : `${referenceField}_${getLabelFieldFromCode(relatedField)}`;\r\n\r\n // Get field type from metadata for the code field\r\n const baseField = getLabelFieldFromCode(relatedField);\r\n const codeFieldMeta = metadata?.get(`${baseField}code`) || metadata?.get(relatedField);\r\n\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, labelFieldName],\r\n valueField: fieldName,\r\n labelField: labelFieldName,\r\n codeField: fieldName,\r\n isCodeField: true,\r\n relatedObjectType,\r\n fieldType: codeFieldMeta?.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // Input is NOT a code field - check if it's a dropdown using metadata\r\n const potentialCodeField = getCodeFieldFromLabel(relatedField);\r\n\r\n // Check if the code field exists and is a dropdown type\r\n const codeFieldMeta = metadata?.get(potentialCodeField);\r\n const isDropdown = codeFieldMeta?.systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n\r\n if (isDropdown) {\r\n // This is a label field that has a corresponding dropdown code field\r\n const codeFieldName = `${referenceField}_${potentialCodeField}`;\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, codeFieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n codeField: codeFieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n fieldType: codeFieldMeta.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // Check if the field itself is a dropdown (for fields that don't follow code pattern)\r\n const fieldIsDropdown = fieldMeta?.systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n\r\n if (fieldIsDropdown) {\r\n // The field itself is a dropdown - check if there's a code variant\r\n const codeFieldName = `${referenceField}_${potentialCodeField}`;\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, codeFieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n codeField: codeFieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n fieldType: fieldMeta.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // Regular field without code/label pairing\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n fieldType: fieldMeta?.systemFieldTypeId,\r\n };\r\n }\r\n\r\n /**\r\n * Expands a list of fields to include related field labels/codes\r\n *\r\n * @param fields - Array of field names to expand\r\n * @param showRealValue - Whether labels are being returned (default: true).\r\n * When true, expands dropdown fields to include both code and label.\r\n * When false, returns fields as-is (no expansion needed).\r\n * @returns Array of field names with related field expansions\r\n *\r\n * @example\r\n * // With showRealValue=true (default)\r\n * resolver.expandFields(['accountid_status', 'accountid_telephone1'])\r\n * // ['accountid_status', 'accountid_statuscode', 'accountid_telephone1']\r\n *\r\n * // With showRealValue=false\r\n * resolver.expandFields(['accountid_status', 'accountid_telephone1'], false)\r\n * // ['accountid_status', 'accountid_telephone1']\r\n */\r\n expandFields(fields: string[], showRealValue: boolean = true): string[] {\r\n const expanded = new Set<string>();\r\n\r\n for (const field of fields) {\r\n const resolution = this.resolve(field, showRealValue);\r\n if (resolution) {\r\n resolution.fields.forEach((f) => expanded.add(f));\r\n } else {\r\n expanded.add(field);\r\n }\r\n }\r\n\r\n return Array.from(expanded);\r\n }\r\n}\r\n\r\n/**\r\n * Resolves a related field without metadata (basic resolution)\r\n * Only handles code fields, does not detect dropdown types without metadata\r\n *\r\n * For full resolution with dropdown detection, use RelatedFieldResolver with metadata\r\n *\r\n * @param fieldName - The related field name to resolve\r\n * @returns Resolution with fields, or null if not a related field\r\n */\r\nexport function resolveRelatedField(fieldName: string): RelatedFieldResolution | null {\r\n const parsed = parseRelatedField(fieldName);\r\n if (!parsed) {\r\n return null;\r\n }\r\n\r\n const { referenceField, relatedField, relatedObjectType } = parsed;\r\n const fieldIsCode = isCodeField(relatedField);\r\n\r\n // For code fields, we always add the label\r\n if (fieldIsCode) {\r\n const labelOnRelated = getLabelFieldForField(relatedField, relatedObjectType);\r\n const labelFieldName = labelOnRelated\r\n ? `${referenceField}_${labelOnRelated}`\r\n : `${referenceField}_${getLabelFieldFromCode(relatedField)}`;\r\n\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, labelFieldName],\r\n valueField: fieldName,\r\n labelField: labelFieldName,\r\n codeField: fieldName,\r\n isCodeField: true,\r\n relatedObjectType,\r\n };\r\n }\r\n\r\n // Without metadata, we can only return the field as-is\r\n // Use RelatedFieldResolver with metadata for dropdown detection\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n };\r\n}\r\n\r\n/**\r\n * Expands a list of fields to include related field labels/codes\r\n * Basic expansion without metadata - only expands code fields\r\n *\r\n * For full expansion with dropdown detection, use RelatedFieldResolver with metadata\r\n *\r\n * @param fields - Array of field names to expand\r\n * @returns Array of field names with related field expansions\r\n */\r\nexport function expandRelatedFields(fields: string[]): string[] {\r\n const expanded = new Set<string>();\r\n\r\n for (const field of fields) {\r\n const resolution = resolveRelatedField(field);\r\n if (resolution) {\r\n resolution.fields.forEach((f) => expanded.add(f));\r\n } else {\r\n expanded.add(field);\r\n }\r\n }\r\n\r\n return Array.from(expanded);\r\n}\r\n\r\n/**\r\n * Gets all related field information for a field name\r\n * Combines parsing and basic resolution into a single call\r\n *\r\n * @param fieldName - The field name to analyze\r\n * @returns Combined info with parsing and resolution, or null if not a related field\r\n */\r\nexport function getRelatedFieldInfo(\r\n fieldName: string,\r\n): (RelatedFieldInfo & RelatedFieldResolution) | null {\r\n const parsed = parseRelatedField(fieldName);\r\n if (!parsed) {\r\n return null;\r\n }\r\n\r\n const resolved = resolveRelatedField(fieldName);\r\n if (!resolved) {\r\n return null;\r\n }\r\n\r\n return { ...parsed, ...resolved };\r\n}\r\n","/**\r\n * Fields to exclude from queries when using '*' (all fields) for specific object types\r\n * These fields cause API errors when queried\r\n */\r\nexport const EXCLUDED_FIELDS_FOR_STAR_QUERY: Record<string, string[]> = {\r\n '7': ['deletedon', 'deletedby'], // Note\r\n '8': ['deletedon', 'deletedby', 's', 'w', 'o', 't', 'description'], // Competitor\r\n '114': ['deletedon', 'deletedby'], // Calendar Resource\r\n '115': ['deletedon', 'deletedby'], // Customer Journey\r\n '116': ['deletedon', 'deletedby'], // Profile\r\n '117': ['deletedon', 'deletedby'], // Landing Page\r\n};\r\n\r\n/**\r\n * Fields to exclude from lookup relation queries across ALL object types\r\n * These fields cause API errors when queried for lookup relationships\r\n */\r\nexport const EXCLUDED_LOOKUP_FIELDS: string[] = ['deletedby', 'deletedon'];\r\n\r\n/**\r\n * Checks if a field should be excluded from star queries for a given object type\r\n *\r\n * @param objectType - The object type ID\r\n * @param fieldName - The field name to check\r\n * @returns True if the field should be excluded\r\n */\r\nexport function isExcludedFromStarQuery(objectType: string | number, fieldName: string): boolean {\r\n const objectTypeStr = String(objectType);\r\n const excludedFields = EXCLUDED_FIELDS_FOR_STAR_QUERY[objectTypeStr];\r\n return excludedFields ? excludedFields.includes(fieldName) : false;\r\n}\r\n\r\n/**\r\n * Gets the list of excluded fields for a given object type\r\n *\r\n * @param objectType - The object type ID\r\n * @returns Array of field names to exclude, or empty array if none\r\n */\r\nexport function getExcludedFieldsForStarQuery(objectType: string | number): string[] {\r\n const objectTypeStr = String(objectType);\r\n return EXCLUDED_FIELDS_FOR_STAR_QUERY[objectTypeStr] || [];\r\n}\r\n","import { FIELD_TYPE_IDS } from '../constants/fieldTypes';\r\n\r\n/**\r\n * Checks if a field is a dropdown type based on its systemFieldTypeId\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a dropdown\r\n */\r\nexport function isDropdownField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n}\r\n\r\n/**\r\n * Checks if a field is a lookup type based on its systemFieldTypeId\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a lookup\r\n */\r\nexport function isLookupField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.LOOKUP;\r\n}\r\n\r\n/**\r\n * Checks if a field is a dropdown or lookup type\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a dropdown or lookup\r\n */\r\nexport function isDropdownOrLookupField(systemFieldTypeId: string): boolean {\r\n return isDropdownField(systemFieldTypeId) || isLookupField(systemFieldTypeId);\r\n}\r\n\r\n/**\r\n * Checks if a field is a text type\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a text field\r\n */\r\nexport function isTextField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.TEXT;\r\n}\r\n\r\n/**\r\n * Checks if a field is a numeric type\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a numeric field\r\n */\r\nexport function isNumericField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.NUMERIC;\r\n}\r\n\r\n/**\r\n * Checks if a field is a date type (date or datetime)\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a date or datetime field\r\n */\r\nexport function isDateField(systemFieldTypeId: string): boolean {\r\n return (\r\n systemFieldTypeId === FIELD_TYPE_IDS.DATE ||\r\n systemFieldTypeId === FIELD_TYPE_IDS.DATETIME\r\n );\r\n}\r\n\r\n/**\r\n * Gets the field type name from its system ID\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns Human-readable field type name, or 'Unknown' if not found\r\n */\r\nexport function getFieldTypeName(systemFieldTypeId: string): string {\r\n const typeMap: Record<string, string> = {\r\n [FIELD_TYPE_IDS.DROPDOWN]: 'Dropdown',\r\n [FIELD_TYPE_IDS.LOOKUP]: 'Lookup',\r\n [FIELD_TYPE_IDS.EMAIL]: 'Email',\r\n [FIELD_TYPE_IDS.TEXT]: 'Text',\r\n [FIELD_TYPE_IDS.URL]: 'URL',\r\n [FIELD_TYPE_IDS.LONG_TEXT]: 'Long Text',\r\n [FIELD_TYPE_IDS.DATETIME]: 'DateTime',\r\n [FIELD_TYPE_IDS.DATE]: 'Date',\r\n [FIELD_TYPE_IDS.HTML]: 'HTML',\r\n [FIELD_TYPE_IDS.TELEPHONE]: 'Telephone',\r\n [FIELD_TYPE_IDS.NUMERIC]: 'Number',\r\n };\r\n return typeMap[systemFieldTypeId] || 'Unknown';\r\n}\r\n","/**\r\n * SDK Adapter for @fireberry/sdk\r\n *\r\n * This module provides utilities to enhance the Fireberry SDK with\r\n * QueryBuilder and field mapping capabilities from fireberry-api-client.\r\n *\r\n * @example\r\n * ```typescript\r\n * import FireberryClientSDK from '@fireberry/sdk/client';\r\n * import { createSDKQueryBuilder, EnhancedSDK } from 'fireberry-api-client/sdk';\r\n *\r\n * // Option 1: Use query builder factory\r\n * const sdk = new FireberryClientSDK();\r\n * await sdk.initializeContext();\r\n *\r\n * const queryBuilder = createSDKQueryBuilder(sdk);\r\n * const results = await queryBuilder(1)\r\n * .select('accountid', 'accountname', 'statuscode')\r\n * .where('statuscode').equals('1')\r\n * .execute();\r\n *\r\n * // Option 2: Use enhanced SDK wrapper\r\n * const enhanced = EnhancedSDK.create(sdk);\r\n * const results = await enhanced\r\n * .query(1)\r\n * .select('accountid', 'accountname')\r\n * .where('statuscode').equals('1')\r\n * .execute();\r\n * ```\r\n *\r\n * @packageDocumentation\r\n */\r\n\r\nimport type {\r\n FireberrySDKClient,\r\n FireberrySDKAPI,\r\n SDKQueryPayload,\r\n SDKResponseData,\r\n SDKContext,\r\n} from '../types/sdk';\r\nimport { QueryBuilder, escapeQueryValue, sanitizeQuery, type ConditionBuilder } from '../utils/queryBuilder';\r\nimport { getLabelFieldForField } from '../utils/fieldMapping';\r\nimport { getObjectIdFieldName, getNameFieldByObjectType } from '../utils/objectMapping';\r\nimport { getExcludedFieldsForStarQuery } from '../constants/excludedFields';\r\nimport {\r\n isDropdownField,\r\n isLookupField,\r\n isDropdownOrLookupField,\r\n} from '../utils/fieldTypes';\r\n\r\n// Re-export types for convenience\r\nexport type {\r\n FireberrySDKClient,\r\n FireberrySDKAPI,\r\n SDKQueryPayload,\r\n SDKResponseData,\r\n SDKContext,\r\n} from '../types/sdk';\r\n\r\n/**\r\n * Condition builder that returns SDKQueryBuilder for fluent chaining\r\n */\r\ninterface SDKConditionBuilder {\r\n equals(value: string | number): SDKQueryBuilder;\r\n notEquals(value: string | number): SDKQueryBuilder;\r\n lessThan(value: string | number): SDKQueryBuilder;\r\n greaterThan(value: string | number): SDKQueryBuilder;\r\n lessThanOrEqual(value: string | number): SDKQueryBuilder;\r\n greaterThanOrEqual(value: string | number): SDKQueryBuilder;\r\n contains(value: string): SDKQueryBuilder;\r\n notContains(value: string): SDKQueryBuilder;\r\n startsWith(value: string): SDKQueryBuilder;\r\n notStartsWith(value: string): SDKQueryBuilder;\r\n isNull(): SDKQueryBuilder;\r\n isNotNull(): SDKQueryBuilder;\r\n}\r\n\r\n/**\r\n * SDK-compatible query builder that executes via the Fireberry SDK\r\n */\r\nexport class SDKQueryBuilder {\r\n private builder: QueryBuilder;\r\n private objectTypeId: number | string;\r\n private sdk: FireberrySDKAPI;\r\n private selectedFields: string[] = [];\r\n private pageSizeValue?: number;\r\n private pageNum: number = 1;\r\n\r\n constructor(sdk: FireberrySDKAPI, objectType: number | string) {\r\n this.sdk = sdk;\r\n this.objectTypeId = objectType;\r\n this.builder = new QueryBuilder();\r\n }\r\n\r\n /**\r\n * Select fields to return\r\n * @param fields - Field names to include in results\r\n */\r\n select(...fields: string[]): this {\r\n this.selectedFields.push(...fields);\r\n return this;\r\n }\r\n\r\n /**\r\n * Select fields with their label fields automatically included\r\n * Useful for dropdown and lookup fields where you want both ID and display value\r\n * @param fields - Field names to include\r\n */\r\n selectWithLabels(...fields: string[]): this {\r\n const objectType = typeof this.objectTypeId === 'string'\r\n ? parseInt(this.objectTypeId, 10)\r\n : this.objectTypeId;\r\n\r\n for (const field of fields) {\r\n if (!this.selectedFields.includes(field)) {\r\n this.selectedFields.push(field);\r\n }\r\n const labelField = getLabelFieldForField(field, objectType);\r\n if (labelField && !this.selectedFields.includes(labelField)) {\r\n this.selectedFields.push(labelField);\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Start a WHERE condition\r\n * @param field - Field name to filter on\r\n */\r\n where(field: string): SDKConditionBuilder {\r\n const innerBuilder = this.builder.where(field);\r\n // Wrap the inner condition builder to return this SDKQueryBuilder\r\n return {\r\n equals: (value: string | number) => { innerBuilder.equals(value); return this; },\r\n notEquals: (value: string | number) => { innerBuilder.notEquals(value); return this; },\r\n lessThan: (value: string | number) => { innerBuilder.lessThan(value); return this; },\r\n greaterThan: (value: string | number) => { innerBuilder.greaterThan(value); return this; },\r\n lessThanOrEqual: (value: string | number) => { innerBuilder.lessThanOrEqual(value); return this; },\r\n greaterThanOrEqual: (value: string | number) => { innerBuilder.greaterThanOrEqual(value); return this; },\r\n contains: (value: string) => { innerBuilder.contains(value); return this; },\r\n notContains: (value: string) => { innerBuilder.notContains(value); return this; },\r\n startsWith: (value: string) => { innerBuilder.startsWith(value); return this; },\r\n notStartsWith: (value: string) => { innerBuilder.notStartsWith(value); return this; },\r\n isNull: () => { innerBuilder.isNull(); return this; },\r\n isNotNull: () => { innerBuilder.isNotNull(); return this; },\r\n };\r\n }\r\n\r\n /**\r\n * Add AND logical operator\r\n */\r\n and(): this {\r\n this.builder.and();\r\n return this;\r\n }\r\n\r\n /**\r\n * Add OR logical operator\r\n */\r\n or(): this {\r\n this.builder.or();\r\n return this;\r\n }\r\n\r\n /**\r\n * Set page size for pagination\r\n * @param size - Number of records per page\r\n */\r\n pageSize(size: number): this {\r\n this.pageSizeValue = size;\r\n return this;\r\n }\r\n\r\n /**\r\n * Set page number for pagination\r\n * @param page - Page number (1-based)\r\n */\r\n page(page: number): this {\r\n this.pageNum = page;\r\n return this;\r\n }\r\n\r\n /**\r\n * Build the query payload without executing\r\n * Useful if you want to modify the payload before sending\r\n */\r\n toQueryPayload(): SDKQueryPayload {\r\n const payload: SDKQueryPayload = {\r\n fields: this.selectedFields.length > 0\r\n ? this.selectedFields.join(',')\r\n : '*',\r\n query: this.builder.build(),\r\n };\r\n\r\n if (this.pageSizeValue !== undefined) {\r\n payload.page_size = this.pageSizeValue;\r\n }\r\n\r\n if (this.pageNum > 1) {\r\n payload.page_number = this.pageNum;\r\n }\r\n\r\n return payload;\r\n }\r\n\r\n /**\r\n * Execute the query via the SDK\r\n */\r\n async execute<T = Record<string, unknown>>(): Promise<SDKResponseData<T>> {\r\n const payload = this.toQueryPayload();\r\n return this.sdk.query(this.objectTypeId, payload) as Promise<SDKResponseData<T>>;\r\n }\r\n}\r\n\r\n/**\r\n * Creates a query builder factory bound to a Fireberry SDK instance\r\n *\r\n * @param sdk - Fireberry SDK client or API instance\r\n * @returns Factory function that creates SDKQueryBuilder instances\r\n *\r\n * @example\r\n * ```typescript\r\n * import FireberryClientSDK from '@fireberry/sdk/client';\r\n * import { createSDKQueryBuilder } from 'fireberry-api-client/sdk';\r\n *\r\n * const sdk = new FireberryClientSDK();\r\n * await sdk.initializeContext();\r\n *\r\n * const queryBuilder = createSDKQueryBuilder(sdk);\r\n *\r\n * // Query accounts where status is active\r\n * const results = await queryBuilder(1) // 1 = Account object type\r\n * .select('accountid', 'accountname', 'statuscode', 'status')\r\n * .where('statuscode').equals('1')\r\n * .pageSize(50)\r\n * .execute();\r\n * ```\r\n */\r\nexport function createSDKQueryBuilder(\r\n sdk: FireberrySDKClient | FireberrySDKAPI\r\n): (objectType: number | string) => SDKQueryBuilder {\r\n const api = 'api' in sdk ? sdk.api : sdk;\r\n\r\n return (objectType: number | string) => new SDKQueryBuilder(api, objectType);\r\n}\r\n\r\n/**\r\n * Enhanced SDK wrapper that combines Fireberry SDK with utility functions\r\n * Provides a more feature-rich API for working with Fireberry data\r\n */\r\nexport class EnhancedSDK<TData = Record<string, unknown>> {\r\n private sdk: FireberrySDKClient<TData>;\r\n\r\n private constructor(sdk: FireberrySDKClient<TData>) {\r\n this.sdk = sdk;\r\n }\r\n\r\n /**\r\n * Create an EnhancedSDK wrapper around an existing SDK instance\r\n * The SDK should already be initialized with context\r\n *\r\n * @param sdk - Initialized Fireberry SDK client\r\n */\r\n static create<T = Record<string, unknown>>(\r\n sdk: FireberrySDKClient<T>\r\n ): EnhancedSDK<T> {\r\n return new EnhancedSDK(sdk);\r\n }\r\n\r\n /**\r\n * Get the current context (user and record info)\r\n */\r\n get context(): SDKContext | null {\r\n return this.sdk.context;\r\n }\r\n\r\n /**\r\n * Get the underlying SDK API for direct access\r\n */\r\n get api(): FireberrySDKAPI<TData> {\r\n return this.sdk.api;\r\n }\r\n\r\n /**\r\n * Get the current user ID from context\r\n */\r\n get userId(): string | undefined {\r\n return this.context?.user?.id;\r\n }\r\n\r\n /**\r\n * Get the current user's full name from context\r\n */\r\n get userFullName(): string | undefined {\r\n return this.context?.user?.fullName;\r\n }\r\n\r\n /**\r\n * Get the current record ID from context\r\n */\r\n get recordId(): string | undefined {\r\n return this.context?.record?.id;\r\n }\r\n\r\n /**\r\n * Get the current record's object type from context\r\n */\r\n get recordType(): number | undefined {\r\n return this.context?.record?.type;\r\n }\r\n\r\n /**\r\n * Start building a query for an object type\r\n *\r\n * @param objectType - Object type ID (e.g., 1 for Account, 2 for Contact)\r\n * @returns SDKQueryBuilder for fluent query construction\r\n *\r\n * @example\r\n * ```typescript\r\n * const results = await enhanced\r\n * .query(1)\r\n * .select('accountid', 'accountname')\r\n * .where('ownerid').equals(enhanced.userId)\r\n * .execute();\r\n * ```\r\n */\r\n query(objectType: number | string): SDKQueryBuilder {\r\n // Cast is safe - SDKQueryBuilder only uses the query method which has the same signature\r\n return new SDKQueryBuilder(this.sdk.api as FireberrySDKAPI, objectType);\r\n }\r\n\r\n /**\r\n * Get the primary key field name for an object type\r\n *\r\n * @param objectType - Object type ID\r\n * @returns Primary key field name (e.g., 'accountid' for type 1)\r\n *\r\n * @example\r\n * ```typescript\r\n * const idField = enhanced.getIdField(1); // 'accountid'\r\n * const idField = enhanced.getIdField(2); // 'contactid'\r\n * ```\r\n */\r\n getIdField(objectType: number | string): string {\r\n return getObjectIdFieldName(objectType);\r\n }\r\n\r\n /**\r\n * Get the display name field for an object type\r\n *\r\n * @param objectType - Object type ID\r\n * @returns Name field (e.g., 'accountname' for type 1, 'fullname' for type 2)\r\n *\r\n * @example\r\n * ```typescript\r\n * const nameField = enhanced.getNameField(1); // 'accountname'\r\n * const nameField = enhanced.getNameField(2); // 'fullname'\r\n * ```\r\n */\r\n getNameField(objectType: number | string): string {\r\n return getNameFieldByObjectType(objectType);\r\n }\r\n\r\n /**\r\n * Get the label field for a dropdown or lookup field\r\n *\r\n * @param fieldName - API field name\r\n * @param objectType - Object type ID\r\n * @returns Label field name, or empty string if not applicable\r\n *\r\n * @example\r\n * ```typescript\r\n * const labelField = enhanced.getLabelField('statuscode', 1); // 'status'\r\n * const labelField = enhanced.getLabelField('ownerid', 1); // 'ownername'\r\n * ```\r\n */\r\n getLabelField(fieldName: string, objectType: number | string): string {\r\n return getLabelFieldForField(fieldName, objectType);\r\n }\r\n\r\n /**\r\n * Get fields that should be excluded from * (star) queries for an object type\r\n * Some fields cause API errors when included in broad queries\r\n *\r\n * @param objectType - Object type ID\r\n * @returns Array of field names to exclude\r\n */\r\n getExcludedFields(objectType: number | string): string[] {\r\n return getExcludedFieldsForStarQuery(objectType);\r\n }\r\n\r\n /**\r\n * Expand field list to include label fields for dropdowns and lookups\r\n *\r\n * @param fields - Original field list\r\n * @param objectType - Object type ID\r\n * @returns Expanded field list with label fields\r\n *\r\n * @example\r\n * ```typescript\r\n * const fields = enhanced.expandFieldsWithLabels(['statuscode', 'ownerid'], 1);\r\n * // Returns: ['statuscode', 'status', 'ownerid', 'ownername']\r\n * ```\r\n */\r\n expandFieldsWithLabels(fields: string[], objectType: number | string): string[] {\r\n const result: string[] = [];\r\n for (const field of fields) {\r\n if (!result.includes(field)) {\r\n result.push(field);\r\n }\r\n const labelField = getLabelFieldForField(field, objectType);\r\n if (labelField && !result.includes(labelField)) {\r\n result.push(labelField);\r\n }\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Create a record\r\n *\r\n * @param objectType - Object type ID\r\n * @param data - Record data\r\n */\r\n create<T extends Record<string, unknown>>(\r\n objectType: number | string,\r\n data: T\r\n ): Promise<SDKResponseData<TData>> {\r\n return this.sdk.api.create(objectType, data);\r\n }\r\n\r\n /**\r\n * Update a record\r\n *\r\n * @param objectType - Object type ID\r\n * @param recordId - Record ID to update\r\n * @param data - Updated field values\r\n */\r\n update<T extends Record<string, unknown>>(\r\n objectType: number | string,\r\n recordId: string,\r\n data: T\r\n ): Promise<SDKResponseData<TData>> {\r\n return this.sdk.api.update(objectType, recordId, data);\r\n }\r\n\r\n /**\r\n * Delete a record\r\n *\r\n * @param objectType - Object type ID\r\n * @param recordId - Record ID to delete\r\n */\r\n delete(\r\n objectType: number | string,\r\n recordId: string\r\n ): Promise<SDKResponseData<TData>> {\r\n return this.sdk.api.delete(objectType, recordId);\r\n }\r\n\r\n /**\r\n * Clean up the underlying SDK\r\n */\r\n destroy(): void {\r\n this.sdk.destroy();\r\n }\r\n}\r\n\r\n// Re-export utilities that work standalone with the SDK\r\nexport {\r\n // Query utilities\r\n QueryBuilder,\r\n escapeQueryValue,\r\n sanitizeQuery,\r\n type ConditionBuilder,\r\n // Field utilities\r\n getLabelFieldForField,\r\n getObjectIdFieldName,\r\n getNameFieldByObjectType,\r\n getExcludedFieldsForStarQuery,\r\n // Type detection\r\n isDropdownField,\r\n isLookupField,\r\n isDropdownOrLookupField,\r\n};\r\n\r\n// Re-export constants\r\nexport {\r\n FIELD_TYPE_IDS,\r\n FIELD_TYPE_MAPPINGS,\r\n OBJECT_ID_MAP,\r\n OBJECT_NAME_MAP,\r\n EXCLUDED_FIELDS_FOR_STAR_QUERY,\r\n} from '../constants';\r\n"]}
1
+ {"version":3,"sources":["../../src/constants/objectIds.ts","../../src/utils/queryBuilder.ts","../../src/utils/fieldMapping.ts","../../src/constants/objectNames.ts","../../src/constants/fieldTypes.ts","../../src/utils/relatedFieldMapping.ts","../../src/constants/excludedFields.ts","../../src/utils/fieldTypes.ts","../../src/sdk/index.ts"],"names":[],"mappings":";AAKO,IAAM,aAAA,GAAwC;AAAA,EACnD,CAAA,EAAG,WAAA;AAAA;AAAA,EACH,CAAA,EAAG,WAAA;AAAA;AAAA,EACH,CAAA,EAAG,QAAA;AAAA;AAAA,EACH,CAAA,EAAG,eAAA;AAAA;AAAA,EACH,CAAA,EAAG,SAAA;AAAA;AAAA,EACH,CAAA,EAAG,YAAA;AAAA;AAAA,EACH,CAAA,EAAG,QAAA;AAAA;AAAA,EACH,CAAA,EAAG,cAAA;AAAA;AAAA,EACH,CAAA,EAAG,WAAA;AAAA;AAAA,EACH,EAAA,EAAI,QAAA;AAAA;AAAA,EACJ,EAAA,EAAI,SAAA;AAAA;AAAA,EACJ,EAAA,EAAI,YAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,gBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,iBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,gBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA;AAAA,EACJ,EAAA,EAAI,iBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,YAAA;AAAA;AAAA,EACJ,EAAA,EAAI,kBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,UAAA;AAAA;AAAA,EACJ,EAAA,EAAI,YAAA;AAAA;AAAA,EACJ,EAAA,EAAI,QAAA;AAAA;AAAA,EACJ,EAAA,EAAI,YAAA;AAAA;AAAA,EACJ,EAAA,EAAI,gBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,QAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,sBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,iBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,mBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,UAAA;AAAA;AAAA,EACJ,GAAA,EAAK,WAAA;AAAA;AAAA,EACL,GAAA,EAAK,mBAAA;AAAA;AAAA,EACL,GAAA,EAAK,eAAA;AAAA;AAAA,EACL,GAAA,EAAK,gBAAA;AAAA;AAAA,EACL,GAAA,EAAK,aAAA;AAAA;AAAA,EACL,GAAA,EAAK,gBAAA;AAAA;AAAA,EACL,GAAA,EAAK,sBAAA;AAAA;AAAA,EACL,GAAA,EAAK,YAAA;AAAA;AAAA,EACL,GAAA,EAAK,eAAA;AAAA;AAAA,EACL,GAAA,EAAK,wBAAA;AAAA;AAAA,EACL,GAAA,EAAK,oBAAA;AAAA;AAAA,EACL,GAAA,EAAK,WAAA;AAAA;AAAA,EACL,GAAA,EAAK,WAAA;AAAA;AAAA,EACL,GAAA,EAAK;AAAA;AACP;AAQO,SAAS,qBAAqB,YAAA,EAAuC;AAC1E,EAAA,MAAM,gBACJ,OAAO,YAAA,KAAiB,WAAW,QAAA,CAAS,YAAA,EAAc,EAAE,CAAA,GAAI,YAAA;AAGlE,EAAA,IAAI,aAAA,CAAc,aAAa,CAAA,EAAG;AAChC,IAAA,OAAO,cAAc,aAAa,CAAA;AAAA,EACpC;AAGA,EAAA,IAAI,iBAAiB,GAAA,EAAM;AACzB,IAAA,OAAO,eAAe,aAAa,CAAA,EAAA,CAAA;AAAA,EACrC;AAGA,EAAA,OAAO,IAAA;AACT;;;AC7EA,IAAM,eAAA,GAAkB,qBAAA;AAejB,SAAS,WAAW,KAAA,EAAwB;AACjD,EAAA,OAAO,eAAA,CAAgB,KAAK,KAAK,CAAA;AACnC;AAeO,SAAS,OAAA,CAAQ,SAAiB,IAAA,EAAsB;AAE7D,EAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,KAAA,CAAM,OAAO,EAAE,CAAC,CAAA;AACzC,EAAA,MAAM,IAAA,mBAAO,IAAI,IAAA,CAAK,QAAA,GAAW,WAAW,CAAA;AAC5C,EAAA,IAAA,CAAK,OAAA,CAAQ,IAAA,CAAK,OAAA,EAAQ,GAAI,IAAI,CAAA;AAElC,EAAA,MAAM,IAAA,GAAO,KAAK,WAAA,EAAY;AAC9B,EAAA,MAAM,KAAA,GAAQ,OAAO,IAAA,CAAK,QAAA,KAAa,CAAC,CAAA,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AACzD,EAAA,MAAM,GAAA,GAAM,OAAO,IAAA,CAAK,OAAA,EAAS,CAAA,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AAElD,EAAA,OAAO,CAAA,EAAG,IAAI,CAAA,CAAA,EAAI,KAAK,IAAI,GAAG,CAAA,CAAA;AAChC;AAUO,SAAS,iBAAiB,KAAA,EAAuB;AACtD,EAAA,IAAI,CAAC,KAAA,EAAO;AACV,IAAA,OAAO,EAAA;AAAA,EACT;AAEA,EAAA,IAAI,OAAA,GAAU,KAAA,CAAM,OAAA,CAAQ,KAAA,EAAO,MAAM,CAAA;AAEzC,EAAA,OAAA,GAAU,OAAA,CAAQ,OAAA,CAAQ,KAAA,EAAO,KAAK,CAAA;AACtC,EAAA,OAAA,GAAU,OAAA,CAAQ,OAAA,CAAQ,KAAA,EAAO,KAAK,CAAA;AAGtC,EAAA,OAAA,GAAU,OAAA,CAAQ,OAAA,CAAQ,UAAA,EAAY,MAAM,CAAA;AAC5C,EAAA,OAAA,GAAU,OAAA,CAAQ,OAAA,CAAQ,WAAA,EAAa,OAAO,CAAA;AAC9C,EAAA,OAAO,OAAA;AACT;AASO,SAAS,cAAc,KAAA,EAAuB;AACnD,EAAA,IAAI,CAAC,KAAA,EAAO;AACV,IAAA,OAAO,EAAA;AAAA,EACT;AAIA,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA;AAAA,IACZ,oDAAA;AAAA,IACA;AAAA,GACF;AAGA,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA;AAAA,IACZ,oEAAA;AAAA,IACA;AAAA,GACF;AAGA,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA;AAAA,IACZ,wHAAA;AAAA,IACA;AAAA,GACF;AAGA,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA;AAAA,IACZ,oGAAA;AAAA,IACA;AAAA,GACF;AAGA,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,uBAAA,EAAyB,EAAE,CAAA;AACjD,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,oBAAA,EAAsB,EAAE,CAAA;AAG9C,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,+BAAA,EAAiC,EAAE,CAAA;AAEzD,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,2CAAA,EAA6C,EAAE,CAAA;AAErE,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,wBAAA,EAA0B,EAAE,CAAA;AAElD,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,UAAA,EAAY,EAAE,CAAA;AAEpC,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,mBAAA,EAAqB,EAAE,CAAA;AAC7C,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,mBAAA,EAAqB,EAAE,CAAA;AAE7C,EAAA,MAAM,aAAA,GAAgB,yBAAA;AACtB,EAAA,OAAO,aAAA,CAAc,IAAA,CAAK,KAAK,CAAA,EAAG;AAChC,IAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,aAAA,EAAe,MAAM,CAAA;AAAA,EAC7C;AAEA,EAAA,KAAA,GAAQ,KAAA,CAAM,OAAA,CAAQ,MAAA,EAAQ,GAAG,CAAA;AACjC,EAAA,OAAO,MAAM,IAAA,EAAK;AACpB;AAKO,SAAS,QAAA,GAAmB;AACjC,EAAA,MAAM,GAAA,uBAAU,IAAA,EAAK;AACrB,EAAA,MAAM,IAAA,GAAO,IAAI,WAAA,EAAY;AAC7B,EAAA,MAAM,KAAA,GAAQ,OAAO,GAAA,CAAI,QAAA,KAAa,CAAC,CAAA,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AACxD,EAAA,MAAM,GAAA,GAAM,OAAO,GAAA,CAAI,OAAA,EAAS,CAAA,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACjD,EAAA,OAAO,CAAA,EAAG,IAAI,CAAA,CAAA,EAAI,KAAK,IAAI,GAAG,CAAA,CAAA;AAChC;AAKO,SAAS,cAAA,GAAyB;AACvC,EAAA,MAAM,GAAA,uBAAU,IAAA,EAAK;AACrB,EAAA,MAAM,GAAA,GAAM,IAAI,MAAA,EAAO;AAEvB,EAAA,MAAM,OAAO,GAAA,CAAI,OAAA,KAAY,GAAA,IAAO,GAAA,KAAQ,IAAI,EAAA,GAAK,CAAA,CAAA;AACrD,EAAA,MAAM,MAAA,GAAS,IAAI,IAAA,CAAK,GAAG,CAAA;AAC3B,EAAA,MAAA,CAAO,QAAQ,IAAI,CAAA;AACnB,EAAA,MAAM,IAAA,GAAO,OAAO,WAAA,EAAY;AAChC,EAAA,MAAM,KAAA,GAAQ,OAAO,MAAA,CAAO,QAAA,KAAa,CAAC,CAAA,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AAC3D,EAAA,MAAM,MAAA,GAAS,OAAO,MAAA,CAAO,OAAA,EAAS,CAAA,CAAE,QAAA,CAAS,GAAG,GAAG,CAAA;AACvD,EAAA,OAAO,CAAA,EAAG,IAAI,CAAA,CAAA,EAAI,KAAK,IAAI,MAAM,CAAA,CAAA;AACnC;AAKO,SAAS,eAAA,GAA0B;AACxC,EAAA,MAAM,GAAA,uBAAU,IAAA,EAAK;AACrB,EAAA,MAAM,IAAA,GAAO,IAAI,WAAA,EAAY;AAC7B,EAAA,MAAM,KAAA,GAAQ,OAAO,GAAA,CAAI,QAAA,KAAa,CAAC,CAAA,CAAE,QAAA,CAAS,CAAA,EAAG,GAAG,CAAA;AACxD,EAAA,OAAO,CAAA,EAAG,IAAI,CAAA,CAAA,EAAI,KAAK,CAAA,GAAA,CAAA;AACzB;AAmHO,IAAM,eAAN,MAAmB;AAAA,EAChB,aAA+B,EAAC;AAAA,EAChC,gBAAkC,EAAC;AAAA,EACnC,YAAA,GAA8B,IAAA;AAAA,EAC9B,iBAA2B,EAAC;AAAA,EAC5B,YAAA,GAA8B,IAAA;AAAA,EAC9B,WAAA,GAA6B,IAAA;AAAA,EAC7B,aAAA,GAAgC,MAAA;AAAA,EAChC,UAAA,GAA4B,IAAA;AAAA,EAC5B,UAAA,GAAqB,CAAA;AAAA,EACrB,iBAAA,GAA6B,IAAA;AAAA,EAC7B,MAAA,GAA6B,IAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAMrC,YAAY,MAAA,EAAsB;AAChC,IAAA,IAAA,CAAK,SAAS,MAAA,IAAU,IAAA;AAAA,EAC1B;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,WAAW,UAAA,EAAmC;AAC5C,IAAA,IAAA,CAAK,YAAA,GAAe,OAAO,UAAU,CAAA;AACrC,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,UAAU,MAAA,EAAwB;AAChC,IAAA,IAAA,CAAK,cAAA,CAAe,IAAA,CAAK,GAAG,MAAM,CAAA;AAClC,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAM,KAAA,EAAiC;AACrC,IAAA,IAAA,CAAK,YAAA,GAAe,KAAA;AACpB,IAAA,OAAO,KAAK,sBAAA,EAAuB;AAAA,EACrC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAqBA,UAAU,KAAA,EAAqC;AAC7C,IAAA,OAAO,IAAA,CAAK,2BAA2B,KAAK,CAAA;AAAA,EAC9C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsBA,QAAQ,KAAA,EAA8B;AACpC,IAAA,IAAI,CAAC,KAAK,YAAA,EAAc;AACtB,MAAA,MAAM,IAAI,MAAM,2EAA2E,CAAA;AAAA,IAC7F;AACA,IAAA,MAAM,OAAA,GAAU,oBAAA,CAAqB,IAAA,CAAK,YAAY,CAAA;AACtD,IAAA,IAAA,CAAK,YAAA,CAAa,OAAA,EAAS,GAAA,EAAK,MAAA,CAAO,KAAK,CAAC,CAAA;AAC7C,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EA2BA,SAAS,MAAA,EAAmC;AAC1C,IAAA,IAAI,CAAC,KAAK,YAAA,EAAc;AACtB,MAAA,MAAM,IAAI,MAAM,4EAA4E,CAAA;AAAA,IAC9F;AACA,IAAA,IAAI,CAAC,MAAA,IAAU,MAAA,CAAO,MAAA,KAAW,CAAA,EAAG;AAClC,MAAA,MAAM,IAAI,MAAM,4CAA4C,CAAA;AAAA,IAC9D;AACA,IAAA,MAAM,OAAA,GAAU,oBAAA,CAAqB,IAAA,CAAK,YAAY,CAAA;AAGtD,IAAA,IAAA,CAAK,aAAa,OAAA,EAAS,GAAA,EAAK,OAAO,MAAA,CAAO,CAAC,CAAC,CAAC,CAAA;AAGjD,IAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,MAAA,CAAO,QAAQ,CAAA,EAAA,EAAK;AACtC,MAAA,IAAA,CAAK,aAAA,CAAc,KAAK,IAAI,CAAA;AAC5B,MAAA,IAAA,CAAK,aAAa,OAAA,EAAS,GAAA,EAAK,OAAO,MAAA,CAAO,CAAC,CAAC,CAAC,CAAA;AAAA,IACnD;AAEA,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAmBA,OAAA,CAAQ,OAAe,MAAA,EAAmC;AACxD,IAAA,IAAI,CAAC,MAAA,IAAU,MAAA,CAAO,MAAA,KAAW,CAAA,EAAG;AAClC,MAAA,MAAM,IAAI,MAAM,wCAAwC,CAAA;AAAA,IAC1D;AAEA,IAAA,IAAA,CAAK,aAAa,KAAA,EAAO,GAAA,EAAK,OAAO,MAAA,CAAO,CAAC,CAAC,CAAC,CAAA;AAE/C,IAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,MAAA,CAAO,QAAQ,CAAA,EAAA,EAAK;AACtC,MAAA,IAAA,CAAK,aAAA,CAAc,KAAK,IAAI,CAAA;AAC5B,MAAA,IAAA,CAAK,aAAa,KAAA,EAAO,GAAA,EAAK,OAAO,MAAA,CAAO,CAAC,CAAC,CAAC,CAAA;AAAA,IACjD;AACA,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,GAAA,GAAY;AACV,IAAA,IAAI,IAAA,CAAK,UAAA,CAAW,MAAA,GAAS,CAAA,EAAG;AAC9B,MAAA,IAAA,CAAK,aAAA,CAAc,KAAK,KAAK,CAAA;AAAA,IAC/B;AACA,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,EAAA,GAAW;AACT,IAAA,IAAI,IAAA,CAAK,UAAA,CAAW,MAAA,GAAS,CAAA,EAAG;AAC9B,MAAA,IAAA,CAAK,aAAA,CAAc,KAAK,IAAI,CAAA;AAAA,IAC9B;AACA,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAA,CAAO,KAAA,EAAe,SAAA,GAA4B,MAAA,EAAc;AAC9D,IAAA,IAAA,CAAK,WAAA,GAAc,KAAA;AACnB,IAAA,IAAA,CAAK,aAAA,GAAgB,SAAA;AACrB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAM,KAAA,EAAqB;AACzB,IAAA,IAAA,CAAK,UAAA,GAAa,KAAA;AAClB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,KAAK,IAAA,EAAoB;AACvB,IAAA,IAAA,CAAK,UAAA,GAAa,IAAA;AAClB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,cAAc,IAAA,EAAqB;AACjC,IAAA,IAAA,CAAK,iBAAA,GAAoB,IAAA;AACzB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,KAAA,GAAgB;AACd,IAAA,IAAI,IAAA,CAAK,UAAA,CAAW,MAAA,KAAW,CAAA,EAAG;AAChC,MAAA,OAAO,EAAA;AAAA,IACT;AAEA,IAAA,MAAM,QAAkB,EAAC;AAEzB,IAAA,KAAA,IAAS,IAAI,CAAA,EAAG,CAAA,GAAI,IAAA,CAAK,UAAA,CAAW,QAAQ,CAAA,EAAA,EAAK;AAC/C,MAAA,MAAM,SAAA,GAAY,IAAA,CAAK,UAAA,CAAW,CAAC,CAAA;AACnC,MAAA,IAAI,YAAA;AAEJ,MAAA,IAAI,SAAA,CAAU,QAAA,KAAa,SAAA,IAAa,SAAA,CAAU,aAAa,aAAA,EAAe;AAC5E,QAAA,YAAA,GAAe,CAAA,CAAA,EAAI,SAAA,CAAU,KAAK,CAAA,CAAA,EAAI,UAAU,QAAQ,CAAA,CAAA,CAAA;AAAA,MAC1D,CAAA,MAAO;AACL,QAAA,MAAM,YAAA,GAAe,gBAAA,CAAiB,SAAA,CAAU,KAAA,IAAS,EAAE,CAAA;AAC3D,QAAA,YAAA,GAAe,IAAI,SAAA,CAAU,KAAK,IAAI,SAAA,CAAU,QAAQ,IAAI,YAAY,CAAA,CAAA,CAAA;AAAA,MAC1E;AAEA,MAAA,KAAA,CAAM,KAAK,YAAY,CAAA;AAGvB,MAAA,IAAI,CAAA,GAAI,IAAA,CAAK,aAAA,CAAc,MAAA,EAAQ;AACjC,QAAA,KAAA,CAAM,IAAA,CAAK,IAAA,CAAK,aAAA,CAAc,CAAC,CAAC,CAAA;AAAA,MAClC;AAAA,IACF;AAEA,IAAA,OAAO,KAAA,CAAM,KAAK,GAAG,CAAA;AAAA,EACvB;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,SAAA,GAAsB;AACpB,IAAA,OAAO,CAAC,GAAG,IAAA,CAAK,cAAc,CAAA;AAAA,EAChC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAwBA,YAAA,GAA4F;AAC1F,IAAA,MAAM,OAAA,GAAuF;AAAA,MAC3F,MAAA,EAAQ,KAAK,cAAA,CAAe,MAAA,GAAS,IAAI,IAAA,CAAK,cAAA,CAAe,IAAA,CAAK,GAAG,CAAA,GAAI,GAAA;AAAA,MACzE,KAAA,EAAO,KAAK,KAAA;AAAM,KACpB;AAEA,IAAA,IAAI,IAAA,CAAK,eAAe,IAAA,EAAM;AAC5B,MAAA,OAAA,CAAQ,YAAY,IAAA,CAAK,UAAA;AAAA,IAC3B;AAEA,IAAA,IAAI,IAAA,CAAK,aAAa,CAAA,EAAG;AACvB,MAAA,OAAA,CAAQ,cAAc,IAAA,CAAK,UAAA;AAAA,IAC7B;AAEA,IAAA,OAAO,OAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAkBA,MAAM,MAAM,MAAA,EAAuC;AACjD,IAAA,IAAI,CAAC,KAAK,MAAA,EAAQ;AAChB,MAAA,MAAM,IAAI,MAAM,+FAA+F,CAAA;AAAA,IACjH;AAEA,IAAA,IAAI,CAAC,KAAK,YAAA,EAAc;AACtB,MAAA,MAAM,IAAI,MAAM,8DAA8D,CAAA;AAAA,IAChF;AAGA,IAAA,MAAM,OAAA,GAAU,oBAAA,CAAqB,IAAA,CAAK,YAAY,CAAA;AAEtD,IAAA,MAAM,MAAA,GAAS,MAAM,IAAA,CAAK,MAAA,CAAO,KAAA,CAAM;AAAA,MACrC,YAAY,IAAA,CAAK,YAAA;AAAA,MACjB,MAAA,EAAQ,CAAC,OAAO,CAAA;AAAA,MAChB,KAAA,EAAO,KAAK,KAAA,EAAM;AAAA,MAClB,aAAA,EAAe,KAAA;AAAA;AAAA,MACf;AAAA,KACD,CAAA;AAED,IAAA,OAAO,MAAA,CAAO,KAAA;AAAA,EAChB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAoBA,MAAM,MAAM,MAAA,EAA+D;AAEzE,IAAA,MAAM,gBAAgB,IAAA,CAAK,UAAA;AAC3B,IAAA,IAAA,CAAK,UAAA,GAAa,CAAA;AAElB,IAAA,IAAI;AACF,MAAA,MAAM,MAAA,GAAS,MAAM,IAAA,CAAK,OAAA,CAAQ,MAAM,CAAA;AACxC,MAAA,OAAO,MAAA,CAAO,OAAA,CAAQ,CAAC,CAAA,IAAK,IAAA;AAAA,IAC9B,CAAA,SAAE;AAEA,MAAA,IAAA,CAAK,UAAA,GAAa,aAAA;AAAA,IACpB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,MAAM,QAAQ,MAAA,EAA4C;AACxD,IAAA,IAAI,CAAC,KAAK,MAAA,EAAQ;AAChB,MAAA,MAAM,IAAI,MAAM,+FAA+F,CAAA;AAAA,IACjH;AAEA,IAAA,IAAI,CAAC,KAAK,YAAA,EAAc;AACtB,MAAA,MAAM,IAAI,MAAM,8DAA8D,CAAA;AAAA,IAChF;AAEA,IAAA,MAAM,YAAA,GAAoD;AAAA,MACxD,YAAY,IAAA,CAAK,YAAA;AAAA,MACjB,MAAA,EAAQ,KAAK,cAAA,CAAe,MAAA,GAAS,IAAI,IAAA,CAAK,cAAA,GAAiB,CAAC,GAAG,CAAA;AAAA,MACnE,KAAA,EAAO,KAAK,KAAA,EAAM;AAAA,MAClB,eAAe,IAAA,CAAK;AAAA,KACtB;AAEA,IAAA,IAAI,KAAK,WAAA,EAAa;AACpB,MAAA,YAAA,CAAa,SAAS,IAAA,CAAK,WAAA;AAC3B,MAAA,YAAA,CAAa,WAAW,IAAA,CAAK,aAAA;AAAA,IAC/B;AAEA,IAAA,IAAI,IAAA,CAAK,eAAe,IAAA,EAAM;AAC5B,MAAA,YAAA,CAAa,QAAQ,IAAA,CAAK,UAAA;AAAA,IAC5B;AAEA,IAAA,IAAI,IAAA,CAAK,aAAa,CAAA,EAAG;AACvB,MAAA,YAAA,CAAa,OAAO,IAAA,CAAK,UAAA;AAAA,IAC3B;AAEA,IAAA,IAAI,MAAA,EAAQ;AACV,MAAA,YAAA,CAAa,MAAA,GAAS,MAAA;AAAA,IACxB;AAEA,IAAA,OAAO,IAAA,CAAK,MAAA,CAAO,KAAA,CAAM,YAAY,CAAA;AAAA,EACvC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsBA,MAAM,iBAAiB,MAAA,EAAwD;AAC7E,IAAA,IAAI,CAAC,KAAK,MAAA,EAAQ;AAChB,MAAA,MAAM,IAAI,MAAM,+FAA+F,CAAA;AAAA,IACjH;AAEA,IAAA,IAAI,CAAC,KAAK,YAAA,EAAc;AACtB,MAAA,MAAM,IAAI,MAAM,8DAA8D,CAAA;AAAA,IAChF;AAEA,IAAA,MAAM,SAAA,GAAY,KAAK,GAAA,EAAI;AAC3B,IAAA,MAAM,MAAA,GAAS,KAAK,cAAA,CAAe,MAAA,GAAS,IAAI,IAAA,CAAK,cAAA,GAAiB,CAAC,GAAG,CAAA;AAC1E,IAAA,MAAM,WAAA,GAAc,KAAK,KAAA,EAAM;AAE/B,IAAA,MAAM,YAAA,GAAoD;AAAA,MACxD,YAAY,IAAA,CAAK,YAAA;AAAA,MACjB,MAAA;AAAA,MACA,KAAA,EAAO,WAAA;AAAA,MACP,eAAe,IAAA,CAAK;AAAA,KACtB;AAEA,IAAA,IAAI,KAAK,WAAA,EAAa;AACpB,MAAA,YAAA,CAAa,SAAS,IAAA,CAAK,WAAA;AAC3B,MAAA,YAAA,CAAa,WAAW,IAAA,CAAK,aAAA;AAAA,IAC/B;AAEA,IAAA,IAAI,IAAA,CAAK,eAAe,IAAA,EAAM;AAC5B,MAAA,YAAA,CAAa,QAAQ,IAAA,CAAK,UAAA;AAAA,IAC5B;AAEA,IAAA,IAAI,IAAA,CAAK,aAAa,CAAA,EAAG;AACvB,MAAA,YAAA,CAAa,OAAO,IAAA,CAAK,UAAA;AAAA,IAC3B;AAEA,IAAA,IAAI,MAAA,EAAQ;AACV,MAAA,YAAA,CAAa,MAAA,GAAS,MAAA;AAAA,IACxB;AAEA,IAAA,MAAM,MAAA,GAAS,MAAM,IAAA,CAAK,MAAA,CAAO,MAAM,YAAY,CAAA;AACnD,IAAA,MAAM,eAAA,GAAkB,IAAA,CAAK,GAAA,EAAI,GAAI,SAAA;AAErC,IAAA,MAAM,QAAA,GAA0B;AAAA,MAC9B,YAAY,IAAA,CAAK,YAAA;AAAA,MACjB,MAAA;AAAA,MACA,WAAA;AAAA,MACA,YAAY,IAAA,CAAK,UAAA;AAAA,MACjB,QAAA,EAAU,KAAK,UAAA,IAAc,GAAA;AAAA,MAC7B,QAAA,EAAU,IAAA;AAAA;AAAA,MACV;AAAA,KACF;AAEA,IAAA,IAAI,KAAK,WAAA,EAAa;AACpB,MAAA,QAAA,CAAS,SAAS,IAAA,CAAK,WAAA;AACvB,MAAA,QAAA,CAAS,WAAW,IAAA,CAAK,aAAA;AAAA,IAC3B;AAEA,IAAA,IAAI,IAAA,CAAK,eAAe,IAAA,EAAM;AAC5B,MAAA,QAAA,CAAS,QAAQ,IAAA,CAAK,UAAA;AAAA,IACxB;AAEA,IAAA,OAAO;AAAA,MACL,GAAG,MAAA;AAAA,MACH;AAAA,KACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAsBA,OAAA,GAA8B;AAC5B,IAAA,MAAM,WAAqB,EAAC;AAC5B,IAAA,MAAM,cAAwB,EAAC;AAG/B,IAAA,IAAI,CAAC,KAAK,YAAA,EAAc;AACtB,MAAA,QAAA,CAAS,KAAK,8DAA8D,CAAA;AAAA,IAC9E;AAGA,IAAA,MAAM,MAAA,GAAS,KAAK,cAAA,CAAe,MAAA,GAAS,IAAI,IAAA,CAAK,cAAA,GAAiB,CAAC,GAAG,CAAA;AAC1E,IAAA,MAAM,eAAe,MAAA,CAAO,QAAA,CAAS,GAAG,CAAA,IAAK,OAAO,MAAA,KAAW,CAAA;AAE/D,IAAA,IAAI,YAAA,EAAc;AAChB,MAAA,QAAA,CAAS,KAAK,+EAA+E,CAAA;AAC7F,MAAA,WAAA,CAAY,KAAK,qEAAqE,CAAA;AAAA,IACxF;AAGA,IAAA,MAAM,WAAA,GAAc,KAAK,KAAA,EAAM;AAC/B,IAAA,MAAM,cAAA,GAAiB,KAAK,UAAA,CAAW,MAAA;AAEvC,IAAA,IAAI,cAAA,KAAmB,CAAA,IAAK,CAAC,IAAA,CAAK,UAAA,EAAY;AAC5C,MAAA,QAAA,CAAS,KAAK,8EAA8E,CAAA;AAC5F,MAAA,WAAA,CAAY,KAAK,qEAAqE,CAAA;AAAA,IACxF;AAGA,IAAA,MAAM,mBAAA,GAAsB,KAAK,UAAA,CAAW,IAAA;AAAA,MAC1C,CAAC,MAAM,CAAA,CAAE,QAAA,KAAa,gBAAgB,CAAA,CAAE,KAAA,EAAO,WAAW,GAAG;AAAA,KAC/D;AACA,IAAA,IAAI,mBAAA,EAAqB;AACvB,MAAA,QAAA,CAAS,KAAK,qEAAqE,CAAA;AAAA,IACrF;AAGA,IAAA,MAAM,kBAAkB,IAAA,CAAK,aAAA,CAAc,KAAK,CAAC,EAAA,KAAO,OAAO,IAAI,CAAA;AACnE,IAAA,IAAI,eAAA,IAAmB,iBAAiB,CAAA,EAAG;AACzC,MAAA,QAAA,CAAS,KAAK,sDAAsD,CAAA;AACpE,MAAA,WAAA,CAAY,KAAK,sDAAsD,CAAA;AAAA,IACzE;AAGA,IAAA,MAAM,SAAA,GAAY,KAAK,WAAA,IAAe,YAAA;AACtC,IAAA,MAAM,gBAAgB,IAAA,CAAK,aAAA;AAG3B,IAAA,IAAI,iBAAA,GAAoB,CAAA;AACxB,IAAA,MAAM,QAAA,GAAW,GAAA;AACjB,IAAA,MAAM,YAAA,GAAe,KAAK,UAAA,KAAe,IAAA;AAEzC,IAAA,IAAI,YAAA,IAAgB,mBAAmB,CAAA,EAAG;AAExC,MAAA,iBAAA,GAAoB,EAAA;AACpB,MAAA,QAAA,CAAS,KAAK,mEAAmE,CAAA;AAAA,IACnF,CAAA,MAAA,IAAW,IAAA,CAAK,UAAA,KAAe,IAAA,EAAM;AACnC,MAAA,iBAAA,GAAoB,IAAA,CAAK,IAAA,CAAK,IAAA,CAAK,UAAA,GAAa,QAAQ,CAAA;AAAA,IAC1D;AAGA,IAAA,IAAI,CAAC,IAAA,CAAK,WAAA,IAAe,cAAA,GAAiB,CAAA,EAAG;AAC3C,MAAA,WAAA,CAAY,KAAK,uDAAuD,CAAA;AAAA,IAC1E;AAGA,IAAA,IAAI,KAAK,iBAAA,EAAmB;AAC1B,MAAA,WAAA,CAAY,KAAK,qFAAqF,CAAA;AAAA,IACxG;AAEA,IAAA,OAAO;AAAA,MACL,UAAA,EAAY,KAAK,YAAA,IAAgB,WAAA;AAAA,MACjC,OAAO,WAAA,IAAe,iBAAA;AAAA,MACtB,MAAA;AAAA,MACA,YAAA;AAAA,MACA,YAAA;AAAA,MACA,OAAO,IAAA,CAAK,UAAA;AAAA,MACZ,QAAA;AAAA,MACA,OAAA,EAAS;AAAA,QACP,KAAA,EAAO,SAAA;AAAA,QACP,SAAA,EAAW;AAAA,OACb;AAAA,MACA,iBAAA;AAAA,MACA,QAAA;AAAA,MACA,WAAA;AAAA,MACA,cAAA;AAAA,MACA,eAAe,IAAA,CAAK;AAAA,KACtB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,sBAAA,GAA2C;AACjD,IAAA,MAAM,QAAQ,IAAA,CAAK,YAAA;AAEnB,IAAA,OAAO;AAAA,MACL,MAAA,EAAQ,CAAC,KAAA,KAAyC;AAChD,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,MAAA,CAAO,KAAK,CAAC,CAAA;AAC3C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,SAAA,EAAW,CAAC,KAAA,KAAyC;AACnD,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,MAAA,CAAO,KAAK,CAAC,CAAA;AAC5C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,EAAA,EAAI,CAAC,MAAA,KAA8C;AACjD,QAAA,IAAI,CAAC,MAAA,IAAU,MAAA,CAAO,MAAA,KAAW,CAAA,EAAG;AAClC,UAAA,MAAM,IAAI,MAAM,mCAAmC,CAAA;AAAA,QACrD;AAEA,QAAA,IAAA,CAAK,aAAa,KAAA,EAAO,GAAA,EAAK,OAAO,MAAA,CAAO,CAAC,CAAC,CAAC,CAAA;AAE/C,QAAA,KAAA,IAAS,CAAA,GAAI,CAAA,EAAG,CAAA,GAAI,MAAA,CAAO,QAAQ,CAAA,EAAA,EAAK;AACtC,UAAA,IAAA,CAAK,aAAA,CAAc,KAAK,IAAI,CAAA;AAC5B,UAAA,IAAA,CAAK,aAAa,KAAA,EAAO,GAAA,EAAK,OAAO,MAAA,CAAO,CAAC,CAAC,CAAC,CAAA;AAAA,QACjD;AACA,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,QAAA,EAAU,CAAC,KAAA,KAAyC;AAClD,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,MAAA,CAAO,KAAK,CAAC,CAAA;AAC3C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,WAAA,EAAa,CAAC,KAAA,KAAyC;AACrD,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,MAAA,CAAO,KAAK,CAAC,CAAA;AAC3C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,eAAA,EAAiB,CAAC,KAAA,KAAyC;AACzD,QAAA,MAAM,QAAA,GAAW,OAAO,KAAK,CAAA;AAI7B,QAAA,IAAI,UAAA,CAAW,QAAQ,CAAA,EAAG;AACxB,UAAA,MAAM,OAAA,GAAU,OAAA,CAAQ,QAAA,EAAU,CAAC,CAAA;AACnC,UAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,OAAO,CAAA;AAAA,QACvC,CAAA,MAAO;AACL,UAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,QAAQ,CAAA;AAAA,QACzC;AACA,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,kBAAA,EAAoB,CAAC,KAAA,KAAyC;AAC5D,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,MAAA,CAAO,KAAK,CAAC,CAAA;AAC5C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,QAAA,EAAU,CAAC,KAAA,KAAgC;AAEzC,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,YAAA,EAAc,CAAA,CAAA,EAAI,KAAK,CAAA,CAAE,CAAA;AAClD,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,WAAA,EAAa,CAAC,KAAA,KAAgC;AAE5C,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,gBAAA,EAAkB,CAAA,CAAA,EAAI,KAAK,CAAA,CAAE,CAAA;AACtD,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,UAAA,EAAY,CAAC,KAAA,KAAgC;AAC3C,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,YAAA,EAAc,KAAK,CAAA;AAC5C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,aAAA,EAAe,CAAC,KAAA,KAAgC;AAC9C,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,gBAAA,EAAkB,KAAK,CAAA;AAChD,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,QAAQ,MAAoB;AAC1B,QAAA,IAAA,CAAK,YAAA,CAAa,OAAO,SAAS,CAAA;AAClC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,WAAW,MAAoB;AAC7B,QAAA,IAAA,CAAK,YAAA,CAAa,OAAO,aAAa,CAAA;AACtC,QAAA,OAAO,IAAA;AAAA,MACT;AAAA,KACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,2BAA2B,KAAA,EAAqC;AACtE,IAAA,OAAO;AAAA,MACL,OAAO,MAAoB;AACzB,QAAA,MAAM,QAAQ,QAAA,EAAS;AAEvB,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,KAAK,CAAA;AACpC,QAAA,IAAA,CAAK,aAAA,CAAc,KAAK,KAAK,CAAA;AAC7B,QAAA,IAAA,CAAK,aAAa,KAAA,EAAO,GAAA,EAAK,OAAA,CAAQ,KAAA,EAAO,CAAC,CAAC,CAAA;AAC/C,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,UAAU,MAAoB;AAC5B,QAAA,MAAM,cAAc,cAAA,EAAe;AACnC,QAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,QAAA,EAAS,EAAG,CAAC,CAAA;AAEtC,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,WAAW,CAAA;AAC1C,QAAA,IAAA,CAAK,aAAA,CAAc,KAAK,KAAK,CAAA;AAC7B,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,QAAQ,CAAA;AACtC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,WAAW,MAAoB;AAC7B,QAAA,MAAM,eAAe,eAAA,EAAgB;AACrC,QAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,QAAA,EAAS,EAAG,CAAC,CAAA;AAEtC,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,YAAY,CAAA;AAC3C,QAAA,IAAA,CAAK,aAAA,CAAc,KAAK,KAAK,CAAA;AAC7B,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,QAAQ,CAAA;AACtC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,OAAA,EAAS,CAAC,SAAA,EAAmB,OAAA,KAAkC;AAE7D,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,SAAS,CAAA;AACxC,QAAA,IAAA,CAAK,aAAA,CAAc,KAAK,KAAK,CAAA;AAE7B,QAAA,MAAM,cAAc,UAAA,CAAW,OAAO,IAAI,OAAA,CAAQ,OAAA,EAAS,CAAC,CAAA,GAAI,OAAA;AAChE,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,WAAW,CAAA;AACzC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,OAAA,EAAS,CAAC,IAAA,KAA+B;AACvC,QAAA,MAAM,QAAQ,QAAA,EAAS;AACvB,QAAA,MAAM,SAAA,GAAY,OAAA,CAAQ,KAAA,EAAO,CAAC,IAAI,CAAA;AACtC,QAAA,MAAM,QAAA,GAAW,OAAA,CAAQ,KAAA,EAAO,CAAC,CAAA;AAEjC,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,SAAS,CAAA;AACxC,QAAA,IAAA,CAAK,aAAA,CAAc,KAAK,KAAK,CAAA;AAC7B,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,QAAQ,CAAA;AACtC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,MAAA,EAAQ,CAAC,IAAA,KAA+B;AACtC,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,IAAI,CAAA;AAClC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,KAAA,EAAO,CAAC,IAAA,KAA+B;AAIrC,QAAA,MAAM,UAAU,UAAA,CAAW,IAAI,IAAI,OAAA,CAAQ,IAAA,EAAM,CAAC,CAAA,GAAI,IAAA;AACtD,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,OAAO,CAAA;AACtC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,UAAA,EAAY,CAAC,IAAA,KAA+B;AAE1C,QAAA,MAAM,UAAU,UAAA,CAAW,IAAI,IAAI,OAAA,CAAQ,IAAA,EAAM,CAAC,CAAA,GAAI,IAAA;AACtD,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,GAAA,EAAK,OAAO,CAAA;AACrC,QAAA,OAAO,IAAA;AAAA,MACT,CAAA;AAAA,MACA,SAAA,EAAW,CAAC,IAAA,KAA+B;AACzC,QAAA,IAAA,CAAK,YAAA,CAAa,KAAA,EAAO,IAAA,EAAM,IAAI,CAAA;AACnC,QAAA,OAAO,IAAA;AAAA,MACT;AAAA,KACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,YAAA,CAAa,KAAA,EAAe,QAAA,EAAyB,KAAA,EAAsB;AACjF,IAAA,IAAA,CAAK,WAAW,IAAA,CAAK,EAAE,KAAA,EAAO,QAAA,EAAU,OAAO,CAAA;AAC/C,IAAA,IAAA,CAAK,YAAA,GAAe,IAAA;AAAA,EACtB;AACF;;;AC7iCA,IAAM,4BAAA,GAAuD;AAAA,EAC3D,QAAA,EAAU,aAAA;AAAA,EACV,YAAA,EAAc,iBAAA;AAAA,EACd,QAAA,EAAU,UAAA;AAAA;AAAA,EACV,MAAA,EAAQ;AAAA;AACV,CAAA;AAMA,IAAM,kBAAA,GAA+B;AAAA,EACnC,gBAAA;AAAA;AAAA,EACA,WAAA;AAAA;AAAA,EACA;AAAA;AACF,CAAA;AAMA,IAAM,oBAAA,GAAiC;AAAA,EACrC;AAAA;AACF,CAAA;AAOA,IAAM,0BAAA,GAAuC;AAAA,EAC3C,eAAA;AAAA;AAAA,EACA,iBAAA;AAAA;AAAA,EACA,WAAA;AAAA;AAAA,EACA,WAAA;AAAA;AAAA,EACA,mBAAA;AAAA;AAAA,EACA,eAAA;AAAA;AAAA,EACA,gBAAA;AAAA;AAAA,EACA,gBAAA;AAAA;AAAA,EACA,eAAA;AAAA;AAAA,EACA,WAAA;AAAA;AAAA,EACA,UAAA;AAAA;AAAA,EACA;AAAA;AACF,CAAA;AAMA,IAAM,qBAAA,GAGF;AAAA;AAAA,EAEF,CAAA,EAAG;AAAA,IACD,kBAAA,EAAoB,CAAC,kBAAA,EAAoB,kBAAkB;AAAA,GAC7D;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,SAAA,EAAW,WAAA,EAAa,mBAAmB,cAAc,CAAA;AAAA,IAC5E,kBAAA,EAAoB;AAAA,MAClB,mBAAA;AAAA,MACA,gBAAA;AAAA,MACA,cAAA;AAAA,MACA,kBAAA;AAAA,MACA,sBAAA;AAAA,MACA,sBAAA;AAAA,MACA,gBAAA;AAAA,MACA,gBAAA;AAAA,MACA,kBAAA;AAAA,MACA,kBAAA;AAAA,MACA,YAAA;AAAA,MACA,YAAA;AAAA,MACA,aAAA;AAAA,MACA,aAAA;AAAA,MACA;AAAA;AACF,GACF;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,kBAAA,EAAoB,CAAC,cAAc;AAAA;AAAA,GACrC;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,YAAA,EAAc,WAAA,EAAa,SAAS;AAAA,GACzD;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,YAAY;AAAA,GACjC;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,YAAY;AAAA,GACjC;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,YAAA,EAAc,SAAS;AAAA,GAC5C;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB;AAAA,MAChB,YAAA;AAAA,MACA,kBAAA;AAAA,MACA,WAAA;AAAA,MACA,SAAA;AAAA,MACA;AAAA,KACF;AAAA,IACA,kBAAA,EAAoB;AAAA,MAClB,gBAAA;AAAA,MACA,aAAA;AAAA,MACA,mBAAA;AAAA,MACA,cAAA;AAAA,MACA,WAAA;AAAA,MACA;AAAA;AACF,GACF;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB;AAAA,MAChB,SAAA;AAAA,MACA,iBAAA;AAAA,MACA,YAAA;AAAA,MACA,kBAAA;AAAA,MACA,mBAAA;AAAA,MACA;AAAA,KACF;AAAA,IACA,kBAAA,EAAoB,CAAC,cAAA,EAAgB,mBAAA,EAAqB,aAAa,gBAAgB;AAAA,GACzF;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,WAAA,EAAa,YAAA,EAAc,WAAW,eAAe,CAAA;AAAA,IACxE,kBAAA,EAAoB,CAAC,gBAAA,EAAkB,WAAA,EAAa,qBAAqB,cAAc;AAAA,GACzF;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,aAAA,EAAe,YAAA,EAAc,WAAW,WAAW,CAAA;AAAA,IACtE,kBAAA,EAAoB,CAAC,WAAA,EAAa,cAAc;AAAA,GAClD;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,YAAA,EAAc,SAAA,EAAW,mBAAmB,WAAW,CAAA;AAAA,IAC1E,kBAAA,EAAoB,CAAC,mBAAA,EAAqB,WAAA,EAAa,kBAAkB,cAAc;AAAA,GACzF;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,SAAA,EAAW,YAAA,EAAc,WAAW,CAAA;AAAA,IACvD,kBAAA,EAAoB,CAAC,gBAAA,EAAkB,mBAAA,EAAqB,gBAAgB,WAAW;AAAA,GACzF;AAAA;AAAA,EAEA,EAAA,EAAI;AAAA,IACF,gBAAA,EAAkB,CAAC,WAAA,EAAa,SAAA,EAAW,YAAY,CAAA;AAAA,IACvD,kBAAA,EAAoB,CAAC,WAAA,EAAa,mBAAA,EAAqB,gBAAgB,gBAAgB;AAAA,GACzF;AAAA;AAAA,EAEA,GAAA,EAAK;AAAA,IACH,gBAAA,EAAkB,CAAC,WAAA,EAAa,QAAA,EAAU,aAAa,SAAS;AAAA,GAClE;AAAA;AAAA,EAEA,GAAA,EAAK;AAAA,IACH,gBAAA,EAAkB,CAAC,SAAS;AAAA,GAC9B;AAAA;AAAA,EAEA,GAAA,EAAK;AAAA,IACH,gBAAA,EAAkB,CAAC,WAAA,EAAa,SAAS,CAAA;AAAA,IACzC,kBAAA,EAAoB,CAAC,gBAAA,EAAkB,UAAA,EAAY,YAAY;AAAA,GACjE;AAAA;AAAA,EAEA,GAAA,EAAK;AAAA,IACH,gBAAA,EAAkB,CAAC,QAAA,EAAU,SAAA,EAAW,aAAa,WAAW;AAAA,GAClE;AAAA;AAAA,EAEA,GAAA,EAAK;AAAA,IACH,gBAAA,EAAkB,CAAC,SAAS;AAAA,GAC9B;AAAA;AAAA,EAEA,GAAA,EAAK;AAAA,IACH,gBAAA,EAAkB,CAAC,SAAS;AAAA;AAEhC,CAAA;AAgBO,SAAS,qBAAA,CAAsB,WAAmB,UAAA,EAAqC;AAE5F,EAAA,IAAI,4BAAA,CAA6B,SAAS,CAAA,EAAG;AAC3C,IAAA,OAAO,6BAA6B,SAAS,CAAA;AAAA,EAC/C;AAGA,EAAA,IAAI,0BAAA,CAA2B,QAAA,CAAS,SAAS,CAAA,EAAG;AAClD,IAAA,OAAO,EAAA;AAAA,EACT;AAGA,EAAA,IAAI,qBAAA,CAAsB,IAAA,CAAK,SAAS,CAAA,EAAG;AACzC,IAAA,OAAO,MAAA;AAAA,EACT;AAGA,EAAA,IAAI,SAAA,CAAU,UAAA,CAAW,KAAK,CAAA,EAAG;AAC/B,IAAA,OAAO,GAAG,SAAS,CAAA,IAAA,CAAA;AAAA,EACrB;AAGA,EAAA,MAAM,gBACJ,OAAO,UAAA,KAAe,WAAW,QAAA,CAAS,UAAA,EAAY,EAAE,CAAA,GAAI,UAAA;AAC9D,EAAA,MAAM,SAAA,GAAY,qBAAA,CAAsB,aAAa,CAAA,IAAK,IAAA;AAG1D,EAAA,MAAM,iBAAiB,aAAA,IAAiB,GAAA;AACxC,EAAA,MAAM,sBAAA,GAAyB,cAAA,GAAiB,CAAC,SAAS,IAAI,EAAC;AAG/D,EAAA,MAAM,kBAAA,GAAqB;AAAA,IACzB,GAAG,oBAAA;AAAA,IACH,GAAI,SAAA,EAAW,kBAAA,IAAsB;AAAC,GACxC;AACA,EAAA,MAAM,gBAAA,GAAmB;AAAA,IACvB,GAAG,kBAAA;AAAA,IACH,GAAI,SAAA,EAAW,gBAAA,IAAoB,EAAC;AAAA,IACpC,GAAG;AAAA,GACL;AAGA,EAAA,IAAI,SAAA,CAAU,SAAS,MAAM,CAAA,IAAK,CAAC,kBAAA,CAAmB,QAAA,CAAS,SAAS,CAAA,EAAG;AACzE,IAAA,OAAO,SAAA,CAAU,KAAA,CAAM,CAAA,EAAG,EAAE,CAAA;AAAA,EAC9B;AAGA,EAAA,IAAI,SAAA,CAAU,SAAS,IAAI,CAAA,IAAK,CAAC,gBAAA,CAAiB,QAAA,CAAS,SAAS,CAAA,EAAG;AACrE,IAAA,OAAO,SAAA,CAAU,KAAA,CAAM,CAAA,EAAG,EAAE,CAAA,GAAI,MAAA;AAAA,EAClC;AAGA,EAAA,OAAO,GAAG,SAAS,CAAA,IAAA,CAAA;AACrB;;;ACtPO,IAAM,eAAA,GAA0C;AAAA,EACrD,CAAA,EAAG,aAAA;AAAA;AAAA,EACH,CAAA,EAAG,UAAA;AAAA;AAAA,EACH,CAAA,EAAG,UAAA;AAAA;AAAA,EACH,CAAA,EAAG,MAAA;AAAA;AAAA,EACH,CAAA,EAAG,OAAA;AAAA;AAAA,EACH,CAAA,EAAG,SAAA;AAAA;AAAA,EACH,CAAA,EAAG,UAAA;AAAA;AAAA,EACH,CAAA,EAAG,gBAAA;AAAA;AAAA,EACH,CAAA,EAAG,UAAA;AAAA;AAAA,EACH,EAAA,EAAI,SAAA;AAAA;AAAA,EACJ,EAAA,EAAI,aAAA;AAAA;AAAA,EACJ,EAAA,EAAI,gBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,MAAA;AAAA;AAAA,EACJ,EAAA,EAAI,aAAA;AAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA;AAAA,EACJ,EAAA,EAAI,MAAA;AAAA;AAAA,EACJ,EAAA,EAAI,SAAA;AAAA;AAAA,EACJ,EAAA,EAAI,MAAA;AAAA;AAAA,EACJ,EAAA,EAAI,cAAA;AAAA;AAAA,EACJ,EAAA,EAAI,WAAA;AAAA;AAAA,EACJ,EAAA,EAAI,aAAA;AAAA;AAAA,EACJ,EAAA,EAAI,UAAA;AAAA;AAAA,EACJ,EAAA,EAAI,MAAA;AAAA;AAAA,EACJ,EAAA,EAAI,UAAA;AAAA;AAAA,EACJ,EAAA,EAAI,cAAA;AAAA;AAAA,EACJ,EAAA,EAAI,aAAA;AAAA;AAAA,EACJ,EAAA,EAAI,OAAA;AAAA;AAAA,EACJ,EAAA,EAAI,aAAA;AAAA;AAAA,EACJ,EAAA,EAAI,UAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,gBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,eAAA;AAAA;AAAA,EACJ,EAAA,EAAI,MAAA;AAAA;AAAA,EACJ,EAAA,EAAI,gBAAA;AAAA;AAAA,EACJ,EAAA,EAAI,MAAA;AAAA;AAAA,EACJ,GAAA,EAAK,UAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,mBAAA;AAAA;AAAA,EACL,GAAA,EAAK,SAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK,MAAA;AAAA;AAAA,EACL,GAAA,EAAK;AAAA;AACP;AAQO,SAAS,yBAAyB,YAAA,EAAuC;AAC9E,EAAA,MAAM,gBACJ,OAAO,YAAA,KAAiB,WAAW,QAAA,CAAS,YAAA,EAAc,EAAE,CAAA,GAAI,YAAA;AAGlE,EAAA,IAAI,eAAA,CAAgB,aAAa,CAAA,EAAG;AAClC,IAAA,OAAO,gBAAgB,aAAa,CAAA;AAAA,EACtC;AAGA,EAAA,IAAI,iBAAiB,GAAA,EAAM;AACzB,IAAA,OAAO,MAAA;AAAA,EACT;AAGA,EAAA,OAAO,MAAA;AACT;;;AChFO,IAAM,cAAA,GAAiB;AAAA,EAC5B,QAAA,EAAU,sCAAA;AAAA,EACV,MAAA,EAAQ,sCAAA;AAAA,EACR,KAAA,EAAO,sCAAA;AAAA,EACP,IAAA,EAAM,sCAAA;AAAA,EACN,GAAA,EAAK,sCAAA;AAAA,EACL,SAAA,EAAW,sCAAA;AAAA,EACX,QAAA,EAAU,sCAAA;AAAA,EACV,IAAA,EAAM,sCAAA;AAAA,EACN,IAAA,EAAM,sCAAA;AAAA,EACN,SAAA,EAAW,sCAAA;AAAA,EACX,OAAA,EAAS;AACX;AAMO,IAAM,mBAAA,GAA8C;AAAA,EACzD,CAAC,cAAA,CAAe,QAAQ,GAAG,UAAA;AAAA,EAC3B,CAAC,cAAA,CAAe,KAAK,GAAG,OAAA;AAAA,EACxB,CAAC,cAAA,CAAe,IAAI,GAAG,MAAA;AAAA,EACvB,CAAC,cAAA,CAAe,MAAM,GAAG,QAAA;AAAA,EACzB,CAAC,cAAA,CAAe,GAAG,GAAG,KAAA;AAAA,EACtB,CAAC,cAAA,CAAe,SAAS,GAAG,WAAA;AAAA,EAC5B,CAAC,cAAA,CAAe,QAAQ,GAAG,UAAA;AAAA,EAC3B,CAAC,cAAA,CAAe,IAAI,GAAG,MAAA;AAAA,EACvB,CAAC,cAAA,CAAe,IAAI,GAAG,MAAA;AAAA,EACvB,CAAC,cAAA,CAAe,SAAS,GAAG,WAAA;AAAA,EAC5B,CAAC,cAAA,CAAe,OAAO,GAAG;AAC5B;;;ACjB+D,MAAA,CAAO,OAAA,CAAQ,aAAa,CAAA,CAAE,MAAA;AAAA,EAC3F,CAAC,GAAA,EAAK,CAAC,YAAA,EAAc,WAAW,CAAA,KAAM;AACpC,IAAA,GAAA,CAAI,WAAW,CAAA,GAAI,QAAA,CAAS,YAAA,EAAc,EAAE,CAAA;AAC5C,IAAA,OAAO,GAAA;AAAA,EACT,CAAA;AAAA,EACA;AACF;;;ACnBO,IAAM,8BAAA,GAA2D;AAAA,EACtE,GAAA,EAAK,CAAC,WAAA,EAAa,WAAW,CAAA;AAAA;AAAA,EAC9B,GAAA,EAAK,CAAC,WAAA,EAAa,WAAA,EAAa,KAAK,GAAA,EAAK,GAAA,EAAK,KAAK,aAAa,CAAA;AAAA;AAAA,EACjE,KAAA,EAAO,CAAC,WAAA,EAAa,WAAW,CAAA;AAAA;AAAA,EAChC,KAAA,EAAO,CAAC,WAAA,EAAa,WAAW,CAAA;AAAA;AAAA,EAChC,KAAA,EAAO,CAAC,WAAA,EAAa,WAAW,CAAA;AAAA;AAAA,EAChC,KAAA,EAAO,CAAC,WAAA,EAAa,WAAW;AAAA;AAClC;AA2BO,SAAS,8BAA8B,UAAA,EAAuC;AACnF,EAAA,MAAM,aAAA,GAAgB,OAAO,UAAU,CAAA;AACvC,EAAA,OAAO,8BAAA,CAA+B,aAAa,CAAA,IAAK,EAAC;AAC3D;;;ACjCO,SAAS,gBAAgB,iBAAA,EAAoC;AAClE,EAAA,OAAO,sBAAsB,cAAA,CAAe,QAAA;AAC9C;AAQO,SAAS,cAAc,iBAAA,EAAoC;AAChE,EAAA,OAAO,sBAAsB,cAAA,CAAe,MAAA;AAC9C;AAQO,SAAS,wBAAwB,iBAAA,EAAoC;AAC1E,EAAA,OAAO,eAAA,CAAgB,iBAAiB,CAAA,IAAK,aAAA,CAAc,iBAAiB,CAAA;AAC9E;;;ACkDO,IAAM,kBAAN,MAAsB;AAAA,EACnB,OAAA;AAAA,EACA,YAAA;AAAA,EACA,GAAA;AAAA,EACA,iBAA2B,EAAC;AAAA,EAC5B,aAAA;AAAA,EACA,OAAA,GAAkB,CAAA;AAAA,EAE1B,WAAA,CAAY,KAAsB,UAAA,EAA6B;AAC7D,IAAA,IAAA,CAAK,GAAA,GAAM,GAAA;AACX,IAAA,IAAA,CAAK,YAAA,GAAe,UAAA;AACpB,IAAA,IAAA,CAAK,OAAA,GAAU,IAAI,YAAA,EAAa;AAAA,EAClC;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,UAAU,MAAA,EAAwB;AAChC,IAAA,IAAA,CAAK,cAAA,CAAe,IAAA,CAAK,GAAG,MAAM,CAAA;AAClC,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOA,oBAAoB,MAAA,EAAwB;AAC1C,IAAA,MAAM,UAAA,GAAa,OAAO,IAAA,CAAK,YAAA,KAAiB,QAAA,GAC5C,SAAS,IAAA,CAAK,YAAA,EAAc,EAAE,CAAA,GAC9B,IAAA,CAAK,YAAA;AAET,IAAA,KAAA,MAAW,SAAS,MAAA,EAAQ;AAC1B,MAAA,IAAI,CAAC,IAAA,CAAK,cAAA,CAAe,QAAA,CAAS,KAAK,CAAA,EAAG;AACxC,QAAA,IAAA,CAAK,cAAA,CAAe,KAAK,KAAK,CAAA;AAAA,MAChC;AACA,MAAA,MAAM,UAAA,GAAa,qBAAA,CAAsB,KAAA,EAAO,UAAU,CAAA;AAC1D,MAAA,IAAI,cAAc,CAAC,IAAA,CAAK,cAAA,CAAe,QAAA,CAAS,UAAU,CAAA,EAAG;AAC3D,QAAA,IAAA,CAAK,cAAA,CAAe,KAAK,UAAU,CAAA;AAAA,MACrC;AAAA,IACF;AACA,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAM,KAAA,EAAoC;AACxC,IAAA,MAAM,YAAA,GAAe,IAAA,CAAK,OAAA,CAAQ,KAAA,CAAM,KAAK,CAAA;AAE7C,IAAA,OAAO;AAAA,MACL,MAAA,EAAQ,CAAC,KAAA,KAA2B;AAAE,QAAA,YAAA,CAAa,OAAO,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MAC/E,SAAA,EAAW,CAAC,KAAA,KAA2B;AAAE,QAAA,YAAA,CAAa,UAAU,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACrF,QAAA,EAAU,CAAC,KAAA,KAA2B;AAAE,QAAA,YAAA,CAAa,SAAS,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACnF,WAAA,EAAa,CAAC,KAAA,KAA2B;AAAE,QAAA,YAAA,CAAa,YAAY,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACzF,eAAA,EAAiB,CAAC,KAAA,KAA2B;AAAE,QAAA,YAAA,CAAa,gBAAgB,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACjG,kBAAA,EAAoB,CAAC,KAAA,KAA2B;AAAE,QAAA,YAAA,CAAa,mBAAmB,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACvG,QAAA,EAAU,CAAC,KAAA,KAAkB;AAAE,QAAA,YAAA,CAAa,SAAS,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MAC1E,WAAA,EAAa,CAAC,KAAA,KAAkB;AAAE,QAAA,YAAA,CAAa,YAAY,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MAChF,UAAA,EAAY,CAAC,KAAA,KAAkB;AAAE,QAAA,YAAA,CAAa,WAAW,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MAC9E,aAAA,EAAe,CAAC,KAAA,KAAkB;AAAE,QAAA,YAAA,CAAa,cAAc,KAAK,CAAA;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACpF,QAAQ,MAAM;AAAE,QAAA,YAAA,CAAa,MAAA,EAAO;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM,CAAA;AAAA,MACpD,WAAW,MAAM;AAAE,QAAA,YAAA,CAAa,SAAA,EAAU;AAAG,QAAA,OAAO,IAAA;AAAA,MAAM;AAAA,KAC5D;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,GAAA,GAAY;AACV,IAAA,IAAA,CAAK,QAAQ,GAAA,EAAI;AACjB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,EAAA,GAAW;AACT,IAAA,IAAA,CAAK,QAAQ,EAAA,EAAG;AAChB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,SAAS,IAAA,EAAoB;AAC3B,IAAA,IAAA,CAAK,aAAA,GAAgB,IAAA;AACrB,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,KAAK,IAAA,EAAoB;AACvB,IAAA,IAAA,CAAK,OAAA,GAAU,IAAA;AACf,IAAA,OAAO,IAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,cAAA,GAAkC;AAChC,IAAA,MAAM,OAAA,GAA2B;AAAA,MAC/B,MAAA,EAAQ,KAAK,cAAA,CAAe,MAAA,GAAS,IACjC,IAAA,CAAK,cAAA,CAAe,IAAA,CAAK,GAAG,CAAA,GAC5B,GAAA;AAAA,MACJ,KAAA,EAAO,IAAA,CAAK,OAAA,CAAQ,KAAA;AAAM,KAC5B;AAEA,IAAA,IAAI,IAAA,CAAK,kBAAkB,MAAA,EAAW;AACpC,MAAA,OAAA,CAAQ,YAAY,IAAA,CAAK,aAAA;AAAA,IAC3B;AAEA,IAAA,IAAI,IAAA,CAAK,UAAU,CAAA,EAAG;AACpB,MAAA,OAAA,CAAQ,cAAc,IAAA,CAAK,OAAA;AAAA,IAC7B;AAEA,IAAA,OAAO,OAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,MAAM,OAAA,GAAoE;AACxE,IAAA,MAAM,OAAA,GAAU,KAAK,cAAA,EAAe;AACpC,IAAA,OAAO,IAAA,CAAK,GAAA,CAAI,KAAA,CAAM,IAAA,CAAK,cAAc,OAAO,CAAA;AAAA,EAClD;AACF;AA0BO,SAAS,sBACd,GAAA,EACkD;AAClD,EAAA,MAAM,GAAA,GAAM,KAAA,IAAS,GAAA,GAAM,GAAA,CAAI,GAAA,GAAM,GAAA;AAErC,EAAA,OAAO,CAAC,UAAA,KAAgC,IAAI,eAAA,CAAgB,KAAK,UAAU,CAAA;AAC7E;AAMO,IAAM,WAAA,GAAN,MAAM,YAAA,CAA6C;AAAA,EAChD,GAAA;AAAA,EAEA,YAAY,GAAA,EAAgC;AAClD,IAAA,IAAA,CAAK,GAAA,GAAM,GAAA;AAAA,EACb;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,OAAO,OACL,GAAA,EACgB;AAChB,IAAA,OAAO,IAAI,aAAY,GAAG,CAAA;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,OAAA,GAA6B;AAC/B,IAAA,OAAO,KAAK,GAAA,CAAI,OAAA;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,GAAA,GAA8B;AAChC,IAAA,OAAO,KAAK,GAAA,CAAI,GAAA;AAAA,EAClB;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,MAAA,GAA6B;AAC/B,IAAA,OAAO,IAAA,CAAK,SAAS,IAAA,EAAM,EAAA;AAAA,EAC7B;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,YAAA,GAAmC;AACrC,IAAA,OAAO,IAAA,CAAK,SAAS,IAAA,EAAM,QAAA;AAAA,EAC7B;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,QAAA,GAA+B;AACjC,IAAA,OAAO,IAAA,CAAK,SAAS,MAAA,EAAQ,EAAA;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,UAAA,GAAiC;AACnC,IAAA,OAAO,IAAA,CAAK,SAAS,MAAA,EAAQ,IAAA;AAAA,EAC/B;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAiBA,MAAM,UAAA,EAA8C;AAElD,IAAA,OAAO,IAAI,eAAA,CAAgB,IAAA,CAAK,GAAA,CAAI,KAAwB,UAAU,CAAA;AAAA,EACxE;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcA,WAAW,UAAA,EAAqC;AAC9C,IAAA,OAAO,qBAAqB,UAAU,CAAA;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAcA,aAAa,UAAA,EAAqC;AAChD,IAAA,OAAO,yBAAyB,UAAU,CAAA;AAAA,EAC5C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAeA,aAAA,CAAc,WAAmB,UAAA,EAAqC;AACpE,IAAA,OAAO,qBAAA,CAAsB,WAAW,UAAU,CAAA;AAAA,EACpD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,kBAAkB,UAAA,EAAuC;AACvD,IAAA,OAAO,8BAA8B,UAAU,CAAA;AAAA,EACjD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAeA,sBAAA,CAAuB,QAAkB,UAAA,EAAuC;AAC9E,IAAA,MAAM,SAAmB,EAAC;AAC1B,IAAA,KAAA,MAAW,SAAS,MAAA,EAAQ;AAC1B,MAAA,IAAI,CAAC,MAAA,CAAO,QAAA,CAAS,KAAK,CAAA,EAAG;AAC3B,QAAA,MAAA,CAAO,KAAK,KAAK,CAAA;AAAA,MACnB;AACA,MAAA,MAAM,UAAA,GAAa,qBAAA,CAAsB,KAAA,EAAO,UAAU,CAAA;AAC1D,MAAA,IAAI,UAAA,IAAc,CAAC,MAAA,CAAO,QAAA,CAAS,UAAU,CAAA,EAAG;AAC9C,QAAA,MAAA,CAAO,KAAK,UAAU,CAAA;AAAA,MACxB;AAAA,IACF;AACA,IAAA,OAAO,MAAA;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAA,CACE,YACA,IAAA,EACiC;AACjC,IAAA,OAAO,IAAA,CAAK,GAAA,CAAI,GAAA,CAAI,MAAA,CAAO,YAAY,IAAI,CAAA;AAAA,EAC7C;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EASA,MAAA,CACE,UAAA,EACA,QAAA,EACA,IAAA,EACiC;AACjC,IAAA,OAAO,KAAK,GAAA,CAAI,GAAA,CAAI,MAAA,CAAO,UAAA,EAAY,UAAU,IAAI,CAAA;AAAA,EACvD;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,MAAA,CACE,YACA,QAAA,EACiC;AACjC,IAAA,OAAO,IAAA,CAAK,GAAA,CAAI,GAAA,CAAI,MAAA,CAAO,YAAY,QAAQ,CAAA;AAAA,EACjD;AAAA;AAAA;AAAA;AAAA,EAKA,OAAA,GAAgB;AACd,IAAA,IAAA,CAAK,IAAI,OAAA,EAAQ;AAAA,EACnB;AACF","file":"index.js","sourcesContent":["/**\r\n * Object Type ID to Primary Key Field Mapping\r\n * Maps Fireberry object type IDs to their primary key field names\r\n * Generated from actual API responses\r\n */\r\nexport const OBJECT_ID_MAP: Record<number, string> = {\r\n 1: 'accountid', // Account\r\n 2: 'contactid', // Contact\r\n 3: 'leadid', // Lead\r\n 4: 'opportunityid', // Opportunity\r\n 5: 'casesid', // Cases\r\n 6: 'activityid', // Activity\r\n 7: 'noteid', // Note\r\n 8: 'competitorid', // Competitor\r\n 9: 'crmuserid', // CrmUser\r\n 10: 'taskid', // Task\r\n 12: 'quoteid', // Quote\r\n 13: 'crmorderid', // CrmOrder\r\n 14: 'productid', // Product\r\n 17: 'crmorderitemid', // CrmOrderItem\r\n 20: 'emailtemplateid', // EmailTemplate\r\n 23: 'businessunitid', // BusinessUnit\r\n 25: 'orgid', // Org\r\n 27: 'printtemplateid', // PrintTemplate\r\n 28: 'contractid', // Contract\r\n 33: 'accountproductid', // AccountProduct\r\n 46: 'projectid', // Project\r\n 55: 'wfruleid', // WFRule\r\n 58: 'mdobjectid', // MDObject\r\n 64: 'roleid', // Role\r\n 67: 'campaignid', // Campaign\r\n 70: 'crmuserloginid', // CrmUserLogin\r\n 73: 'systemfieldid', // SystemField\r\n 76: 'articleid', // Article\r\n 77: 'linkid', // Link\r\n 78: 'invoiceid', // Invoice\r\n 80: 'invoicereceiptitemid', // InvoiceReceiptItem\r\n 81: 'invoiceid', // InvoiceNo\r\n 82: 'invoiceid', // InvoiceDraft\r\n 83: 'invoiceid', // InvoiceReceipt\r\n 84: 'invoiceid', // InvoiceReno\r\n 85: 'invoiceid', // InvoiceCredit\r\n 86: 'invoiceid', // InvoiceDelivery\r\n 89: 'iprestrictionid', // IpRestriction\r\n 90: 'transactionitemid', // TransactionItem\r\n 93: 'chargeid', // Charge\r\n 100: 'calllogid', // calllog\r\n 101: 'attendanceclockid', // AttendanceClock\r\n 102: 'activitylogid', // ActivityLog\r\n 104: 'conversationid', // Conversation\r\n 105: 'teaminboxid', // TeamInbox\r\n 106: 'texttemplateid', // TextTemplate\r\n 107: 'facebookconnectionid', // FacebookConnection\r\n 109: 'auditlogid', // AuditLog\r\n 110: 'smstemplateid', // SMSTemplate\r\n 111: 'providerverificationid', // ProviderVerification\r\n 114: 'calendarresourceid', // CalendarResource\r\n 115: 'journeyid', // Journey\r\n 116: 'profileid', // Profile\r\n 117: 'landingpageid', // LandingPage\r\n};\r\n\r\n/**\r\n * Gets the primary key field name for a given object type\r\n *\r\n * @param objectTypeId - The numeric object type ID\r\n * @returns The correct ID field name for the object type\r\n */\r\nexport function getObjectIdFieldName(objectTypeId: string | number): string {\r\n const objectTypeNum =\r\n typeof objectTypeId === 'string' ? parseInt(objectTypeId, 10) : objectTypeId;\r\n\r\n // Check if it's a mapped base object\r\n if (OBJECT_ID_MAP[objectTypeNum]) {\r\n return OBJECT_ID_MAP[objectTypeNum];\r\n }\r\n\r\n // For custom objects (1000 and up), use the pattern customobjectXid\r\n if (objectTypeNum >= 1000) {\r\n return `customobject${objectTypeNum}id`;\r\n }\r\n\r\n // Fallback to generic 'id' for unmapped objects\r\n return 'id';\r\n}\r\n","import type { QueryOperator, QueryMetadata, QueryResultWithMetadata, QueryExplainResult } from '../types/query';\r\nimport { getObjectIdFieldName } from '../constants/objectIds';\r\n\r\n/**\r\n * Regular expression to match pure date format (YYYY-MM-DD).\r\n * Does not match datetime formats like YYYY-MM-DDTHH:MM:SS or YYYY-MM-DD HH:MM:SS.\r\n */\r\nconst PURE_DATE_REGEX = /^\\d{4}-\\d{2}-\\d{2}$/;\r\n\r\n/**\r\n * Checks if a value is a pure date string (YYYY-MM-DD format).\r\n * Returns false for datetime formats that include time components.\r\n *\r\n * @param value - The value to check\r\n * @returns True if the value is a pure date string\r\n *\r\n * @example\r\n * isPureDate('2024-01-15') // true\r\n * isPureDate('2024-01-15T10:30:00') // false\r\n * isPureDate('2024-01-15 10:30:00') // false\r\n * isPureDate('123') // false\r\n */\r\nexport function isPureDate(value: string): boolean {\r\n return PURE_DATE_REGEX.test(value);\r\n}\r\n\r\n/**\r\n * Adds a specified number of days to a date string.\r\n * Works with both pure dates (YYYY-MM-DD) and datetime formats.\r\n *\r\n * @param dateStr - The date string to modify\r\n * @param days - Number of days to add (can be negative)\r\n * @returns The modified date in YYYY-MM-DD format\r\n *\r\n * @example\r\n * addDays('2024-01-15', 1) // '2024-01-16'\r\n * addDays('2024-01-31', 1) // '2024-02-01'\r\n * addDays('2024-03-01', -1) // '2024-02-29' (leap year)\r\n */\r\nexport function addDays(dateStr: string, days: number): string {\r\n // Extract just the date part if datetime format\r\n const datePart = dateStr.split(/[T\\s]/)[0];\r\n const date = new Date(datePart + 'T00:00:00');\r\n date.setDate(date.getDate() + days);\r\n\r\n const year = date.getFullYear();\r\n const month = String(date.getMonth() + 1).padStart(2, '0');\r\n const day = String(date.getDate()).padStart(2, '0');\r\n\r\n return `${year}-${month}-${day}`;\r\n}\r\n\r\n/**\r\n * Escapes special characters in query values to prevent query injection.\r\n * This is a security measure to ensure user-provided values cannot modify\r\n * the query structure or inject additional query logic.\r\n *\r\n * @param value - The value to escape\r\n * @returns Escaped value safe for use in Fireberry queries\r\n */\r\nexport function escapeQueryValue(value: string): string {\r\n if (!value) {\r\n return '';\r\n }\r\n // Escape backslashes first to avoid double-escaping\r\n let escaped = value.replace(/\\\\/g, '\\\\\\\\');\r\n // Escape parentheses which could break out of conditions\r\n escaped = escaped.replace(/\\(/g, '\\\\(');\r\n escaped = escaped.replace(/\\)/g, '\\\\)');\r\n // Escape logical operators that could inject additional conditions\r\n // Using word boundaries to only match standalone operators\r\n escaped = escaped.replace(/\\bor\\b/gi, '\\\\or');\r\n escaped = escaped.replace(/\\band\\b/gi, '\\\\and');\r\n return escaped;\r\n}\r\n\r\n/**\r\n * Sanitizes a query string to ensure proper syntax for the Fireberry API\r\n * Handles common syntax issues and removes extraneous elements\r\n *\r\n * @param query - Query string to sanitize\r\n * @returns Sanitized query string\r\n */\r\nexport function sanitizeQuery(query: string): string {\r\n if (!query) {\r\n return '';\r\n }\r\n\r\n // First, protect special operators from being modified\r\n // Temporarily mark is-null and is-not-null operators\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+)\\s+(is-null|is-not-null)\\s*\\)/g,\r\n '($1 __SPECIAL_OPERATOR__$2)',\r\n );\r\n\r\n // Also protect text operators like start-with, not-start-with\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+)\\s+(start-with|not-start-with)\\s+([^)]+)\\s*\\)/g,\r\n '($1 __TEXT_OPERATOR__$2 $3)',\r\n );\r\n\r\n // Fix missing operators: (field value) -> (field = value)\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+(?:field|Field|system|System)[0-9]*)\\s+(?!__SPECIAL_OPERATOR__|__TEXT_OPERATOR__)([^()<>=!]+)\\s*\\)/g,\r\n '($1 = $2)',\r\n );\r\n\r\n // Fix with a more general pattern for any field-value pair without operator\r\n query = query.replace(\r\n /\\(\\s*([a-zA-Z0-9_]+)\\s+(?!__SPECIAL_OPERATOR__|__TEXT_OPERATOR__|<=|>=|!=|<|>|=\\s)([^()<>]+)\\s*\\)/g,\r\n '($1 = $2)',\r\n );\r\n\r\n // Restore special operators\r\n query = query.replace(/__SPECIAL_OPERATOR__/g, '');\r\n query = query.replace(/__TEXT_OPERATOR__/g, '');\r\n\r\n // Remove parentheses containing only a comparison operator\r\n query = query.replace(/\\(\\s*(?:<=|>=|!=|<|>|=)\\s*\\)/g, '');\r\n // Remove parentheses containing only text operators\r\n query = query.replace(/\\(\\s*(?:start-with|not-start-with)\\s*\\)/gi, '');\r\n // Remove parentheses containing only logical operators (AND/OR)\r\n query = query.replace(/\\(\\s*(?:and|or)\\s*\\)/gi, '');\r\n // Remove empty parentheses\r\n query = query.replace(/\\(\\s*\\)/g, '');\r\n // Remove logical operators without operands at start/end\r\n query = query.replace(/^\\s*(and|or)\\s*/gi, '');\r\n query = query.replace(/\\s*(and|or)\\s*$/gi, '');\r\n // Remove redundant nested parentheses: ((x)) -> (x)\r\n const nestedPattern = /\\(\\s*\\(([^()]+)\\)\\s*\\)/g;\r\n while (nestedPattern.test(query)) {\r\n query = query.replace(nestedPattern, '($1)');\r\n }\r\n // Collapse multiple spaces\r\n query = query.replace(/\\s+/g, ' ');\r\n return query.trim();\r\n}\r\n\r\n/**\r\n * Gets today's date in YYYY-MM-DD format\r\n */\r\nexport function getToday(): string {\r\n const now = new Date();\r\n const year = now.getFullYear();\r\n const month = String(now.getMonth() + 1).padStart(2, '0');\r\n const day = String(now.getDate()).padStart(2, '0');\r\n return `${year}-${month}-${day}`;\r\n}\r\n\r\n/**\r\n * Gets the start of current week (Monday) in YYYY-MM-DD format\r\n */\r\nexport function getStartOfWeek(): string {\r\n const now = new Date();\r\n const day = now.getDay();\r\n // Adjust so Monday is first day (day 0 = Sunday, so Monday = 1)\r\n const diff = now.getDate() - day + (day === 0 ? -6 : 1);\r\n const monday = new Date(now);\r\n monday.setDate(diff);\r\n const year = monday.getFullYear();\r\n const month = String(monday.getMonth() + 1).padStart(2, '0');\r\n const dayStr = String(monday.getDate()).padStart(2, '0');\r\n return `${year}-${month}-${dayStr}`;\r\n}\r\n\r\n/**\r\n * Gets the start of current month in YYYY-MM-DD format\r\n */\r\nexport function getStartOfMonth(): string {\r\n const now = new Date();\r\n const year = now.getFullYear();\r\n const month = String(now.getMonth() + 1).padStart(2, '0');\r\n return `${year}-${month}-01`;\r\n}\r\n\r\n/**\r\n * Date condition builder for fluent date query construction\r\n */\r\nexport interface DateConditionBuilder {\r\n /** Records from today */\r\n today(): QueryBuilder;\r\n /** Records from this week (Monday to now) */\r\n thisWeek(): QueryBuilder;\r\n /** Records from this month */\r\n thisMonth(): QueryBuilder;\r\n /** Records between two dates (inclusive) */\r\n between(startDate: string, endDate: string): QueryBuilder;\r\n /** Records from the last N days (including today) */\r\n daysAgo(days: number): QueryBuilder;\r\n /** Records before date */\r\n before(date: string): QueryBuilder;\r\n /** Records after date */\r\n after(date: string): QueryBuilder;\r\n /** Records on or before date */\r\n onOrBefore(date: string): QueryBuilder;\r\n /** Records on or after date */\r\n onOrAfter(date: string): QueryBuilder;\r\n}\r\n\r\n/**\r\n * Condition builder for fluent query construction\r\n */\r\nexport interface ConditionBuilder {\r\n /** Equals comparison (=) */\r\n equals(value: string | number): QueryBuilder;\r\n /** Not equals comparison (!=) */\r\n notEquals(value: string | number): QueryBuilder;\r\n /** IN comparison - matches any of the provided values (joined with OR) */\r\n in(values: (string | number)[]): QueryBuilder;\r\n /** Less than comparison (<) - works with numbers and dates */\r\n lessThan(value: string | number): QueryBuilder;\r\n /** Greater than comparison (>) - works with numbers and dates */\r\n greaterThan(value: string | number): QueryBuilder;\r\n /** Less than or equal (<=) - auto-converts pure dates (YYYY-MM-DD) to < nextDay */\r\n lessThanOrEqual(value: string | number): QueryBuilder;\r\n /** Greater than or equal (>=) - works with numbers and dates */\r\n greaterThanOrEqual(value: string | number): QueryBuilder;\r\n /** Contains value (translates to start-with %value) */\r\n contains(value: string): QueryBuilder;\r\n /** Does not contain value (translates to not-start-with %value) */\r\n notContains(value: string): QueryBuilder;\r\n /** Starts with value (start-with) */\r\n startsWith(value: string): QueryBuilder;\r\n /** Does not start with value (not-start-with) */\r\n notStartsWith(value: string): QueryBuilder;\r\n /** Field is null (is-null) */\r\n isNull(): QueryBuilder;\r\n /** Field is not null (is-not-null) */\r\n isNotNull(): QueryBuilder;\r\n}\r\n\r\n/**\r\n * Internal representation of a query condition\r\n */\r\ninterface QueryCondition {\r\n field: string;\r\n operator: QueryOperator;\r\n value?: string;\r\n}\r\n\r\n/**\r\n * Fluent query builder for constructing Fireberry queries\r\n *\r\n * @example\r\n * ```typescript\r\n * // Build a query string\r\n * const query = new QueryBuilder()\r\n * .where('statuscode').equals('1')\r\n * .and()\r\n * .where('emailaddress1').contains('@example.com')\r\n * .build();\r\n * // Output: \"(statuscode = 1) and (emailaddress1 start-with %@example.com)\"\r\n *\r\n * // With select and execute (requires client)\r\n * const result = await new QueryBuilder(client)\r\n * .objectType('1')\r\n * .select('accountid', 'name', 'emailaddress1')\r\n * .where('statuscode').equals('1')\r\n * .limit(100)\r\n * .execute();\r\n * ```\r\n */\r\n/**\r\n * Client interface for query execution\r\n */\r\ninterface QueryClient {\r\n query(options: {\r\n objectType: string;\r\n fields: string[];\r\n query: string;\r\n showRealValue: boolean;\r\n sortBy?: string;\r\n sortType?: 'asc' | 'desc';\r\n limit?: number;\r\n page?: number;\r\n signal?: AbortSignal;\r\n }): Promise<QueryResult>;\r\n}\r\n\r\n/**\r\n * Query result type\r\n */\r\ninterface QueryResult {\r\n records: Record<string, unknown>[];\r\n total: number;\r\n success: boolean;\r\n}\r\n\r\nexport class QueryBuilder {\r\n private conditions: QueryCondition[] = [];\r\n private joinOperators: ('and' | 'or')[] = [];\r\n private currentField: string | null = null;\r\n private selectedFields: string[] = [];\r\n private objectTypeId: string | null = null;\r\n private sortByField: string | null = null;\r\n private sortDirection: 'asc' | 'desc' = 'desc';\r\n private limitValue: number | null = null;\r\n private pageNumber: number = 1;\r\n private showRealValueFlag: boolean = true;\r\n private client: QueryClient | null = null;\r\n\r\n /**\r\n * Creates a new QueryBuilder\r\n * @param client - Optional FireberryClient for executing queries\r\n */\r\n constructor(client?: QueryClient) {\r\n this.client = client ?? null;\r\n }\r\n\r\n /**\r\n * Sets the object type for the query\r\n * @param objectType - Object type ID (e.g., '1' for Account)\r\n */\r\n objectType(objectType: string | number): this {\r\n this.objectTypeId = String(objectType);\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds fields to select\r\n * @param fields - Field names to select\r\n */\r\n select(...fields: string[]): this {\r\n this.selectedFields.push(...fields);\r\n return this;\r\n }\r\n\r\n /**\r\n * Starts a new WHERE condition\r\n * @param field - Field name to filter on\r\n */\r\n where(field: string): ConditionBuilder {\r\n this.currentField = field;\r\n return this.createConditionBuilder();\r\n }\r\n\r\n /**\r\n * Starts a new WHERE condition for a date field with date-specific helpers\r\n * @param field - Date field name to filter on\r\n *\r\n * @example\r\n * ```typescript\r\n * // Records created today\r\n * qb.whereDate('createdon').today()\r\n *\r\n * // Records from this week\r\n * qb.whereDate('createdon').thisWeek()\r\n *\r\n * // Records from the last 30 days\r\n * qb.whereDate('createdon').daysAgo(30)\r\n *\r\n * // Records between two dates\r\n * qb.whereDate('createdon').between('2024-01-01', '2024-12-31')\r\n * ```\r\n */\r\n whereDate(field: string): DateConditionBuilder {\r\n return this.createDateConditionBuilder(field);\r\n }\r\n\r\n /**\r\n * Adds a WHERE condition for the primary ID field, automatically mapped based on object type\r\n * @param value - The ID value to match\r\n * @throws Error if objectType is not set\r\n *\r\n * @example\r\n * ```typescript\r\n * // Instead of knowing the exact field name:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereId('abc123') // Automatically uses 'accountid' for object type 1\r\n * .execute();\r\n *\r\n * // Equivalent to:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .where('accountid').equals('abc123')\r\n * .execute();\r\n * ```\r\n */\r\n whereId(value: string | number): this {\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type must be set before using whereId(). Call .objectType() first.');\r\n }\r\n const idField = getObjectIdFieldName(this.objectTypeId);\r\n this.addCondition(idField, '=', String(value));\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds WHERE conditions for multiple primary ID values, joined with OR\r\n * @param values - Array of ID values to match\r\n * @throws Error if objectType is not set\r\n * @throws Error if values array is empty\r\n *\r\n * @example\r\n * ```typescript\r\n * // Query multiple accounts by ID:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereIds(['id1', 'id2', 'id3'])\r\n * .execute();\r\n *\r\n * // Generates: (accountid = id1) or (accountid = id2) or (accountid = id3)\r\n *\r\n * // Can be combined with other conditions:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereIds(['id1', 'id2'])\r\n * .and()\r\n * .where('statuscode').equals('1')\r\n * .execute();\r\n * ```\r\n */\r\n whereIds(values: (string | number)[]): this {\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type must be set before using whereIds(). Call .objectType() first.');\r\n }\r\n if (!values || values.length === 0) {\r\n throw new Error('whereIds() requires at least one ID value.');\r\n }\r\n const idField = getObjectIdFieldName(this.objectTypeId);\r\n\r\n // Add first condition\r\n this.addCondition(idField, '=', String(values[0]));\r\n\r\n // Add remaining conditions with OR\r\n for (let i = 1; i < values.length; i++) {\r\n this.joinOperators.push('or');\r\n this.addCondition(idField, '=', String(values[i]));\r\n }\r\n\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds a WHERE IN condition for a field with multiple values, joined with OR\r\n * @param field - Field name to filter on\r\n * @param values - Array of values to match\r\n * @throws Error if values array is empty\r\n *\r\n * @example\r\n * ```typescript\r\n * // Query accounts with specific status codes:\r\n * new QueryBuilder(client)\r\n * .objectType(1)\r\n * .whereIn('statuscode', [1, 2, 3])\r\n * .execute();\r\n *\r\n * // Generates: (statuscode = 1) or (statuscode = 2) or (statuscode = 3)\r\n * ```\r\n */\r\n whereIn(field: string, values: (string | number)[]): this {\r\n if (!values || values.length === 0) {\r\n throw new Error('whereIn() requires at least one value.');\r\n }\r\n // Add first condition\r\n this.addCondition(field, '=', String(values[0]));\r\n // Add remaining conditions with OR\r\n for (let i = 1; i < values.length; i++) {\r\n this.joinOperators.push('or');\r\n this.addCondition(field, '=', String(values[i]));\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds an AND logical operator\r\n */\r\n and(): this {\r\n if (this.conditions.length > 0) {\r\n this.joinOperators.push('and');\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Adds an OR logical operator\r\n */\r\n or(): this {\r\n if (this.conditions.length > 0) {\r\n this.joinOperators.push('or');\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the sort field and direction\r\n * @param field - Field to sort by\r\n * @param direction - Sort direction ('asc' or 'desc')\r\n */\r\n sortBy(field: string, direction: 'asc' | 'desc' = 'desc'): this {\r\n this.sortByField = field;\r\n this.sortDirection = direction;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the maximum number of records to return\r\n * @param count - Maximum record count\r\n */\r\n limit(count: number): this {\r\n this.limitValue = count;\r\n return this;\r\n }\r\n\r\n /**\r\n * Sets the page number for pagination\r\n * @param page - Page number (1-based)\r\n */\r\n page(page: number): this {\r\n this.pageNumber = page;\r\n return this;\r\n }\r\n\r\n /**\r\n * Controls whether to show real values (labels) for dropdown fields\r\n * @param show - Whether to show real values (default: true)\r\n */\r\n showRealValue(show: boolean): this {\r\n this.showRealValueFlag = show;\r\n return this;\r\n }\r\n\r\n /**\r\n * Builds the query string from conditions\r\n * @returns The built query string\r\n */\r\n build(): string {\r\n if (this.conditions.length === 0) {\r\n return '';\r\n }\r\n\r\n const parts: string[] = [];\r\n\r\n for (let i = 0; i < this.conditions.length; i++) {\r\n const condition = this.conditions[i];\r\n let conditionStr: string;\r\n\r\n if (condition.operator === 'is-null' || condition.operator === 'is-not-null') {\r\n conditionStr = `(${condition.field} ${condition.operator})`;\r\n } else {\r\n const escapedValue = escapeQueryValue(condition.value || '');\r\n conditionStr = `(${condition.field} ${condition.operator} ${escapedValue})`;\r\n }\r\n\r\n parts.push(conditionStr);\r\n\r\n // Add join operator if there's a next condition\r\n if (i < this.joinOperators.length) {\r\n parts.push(this.joinOperators[i]);\r\n }\r\n }\r\n\r\n return parts.join(' ');\r\n }\r\n\r\n /**\r\n * Returns the selected fields array\r\n * Useful for inspecting the query configuration\r\n */\r\n getFields(): string[] {\r\n return [...this.selectedFields];\r\n }\r\n\r\n /**\r\n * Converts the query builder state to a payload compatible with @fireberry/sdk\r\n *\r\n * @returns Object with `fields` (comma-separated string) and `query` (filter string)\r\n *\r\n * @example\r\n * ```typescript\r\n * import FireberryClientSDK from '@fireberry/sdk/client';\r\n * import { QueryBuilder } from 'fireberry-api-client';\r\n *\r\n * const sdk = new FireberryClientSDK();\r\n * await sdk.initializeContext();\r\n *\r\n * const payload = new QueryBuilder()\r\n * .select('accountid', 'accountname')\r\n * .where('statuscode').equals('1')\r\n * .toSDKPayload();\r\n *\r\n * // Use with SDK\r\n * const results = await sdk.api.query(1, payload);\r\n * ```\r\n */\r\n toSDKPayload(): { fields: string; query: string; page_size?: number; page_number?: number } {\r\n const payload: { fields: string; query: string; page_size?: number; page_number?: number } = {\r\n fields: this.selectedFields.length > 0 ? this.selectedFields.join(',') : '*',\r\n query: this.build(),\r\n };\r\n\r\n if (this.limitValue !== null) {\r\n payload.page_size = this.limitValue;\r\n }\r\n\r\n if (this.pageNumber > 1) {\r\n payload.page_number = this.pageNumber;\r\n }\r\n\r\n return payload;\r\n }\r\n\r\n /**\r\n * Executes the query and returns the count of matching records\r\n * Uses minimal field selection (ID only) for efficiency\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns Number of matching records\r\n *\r\n * @example\r\n * ```typescript\r\n * const activeCount = await client.queryBuilder()\r\n * .objectType(1)\r\n * .where('statuscode').equals('1')\r\n * .count();\r\n *\r\n * console.log(`Found ${activeCount} active accounts`);\r\n * ```\r\n */\r\n async count(signal?: AbortSignal): Promise<number> {\r\n if (!this.client) {\r\n throw new Error('QueryBuilder requires a client to execute queries. Pass a FireberryClient to the constructor.');\r\n }\r\n\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type is required. Use .objectType() before executing.');\r\n }\r\n\r\n // Use minimal fields for efficiency - just get the ID field\r\n const idField = getObjectIdFieldName(this.objectTypeId);\r\n\r\n const result = await this.client.query({\r\n objectType: this.objectTypeId,\r\n fields: [idField],\r\n query: this.build(),\r\n showRealValue: false, // No need for labels\r\n signal,\r\n });\r\n\r\n return result.total;\r\n }\r\n\r\n /**\r\n * Executes the query and returns the first record or null\r\n * Automatically sets limit to 1 for efficiency\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns First record or null if no records found\r\n *\r\n * @example\r\n * ```typescript\r\n * const account = await client.queryBuilder()\r\n * .objectType(1)\r\n * .where('accountname').equals('Acme Corp')\r\n * .first();\r\n *\r\n * if (account) {\r\n * console.log(account.accountid);\r\n * }\r\n * ```\r\n */\r\n async first(signal?: AbortSignal): Promise<Record<string, unknown> | null> {\r\n // Temporarily set limit to 1 for efficiency\r\n const originalLimit = this.limitValue;\r\n this.limitValue = 1;\r\n\r\n try {\r\n const result = await this.execute(signal);\r\n return result.records[0] ?? null;\r\n } finally {\r\n // Restore original limit\r\n this.limitValue = originalLimit;\r\n }\r\n }\r\n\r\n /**\r\n * Executes the query (requires client to be set)\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns Query results\r\n */\r\n async execute(signal?: AbortSignal): Promise<QueryResult> {\r\n if (!this.client) {\r\n throw new Error('QueryBuilder requires a client to execute queries. Pass a FireberryClient to the constructor.');\r\n }\r\n\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type is required. Use .objectType() before executing.');\r\n }\r\n\r\n const queryOptions: Parameters<QueryClient['query']>[0] = {\r\n objectType: this.objectTypeId,\r\n fields: this.selectedFields.length > 0 ? this.selectedFields : ['*'],\r\n query: this.build(),\r\n showRealValue: this.showRealValueFlag,\r\n };\r\n\r\n if (this.sortByField) {\r\n queryOptions.sortBy = this.sortByField;\r\n queryOptions.sortType = this.sortDirection;\r\n }\r\n\r\n if (this.limitValue !== null) {\r\n queryOptions.limit = this.limitValue;\r\n }\r\n\r\n if (this.pageNumber > 1) {\r\n queryOptions.page = this.pageNumber;\r\n }\r\n\r\n if (signal) {\r\n queryOptions.signal = signal;\r\n }\r\n\r\n return this.client.query(queryOptions);\r\n }\r\n\r\n /**\r\n * Executes the query and returns results with debugging metadata\r\n * Includes query string, fields, pagination info, and execution time\r\n * @param signal - Optional AbortSignal for cancellation\r\n * @returns Query results with metadata\r\n *\r\n * @example\r\n * ```typescript\r\n * const result = await client.queryBuilder()\r\n * .objectType(1)\r\n * .select('accountid', 'accountname')\r\n * .where('statuscode').equals('1')\r\n * .executeWithDebug();\r\n *\r\n * console.log('Query:', result.metadata.queryString);\r\n * console.log('Fields:', result.metadata.fields);\r\n * console.log('Execution time:', result.metadata.executionTimeMs, 'ms');\r\n * console.log('Records:', result.records.length);\r\n * ```\r\n */\r\n async executeWithDebug(signal?: AbortSignal): Promise<QueryResultWithMetadata> {\r\n if (!this.client) {\r\n throw new Error('QueryBuilder requires a client to execute queries. Pass a FireberryClient to the constructor.');\r\n }\r\n\r\n if (!this.objectTypeId) {\r\n throw new Error('Object type is required. Use .objectType() before executing.');\r\n }\r\n\r\n const startTime = Date.now();\r\n const fields = this.selectedFields.length > 0 ? this.selectedFields : ['*'];\r\n const queryString = this.build();\r\n\r\n const queryOptions: Parameters<QueryClient['query']>[0] = {\r\n objectType: this.objectTypeId,\r\n fields,\r\n query: queryString,\r\n showRealValue: this.showRealValueFlag,\r\n };\r\n\r\n if (this.sortByField) {\r\n queryOptions.sortBy = this.sortByField;\r\n queryOptions.sortType = this.sortDirection;\r\n }\r\n\r\n if (this.limitValue !== null) {\r\n queryOptions.limit = this.limitValue;\r\n }\r\n\r\n if (this.pageNumber > 1) {\r\n queryOptions.page = this.pageNumber;\r\n }\r\n\r\n if (signal) {\r\n queryOptions.signal = signal;\r\n }\r\n\r\n const result = await this.client.query(queryOptions);\r\n const executionTimeMs = Date.now() - startTime;\r\n\r\n const metadata: QueryMetadata = {\r\n objectType: this.objectTypeId,\r\n fields,\r\n queryString,\r\n pageNumber: this.pageNumber,\r\n pageSize: this.limitValue ?? 500,\r\n autoPage: true, // Default behavior\r\n executionTimeMs,\r\n };\r\n\r\n if (this.sortByField) {\r\n metadata.sortBy = this.sortByField;\r\n metadata.sortType = this.sortDirection;\r\n }\r\n\r\n if (this.limitValue !== null) {\r\n metadata.limit = this.limitValue;\r\n }\r\n\r\n return {\r\n ...result,\r\n metadata,\r\n };\r\n }\r\n\r\n /**\r\n * Analyzes the query without executing it (dry run)\r\n * Returns information about what the query will do, potential issues, and optimization suggestions\r\n *\r\n * @returns Query analysis with warnings and suggestions\r\n *\r\n * @example\r\n * ```typescript\r\n * const analysis = client.queryBuilder()\r\n * .objectType('1')\r\n * .select('*')\r\n * .where('statuscode').equals('1')\r\n * .explain();\r\n *\r\n * console.log('Query:', analysis.query);\r\n * console.log('Warnings:', analysis.warnings);\r\n * console.log('Suggestions:', analysis.suggestions);\r\n * console.log('Estimated API calls:', analysis.estimatedApiCalls);\r\n * ```\r\n */\r\n explain(): QueryExplainResult {\r\n const warnings: string[] = [];\r\n const suggestions: string[] = [];\r\n\r\n // Check object type\r\n if (!this.objectTypeId) {\r\n warnings.push('Object type is not set. Call .objectType() before executing.');\r\n }\r\n\r\n // Analyze fields\r\n const fields = this.selectedFields.length > 0 ? this.selectedFields : ['*'];\r\n const usesWildcard = fields.includes('*') || fields.length === 0;\r\n\r\n if (usesWildcard) {\r\n warnings.push('Using wildcard (*) fields may include unnecessary data and slow down queries.');\r\n suggestions.push('Consider selecting only the specific fields you need with .select()');\r\n }\r\n\r\n // Analyze query\r\n const queryString = this.build();\r\n const conditionCount = this.conditions.length;\r\n\r\n if (conditionCount === 0 && !this.limitValue) {\r\n warnings.push('No query conditions or limit set. This may return a large number of records.');\r\n suggestions.push('Add filters with .where() or set a .limit() to control result size.');\r\n }\r\n\r\n // Check for potential performance issues\r\n const hasContainsOperator = this.conditions.some(\r\n (c) => c.operator === 'start-with' && c.value?.startsWith('%'),\r\n );\r\n if (hasContainsOperator) {\r\n warnings.push('Contains queries (using % prefix) may be slower than exact matches.');\r\n }\r\n\r\n // Check for OR conditions (may affect performance)\r\n const hasOrConditions = this.joinOperators.some((op) => op === 'or');\r\n if (hasOrConditions && conditionCount > 5) {\r\n warnings.push('Multiple OR conditions may affect query performance.');\r\n suggestions.push('Consider breaking into separate queries if possible.');\r\n }\r\n\r\n // Sorting analysis\r\n const sortField = this.sortByField || 'modifiedon';\r\n const sortDirection = this.sortDirection;\r\n\r\n // Estimate API calls\r\n let estimatedApiCalls = 1;\r\n const pageSize = 500; // Default page size\r\n const willAutoPage = this.limitValue === null; // Auto-page when no limit\r\n\r\n if (willAutoPage && conditionCount === 0) {\r\n // Without conditions, could be many pages\r\n estimatedApiCalls = -1; // Unknown/many\r\n warnings.push('Without filters, query may require many API calls for pagination.');\r\n } else if (this.limitValue !== null) {\r\n estimatedApiCalls = Math.ceil(this.limitValue / pageSize);\r\n }\r\n\r\n // Check for missing index hints\r\n if (!this.sortByField && conditionCount > 0) {\r\n suggestions.push('Consider adding .sortBy() to control result ordering.');\r\n }\r\n\r\n // Check showRealValue impact\r\n if (this.showRealValueFlag) {\r\n suggestions.push('showRealValue is enabled (default). Set .showRealValue(false) if you only need IDs.');\r\n }\r\n\r\n return {\r\n objectType: this.objectTypeId || '(not set)',\r\n query: queryString || '(no conditions)',\r\n fields,\r\n usesWildcard,\r\n willAutoPage,\r\n limit: this.limitValue,\r\n pageSize,\r\n sorting: {\r\n field: sortField,\r\n direction: sortDirection,\r\n },\r\n estimatedApiCalls,\r\n warnings,\r\n suggestions,\r\n conditionCount,\r\n showRealValue: this.showRealValueFlag,\r\n };\r\n }\r\n\r\n /**\r\n * Creates a condition builder for the current field\r\n */\r\n private createConditionBuilder(): ConditionBuilder {\r\n const field = this.currentField!;\r\n\r\n return {\r\n equals: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '=', String(value));\r\n return this;\r\n },\r\n notEquals: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '!=', String(value));\r\n return this;\r\n },\r\n in: (values: (string | number)[]): QueryBuilder => {\r\n if (!values || values.length === 0) {\r\n throw new Error('in() requires at least one value.');\r\n }\r\n // Add first condition\r\n this.addCondition(field, '=', String(values[0]));\r\n // Add remaining conditions with OR\r\n for (let i = 1; i < values.length; i++) {\r\n this.joinOperators.push('or');\r\n this.addCondition(field, '=', String(values[i]));\r\n }\r\n return this;\r\n },\r\n lessThan: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '<', String(value));\r\n return this;\r\n },\r\n greaterThan: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '>', String(value));\r\n return this;\r\n },\r\n lessThanOrEqual: (value: string | number): QueryBuilder => {\r\n const strValue = String(value);\r\n // Fireberry API bug: <= with pure dates (YYYY-MM-DD) behaves like <\r\n // because it interprets the date as midnight (00:00:00).\r\n // Auto-convert to < nextDay for correct \"on or before\" behavior.\r\n if (isPureDate(strValue)) {\r\n const nextDay = addDays(strValue, 1);\r\n this.addCondition(field, '<', nextDay);\r\n } else {\r\n this.addCondition(field, '<=', strValue);\r\n }\r\n return this;\r\n },\r\n greaterThanOrEqual: (value: string | number): QueryBuilder => {\r\n this.addCondition(field, '>=', String(value));\r\n return this;\r\n },\r\n contains: (value: string): QueryBuilder => {\r\n // Contains translates to \"start-with %value\"\r\n this.addCondition(field, 'start-with', `%${value}`);\r\n return this;\r\n },\r\n notContains: (value: string): QueryBuilder => {\r\n // Not contains translates to \"not-start-with %value\"\r\n this.addCondition(field, 'not-start-with', `%${value}`);\r\n return this;\r\n },\r\n startsWith: (value: string): QueryBuilder => {\r\n this.addCondition(field, 'start-with', value);\r\n return this;\r\n },\r\n notStartsWith: (value: string): QueryBuilder => {\r\n this.addCondition(field, 'not-start-with', value);\r\n return this;\r\n },\r\n isNull: (): QueryBuilder => {\r\n this.addCondition(field, 'is-null');\r\n return this;\r\n },\r\n isNotNull: (): QueryBuilder => {\r\n this.addCondition(field, 'is-not-null');\r\n return this;\r\n },\r\n };\r\n }\r\n\r\n /**\r\n * Creates a date condition builder for the specified field\r\n */\r\n private createDateConditionBuilder(field: string): DateConditionBuilder {\r\n return {\r\n today: (): QueryBuilder => {\r\n const today = getToday();\r\n // Records from today: >= today AND < tomorrow\r\n this.addCondition(field, '>=', today);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', addDays(today, 1));\r\n return this;\r\n },\r\n thisWeek: (): QueryBuilder => {\r\n const startOfWeek = getStartOfWeek();\r\n const tomorrow = addDays(getToday(), 1);\r\n // Records from start of week to now: >= monday AND < tomorrow\r\n this.addCondition(field, '>=', startOfWeek);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', tomorrow);\r\n return this;\r\n },\r\n thisMonth: (): QueryBuilder => {\r\n const startOfMonth = getStartOfMonth();\r\n const tomorrow = addDays(getToday(), 1);\r\n // Records from start of month to now: >= first day AND < tomorrow\r\n this.addCondition(field, '>=', startOfMonth);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', tomorrow);\r\n return this;\r\n },\r\n between: (startDate: string, endDate: string): QueryBuilder => {\r\n // Records between dates (inclusive): >= start AND < day after end\r\n this.addCondition(field, '>=', startDate);\r\n this.joinOperators.push('and');\r\n // Use < next day for inclusive end date (handles API quirk)\r\n const dayAfterEnd = isPureDate(endDate) ? addDays(endDate, 1) : endDate;\r\n this.addCondition(field, '<', dayAfterEnd);\r\n return this;\r\n },\r\n daysAgo: (days: number): QueryBuilder => {\r\n const today = getToday();\r\n const startDate = addDays(today, -days);\r\n const tomorrow = addDays(today, 1);\r\n // Records from N days ago to now: >= (today - N) AND < tomorrow\r\n this.addCondition(field, '>=', startDate);\r\n this.joinOperators.push('and');\r\n this.addCondition(field, '<', tomorrow);\r\n return this;\r\n },\r\n before: (date: string): QueryBuilder => {\r\n this.addCondition(field, '<', date);\r\n return this;\r\n },\r\n after: (date: string): QueryBuilder => {\r\n // After a date means > end of that day\r\n // For pure dates, use > date (API treats as > midnight, so actually > end of previous day)\r\n // We need >= next day for \"after\" to work correctly\r\n const nextDay = isPureDate(date) ? addDays(date, 1) : date;\r\n this.addCondition(field, '>=', nextDay);\r\n return this;\r\n },\r\n onOrBefore: (date: string): QueryBuilder => {\r\n // Use < next day for inclusive (handles API quirk with <=)\r\n const nextDay = isPureDate(date) ? addDays(date, 1) : date;\r\n this.addCondition(field, '<', nextDay);\r\n return this;\r\n },\r\n onOrAfter: (date: string): QueryBuilder => {\r\n this.addCondition(field, '>=', date);\r\n return this;\r\n },\r\n };\r\n }\r\n\r\n /**\r\n * Adds a condition to the query\r\n */\r\n private addCondition(field: string, operator: QueryOperator, value?: string): void {\r\n this.conditions.push({ field, operator, value });\r\n this.currentField = null;\r\n }\r\n}\r\n","/**\r\n * Special field name mappings that don't follow standard patterns\r\n * Key: original field name, Value: corresponding label field\r\n */\r\nconst SPECIAL_LABEL_FIELD_MAPPINGS: Record<string, string> = {\r\n objectid: 'objecttitle',\r\n lastactionid: 'lastactiontitle',\r\n wfruleid: 'rulename', // Object 55 (Workflow Rules)\r\n noteid: 'subject', // Object 7 (Note) - noteid maps to subject\r\n};\r\n\r\n/**\r\n * Fields ending with \"id\" that should NOT have \"id\" replaced with \"name\"\r\n * These fields will use the default append \"name\" behavior instead\r\n */\r\nconst EXCLUDED_ID_FIELDS: string[] = [\r\n 'businessunitid', // → businessunitidname (not businessunitname)\r\n 'crmuserid', // → no name field exists\r\n 'languageid', // → languageidname (not languagename)\r\n];\r\n\r\n/**\r\n * Fields ending with \"code\" that should NOT have \"code\" removed\r\n * These fields will use the default append \"name\" behavior instead\r\n */\r\nconst EXCLUDED_CODE_FIELDS: string[] = [\r\n 'duplicaterecordcode', // → duplicaterecordcodename (not duplicaterecord)\r\n];\r\n\r\n/**\r\n * Fields that have NO corresponding label field at all\r\n * These fields should not have any name transformation applied\r\n * Return empty string to signal \"no label field exists\"\r\n */\r\nconst FIELDS_WITHOUT_LABEL_FIELD: string[] = [\r\n 'systemfieldid', // Object 73 - no name field exists\r\n 'fieldobjecttype', // Object 73 - base field doesn't exist, only *name exists\r\n 'invoiceid', // Objects 78, 81 - no name field exists\r\n 'calllogid', // Object 100 - no name field exists (primary key)\r\n 'attendanceclockid', // Object 101 - no name field exists (primary key)\r\n 'activitylogid', // Object 102 - no name field exists (primary key)\r\n 'conversationid', // Object 104 - no name field exists (primary key)\r\n 'texttemplateid', // Text Template - no name field exists (primary key)\r\n 'smstemplateid', // Object 110 - no name field exists (primary key)\r\n 'deletedby', // Object 7 - no name field exists\r\n 'recordid', // Object 7 - no name field exists\r\n 'objecttypecode', // Object 7 - no name field exists\r\n];\r\n\r\n/**\r\n * Object-type specific overrides for field name transformations\r\n * Key: object type ID, Value: { excludedIdFields, excludedCodeFields }\r\n */\r\nconst OBJECT_TYPE_OVERRIDES: Record<\r\n number,\r\n { excludedIdFields?: string[]; excludedCodeFields?: string[] }\r\n> = {\r\n // Account (1) - uses *codename pattern for some fields\r\n 1: {\r\n excludedCodeFields: ['actionstatuscode', 'businesstypecode'],\r\n },\r\n // CRM Orders (13) - uses *idname and *codename patterns\r\n 13: {\r\n excludedIdFields: ['ownerid', 'accountid', 'printtemplateid', 'pcfaccountid'],\r\n excludedCodeFields: [\r\n 'rounddiscountcode',\r\n 'taxincludecode',\r\n 'currencycode',\r\n 'transmissioncode',\r\n 'creditrejectioncode1',\r\n 'creditrejectioncode2',\r\n 'apartmentcode1',\r\n 'apartmentcode2',\r\n 'restrictioncode1',\r\n 'restrictioncode2',\r\n 'casescode1',\r\n 'casescode2',\r\n 'returncode1',\r\n 'returncode2',\r\n 'statuscode',\r\n ],\r\n },\r\n // Products (14)\r\n 14: {\r\n excludedCodeFields: ['categorycode'], // → categoryname (not category)\r\n },\r\n // CRM Order Items (17) - uses *idname pattern\r\n 17: {\r\n excludedIdFields: ['crmorderid', 'productid', 'ownerid'],\r\n },\r\n // Email Templates (20)\r\n 20: {\r\n excludedIdFields: ['mdobjectid'],\r\n },\r\n // Print Templates (27)\r\n 27: {\r\n excludedIdFields: ['mdobjectid'],\r\n },\r\n // System Fields (73) - uses *idname pattern\r\n 73: {\r\n excludedIdFields: ['mdobjectid', 'ownerid'],\r\n },\r\n // Invoices (78) - uses *codename and *idname patterns extensively\r\n 78: {\r\n excludedIdFields: [\r\n 'crmorderid',\r\n 'invoicereceiptid',\r\n 'accountid',\r\n 'ownerid',\r\n 'invoicerenoid',\r\n ],\r\n excludedCodeFields: [\r\n 'taxincludecode',\r\n 'depositcode',\r\n 'rounddiscountcode',\r\n 'currencycode',\r\n 'statecode',\r\n 'invoicetypecode',\r\n ],\r\n },\r\n // Invoice No (81) - uses *codename and *idname patterns extensively\r\n 81: {\r\n excludedIdFields: [\r\n 'ownerid',\r\n 'invoicecreditid',\r\n 'crmorderid',\r\n 'invoicereceiptid',\r\n 'invoicedeliveryid',\r\n 'accountid',\r\n ],\r\n excludedCodeFields: ['currencycode', 'rounddiscountcode', 'statecode', 'taxincludecode'],\r\n },\r\n // Invoice Draft (82) - uses *codename and *idname patterns extensively\r\n 82: {\r\n excludedIdFields: ['accountid', 'crmorderid', 'ownerid', 'invoicerenoid'],\r\n excludedCodeFields: ['taxincludecode', 'statecode', 'rounddiscountcode', 'currencycode'],\r\n },\r\n // Invoice Receipt (83) - uses *codename and *idname patterns\r\n 83: {\r\n excludedIdFields: ['invoicenoid', 'crmorderid', 'ownerid', 'accountid'],\r\n excludedCodeFields: ['statecode', 'currencycode'],\r\n },\r\n // Invoice Tax Receipt (84) - uses *codename and *idname patterns\r\n 84: {\r\n excludedIdFields: ['crmorderid', 'ownerid', 'invoicecreditid', 'accountid'],\r\n excludedCodeFields: ['rounddiscountcode', 'statecode', 'taxincludecode', 'currencycode'],\r\n },\r\n // Invoice Credit (85) - uses *codename and *idname patterns\r\n 85: {\r\n excludedIdFields: ['ownerid', 'crmorderid', 'accountid'],\r\n excludedCodeFields: ['taxincludecode', 'rounddiscountcode', 'currencycode', 'statecode'],\r\n },\r\n // Invoice (86) - uses *codename and *idname patterns\r\n 86: {\r\n excludedIdFields: ['accountid', 'ownerid', 'crmorderid'],\r\n excludedCodeFields: ['statecode', 'rounddiscountcode', 'currencycode', 'taxincludecode'],\r\n },\r\n // Call Log (100) - uses *idname pattern\r\n 100: {\r\n excludedIdFields: ['contactid', 'leadid', 'accountid', 'ownerid'],\r\n },\r\n // Attendance Clock (101) - uses *idname pattern\r\n 101: {\r\n excludedIdFields: ['ownerid'],\r\n },\r\n // Activity Log (102) - uses *idname and *codename patterns\r\n 102: {\r\n excludedIdFields: ['contactid', 'ownerid'],\r\n excludedCodeFields: ['objecttypecode', 'typecode', 'resultcode'],\r\n },\r\n // Conversation (104) - uses *idname pattern\r\n 104: {\r\n excludedIdFields: ['leadid', 'ownerid', 'contactid', 'accountid'],\r\n },\r\n // Text Template (106) - uses *idname pattern\r\n 106: {\r\n excludedIdFields: ['ownerid'],\r\n },\r\n // SMS Template (110) - uses *idname pattern\r\n 110: {\r\n excludedIdFields: ['ownerid'],\r\n },\r\n};\r\n\r\n/**\r\n * Converts a field API name to its corresponding label field.\r\n *\r\n * Rules:\r\n * - Special mappings (e.g., objectid → objecttitle)\r\n * - Fields starting with \"pcf\" (custom fields): append \"name\" → pcf_field → pcf_fieldname\r\n * - Fields ending with \"code\" (unless excluded): remove \"code\" → statuscode → status\r\n * - Fields ending with \"id\" (unless excluded): replace \"id\" with \"name\" → accountid → accountname\r\n * - All other fields: append \"name\"\r\n *\r\n * @param fieldName - The field API name\r\n * @param objectType - The object type ID (required for object-specific overrides)\r\n * @returns The corresponding label field, or empty string if no label field exists\r\n */\r\nexport function getLabelFieldForField(fieldName: string, objectType: string | number): string {\r\n // Check special mappings first\r\n if (SPECIAL_LABEL_FIELD_MAPPINGS[fieldName]) {\r\n return SPECIAL_LABEL_FIELD_MAPPINGS[fieldName];\r\n }\r\n\r\n // Check if field has no label field at all - return empty string\r\n if (FIELDS_WITHOUT_LABEL_FIELD.includes(fieldName)) {\r\n return '';\r\n }\r\n\r\n // Check for custom object primary keys (customobject{N}id pattern) → name\r\n if (/^customobject\\d+id$/.test(fieldName)) {\r\n return 'name';\r\n }\r\n\r\n // Custom fields (pcf prefix) - just append \"name\"\r\n if (fieldName.startsWith('pcf')) {\r\n return `${fieldName}name`;\r\n }\r\n\r\n // Get object-type specific overrides\r\n const objectTypeNum =\r\n typeof objectType === 'string' ? parseInt(objectType, 10) : objectType;\r\n const overrides = OBJECT_TYPE_OVERRIDES[objectTypeNum] || null;\r\n\r\n // For custom objects (1000+), add default exclusions\r\n const isCustomObject = objectTypeNum >= 1000;\r\n const customObjectExclusions = isCustomObject ? ['ownerid'] : [];\r\n\r\n // Combine global and object-specific exclusions\r\n const excludedCodeFields = [\r\n ...EXCLUDED_CODE_FIELDS,\r\n ...(overrides?.excludedCodeFields || []),\r\n ];\r\n const excludedIdFields = [\r\n ...EXCLUDED_ID_FIELDS,\r\n ...(overrides?.excludedIdFields || []),\r\n ...customObjectExclusions,\r\n ];\r\n\r\n // Remove \"code\" suffix (unless excluded) → statuscode → status\r\n if (fieldName.endsWith('code') && !excludedCodeFields.includes(fieldName)) {\r\n return fieldName.slice(0, -4);\r\n }\r\n\r\n // Replace \"id\" suffix with \"name\" (unless excluded) → accountid → accountname\r\n if (fieldName.endsWith('id') && !excludedIdFields.includes(fieldName)) {\r\n return fieldName.slice(0, -2) + 'name';\r\n }\r\n\r\n // Default: append \"name\"\r\n return `${fieldName}name`;\r\n}\r\n","/**\r\n * Object Type ID to Name Field Mapping\r\n * Maps Fireberry object type IDs to their display name field\r\n * Generated from actual API responses\r\n */\r\nexport const OBJECT_NAME_MAP: Record<number, string> = {\r\n 1: 'accountname', // Account\r\n 2: 'fullname', // Contact\r\n 3: 'fullname', // Lead\r\n 4: 'name', // Opportunity\r\n 5: 'title', // Cases\r\n 6: 'subject', // Activity\r\n 7: 'notetext', // Note\r\n 8: 'competitorname', // Competitor\r\n 9: 'fullname', // CrmUser\r\n 10: 'subject', // Task\r\n 12: 'quotenumber', // Quote\r\n 13: 'crmordernumber', // CrmOrder\r\n 14: 'name', // Product\r\n 17: 'productname', // CrmOrderItem\r\n 20: 'title', // EmailTemplate\r\n 23: 'name', // BusinessUnit\r\n 25: 'orgname', // Org\r\n 27: 'name', // PrintTemplate\r\n 28: 'contractname', // Contract\r\n 33: 'productid', // AccountProduct (uses productid as display)\r\n 46: 'projectname', // Project\r\n 55: 'rulename', // WFRule\r\n 58: 'name', // MDObject\r\n 64: 'rolename', // Role\r\n 67: 'campaignname', // Campaign\r\n 70: 'browsername', // CrmUserLogin\r\n 73: 'label', // SystemField (using label as name field)\r\n 76: 'articlename', // Article\r\n 77: 'linkname', // Link\r\n 78: 'invoicenumber', // Invoice\r\n 80: 'documentnumber', // InvoiceReceiptItem\r\n 81: 'invoicenumber', // InvoiceNo\r\n 82: 'invoicenumber', // InvoiceDraft\r\n 83: 'invoicenumber', // InvoiceReceipt\r\n 84: 'invoicenumber', // InvoiceReno\r\n 85: 'invoicenumber', // InvoiceCredit\r\n 86: 'invoicenumber', // InvoiceDelivery\r\n 89: 'name', // IpRestriction\r\n 90: 'documentnumber', // TransactionItem\r\n 93: 'name', // Charge\r\n 100: 'callerid', // calllog\r\n 101: 'name', // AttendanceClock\r\n 102: 'activitylognumber', // ActivityLog\r\n 104: 'subject', // Conversation\r\n 105: 'name', // TeamInbox\r\n 106: 'name', // TextTemplate\r\n 107: 'name', // FacebookConnection\r\n 109: 'name', // AuditLog\r\n 110: 'name', // SMSTemplate\r\n 111: 'name', // ProviderVerification\r\n 114: 'name', // CalendarResource\r\n 115: 'name', // Journey\r\n 116: 'name', // Profile\r\n 117: 'name', // LandingPage\r\n};\r\n\r\n/**\r\n * Gets the display name field for a given object type\r\n *\r\n * @param objectTypeId - The numeric object type ID\r\n * @returns The name field for the object type\r\n */\r\nexport function getNameFieldByObjectType(objectTypeId: string | number): string {\r\n const objectTypeNum =\r\n typeof objectTypeId === 'string' ? parseInt(objectTypeId, 10) : objectTypeId;\r\n\r\n // Check if it's a mapped base object\r\n if (OBJECT_NAME_MAP[objectTypeNum]) {\r\n return OBJECT_NAME_MAP[objectTypeNum];\r\n }\r\n\r\n // For custom objects (1000 and up), use 'name'\r\n if (objectTypeNum >= 1000) {\r\n return 'name';\r\n }\r\n\r\n // Fallback to 'name' for unmapped objects\r\n return 'name';\r\n}\r\n","/**\r\n * Field Type System IDs from Fireberry CRM\r\n * These UUIDs identify different field types in the metadata API\r\n */\r\nexport const FIELD_TYPE_IDS = {\r\n DROPDOWN: 'b4919f2e-2996-48e4-a03c-ba39fb64386c',\r\n LOOKUP: 'a8fcdf65-91bc-46fd-82f6-1234758345a1',\r\n EMAIL: 'c713d2f7-8fa9-43c3-8062-f07486eaf567',\r\n TEXT: 'a1e7ed6f-5083-477b-b44c-9943a6181359',\r\n URL: 'c820d32f-44df-4c2a-9c1e-18734e864fd5',\r\n LONG_TEXT: '80108f9d-1e75-40fa-9fa9-02be4ddc1da1',\r\n DATETIME: 'ce972d02-5013-46d4-9d1d-f09df1ac346a',\r\n DATE: '83bf530c-e04c-462b-9ffc-a46f750fc072',\r\n HTML: 'ed2ad39d-32fc-4585-8f5b-2e93463f050a',\r\n TELEPHONE: '3f62f67a-1cee-403a-bec6-aa02a9804edb',\r\n NUMERIC: '6a34bfe3-fece-4da1-9136-a7b1e5ae3319',\r\n} as const;\r\n\r\n/**\r\n * Human-readable mappings for field types\r\n * Used for display purposes\r\n */\r\nexport const FIELD_TYPE_MAPPINGS: Record<string, string> = {\r\n [FIELD_TYPE_IDS.DROPDOWN]: 'Dropdown',\r\n [FIELD_TYPE_IDS.EMAIL]: 'Email',\r\n [FIELD_TYPE_IDS.TEXT]: 'Text',\r\n [FIELD_TYPE_IDS.LOOKUP]: 'Lookup',\r\n [FIELD_TYPE_IDS.URL]: 'URL',\r\n [FIELD_TYPE_IDS.LONG_TEXT]: 'Long Text',\r\n [FIELD_TYPE_IDS.DATETIME]: 'DateTime',\r\n [FIELD_TYPE_IDS.DATE]: 'Date',\r\n [FIELD_TYPE_IDS.HTML]: 'HTML',\r\n [FIELD_TYPE_IDS.TELEPHONE]: 'Telephone',\r\n [FIELD_TYPE_IDS.NUMERIC]: 'Number',\r\n};\r\n","/**\r\n * Related Field Mapping Utilities\r\n *\r\n * Functions for parsing and resolving related object field references.\r\n * Related fields use the pattern: {referenceField}_{relatedField}\r\n * Example: accountid_telephone1, contactid_fullname, accountid_statuscode\r\n */\r\n\r\nimport { OBJECT_ID_MAP } from '../constants/objectIds';\r\nimport { FIELD_TYPE_IDS } from '../constants/fieldTypes';\r\nimport { getLabelFieldForField } from './fieldMapping';\r\nimport type { FireberryField } from '../types/metadata';\r\n\r\n/**\r\n * Reverse mapping from ID field name to object type ID\r\n * Generated from OBJECT_ID_MAP\r\n */\r\nexport const ID_FIELD_TO_OBJECT_TYPE: Record<string, number> = Object.entries(OBJECT_ID_MAP).reduce(\r\n (acc, [objectTypeId, idFieldName]) => {\r\n acc[idFieldName] = parseInt(objectTypeId, 10);\r\n return acc;\r\n },\r\n {} as Record<string, number>,\r\n);\r\n\r\n/**\r\n * Information about a parsed related field\r\n */\r\nexport interface RelatedFieldInfo {\r\n /** The original field name (e.g., \"accountid_telephone1\") */\r\n originalField: string;\r\n /** The reference/lookup field (e.g., \"accountid\") */\r\n referenceField: string;\r\n /** The field on the related object (e.g., \"telephone1\") */\r\n relatedField: string;\r\n /** The object type ID of the related object (e.g., 1 for Account) */\r\n relatedObjectType: number;\r\n}\r\n\r\n/**\r\n * Result of resolving a related field with its label field(s)\r\n */\r\nexport interface RelatedFieldResolution {\r\n /** The original field name */\r\n originalField: string;\r\n /** All fields that should be selected/returned */\r\n fields: string[];\r\n /** The primary value field */\r\n valueField: string;\r\n /** The label/display field (may be same as valueField for non-code fields) */\r\n labelField: string;\r\n /** The code field if applicable (for dropdown fields) */\r\n codeField?: string;\r\n /** Whether this is a code field (ends with 'code') */\r\n isCodeField: boolean;\r\n /** Related object type ID */\r\n relatedObjectType: number;\r\n /** Field type from metadata (if available) */\r\n fieldType?: string;\r\n}\r\n\r\n/**\r\n * Field metadata map type - maps field name to field info\r\n */\r\nexport type FieldMetadataMap = Map<string, FireberryField>;\r\n\r\n/**\r\n * Gets the object type ID from a reference field name\r\n *\r\n * @param referenceField - The reference field name (e.g., \"accountid\", \"contactid\")\r\n * @returns The object type ID, or null if not a known reference field\r\n *\r\n * @example\r\n * getObjectTypeFromReferenceField('accountid') // 1\r\n * getObjectTypeFromReferenceField('contactid') // 2\r\n * getObjectTypeFromReferenceField('leadid') // 3\r\n * getObjectTypeFromReferenceField('unknown') // null\r\n */\r\nexport function getObjectTypeFromReferenceField(referenceField: string): number | null {\r\n // Check standard mappings\r\n if (ID_FIELD_TO_OBJECT_TYPE[referenceField]) {\r\n return ID_FIELD_TO_OBJECT_TYPE[referenceField];\r\n }\r\n\r\n // Check for custom object pattern: customobject{N}id\r\n const customMatch = referenceField.match(/^customobject(\\d+)id$/);\r\n if (customMatch) {\r\n return parseInt(customMatch[1], 10);\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * Parses a related field name into its components\r\n *\r\n * @param fieldName - The field name to parse (e.g., \"accountid_telephone1\")\r\n * @returns RelatedFieldInfo if it's a valid related field, null otherwise\r\n *\r\n * @example\r\n * parseRelatedField('accountid_telephone1')\r\n * // { originalField: 'accountid_telephone1', referenceField: 'accountid',\r\n * // relatedField: 'telephone1', relatedObjectType: 1 }\r\n *\r\n * parseRelatedField('accountid_statuscode')\r\n * // { originalField: 'accountid_statuscode', referenceField: 'accountid',\r\n * // relatedField: 'statuscode', relatedObjectType: 1 }\r\n *\r\n * parseRelatedField('telephone1') // null (not a related field)\r\n */\r\nexport function parseRelatedField(fieldName: string): RelatedFieldInfo | null {\r\n // Must contain underscore\r\n const underscoreIndex = fieldName.indexOf('_');\r\n if (underscoreIndex === -1) {\r\n return null;\r\n }\r\n\r\n // Try to find a valid reference field by checking progressively longer prefixes\r\n // This handles cases like pcf_accountid_status where we need to find the right split\r\n const parts = fieldName.split('_');\r\n\r\n // Start from the first part and try combinations\r\n for (let i = 0; i < parts.length - 1; i++) {\r\n const potentialRefField = parts.slice(0, i + 1).join('_');\r\n const objectType = getObjectTypeFromReferenceField(potentialRefField);\r\n\r\n if (objectType !== null) {\r\n const relatedField = parts.slice(i + 1).join('_');\r\n if (relatedField) {\r\n return {\r\n originalField: fieldName,\r\n referenceField: potentialRefField,\r\n relatedField,\r\n relatedObjectType: objectType,\r\n };\r\n }\r\n }\r\n }\r\n\r\n return null;\r\n}\r\n\r\n/**\r\n * Determines if a field is a \"code\" field (stores internal value, has label equivalent)\r\n *\r\n * @param fieldName - The field name to check\r\n * @returns True if the field is a code field\r\n */\r\nexport function isCodeField(fieldName: string): boolean {\r\n return fieldName.endsWith('code');\r\n}\r\n\r\n/**\r\n * Gets the code field name for a label field\r\n * For example: status -> statuscode\r\n *\r\n * @param labelField - The label field name\r\n * @returns The code field name\r\n */\r\nexport function getCodeFieldFromLabel(labelField: string): string {\r\n return `${labelField}code`;\r\n}\r\n\r\n/**\r\n * Gets the label field name for a code field\r\n * For example: statuscode -> status\r\n *\r\n * @param codeField - The code field name\r\n * @returns The label field name\r\n */\r\nexport function getLabelFieldFromCode(codeField: string): string {\r\n if (codeField.endsWith('code')) {\r\n return codeField.slice(0, -4);\r\n }\r\n return codeField;\r\n}\r\n\r\n/**\r\n * Checks if a field is a dropdown type based on metadata\r\n *\r\n * @param fieldName - The field name to check\r\n * @param metadata - Field metadata map\r\n * @returns True if the field is a dropdown type\r\n */\r\nexport function isDropdownFieldByMetadata(\r\n fieldName: string,\r\n metadata: FieldMetadataMap,\r\n): boolean {\r\n const field = metadata.get(fieldName);\r\n return field?.systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n}\r\n\r\n/**\r\n * Resolver class for related fields that uses metadata to determine field types\r\n *\r\n * @example\r\n * ```typescript\r\n * // Create resolver with metadata\r\n * const fields = await client.metadata.getFields('1');\r\n * const resolver = new RelatedFieldResolver();\r\n * resolver.setMetadata(1, fields.fields);\r\n *\r\n * // Resolve a related field\r\n * const resolution = resolver.resolve('accountid_statuscode');\r\n * // Returns both statuscode and status fields for dropdown types\r\n * ```\r\n */\r\nexport class RelatedFieldResolver {\r\n private metadataByObjectType: Map<number, FieldMetadataMap> = new Map();\r\n\r\n /**\r\n * Sets field metadata for an object type\r\n *\r\n * @param objectType - The object type ID\r\n * @param fields - Array of field metadata\r\n */\r\n setMetadata(objectType: number, fields: FireberryField[]): void {\r\n const fieldMap: FieldMetadataMap = new Map();\r\n for (const field of fields) {\r\n fieldMap.set(field.fieldName, field);\r\n }\r\n this.metadataByObjectType.set(objectType, fieldMap);\r\n }\r\n\r\n /**\r\n * Gets field metadata for an object type\r\n *\r\n * @param objectType - The object type ID\r\n * @returns Field metadata map, or undefined if not loaded\r\n */\r\n getMetadata(objectType: number): FieldMetadataMap | undefined {\r\n return this.metadataByObjectType.get(objectType);\r\n }\r\n\r\n /**\r\n * Checks if metadata is loaded for an object type\r\n *\r\n * @param objectType - The object type ID\r\n * @returns True if metadata is loaded\r\n */\r\n hasMetadata(objectType: number): boolean {\r\n return this.metadataByObjectType.has(objectType);\r\n }\r\n\r\n /**\r\n * Clears all cached metadata\r\n */\r\n clearMetadata(): void {\r\n this.metadataByObjectType.clear();\r\n }\r\n\r\n /**\r\n * Resolves a related field into all necessary fields for querying and display\r\n * Uses metadata to determine if fields are dropdown types that need code/label pairs\r\n *\r\n * @param fieldName - The related field name to resolve\r\n * @param showRealValue - Whether labels are being returned (default: true).\r\n * When true, expands dropdown fields to include both code and label.\r\n * When false, returns only the requested field (no expansion needed).\r\n * @returns Resolution with all fields and metadata, or null if not a related field\r\n *\r\n * @example\r\n * // With showRealValue=true (default), dropdown fields get code/label pairs\r\n * resolver.resolve('accountid_status')\r\n * // { fields: ['accountid_status', 'accountid_statuscode'], ... }\r\n *\r\n * // With showRealValue=false, no expansion\r\n * resolver.resolve('accountid_status', false)\r\n * // { fields: ['accountid_status'], ... }\r\n *\r\n * // Regular fields stay as-is regardless of showRealValue\r\n * resolver.resolve('accountid_telephone1')\r\n * // { fields: ['accountid_telephone1'], ... }\r\n */\r\n resolve(fieldName: string, showRealValue: boolean = true): RelatedFieldResolution | null {\r\n const parsed = parseRelatedField(fieldName);\r\n if (!parsed) {\r\n return null;\r\n }\r\n\r\n const { referenceField, relatedField, relatedObjectType } = parsed;\r\n const metadata = this.metadataByObjectType.get(relatedObjectType);\r\n const fieldIsCode = isCodeField(relatedField);\r\n const fieldMeta = metadata?.get(relatedField);\r\n\r\n // When showRealValue is false, no expansion needed - just return the field as-is\r\n if (!showRealValue) {\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n isCodeField: fieldIsCode,\r\n relatedObjectType,\r\n fieldType: fieldMeta?.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // For code fields, we want to also get the label\r\n if (fieldIsCode) {\r\n const labelOnRelated = getLabelFieldForField(relatedField, relatedObjectType);\r\n const labelFieldName = labelOnRelated\r\n ? `${referenceField}_${labelOnRelated}`\r\n : `${referenceField}_${getLabelFieldFromCode(relatedField)}`;\r\n\r\n // Get field type from metadata for the code field\r\n const baseField = getLabelFieldFromCode(relatedField);\r\n const codeFieldMeta = metadata?.get(`${baseField}code`) || metadata?.get(relatedField);\r\n\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, labelFieldName],\r\n valueField: fieldName,\r\n labelField: labelFieldName,\r\n codeField: fieldName,\r\n isCodeField: true,\r\n relatedObjectType,\r\n fieldType: codeFieldMeta?.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // Input is NOT a code field - check if it's a dropdown using metadata\r\n const potentialCodeField = getCodeFieldFromLabel(relatedField);\r\n\r\n // Check if the code field exists and is a dropdown type\r\n const codeFieldMeta = metadata?.get(potentialCodeField);\r\n const isDropdown = codeFieldMeta?.systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n\r\n if (isDropdown) {\r\n // This is a label field that has a corresponding dropdown code field\r\n const codeFieldName = `${referenceField}_${potentialCodeField}`;\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, codeFieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n codeField: codeFieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n fieldType: codeFieldMeta.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // Check if the field itself is a dropdown (for fields that don't follow code pattern)\r\n const fieldIsDropdown = fieldMeta?.systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n\r\n if (fieldIsDropdown) {\r\n // The field itself is a dropdown - check if there's a code variant\r\n const codeFieldName = `${referenceField}_${potentialCodeField}`;\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, codeFieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n codeField: codeFieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n fieldType: fieldMeta.systemFieldTypeId,\r\n };\r\n }\r\n\r\n // Regular field without code/label pairing\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n fieldType: fieldMeta?.systemFieldTypeId,\r\n };\r\n }\r\n\r\n /**\r\n * Expands a list of fields to include related field labels/codes\r\n *\r\n * @param fields - Array of field names to expand\r\n * @param showRealValue - Whether labels are being returned (default: true).\r\n * When true, expands dropdown fields to include both code and label.\r\n * When false, returns fields as-is (no expansion needed).\r\n * @returns Array of field names with related field expansions\r\n *\r\n * @example\r\n * // With showRealValue=true (default)\r\n * resolver.expandFields(['accountid_status', 'accountid_telephone1'])\r\n * // ['accountid_status', 'accountid_statuscode', 'accountid_telephone1']\r\n *\r\n * // With showRealValue=false\r\n * resolver.expandFields(['accountid_status', 'accountid_telephone1'], false)\r\n * // ['accountid_status', 'accountid_telephone1']\r\n */\r\n expandFields(fields: string[], showRealValue: boolean = true): string[] {\r\n const expanded = new Set<string>();\r\n\r\n for (const field of fields) {\r\n const resolution = this.resolve(field, showRealValue);\r\n if (resolution) {\r\n resolution.fields.forEach((f) => expanded.add(f));\r\n } else {\r\n expanded.add(field);\r\n }\r\n }\r\n\r\n return Array.from(expanded);\r\n }\r\n}\r\n\r\n/**\r\n * Resolves a related field without metadata (basic resolution)\r\n * Only handles code fields, does not detect dropdown types without metadata\r\n *\r\n * For full resolution with dropdown detection, use RelatedFieldResolver with metadata\r\n *\r\n * @param fieldName - The related field name to resolve\r\n * @returns Resolution with fields, or null if not a related field\r\n */\r\nexport function resolveRelatedField(fieldName: string): RelatedFieldResolution | null {\r\n const parsed = parseRelatedField(fieldName);\r\n if (!parsed) {\r\n return null;\r\n }\r\n\r\n const { referenceField, relatedField, relatedObjectType } = parsed;\r\n const fieldIsCode = isCodeField(relatedField);\r\n\r\n // For code fields, we always add the label\r\n if (fieldIsCode) {\r\n const labelOnRelated = getLabelFieldForField(relatedField, relatedObjectType);\r\n const labelFieldName = labelOnRelated\r\n ? `${referenceField}_${labelOnRelated}`\r\n : `${referenceField}_${getLabelFieldFromCode(relatedField)}`;\r\n\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName, labelFieldName],\r\n valueField: fieldName,\r\n labelField: labelFieldName,\r\n codeField: fieldName,\r\n isCodeField: true,\r\n relatedObjectType,\r\n };\r\n }\r\n\r\n // Without metadata, we can only return the field as-is\r\n // Use RelatedFieldResolver with metadata for dropdown detection\r\n return {\r\n originalField: fieldName,\r\n fields: [fieldName],\r\n valueField: fieldName,\r\n labelField: fieldName,\r\n isCodeField: false,\r\n relatedObjectType,\r\n };\r\n}\r\n\r\n/**\r\n * Expands a list of fields to include related field labels/codes\r\n * Basic expansion without metadata - only expands code fields\r\n *\r\n * For full expansion with dropdown detection, use RelatedFieldResolver with metadata\r\n *\r\n * @param fields - Array of field names to expand\r\n * @returns Array of field names with related field expansions\r\n */\r\nexport function expandRelatedFields(fields: string[]): string[] {\r\n const expanded = new Set<string>();\r\n\r\n for (const field of fields) {\r\n const resolution = resolveRelatedField(field);\r\n if (resolution) {\r\n resolution.fields.forEach((f) => expanded.add(f));\r\n } else {\r\n expanded.add(field);\r\n }\r\n }\r\n\r\n return Array.from(expanded);\r\n}\r\n\r\n/**\r\n * Gets all related field information for a field name\r\n * Combines parsing and basic resolution into a single call\r\n *\r\n * @param fieldName - The field name to analyze\r\n * @returns Combined info with parsing and resolution, or null if not a related field\r\n */\r\nexport function getRelatedFieldInfo(\r\n fieldName: string,\r\n): (RelatedFieldInfo & RelatedFieldResolution) | null {\r\n const parsed = parseRelatedField(fieldName);\r\n if (!parsed) {\r\n return null;\r\n }\r\n\r\n const resolved = resolveRelatedField(fieldName);\r\n if (!resolved) {\r\n return null;\r\n }\r\n\r\n return { ...parsed, ...resolved };\r\n}\r\n","/**\r\n * Fields to exclude from queries when using '*' (all fields) for specific object types\r\n * These fields cause API errors when queried\r\n */\r\nexport const EXCLUDED_FIELDS_FOR_STAR_QUERY: Record<string, string[]> = {\r\n '7': ['deletedon', 'deletedby'], // Note\r\n '8': ['deletedon', 'deletedby', 's', 'w', 'o', 't', 'description'], // Competitor\r\n '114': ['deletedon', 'deletedby'], // Calendar Resource\r\n '115': ['deletedon', 'deletedby'], // Customer Journey\r\n '116': ['deletedon', 'deletedby'], // Profile\r\n '117': ['deletedon', 'deletedby'], // Landing Page\r\n};\r\n\r\n/**\r\n * Fields to exclude from lookup relation queries across ALL object types\r\n * These fields cause API errors when queried for lookup relationships\r\n */\r\nexport const EXCLUDED_LOOKUP_FIELDS: string[] = ['deletedby', 'deletedon'];\r\n\r\n/**\r\n * Checks if a field should be excluded from star queries for a given object type\r\n *\r\n * @param objectType - The object type ID\r\n * @param fieldName - The field name to check\r\n * @returns True if the field should be excluded\r\n */\r\nexport function isExcludedFromStarQuery(objectType: string | number, fieldName: string): boolean {\r\n const objectTypeStr = String(objectType);\r\n const excludedFields = EXCLUDED_FIELDS_FOR_STAR_QUERY[objectTypeStr];\r\n return excludedFields ? excludedFields.includes(fieldName) : false;\r\n}\r\n\r\n/**\r\n * Gets the list of excluded fields for a given object type\r\n *\r\n * @param objectType - The object type ID\r\n * @returns Array of field names to exclude, or empty array if none\r\n */\r\nexport function getExcludedFieldsForStarQuery(objectType: string | number): string[] {\r\n const objectTypeStr = String(objectType);\r\n return EXCLUDED_FIELDS_FOR_STAR_QUERY[objectTypeStr] || [];\r\n}\r\n","import { FIELD_TYPE_IDS } from '../constants/fieldTypes';\r\n\r\n/**\r\n * Checks if a field is a dropdown type based on its systemFieldTypeId\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a dropdown\r\n */\r\nexport function isDropdownField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.DROPDOWN;\r\n}\r\n\r\n/**\r\n * Checks if a field is a lookup type based on its systemFieldTypeId\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a lookup\r\n */\r\nexport function isLookupField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.LOOKUP;\r\n}\r\n\r\n/**\r\n * Checks if a field is a dropdown or lookup type\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a dropdown or lookup\r\n */\r\nexport function isDropdownOrLookupField(systemFieldTypeId: string): boolean {\r\n return isDropdownField(systemFieldTypeId) || isLookupField(systemFieldTypeId);\r\n}\r\n\r\n/**\r\n * Checks if a field is a text type\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a text field\r\n */\r\nexport function isTextField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.TEXT;\r\n}\r\n\r\n/**\r\n * Checks if a field is a numeric type\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a numeric field\r\n */\r\nexport function isNumericField(systemFieldTypeId: string): boolean {\r\n return systemFieldTypeId === FIELD_TYPE_IDS.NUMERIC;\r\n}\r\n\r\n/**\r\n * Checks if a field is a date type (date or datetime)\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns True if the field is a date or datetime field\r\n */\r\nexport function isDateField(systemFieldTypeId: string): boolean {\r\n return (\r\n systemFieldTypeId === FIELD_TYPE_IDS.DATE ||\r\n systemFieldTypeId === FIELD_TYPE_IDS.DATETIME\r\n );\r\n}\r\n\r\n/**\r\n * Gets the field type name from its system ID\r\n *\r\n * @param systemFieldTypeId - The system field type ID from metadata\r\n * @returns Human-readable field type name, or 'Unknown' if not found\r\n */\r\nexport function getFieldTypeName(systemFieldTypeId: string): string {\r\n const typeMap: Record<string, string> = {\r\n [FIELD_TYPE_IDS.DROPDOWN]: 'Dropdown',\r\n [FIELD_TYPE_IDS.LOOKUP]: 'Lookup',\r\n [FIELD_TYPE_IDS.EMAIL]: 'Email',\r\n [FIELD_TYPE_IDS.TEXT]: 'Text',\r\n [FIELD_TYPE_IDS.URL]: 'URL',\r\n [FIELD_TYPE_IDS.LONG_TEXT]: 'Long Text',\r\n [FIELD_TYPE_IDS.DATETIME]: 'DateTime',\r\n [FIELD_TYPE_IDS.DATE]: 'Date',\r\n [FIELD_TYPE_IDS.HTML]: 'HTML',\r\n [FIELD_TYPE_IDS.TELEPHONE]: 'Telephone',\r\n [FIELD_TYPE_IDS.NUMERIC]: 'Number',\r\n };\r\n return typeMap[systemFieldTypeId] || 'Unknown';\r\n}\r\n","/**\r\n * SDK Adapter for @fireberry/sdk\r\n *\r\n * This module provides utilities to enhance the Fireberry SDK with\r\n * QueryBuilder and field mapping capabilities from fireberry-api-client.\r\n *\r\n * @example\r\n * ```typescript\r\n * import FireberryClientSDK from '@fireberry/sdk/client';\r\n * import { createSDKQueryBuilder, EnhancedSDK } from 'fireberry-api-client/sdk';\r\n *\r\n * // Option 1: Use query builder factory\r\n * const sdk = new FireberryClientSDK();\r\n * await sdk.initializeContext();\r\n *\r\n * const queryBuilder = createSDKQueryBuilder(sdk);\r\n * const results = await queryBuilder(1)\r\n * .select('accountid', 'accountname', 'statuscode')\r\n * .where('statuscode').equals('1')\r\n * .execute();\r\n *\r\n * // Option 2: Use enhanced SDK wrapper\r\n * const enhanced = EnhancedSDK.create(sdk);\r\n * const results = await enhanced\r\n * .query(1)\r\n * .select('accountid', 'accountname')\r\n * .where('statuscode').equals('1')\r\n * .execute();\r\n * ```\r\n *\r\n * @packageDocumentation\r\n */\r\n\r\nimport type {\r\n FireberrySDKClient,\r\n FireberrySDKAPI,\r\n SDKQueryPayload,\r\n SDKResponseData,\r\n SDKContext,\r\n} from '../types/sdk';\r\nimport { QueryBuilder, escapeQueryValue, sanitizeQuery, type ConditionBuilder } from '../utils/queryBuilder';\r\nimport { getLabelFieldForField } from '../utils/fieldMapping';\r\nimport { getObjectIdFieldName, getNameFieldByObjectType } from '../utils/objectMapping';\r\nimport { getExcludedFieldsForStarQuery } from '../constants/excludedFields';\r\nimport {\r\n isDropdownField,\r\n isLookupField,\r\n isDropdownOrLookupField,\r\n} from '../utils/fieldTypes';\r\n\r\n// Re-export types for convenience\r\nexport type {\r\n FireberrySDKClient,\r\n FireberrySDKAPI,\r\n SDKQueryPayload,\r\n SDKResponseData,\r\n SDKContext,\r\n} from '../types/sdk';\r\n\r\n/**\r\n * Condition builder that returns SDKQueryBuilder for fluent chaining\r\n */\r\ninterface SDKConditionBuilder {\r\n equals(value: string | number): SDKQueryBuilder;\r\n notEquals(value: string | number): SDKQueryBuilder;\r\n lessThan(value: string | number): SDKQueryBuilder;\r\n greaterThan(value: string | number): SDKQueryBuilder;\r\n lessThanOrEqual(value: string | number): SDKQueryBuilder;\r\n greaterThanOrEqual(value: string | number): SDKQueryBuilder;\r\n contains(value: string): SDKQueryBuilder;\r\n notContains(value: string): SDKQueryBuilder;\r\n startsWith(value: string): SDKQueryBuilder;\r\n notStartsWith(value: string): SDKQueryBuilder;\r\n isNull(): SDKQueryBuilder;\r\n isNotNull(): SDKQueryBuilder;\r\n}\r\n\r\n/**\r\n * SDK-compatible query builder that executes via the Fireberry SDK\r\n */\r\nexport class SDKQueryBuilder {\r\n private builder: QueryBuilder;\r\n private objectTypeId: number | string;\r\n private sdk: FireberrySDKAPI;\r\n private selectedFields: string[] = [];\r\n private pageSizeValue?: number;\r\n private pageNum: number = 1;\r\n\r\n constructor(sdk: FireberrySDKAPI, objectType: number | string) {\r\n this.sdk = sdk;\r\n this.objectTypeId = objectType;\r\n this.builder = new QueryBuilder();\r\n }\r\n\r\n /**\r\n * Select fields to return\r\n * @param fields - Field names to include in results\r\n */\r\n select(...fields: string[]): this {\r\n this.selectedFields.push(...fields);\r\n return this;\r\n }\r\n\r\n /**\r\n * Select fields with their label fields automatically included\r\n * Useful for dropdown and lookup fields where you want both ID and display value\r\n * @param fields - Field names to include\r\n */\r\n selectWithLabels(...fields: string[]): this {\r\n const objectType = typeof this.objectTypeId === 'string'\r\n ? parseInt(this.objectTypeId, 10)\r\n : this.objectTypeId;\r\n\r\n for (const field of fields) {\r\n if (!this.selectedFields.includes(field)) {\r\n this.selectedFields.push(field);\r\n }\r\n const labelField = getLabelFieldForField(field, objectType);\r\n if (labelField && !this.selectedFields.includes(labelField)) {\r\n this.selectedFields.push(labelField);\r\n }\r\n }\r\n return this;\r\n }\r\n\r\n /**\r\n * Start a WHERE condition\r\n * @param field - Field name to filter on\r\n */\r\n where(field: string): SDKConditionBuilder {\r\n const innerBuilder = this.builder.where(field);\r\n // Wrap the inner condition builder to return this SDKQueryBuilder\r\n return {\r\n equals: (value: string | number) => { innerBuilder.equals(value); return this; },\r\n notEquals: (value: string | number) => { innerBuilder.notEquals(value); return this; },\r\n lessThan: (value: string | number) => { innerBuilder.lessThan(value); return this; },\r\n greaterThan: (value: string | number) => { innerBuilder.greaterThan(value); return this; },\r\n lessThanOrEqual: (value: string | number) => { innerBuilder.lessThanOrEqual(value); return this; },\r\n greaterThanOrEqual: (value: string | number) => { innerBuilder.greaterThanOrEqual(value); return this; },\r\n contains: (value: string) => { innerBuilder.contains(value); return this; },\r\n notContains: (value: string) => { innerBuilder.notContains(value); return this; },\r\n startsWith: (value: string) => { innerBuilder.startsWith(value); return this; },\r\n notStartsWith: (value: string) => { innerBuilder.notStartsWith(value); return this; },\r\n isNull: () => { innerBuilder.isNull(); return this; },\r\n isNotNull: () => { innerBuilder.isNotNull(); return this; },\r\n };\r\n }\r\n\r\n /**\r\n * Add AND logical operator\r\n */\r\n and(): this {\r\n this.builder.and();\r\n return this;\r\n }\r\n\r\n /**\r\n * Add OR logical operator\r\n */\r\n or(): this {\r\n this.builder.or();\r\n return this;\r\n }\r\n\r\n /**\r\n * Set page size for pagination\r\n * @param size - Number of records per page\r\n */\r\n pageSize(size: number): this {\r\n this.pageSizeValue = size;\r\n return this;\r\n }\r\n\r\n /**\r\n * Set page number for pagination\r\n * @param page - Page number (1-based)\r\n */\r\n page(page: number): this {\r\n this.pageNum = page;\r\n return this;\r\n }\r\n\r\n /**\r\n * Build the query payload without executing\r\n * Useful if you want to modify the payload before sending\r\n */\r\n toQueryPayload(): SDKQueryPayload {\r\n const payload: SDKQueryPayload = {\r\n fields: this.selectedFields.length > 0\r\n ? this.selectedFields.join(',')\r\n : '*',\r\n query: this.builder.build(),\r\n };\r\n\r\n if (this.pageSizeValue !== undefined) {\r\n payload.page_size = this.pageSizeValue;\r\n }\r\n\r\n if (this.pageNum > 1) {\r\n payload.page_number = this.pageNum;\r\n }\r\n\r\n return payload;\r\n }\r\n\r\n /**\r\n * Execute the query via the SDK\r\n */\r\n async execute<T = Record<string, unknown>>(): Promise<SDKResponseData<T>> {\r\n const payload = this.toQueryPayload();\r\n return this.sdk.query(this.objectTypeId, payload) as Promise<SDKResponseData<T>>;\r\n }\r\n}\r\n\r\n/**\r\n * Creates a query builder factory bound to a Fireberry SDK instance\r\n *\r\n * @param sdk - Fireberry SDK client or API instance\r\n * @returns Factory function that creates SDKQueryBuilder instances\r\n *\r\n * @example\r\n * ```typescript\r\n * import FireberryClientSDK from '@fireberry/sdk/client';\r\n * import { createSDKQueryBuilder } from 'fireberry-api-client/sdk';\r\n *\r\n * const sdk = new FireberryClientSDK();\r\n * await sdk.initializeContext();\r\n *\r\n * const queryBuilder = createSDKQueryBuilder(sdk);\r\n *\r\n * // Query accounts where status is active\r\n * const results = await queryBuilder(1) // 1 = Account object type\r\n * .select('accountid', 'accountname', 'statuscode', 'status')\r\n * .where('statuscode').equals('1')\r\n * .pageSize(50)\r\n * .execute();\r\n * ```\r\n */\r\nexport function createSDKQueryBuilder(\r\n sdk: FireberrySDKClient | FireberrySDKAPI\r\n): (objectType: number | string) => SDKQueryBuilder {\r\n const api = 'api' in sdk ? sdk.api : sdk;\r\n\r\n return (objectType: number | string) => new SDKQueryBuilder(api, objectType);\r\n}\r\n\r\n/**\r\n * Enhanced SDK wrapper that combines Fireberry SDK with utility functions\r\n * Provides a more feature-rich API for working with Fireberry data\r\n */\r\nexport class EnhancedSDK<TData = Record<string, unknown>> {\r\n private sdk: FireberrySDKClient<TData>;\r\n\r\n private constructor(sdk: FireberrySDKClient<TData>) {\r\n this.sdk = sdk;\r\n }\r\n\r\n /**\r\n * Create an EnhancedSDK wrapper around an existing SDK instance\r\n * The SDK should already be initialized with context\r\n *\r\n * @param sdk - Initialized Fireberry SDK client\r\n */\r\n static create<T = Record<string, unknown>>(\r\n sdk: FireberrySDKClient<T>\r\n ): EnhancedSDK<T> {\r\n return new EnhancedSDK(sdk);\r\n }\r\n\r\n /**\r\n * Get the current context (user and record info)\r\n */\r\n get context(): SDKContext | null {\r\n return this.sdk.context;\r\n }\r\n\r\n /**\r\n * Get the underlying SDK API for direct access\r\n */\r\n get api(): FireberrySDKAPI<TData> {\r\n return this.sdk.api;\r\n }\r\n\r\n /**\r\n * Get the current user ID from context\r\n */\r\n get userId(): string | undefined {\r\n return this.context?.user?.id;\r\n }\r\n\r\n /**\r\n * Get the current user's full name from context\r\n */\r\n get userFullName(): string | undefined {\r\n return this.context?.user?.fullName;\r\n }\r\n\r\n /**\r\n * Get the current record ID from context\r\n */\r\n get recordId(): string | undefined {\r\n return this.context?.record?.id;\r\n }\r\n\r\n /**\r\n * Get the current record's object type from context\r\n */\r\n get recordType(): number | undefined {\r\n return this.context?.record?.type;\r\n }\r\n\r\n /**\r\n * Start building a query for an object type\r\n *\r\n * @param objectType - Object type ID (e.g., 1 for Account, 2 for Contact)\r\n * @returns SDKQueryBuilder for fluent query construction\r\n *\r\n * @example\r\n * ```typescript\r\n * const results = await enhanced\r\n * .query(1)\r\n * .select('accountid', 'accountname')\r\n * .where('ownerid').equals(enhanced.userId)\r\n * .execute();\r\n * ```\r\n */\r\n query(objectType: number | string): SDKQueryBuilder {\r\n // Cast is safe - SDKQueryBuilder only uses the query method which has the same signature\r\n return new SDKQueryBuilder(this.sdk.api as FireberrySDKAPI, objectType);\r\n }\r\n\r\n /**\r\n * Get the primary key field name for an object type\r\n *\r\n * @param objectType - Object type ID\r\n * @returns Primary key field name (e.g., 'accountid' for type 1)\r\n *\r\n * @example\r\n * ```typescript\r\n * const idField = enhanced.getIdField(1); // 'accountid'\r\n * const idField = enhanced.getIdField(2); // 'contactid'\r\n * ```\r\n */\r\n getIdField(objectType: number | string): string {\r\n return getObjectIdFieldName(objectType);\r\n }\r\n\r\n /**\r\n * Get the display name field for an object type\r\n *\r\n * @param objectType - Object type ID\r\n * @returns Name field (e.g., 'accountname' for type 1, 'fullname' for type 2)\r\n *\r\n * @example\r\n * ```typescript\r\n * const nameField = enhanced.getNameField(1); // 'accountname'\r\n * const nameField = enhanced.getNameField(2); // 'fullname'\r\n * ```\r\n */\r\n getNameField(objectType: number | string): string {\r\n return getNameFieldByObjectType(objectType);\r\n }\r\n\r\n /**\r\n * Get the label field for a dropdown or lookup field\r\n *\r\n * @param fieldName - API field name\r\n * @param objectType - Object type ID\r\n * @returns Label field name, or empty string if not applicable\r\n *\r\n * @example\r\n * ```typescript\r\n * const labelField = enhanced.getLabelField('statuscode', 1); // 'status'\r\n * const labelField = enhanced.getLabelField('ownerid', 1); // 'ownername'\r\n * ```\r\n */\r\n getLabelField(fieldName: string, objectType: number | string): string {\r\n return getLabelFieldForField(fieldName, objectType);\r\n }\r\n\r\n /**\r\n * Get fields that should be excluded from * (star) queries for an object type\r\n * Some fields cause API errors when included in broad queries\r\n *\r\n * @param objectType - Object type ID\r\n * @returns Array of field names to exclude\r\n */\r\n getExcludedFields(objectType: number | string): string[] {\r\n return getExcludedFieldsForStarQuery(objectType);\r\n }\r\n\r\n /**\r\n * Expand field list to include label fields for dropdowns and lookups\r\n *\r\n * @param fields - Original field list\r\n * @param objectType - Object type ID\r\n * @returns Expanded field list with label fields\r\n *\r\n * @example\r\n * ```typescript\r\n * const fields = enhanced.expandFieldsWithLabels(['statuscode', 'ownerid'], 1);\r\n * // Returns: ['statuscode', 'status', 'ownerid', 'ownername']\r\n * ```\r\n */\r\n expandFieldsWithLabels(fields: string[], objectType: number | string): string[] {\r\n const result: string[] = [];\r\n for (const field of fields) {\r\n if (!result.includes(field)) {\r\n result.push(field);\r\n }\r\n const labelField = getLabelFieldForField(field, objectType);\r\n if (labelField && !result.includes(labelField)) {\r\n result.push(labelField);\r\n }\r\n }\r\n return result;\r\n }\r\n\r\n /**\r\n * Create a record\r\n *\r\n * @param objectType - Object type ID\r\n * @param data - Record data\r\n */\r\n create<T extends Record<string, unknown>>(\r\n objectType: number | string,\r\n data: T\r\n ): Promise<SDKResponseData<TData>> {\r\n return this.sdk.api.create(objectType, data);\r\n }\r\n\r\n /**\r\n * Update a record\r\n *\r\n * @param objectType - Object type ID\r\n * @param recordId - Record ID to update\r\n * @param data - Updated field values\r\n */\r\n update<T extends Record<string, unknown>>(\r\n objectType: number | string,\r\n recordId: string,\r\n data: T\r\n ): Promise<SDKResponseData<TData>> {\r\n return this.sdk.api.update(objectType, recordId, data);\r\n }\r\n\r\n /**\r\n * Delete a record\r\n *\r\n * @param objectType - Object type ID\r\n * @param recordId - Record ID to delete\r\n */\r\n delete(\r\n objectType: number | string,\r\n recordId: string\r\n ): Promise<SDKResponseData<TData>> {\r\n return this.sdk.api.delete(objectType, recordId);\r\n }\r\n\r\n /**\r\n * Clean up the underlying SDK\r\n */\r\n destroy(): void {\r\n this.sdk.destroy();\r\n }\r\n}\r\n\r\n// Re-export utilities that work standalone with the SDK\r\nexport {\r\n // Query utilities\r\n QueryBuilder,\r\n escapeQueryValue,\r\n sanitizeQuery,\r\n type ConditionBuilder,\r\n // Field utilities\r\n getLabelFieldForField,\r\n getObjectIdFieldName,\r\n getNameFieldByObjectType,\r\n getExcludedFieldsForStarQuery,\r\n // Type detection\r\n isDropdownField,\r\n isLookupField,\r\n isDropdownOrLookupField,\r\n};\r\n\r\n// Re-export constants\r\nexport {\r\n FIELD_TYPE_IDS,\r\n FIELD_TYPE_MAPPINGS,\r\n OBJECT_ID_MAP,\r\n OBJECT_NAME_MAP,\r\n EXCLUDED_FIELDS_FOR_STAR_QUERY,\r\n} from '../constants';\r\n"]}