@payloadcms/plugin-import-export 3.84.1 → 3.85.0
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.
- package/dist/export/batchProcessor.d.ts +9 -1
- package/dist/export/batchProcessor.d.ts.map +1 -1
- package/dist/export/batchProcessor.js +57 -15
- package/dist/export/batchProcessor.js.map +1 -1
- package/dist/export/createExport.d.ts.map +1 -1
- package/dist/export/createExport.js +98 -20
- package/dist/export/createExport.js.map +1 -1
- package/dist/export/handlePreview.d.ts.map +1 -1
- package/dist/export/handlePreview.js +38 -13
- package/dist/export/handlePreview.js.map +1 -1
- package/dist/exports/types.d.ts +1 -1
- package/dist/exports/types.d.ts.map +1 -1
- package/dist/exports/types.js.map +1 -1
- package/dist/import/batchProcessor.d.ts +14 -2
- package/dist/import/batchProcessor.d.ts.map +1 -1
- package/dist/import/batchProcessor.js +49 -27
- package/dist/import/batchProcessor.js.map +1 -1
- package/dist/import/createImport.d.ts +1 -10
- package/dist/import/createImport.d.ts.map +1 -1
- package/dist/import/createImport.js +33 -52
- package/dist/import/createImport.js.map +1 -1
- package/dist/import/handlePreview.d.ts.map +1 -1
- package/dist/import/handlePreview.js +32 -6
- package/dist/import/handlePreview.js.map +1 -1
- package/dist/index.d.ts +58 -3
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +18 -1
- package/dist/index.js.map +1 -1
- package/dist/types.d.ts +218 -39
- package/dist/types.d.ts.map +1 -1
- package/dist/types.js.map +1 -1
- package/dist/utilities/applyFieldHooks.d.ts +23 -0
- package/dist/utilities/applyFieldHooks.d.ts.map +1 -0
- package/dist/utilities/applyFieldHooks.js +118 -0
- package/dist/utilities/applyFieldHooks.js.map +1 -0
- package/dist/utilities/applyFieldHooks.spec.js +205 -0
- package/dist/utilities/applyFieldHooks.spec.js.map +1 -0
- package/dist/utilities/collectDisabledFieldPaths.d.ts.map +1 -1
- package/dist/utilities/collectDisabledFieldPaths.js +1 -1
- package/dist/utilities/collectDisabledFieldPaths.js.map +1 -1
- package/dist/utilities/flattenObject.d.ts +8 -6
- package/dist/utilities/flattenObject.d.ts.map +1 -1
- package/dist/utilities/flattenObject.js +95 -75
- package/dist/utilities/flattenObject.js.map +1 -1
- package/dist/utilities/flattenObject.spec.js +158 -0
- package/dist/utilities/flattenObject.spec.js.map +1 -0
- package/dist/utilities/flattenedFields.d.ts +21 -0
- package/dist/utilities/flattenedFields.d.ts.map +1 -0
- package/dist/utilities/flattenedFields.js +34 -0
- package/dist/utilities/flattenedFields.js.map +1 -0
- package/dist/utilities/getExportFieldFunctions.d.ts +5 -5
- package/dist/utilities/getExportFieldFunctions.d.ts.map +1 -1
- package/dist/utilities/getExportFieldFunctions.js +92 -98
- package/dist/utilities/getExportFieldFunctions.js.map +1 -1
- package/dist/utilities/getExportFieldFunctions.spec.js +50 -0
- package/dist/utilities/getExportFieldFunctions.spec.js.map +1 -0
- package/dist/utilities/getImportFieldFunctions.d.ts +5 -5
- package/dist/utilities/getImportFieldFunctions.d.ts.map +1 -1
- package/dist/utilities/getImportFieldFunctions.js +103 -103
- package/dist/utilities/getImportFieldFunctions.js.map +1 -1
- package/dist/utilities/getImportFieldFunctions.spec.js +167 -0
- package/dist/utilities/getImportFieldFunctions.spec.js.map +1 -0
- package/dist/utilities/isPlainObject.d.ts +2 -0
- package/dist/utilities/isPlainObject.d.ts.map +1 -0
- package/dist/utilities/isPlainObject.js +3 -0
- package/dist/utilities/isPlainObject.js.map +1 -0
- package/dist/utilities/legacyHookDispatch.spec.js +227 -0
- package/dist/utilities/legacyHookDispatch.spec.js.map +1 -0
- package/dist/utilities/polymorphicRel.d.ts +14 -0
- package/dist/utilities/polymorphicRel.d.ts.map +1 -0
- package/dist/utilities/polymorphicRel.js +17 -0
- package/dist/utilities/polymorphicRel.js.map +1 -0
- package/dist/utilities/processRichTextField.js.map +1 -1
- package/dist/utilities/removeDisabledFields.js.map +1 -1
- package/dist/utilities/setNestedValue.d.ts.map +1 -1
- package/dist/utilities/setNestedValue.js +10 -8
- package/dist/utilities/setNestedValue.js.map +1 -1
- package/dist/utilities/siblingDoc.spec.js +278 -0
- package/dist/utilities/siblingDoc.spec.js.map +1 -0
- package/dist/utilities/unflattenObject.d.ts +4 -3
- package/dist/utilities/unflattenObject.d.ts.map +1 -1
- package/dist/utilities/unflattenObject.js +57 -169
- package/dist/utilities/unflattenObject.js.map +1 -1
- package/dist/utilities/unflattenObject.spec.js +33 -0
- package/dist/utilities/unflattenObject.spec.js.map +1 -1
- package/dist/utilities/unflattenPostProcess.d.ts +11 -0
- package/dist/utilities/unflattenPostProcess.d.ts.map +1 -0
- package/dist/utilities/unflattenPostProcess.js +148 -0
- package/dist/utilities/unflattenPostProcess.js.map +1 -0
- package/package.json +7 -7
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/utilities/unflattenObject.ts"],"sourcesContent":["import type { FlattenedField, PayloadRequest } from 'payload'\n\nimport type { FromCSVFunction } from '../types.js'\n\nimport { processRichTextField } from './processRichTextField.js'\n\ntype UnflattenArgs = {\n data: Record<string, unknown>\n fields: FlattenedField[]\n fromCSVFunctions?: Record<string, FromCSVFunction>\n req: PayloadRequest\n}\n\n/**\n * Converts flattened CSV data back into a nested document structure.\n *\n * The algorithm:\n * 1. Sorts keys to ensure array indices are processed in order\n * 2. For each flattened key (e.g., \"blocks_0_hero_title\"), splits by underscore into path segments\n * 3. Traverses/builds the nested structure, handling:\n * - Arrays (numeric segments like \"0\", \"1\")\n * - Blocks (blockType detection from slug patterns)\n * - Polymorphic relationships (_relationTo and _id suffix pairs)\n * - Regular nested objects\n * 4. Post-processes to handle localized fields, hasMany conversions, and relationship transforms\n */\nexport const unflattenObject = ({\n data,\n fields,\n fromCSVFunctions = {},\n req,\n}: UnflattenArgs): Record<string, unknown> => {\n if (!data || typeof data !== 'object') {\n return {}\n }\n\n const result: Record<string, unknown> = {}\n\n // Sort keys to ensure array indices are processed in order\n const sortedKeys = Object.keys(data).sort((a, b) => {\n // Extract array indices from flattened keys (e.g., \"field_0_subfield\" -> \"0\")\n const aMatch = a.match(/_(\\d+)(?:_|$)/)\n const bMatch = b.match(/_(\\d+)(?:_|$)/)\n\n if (aMatch && bMatch && aMatch.index !== undefined && bMatch.index !== undefined) {\n const aBase = a.substring(0, aMatch.index)\n const bBase = b.substring(0, bMatch.index)\n\n if (aBase === bBase) {\n return (parseInt(aMatch?.[1] ?? '0', 10) || 0) - (parseInt(bMatch?.[1] ?? '0', 10) || 0)\n }\n }\n\n return a.localeCompare(b)\n })\n\n for (const flatKey of sortedKeys) {\n let value = data[flatKey]\n\n // Skip undefined values but keep null for required field validation\n if (value === undefined) {\n continue\n }\n\n // Preserve system fields with underscore prefix (like _status) without splitting\n if (flatKey === '_status') {\n result[flatKey] = value\n continue\n }\n\n // Check if this is a _relationTo key for a polymorphic relationship\n if (flatKey.endsWith('_relationTo')) {\n const baseKey = flatKey.replace(/_relationTo$/, '')\n const idKey = `${baseKey}_id`\n\n // Check if this is a polymorphic relationship field\n const isPolymorphic = fields.some(\n (field) =>\n field.name === baseKey &&\n field.type === 'relationship' &&\n 'relationTo' in field &&\n Array.isArray(field.relationTo),\n )\n\n if (isPolymorphic) {\n if (baseKey in result) {\n continue\n }\n\n // If the corresponding _id key is undefined, skip processing entirely\n // This prevents creating empty objects when we should preserve existing data\n if (!(idKey in data) || data[idKey] === undefined) {\n continue\n }\n }\n }\n\n // Check if this is a _id key for a polymorphic relationship where _relationTo is undefined\n if (flatKey.endsWith('_id')) {\n const baseKey = flatKey.replace(/_id$/, '')\n const relationToKey = `${baseKey}_relationTo`\n\n // Check if this is a polymorphic relationship field\n const isPolymorphic = fields.some(\n (field) =>\n field.name === baseKey &&\n field.type === 'relationship' &&\n 'relationTo' in field &&\n Array.isArray(field.relationTo),\n )\n\n if (isPolymorphic) {\n // If the corresponding _relationTo key is undefined, skip processing entirely\n // This prevents creating empty objects when we should preserve existing data\n if (!(relationToKey in data) || data[relationToKey] === undefined) {\n continue\n }\n }\n }\n\n // Apply fromCSV function if available\n if (fromCSVFunctions[flatKey]) {\n value = fromCSVFunctions[flatKey]({\n columnName: flatKey,\n data,\n value,\n })\n }\n\n // Example: \"blocks_0_content_text\" -> [\"blocks\", \"0\", \"content\", \"text\"]\n const pathSegments = flatKey.split('_')\n let currentObject: Record<string, unknown> = result\n\n for (let i = 0; i < pathSegments.length; i++) {\n const segment = pathSegments[i]\n if (!segment) {\n continue\n } // Skip empty segments\n\n const nextSegment = pathSegments[i + 1]\n const isLast = i === pathSegments.length - 1\n\n // Check if next segment is a numeric array index (e.g., \"0\", \"1\", \"2\")\n const isArrayIndex = nextSegment !== undefined && /^\\d+$/.test(nextSegment)\n\n if (isLast) {\n // Special handling for blockType suffix in blocks\n if (segment === 'blockType' && i >= 3) {\n // Pattern: blocks_0_hero_blockType -> set blockType on the block\n const blockFieldName = pathSegments[0] // 'blocks'\n const isBlockField = fields.some(\n (field) => field.name === blockFieldName && field.type === 'blocks',\n )\n\n if (isBlockField && pathSegments[1]?.match(/^\\d+$/)) {\n const parent = getParentObject(result, pathSegments.slice(0, 2))\n if (parent && typeof parent === 'object') {\n parent.blockType = value\n }\n continue\n }\n }\n\n // Special handling for relationship fields with _id suffix\n if (segment === 'id' && i > 0) {\n const parentKey = pathSegments[i - 1]\n const isPreviousSegmentArrayIndex = parentKey ? /^\\d+$/.test(parentKey) : false\n\n if (!isPreviousSegmentArrayIndex) {\n // Check if this is a relationship field\n const isRelationship = fields.some(\n (field) => field.name === parentKey && field.type === 'relationship',\n )\n\n if (isRelationship) {\n // Check if this is a polymorphic relationship field\n const field = fields.find((f) => f.name === parentKey && f.type === 'relationship')\n const isPolymorphic =\n field && 'relationTo' in field && Array.isArray(field.relationTo)\n\n if (isPolymorphic) {\n const relationToKey = pathSegments.slice(0, i).concat('relationTo').join('_')\n const relationToValue = data[relationToKey]\n\n const parent = getParentObject(result, pathSegments.slice(0, i - 1))\n if (parent && parentKey && typeof parent === 'object') {\n // Both fields must be defined to create/update the relationship\n // If either is undefined, skip the field entirely (preserve existing data)\n if (value !== undefined && relationToValue !== undefined) {\n // Check if both are explicitly null\n if (relationToValue === null && value === null) {\n // Only set to null if explicitly null (user typed \"null\" in CSV)\n parent[parentKey] = null\n } else if (relationToValue || value) {\n // At least one has a value, create the relationship\n parent[parentKey] = {\n relationTo: relationToValue,\n value, // This will be transformed to proper format in postProcess\n }\n }\n // If both are empty strings, don't set the field (handled by not meeting the above conditions)\n }\n // If either is undefined, don't set the field at all (preserve existing data)\n }\n continue\n } else if (!isPolymorphic) {\n const parent = getParentObject(result, pathSegments.slice(0, i - 1))\n if (parent && parentKey && typeof parent === 'object') {\n parent[parentKey] = value\n }\n continue\n }\n }\n }\n }\n\n // _relationTo suffix is handled when processing the _id field above\n if (segment === 'relationTo' && i > 0) {\n const parentKey = pathSegments[i - 1]\n if (parentKey && !parentKey.match(/^\\d+$/)) {\n const field = fields.find((f) => f.name === parentKey && f.type === 'relationship')\n const isPolymorphic = field && 'relationTo' in field && Array.isArray(field.relationTo)\n\n if (isPolymorphic) {\n // For polymorphic relationships, this is handled when processing the _id field\n // Skip it entirely\n continue\n }\n }\n }\n\n currentObject[segment] = value\n } else if (isArrayIndex && nextSegment !== undefined) {\n if (!currentObject[segment] || !Array.isArray(currentObject[segment])) {\n currentObject[segment] = []\n }\n\n const arrayIndex = parseInt(nextSegment)\n const arr = currentObject[segment] as unknown[]\n\n // Ensure array has sufficient length\n while (arr.length <= arrayIndex) {\n arr.push(null)\n }\n\n // Handle array of objects\n if (arr[arrayIndex] === null || arr[arrayIndex] === undefined) {\n arr[arrayIndex] = {}\n }\n\n // Handle blocks field with block slug pattern (e.g., blocks_0_hero_title)\n const isBlocksField = fields.some((f) => f.name === segment && f.type === 'blocks')\n if (isBlocksField && i + 3 < pathSegments.length) {\n const blockSlug = pathSegments[i + 2]\n const blockFieldName = pathSegments[i + 3]\n\n if (blockSlug && blockFieldName) {\n const blockObject = arr[arrayIndex] as Record<string, unknown>\n blockObject.blockType = blockSlug\n\n if (i + 3 === pathSegments.length - 1) {\n blockObject[blockFieldName] = value\n } else {\n if (!blockObject[blockFieldName] || typeof blockObject[blockFieldName] !== 'object') {\n blockObject[blockFieldName] = {}\n }\n currentObject = blockObject[blockFieldName] as Record<string, unknown>\n i = i + 3\n continue\n }\n break\n }\n }\n\n if (i + 2 === pathSegments.length - 1) {\n const lastSegment = pathSegments[pathSegments.length - 1]\n if (lastSegment && arr[arrayIndex] && typeof arr[arrayIndex] === 'object') {\n ;(arr[arrayIndex] as Record<string, unknown>)[lastSegment] = value\n }\n break\n } else if (i + 1 === pathSegments.length - 1) {\n // Direct array value (e.g., tags_0 = \"value\")\n arr[arrayIndex] = value\n break\n } else {\n currentObject = arr[arrayIndex] as Record<string, unknown>\n i++\n }\n } else {\n // Skip if already set to null (polymorphic relationship already processed)\n if (currentObject[segment] === null && isLast && segment === 'relationTo') {\n continue\n }\n\n if (\n !currentObject[segment] ||\n typeof currentObject[segment] !== 'object' ||\n Array.isArray(currentObject[segment])\n ) {\n currentObject[segment] = {}\n }\n\n // Handle polymorphic relationship arrays\n if (segment === 'relationTo' && i > 0 && pathSegments[i - 1]?.match(/^\\d+$/)) {\n currentObject[segment] = value\n } else if (\n typeof currentObject[segment] === 'object' &&\n !Array.isArray(currentObject[segment]) &&\n currentObject[segment] !== null\n ) {\n currentObject = currentObject[segment] as Record<string, unknown>\n }\n }\n }\n }\n\n try {\n // Post-process to handle special structures\n postProcessDocument(result, fields)\n } catch (err) {\n // Log but don't throw - return partially processed result\n\n req.payload.logger.error({\n err,\n msg: '[plugin-import-export] Error in postProcessDocument',\n })\n }\n\n return result\n}\n\nconst getParentObject = (\n obj: Record<string, unknown>,\n segments: string[],\n): Record<string, unknown> | undefined => {\n let current: Record<string, unknown> = obj\n\n for (let i = 0; i < segments.length; i++) {\n const segment = segments[i]\n const nextSegment = segments[i + 1]\n\n if (!segment) {\n continue\n }\n\n if (nextSegment && /^\\d+$/.test(nextSegment)) {\n const arrayIndex = parseInt(nextSegment)\n const arr = current[segment] as unknown[]\n\n if (Array.isArray(arr) && arr[arrayIndex]) {\n current = arr[arrayIndex] as Record<string, unknown>\n i++ // Skip the index\n } else {\n return undefined\n }\n } else {\n const next = current[segment]\n if (typeof next === 'object' && next !== null && !Array.isArray(next)) {\n current = next as Record<string, unknown>\n } else {\n return undefined\n }\n }\n }\n\n return current\n}\n\n/**\n * Post-processes the unflattened document to handle special field types:\n * - Localized fields: transforms field_locale keys to nested { field: { locale: value } }\n * - Number hasMany: converts comma-separated strings or arrays to number arrays\n * - Relationship hasMany: converts comma-separated IDs to arrays\n * - Polymorphic relationships: transforms flat {relationTo, id} to {relationTo, value}\n * - Rich text fields: ensures proper data structure\n */\nconst postProcessDocument = (doc: Record<string, unknown>, fields: FlattenedField[]): void => {\n const localizedFields = fields.filter((field) => field.localized)\n const processedLocalizedFields = new Set<string>()\n\n for (const field of localizedFields) {\n if (processedLocalizedFields.has(field.name)) {\n continue\n }\n\n // Look for all locale-specific keys for this field\n const localePattern = new RegExp(`^${field.name}_([a-z]{2}(?:_[A-Z]{2})?)$`)\n const localeData: Record<string, unknown> = {}\n const keysToDelete: string[] = []\n\n for (const [key, value] of Object.entries(doc)) {\n const match = key.match(localePattern)\n if (match && match[1]) {\n const locale = match[1]\n localeData[locale] = value\n keysToDelete.push(key)\n }\n }\n\n // If we found locale-specific data, restructure it as Payload expects\n if (Object.keys(localeData).length > 0) {\n // Payload stores localized fields as nested objects: { field: { en: 'value', es: 'value' } }\n doc[field.name] = localeData\n keysToDelete.forEach((key) => delete doc[key])\n processedLocalizedFields.add(field.name)\n }\n }\n\n // Handle number fields with hasMany - convert string arrays to number arrays\n const numberFields = fields.filter((field) => field.type === 'number' && field.hasMany)\n for (const field of numberFields) {\n const value = doc[field.name]\n\n // Skip if field doesn't exist in document\n if (!(field.name in doc)) {\n continue\n }\n\n // Handle comma-separated string (e.g., \"1,2,3,4,5\")\n if (typeof value === 'string' && value.includes(',')) {\n doc[field.name] = value\n .split(',')\n .map((v) => v.trim())\n .filter((v) => v !== '')\n .map((v) => {\n const num = parseFloat(v)\n return isNaN(num) ? 0 : num\n })\n }\n // Handle array of values from indexed columns (e.g., field_0, field_1, etc.)\n else if (Array.isArray(value)) {\n // Filter out null, undefined, and empty string values, then convert to numbers\n doc[field.name] = value\n .filter((v) => v !== null && v !== undefined && v !== '')\n .map((v) => {\n if (typeof v === 'string') {\n const num = parseFloat(v)\n return isNaN(num) ? 0 : num\n }\n return v\n })\n }\n // Handle single value for hasMany (convert to array)\n else if (value !== null && value !== undefined && value !== '') {\n const num = typeof value === 'string' ? parseFloat(value) : value\n doc[field.name] = isNaN(num as number) ? [] : [num]\n }\n // Handle empty/null values - convert to empty array for hasMany\n else {\n doc[field.name] = []\n }\n }\n\n // Handle relationship fields with hasMany - convert comma-separated IDs to arrays\n const relationshipFields = fields.filter(\n (field) =>\n (field.type === 'relationship' || field.type === 'upload') &&\n field.hasMany === true &&\n !Array.isArray(field.relationTo), // Skip polymorphic for now, handled separately\n )\n for (const field of relationshipFields) {\n const value = doc[field.name]\n\n // Handle comma-separated string of IDs (e.g., \"id1,id2,id3\")\n if (typeof value === 'string' && value.includes(',')) {\n doc[field.name] = value\n .split(',')\n .map((v) => v.trim())\n .filter((v) => v !== '')\n }\n // Keep array as-is if already an array\n else if (Array.isArray(value)) {\n doc[field.name] = value.filter((v) => v !== null && v !== undefined && v !== '')\n }\n // Convert single value to array for hasMany\n else if (value !== null && value !== undefined && value !== '') {\n doc[field.name] = [value]\n }\n }\n\n // Handle polymorphic relationships - transform from flat structure to proper format\n for (const [key, value] of Object.entries(doc)) {\n // Handle arrays of polymorphic relationships\n if (Array.isArray(value)) {\n // Check if this array contains polymorphic relationship objects\n const hasPolymorphicItems = value.some(\n (item) => typeof item === 'object' && item !== null && 'relationTo' in item,\n )\n\n if (hasPolymorphicItems) {\n // Filter out null/invalid polymorphic items and transform valid ones\n const processedArray = []\n for (let i = 0; i < value.length; i++) {\n const item = value[i]\n if (typeof item === 'object' && item !== null && 'relationTo' in item) {\n const typedItem = item as Record<string, unknown>\n\n // Skip if both relationTo and value/id are null/empty\n if (!typedItem.relationTo || (!typedItem.id && !typedItem.value)) {\n continue\n }\n\n // Transform from {relationTo: 'collection', id: '123'} to {relationTo: 'collection', value: '123'}\n if ('id' in typedItem) {\n typedItem.value = typedItem.id\n delete typedItem.id\n }\n\n processedArray.push(typedItem)\n } else if (item !== null && item !== undefined) {\n processedArray.push(item)\n }\n }\n\n // Update the array with filtered results\n if (value.length !== processedArray.length) {\n doc[key] = processedArray.length > 0 ? processedArray : []\n }\n }\n // For non-polymorphic arrays, preserve null placeholders for sparse arrays\n }\n // Handle single polymorphic relationships\n else if (typeof value === 'object' && value !== null && !Array.isArray(value)) {\n // Check if this is a single polymorphic relationship\n if ('relationTo' in value && ('id' in value || 'value' in value)) {\n const typedValue = value as Record<string, unknown>\n\n // If both relationTo and value are null/empty, set the whole field to null\n if (!typedValue.relationTo || (!typedValue.id && !typedValue.value)) {\n doc[key] = null\n } else {\n // If it has 'id', transform to 'value'\n if ('id' in typedValue && !('value' in typedValue)) {\n typedValue.value = typedValue.id\n delete typedValue.id\n }\n }\n } else {\n // Recursively process nested objects\n postProcessDocument(value as Record<string, unknown>, fields)\n }\n }\n }\n\n // Process rich text fields to ensure proper data types\n const richTextFields = fields.filter((field) => field.type === 'richText')\n for (const field of richTextFields) {\n if (field.name in doc && doc[field.name]) {\n doc[field.name] = processRichTextField(doc[field.name])\n }\n }\n\n // Also process rich text fields in blocks\n const blockFields = fields.filter((field) => field.type === 'blocks')\n for (const field of blockFields) {\n if (field.name in doc && Array.isArray(doc[field.name])) {\n const blocks = doc[field.name] as any[]\n for (const block of blocks) {\n if (!block || typeof block !== 'object') {\n continue\n }\n\n // Look for richText fields directly in the block\n for (const [key, value] of Object.entries(block)) {\n if (key === 'richText' || (typeof key === 'string' && key.includes('richText'))) {\n block[key] = processRichTextField(value)\n }\n }\n }\n }\n }\n}\n"],"names":["processRichTextField","unflattenObject","data","fields","fromCSVFunctions","req","result","sortedKeys","Object","keys","sort","a","b","aMatch","match","bMatch","index","undefined","aBase","substring","bBase","parseInt","localeCompare","flatKey","value","endsWith","baseKey","replace","idKey","isPolymorphic","some","field","name","type","Array","isArray","relationTo","relationToKey","columnName","pathSegments","split","currentObject","i","length","segment","nextSegment","isLast","isArrayIndex","test","blockFieldName","isBlockField","parent","getParentObject","slice","blockType","parentKey","isPreviousSegmentArrayIndex","isRelationship","find","f","concat","join","relationToValue","arrayIndex","arr","push","isBlocksField","blockSlug","blockObject","lastSegment","postProcessDocument","err","payload","logger","error","msg","obj","segments","current","next","doc","localizedFields","filter","localized","processedLocalizedFields","Set","has","localePattern","RegExp","localeData","keysToDelete","key","entries","locale","forEach","add","numberFields","hasMany","includes","map","v","trim","num","parseFloat","isNaN","relationshipFields","hasPolymorphicItems","item","processedArray","typedItem","id","typedValue","richTextFields","blockFields","blocks","block"],"mappings":"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"}
|
|
1
|
+
{"version":3,"sources":["../../src/utilities/unflattenObject.ts"],"sourcesContent":["import type { FlattenedField, PayloadRequest } from 'payload'\n\nimport type { ImportFieldHookEntry } from '../types.js'\n\nimport { getNestedFlattenedFields } from './flattenedFields.js'\nimport { postProcessDocument } from './unflattenPostProcess.js'\n\ntype UnflattenArgs = {\n data: Record<string, unknown>\n fields: FlattenedField[]\n format?: 'csv' | 'json' | ({} & string)\n importFieldHooks?: Record<string, ImportFieldHookEntry>\n req: PayloadRequest\n}\n\nconst indexSegment = /^\\d+$/\n\nconst collectArrayLikeNames = (fields: FlattenedField[], into: Set<string>): void => {\n for (const field of fields) {\n if (!('name' in field) || !field.name) {\n continue\n }\n if (field.type === 'array' || field.type === 'blocks') {\n into.add(field.name)\n }\n const nested = getNestedFlattenedFields(field)\n if (nested) {\n collectArrayLikeNames(nested, into)\n }\n }\n}\n\n/**\n * Drops numeric array-index segments from a flat key so a runtime key like\n * `items_0_note` matches the static, index-free key a user hook is registered\n * under. A digit-only segment is only stripped when the segment immediately\n * before it names an array or blocks field, so a literal field name like\n * `2024` is preserved.\n */\nconst toLogicalKey = (flatKey: string, arrayLikeNames: Set<string>): string => {\n const segments = flatKey.split('_')\n return segments\n .filter((seg, i) => !indexSegment.test(seg) || !arrayLikeNames.has(segments[i - 1] ?? ''))\n .join('_')\n}\n\n/**\n * Converts flattened CSV data back into a nested document structure.\n *\n * The algorithm:\n * 1. Sorts keys to ensure array indices are processed in order\n * 2. For each flattened key (e.g., \"blocks_0_hero_title\"), splits by underscore into path segments\n * 3. Traverses/builds the nested structure, handling:\n * - Arrays (numeric segments like \"0\", \"1\")\n * - Blocks (blockType detection from slug patterns)\n * - Polymorphic relationships (_relationTo and _id suffix pairs)\n * - Regular nested objects\n * 4. Post-processes to handle localized fields, hasMany conversions, and relationship transforms\n */\nexport const unflattenObject = ({\n data,\n fields,\n format = 'csv',\n importFieldHooks = {},\n req,\n}: UnflattenArgs): Record<string, unknown> => {\n if (!data || typeof data !== 'object') {\n return {}\n }\n\n const result: Record<string, unknown> = {}\n\n const arrayLikeNames = new Set<string>()\n collectArrayLikeNames(fields, arrayLikeNames)\n\n // Sort keys to ensure array indices are processed in order\n const sortedKeys = Object.keys(data).sort((a, b) => {\n // Extract array indices from flattened keys (e.g., \"field_0_subfield\" -> \"0\")\n const aMatch = a.match(/_(\\d+)(?:_|$)/)\n const bMatch = b.match(/_(\\d+)(?:_|$)/)\n\n if (aMatch && bMatch && aMatch.index !== undefined && bMatch.index !== undefined) {\n const aBase = a.substring(0, aMatch.index)\n const bBase = b.substring(0, bMatch.index)\n\n if (aBase === bBase) {\n return (parseInt(aMatch?.[1] ?? '0', 10) || 0) - (parseInt(bMatch?.[1] ?? '0', 10) || 0)\n }\n }\n\n return a.localeCompare(b)\n })\n\n for (const flatKey of sortedKeys) {\n let value = data[flatKey]\n\n // Skip undefined values but keep null for required field validation\n if (value === undefined) {\n continue\n }\n\n // Preserve system fields with underscore prefix (like _status) without splitting\n if (flatKey === '_status') {\n result[flatKey] = value\n continue\n }\n\n // Check if this is a _relationTo key for a polymorphic relationship\n if (flatKey.endsWith('_relationTo')) {\n const baseKey = flatKey.replace(/_relationTo$/, '')\n const idKey = `${baseKey}_id`\n\n // Check if this is a polymorphic relationship field\n const isPolymorphic = fields.some(\n (field) =>\n field.name === baseKey &&\n field.type === 'relationship' &&\n 'relationTo' in field &&\n Array.isArray(field.relationTo),\n )\n\n if (isPolymorphic) {\n if (baseKey in result) {\n continue\n }\n\n // If the corresponding _id key is undefined, skip processing entirely\n // This prevents creating empty objects when we should preserve existing data\n if (!(idKey in data) || data[idKey] === undefined) {\n continue\n }\n }\n }\n\n // Check if this is a _id key for a polymorphic relationship where _relationTo is undefined\n if (flatKey.endsWith('_id')) {\n const baseKey = flatKey.replace(/_id$/, '')\n const relationToKey = `${baseKey}_relationTo`\n\n // Check if this is a polymorphic relationship field\n const isPolymorphic = fields.some(\n (field) =>\n field.name === baseKey &&\n field.type === 'relationship' &&\n 'relationTo' in field &&\n Array.isArray(field.relationTo),\n )\n\n if (isPolymorphic) {\n // If the corresponding _relationTo key is undefined, skip processing entirely\n // This prevents creating empty objects when we should preserve existing data\n if (!(relationToKey in data) || data[relationToKey] === undefined) {\n continue\n }\n }\n }\n\n const importHookEntry =\n importFieldHooks[flatKey] ?? importFieldHooks[toLogicalKey(flatKey, arrayLikeNames)]\n if (importHookEntry) {\n try {\n if (importHookEntry.type === 'beforeImport') {\n value = importHookEntry.fn({\n columnName: flatKey,\n data,\n format,\n siblingData: data,\n siblingDoc: data,\n value,\n })\n } else {\n value = importHookEntry.fn({\n columnName: flatKey,\n data,\n value,\n })\n }\n } catch (error) {\n req.payload.logger.error({\n err: error,\n msg: `[plugin-import-export] Field-level beforeImport hook for \"${flatKey}\" threw — falling back to original value`,\n })\n // Keep the original value so the row is not dropped — downstream\n // validation will surface any deeper issue per-row.\n }\n }\n\n // Example: \"blocks_0_content_text\" -> [\"blocks\", \"0\", \"content\", \"text\"]\n const pathSegments = flatKey.split('_')\n let currentObject: Record<string, unknown> = result\n\n for (let i = 0; i < pathSegments.length; i++) {\n const segment = pathSegments[i]\n if (!segment) {\n continue\n } // Skip empty segments\n\n const nextSegment = pathSegments[i + 1]\n const isLast = i === pathSegments.length - 1\n\n // Check if next segment is a numeric array index (e.g., \"0\", \"1\", \"2\")\n const isArrayIndex = nextSegment !== undefined && /^\\d+$/.test(nextSegment)\n\n if (isLast) {\n // Special handling for blockType suffix in blocks\n if (segment === 'blockType' && i >= 3) {\n // Pattern: blocks_0_hero_blockType -> set blockType on the block\n const blockFieldName = pathSegments[0] // 'blocks'\n const isBlockField = fields.some(\n (field) => field.name === blockFieldName && field.type === 'blocks',\n )\n\n if (isBlockField && pathSegments[1]?.match(/^\\d+$/)) {\n const parent = getParentObject(result, pathSegments.slice(0, 2))\n if (parent && typeof parent === 'object') {\n parent.blockType = value\n }\n continue\n }\n }\n\n // Special handling for relationship fields with _id suffix\n if (segment === 'id' && i > 0) {\n const parentKey = pathSegments[i - 1]\n const isPreviousSegmentArrayIndex = parentKey ? /^\\d+$/.test(parentKey) : false\n\n if (!isPreviousSegmentArrayIndex) {\n // Check if this is a relationship field\n const isRelationship = fields.some(\n (field) => field.name === parentKey && field.type === 'relationship',\n )\n\n if (isRelationship) {\n // Check if this is a polymorphic relationship field\n const field = fields.find((f) => f.name === parentKey && f.type === 'relationship')\n const isPolymorphic =\n field && 'relationTo' in field && Array.isArray(field.relationTo)\n\n if (isPolymorphic) {\n const relationToKey = pathSegments.slice(0, i).concat('relationTo').join('_')\n const relationToValue = data[relationToKey]\n\n const parent = getParentObject(result, pathSegments.slice(0, i - 1))\n if (parent && parentKey && typeof parent === 'object') {\n // Both fields must be defined to create/update the relationship\n // If either is undefined, skip the field entirely (preserve existing data)\n if (value !== undefined && relationToValue !== undefined) {\n // Check if both are explicitly null\n if (relationToValue === null && value === null) {\n // Only set to null if explicitly null (user typed \"null\" in CSV)\n parent[parentKey] = null\n } else if (relationToValue || value) {\n // At least one has a value, create the relationship\n parent[parentKey] = {\n relationTo: relationToValue,\n value, // This will be transformed to proper format in postProcess\n }\n }\n // If both are empty strings, don't set the field (handled by not meeting the above conditions)\n }\n // If either is undefined, don't set the field at all (preserve existing data)\n }\n continue\n } else if (!isPolymorphic) {\n const parent = getParentObject(result, pathSegments.slice(0, i - 1))\n if (parent && parentKey && typeof parent === 'object') {\n parent[parentKey] = value\n }\n continue\n }\n }\n }\n }\n\n // _relationTo suffix is handled when processing the _id field above\n if (segment === 'relationTo' && i > 0) {\n const parentKey = pathSegments[i - 1]\n if (parentKey && !parentKey.match(/^\\d+$/)) {\n const field = fields.find((f) => f.name === parentKey && f.type === 'relationship')\n const isPolymorphic = field && 'relationTo' in field && Array.isArray(field.relationTo)\n\n if (isPolymorphic) {\n // For polymorphic relationships, this is handled when processing the _id field\n // Skip it entirely\n continue\n }\n }\n }\n\n currentObject[segment] = value\n } else if (isArrayIndex && nextSegment !== undefined) {\n if (!currentObject[segment] || !Array.isArray(currentObject[segment])) {\n currentObject[segment] = []\n }\n\n const arrayIndex = parseInt(nextSegment)\n const arr = currentObject[segment] as unknown[]\n\n // Ensure array has sufficient length\n while (arr.length <= arrayIndex) {\n arr.push(null)\n }\n\n // Handle array of objects\n if (arr[arrayIndex] === null || arr[arrayIndex] === undefined) {\n arr[arrayIndex] = {}\n }\n\n // Handle blocks field with block slug pattern (e.g., blocks_0_hero_title)\n const isBlocksField = fields.some((f) => f.name === segment && f.type === 'blocks')\n if (isBlocksField && i + 3 < pathSegments.length) {\n const blockSlug = pathSegments[i + 2]\n const blockFieldName = pathSegments[i + 3]\n\n if (blockSlug && blockFieldName) {\n const blockObject = arr[arrayIndex] as Record<string, unknown>\n blockObject.blockType = blockSlug\n\n if (i + 3 === pathSegments.length - 1) {\n blockObject[blockFieldName] = value\n } else {\n if (!blockObject[blockFieldName] || typeof blockObject[blockFieldName] !== 'object') {\n blockObject[blockFieldName] = {}\n }\n currentObject = blockObject[blockFieldName] as Record<string, unknown>\n i = i + 3\n continue\n }\n break\n }\n }\n\n if (i + 2 === pathSegments.length - 1) {\n const lastSegment = pathSegments[pathSegments.length - 1]\n if (lastSegment && arr[arrayIndex] && typeof arr[arrayIndex] === 'object') {\n ;(arr[arrayIndex] as Record<string, unknown>)[lastSegment] = value\n }\n break\n } else if (i + 1 === pathSegments.length - 1) {\n // Direct array value (e.g., tags_0 = \"value\")\n arr[arrayIndex] = value\n break\n } else {\n currentObject = arr[arrayIndex] as Record<string, unknown>\n i++\n }\n } else {\n // Skip if already set to null (polymorphic relationship already processed)\n if (currentObject[segment] === null && isLast && segment === 'relationTo') {\n continue\n }\n\n if (\n !currentObject[segment] ||\n typeof currentObject[segment] !== 'object' ||\n Array.isArray(currentObject[segment])\n ) {\n currentObject[segment] = {}\n }\n\n // Handle polymorphic relationship arrays\n if (segment === 'relationTo' && i > 0 && pathSegments[i - 1]?.match(/^\\d+$/)) {\n currentObject[segment] = value\n } else if (\n typeof currentObject[segment] === 'object' &&\n !Array.isArray(currentObject[segment]) &&\n currentObject[segment] !== null\n ) {\n currentObject = currentObject[segment] as Record<string, unknown>\n }\n }\n }\n }\n\n try {\n // Post-process to handle special structures\n postProcessDocument(result, fields)\n } catch (err) {\n // Log but don't throw - return partially processed result\n\n req.payload.logger.error({\n err,\n msg: '[plugin-import-export] Error in postProcessDocument',\n })\n }\n\n return result\n}\n\nconst getParentObject = (\n obj: Record<string, unknown>,\n segments: string[],\n): Record<string, unknown> | undefined => {\n let current: Record<string, unknown> = obj\n\n for (let i = 0; i < segments.length; i++) {\n const segment = segments[i]\n const nextSegment = segments[i + 1]\n\n if (!segment) {\n continue\n }\n\n if (nextSegment && /^\\d+$/.test(nextSegment)) {\n const arrayIndex = parseInt(nextSegment)\n const arr = current[segment] as unknown[]\n\n if (Array.isArray(arr) && arr[arrayIndex]) {\n current = arr[arrayIndex] as Record<string, unknown>\n i++ // Skip the index\n } else {\n return undefined\n }\n } else {\n const next = current[segment]\n if (typeof next === 'object' && next !== null && !Array.isArray(next)) {\n current = next as Record<string, unknown>\n } else {\n return undefined\n }\n }\n }\n\n return current\n}\n"],"names":["getNestedFlattenedFields","postProcessDocument","indexSegment","collectArrayLikeNames","fields","into","field","name","type","add","nested","toLogicalKey","flatKey","arrayLikeNames","segments","split","filter","seg","i","test","has","join","unflattenObject","data","format","importFieldHooks","req","result","Set","sortedKeys","Object","keys","sort","a","b","aMatch","match","bMatch","index","undefined","aBase","substring","bBase","parseInt","localeCompare","value","endsWith","baseKey","replace","idKey","isPolymorphic","some","Array","isArray","relationTo","relationToKey","importHookEntry","fn","columnName","siblingData","siblingDoc","error","payload","logger","err","msg","pathSegments","currentObject","length","segment","nextSegment","isLast","isArrayIndex","blockFieldName","isBlockField","parent","getParentObject","slice","blockType","parentKey","isPreviousSegmentArrayIndex","isRelationship","find","f","concat","relationToValue","arrayIndex","arr","push","isBlocksField","blockSlug","blockObject","lastSegment","obj","current","next"],"mappings":"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"}
|
|
@@ -627,6 +627,39 @@ describe('unflattenObject', ()=>{
|
|
|
627
627
|
});
|
|
628
628
|
});
|
|
629
629
|
});
|
|
630
|
+
describe('field-hook key resolution', ()=>{
|
|
631
|
+
it('should fire an import hook on a digit-only field name nested inside an array', ()=>{
|
|
632
|
+
const fields = [
|
|
633
|
+
{
|
|
634
|
+
name: 'archives',
|
|
635
|
+
type: 'array',
|
|
636
|
+
flattenedFields: [
|
|
637
|
+
{
|
|
638
|
+
name: '2024',
|
|
639
|
+
type: 'text'
|
|
640
|
+
}
|
|
641
|
+
]
|
|
642
|
+
}
|
|
643
|
+
];
|
|
644
|
+
const importFieldHooks = {
|
|
645
|
+
archives_2024: {
|
|
646
|
+
type: 'beforeImport',
|
|
647
|
+
fn: ({ value })=>`transformed-${value}`
|
|
648
|
+
}
|
|
649
|
+
};
|
|
650
|
+
const data = {
|
|
651
|
+
archives_0_2024: 'raw'
|
|
652
|
+
};
|
|
653
|
+
const result = unflattenObject({
|
|
654
|
+
data,
|
|
655
|
+
fields,
|
|
656
|
+
importFieldHooks,
|
|
657
|
+
req: mockReq
|
|
658
|
+
});
|
|
659
|
+
const archives = result.archives;
|
|
660
|
+
expect(archives[0]['2024']).toBe('transformed-raw');
|
|
661
|
+
});
|
|
662
|
+
});
|
|
630
663
|
describe('edge cases', ()=>{
|
|
631
664
|
it('should handle empty data', ()=>{
|
|
632
665
|
const result = unflattenObject({
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../../src/utilities/unflattenObject.spec.ts"],"sourcesContent":["import { FlattenedField, PayloadRequest } from 'payload'\n\nimport { unflattenObject } from './unflattenObject.js'\n\nimport { describe, it, expect, vi } from 'vitest'\n\ndescribe('unflattenObject', () => {\n const mockReq = {\n payload: {\n logger: {\n error: vi.fn(),\n },\n },\n } as unknown as PayloadRequest\n\n describe('hasMany number fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'hasManyNumber',\n type: 'number',\n hasMany: true,\n } as FlattenedField,\n ]\n\n it('should handle comma-separated number strings', () => {\n const data = {\n hasManyNumber: '1,2,3,5,8',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 2, 3, 5, 8],\n })\n })\n\n it('should handle comma-separated numbers with spaces', () => {\n const data = {\n hasManyNumber: ' 10 , 20 , 30 ',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [10, 20, 30],\n })\n })\n\n it('should filter out empty values in comma-separated strings', () => {\n const data = {\n hasManyNumber: '1,,3,,5',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 3, 5],\n })\n })\n\n it('should handle single number values', () => {\n const data = {\n hasManyNumber: 42,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [42],\n })\n })\n\n it('should handle single string number values', () => {\n const data = {\n hasManyNumber: '42',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [42],\n })\n })\n\n it('should handle indexed array format', () => {\n const data = {\n hasManyNumber_0: 1,\n hasManyNumber_1: 2,\n hasManyNumber_2: 3,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 2, 3],\n })\n })\n\n it('should filter out null and empty values from indexed arrays', () => {\n const data = {\n hasManyNumber_0: 1,\n hasManyNumber_1: null,\n hasManyNumber_2: '',\n hasManyNumber_3: 3,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 3],\n })\n })\n\n it('should handle empty, null, and undefined values', () => {\n // explicit null gets converted to empty array in postProcess for hasMany\n expect(unflattenObject({ data: { hasManyNumber: null }, fields, req: mockReq })).toEqual({\n hasManyNumber: [],\n })\n // undefined is skipped entirely (preserves existing data)\n expect(unflattenObject({ data: { hasManyNumber: undefined }, fields, req: mockReq })).toEqual(\n {},\n )\n })\n })\n\n describe('hasMany relationship fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'hasManyRelationship',\n type: 'relationship',\n hasMany: true,\n relationTo: 'posts',\n } as FlattenedField,\n ]\n\n it('should handle comma-separated ID strings', () => {\n const data = {\n hasManyRelationship: 'id1,id2,id3',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id2', 'id3'],\n })\n })\n\n it('should handle comma-separated IDs with spaces', () => {\n const data = {\n hasManyRelationship: ' id1 , id2 , id3 ',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id2', 'id3'],\n })\n })\n\n it('should filter out empty values in comma-separated IDs', () => {\n const data = {\n hasManyRelationship: 'id1,,id3,,id5',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id3', 'id5'],\n })\n })\n\n it('should handle single ID values', () => {\n const data = {\n hasManyRelationship: 'singleId',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['singleId'],\n })\n })\n\n it('should handle indexed array format', () => {\n const data = {\n hasManyRelationship_0: 'id1',\n hasManyRelationship_1: 'id2',\n hasManyRelationship_2: 'id3',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id2', 'id3'],\n })\n })\n\n it('should handle MongoDB ObjectIDs', () => {\n const data = {\n hasManyRelationship: '507f1f77bcf86cd799439011,507f191e810c19729de860ea',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['507f1f77bcf86cd799439011', '507f191e810c19729de860ea'],\n })\n })\n })\n\n describe('localized fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'title',\n type: 'text',\n localized: true,\n } as FlattenedField,\n ]\n\n it('should transform locale-specific keys to nested structure', () => {\n const data = {\n title_en: 'English Title',\n title_es: 'Título en Español',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n title: {\n en: 'English Title',\n es: 'Título en Español',\n },\n })\n })\n\n it('should handle missing locales', () => {\n const data = {\n title_en: 'English Title',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n title: {\n en: 'English Title',\n },\n })\n })\n })\n\n describe('blocks fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'blocks',\n type: 'blocks',\n } as FlattenedField,\n ]\n\n it('should handle block fields with blockType', () => {\n const data = {\n blocks_0_hero_title: 'Hero Title',\n blocks_0_hero_subtitle: 'Hero Subtitle',\n blocks_0_hero_blockType: 'hero',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n blocks: [\n {\n blockType: 'hero',\n title: 'Hero Title',\n subtitle: 'Hero Subtitle',\n },\n ],\n })\n })\n\n it('should handle multiple blocks', () => {\n const data = {\n blocks_0_hero_title: 'Hero Title',\n blocks_0_hero_blockType: 'hero',\n blocks_1_text_content: 'Text Content',\n blocks_1_text_blockType: 'text',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n blocks: [\n {\n blockType: 'hero',\n title: 'Hero Title',\n },\n {\n blockType: 'text',\n content: 'Text Content',\n },\n ],\n })\n })\n })\n\n describe('array fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'items',\n type: 'array',\n } as FlattenedField,\n ]\n\n it('should handle indexed array objects', () => {\n const data = {\n items_0_name: 'Item 1',\n items_0_value: 10,\n items_1_name: 'Item 2',\n items_1_value: 20,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n items: [\n { name: 'Item 1', value: 10 },\n { name: 'Item 2', value: 20 },\n ],\n })\n })\n\n it('should handle sparse arrays', () => {\n const data = {\n items_0_name: 'Item 1',\n items_2_name: 'Item 3',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n items: [{ name: 'Item 1' }, null, { name: 'Item 3' }],\n })\n })\n })\n\n describe('group fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'group',\n type: 'group',\n } as FlattenedField,\n ]\n\n it('should handle nested group fields', () => {\n const data = {\n group_field1: 'Value 1',\n group_field2: 'Value 2',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n group: {\n field1: 'Value 1',\n field2: 'Value 2',\n },\n })\n })\n })\n\n describe('polymorphic relationships', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphic',\n type: 'relationship',\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n it('should handle polymorphic relationship with id and relationTo', () => {\n const data = {\n polymorphic_id: '123',\n polymorphic_relationTo: 'posts',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphic: {\n relationTo: 'posts',\n value: '123',\n },\n })\n })\n\n it('should handle explicitly null polymorphic relationships', () => {\n const data = {\n polymorphic_id: null,\n polymorphic_relationTo: null,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphic: null,\n })\n })\n\n it('should skip polymorphic relationships with undefined values', () => {\n const data = {\n polymorphic_id: undefined,\n polymorphic_relationTo: undefined,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n // Both undefined means field is not set (preserves existing data)\n expect(result).toEqual({})\n })\n\n it('should skip polymorphic relationship with undefined id', () => {\n const data = {\n polymorphic_id: undefined,\n polymorphic_relationTo: 'posts',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n // Undefined ID means don't update this field\n expect(result).toEqual({})\n })\n\n it('should skip polymorphic relationship with undefined relationTo', () => {\n const data = {\n polymorphic_id: '123',\n polymorphic_relationTo: undefined,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n // Undefined relationTo means don't update this field\n expect(result).toEqual({})\n })\n\n it('should handle polymorphic hasMany relationships', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphicMany',\n type: 'relationship',\n hasMany: true,\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n const data = {\n polymorphicMany_0_id: '123',\n polymorphicMany_0_relationTo: 'posts',\n polymorphicMany_1_id: '456',\n polymorphicMany_1_relationTo: 'pages',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphicMany: [\n {\n relationTo: 'posts',\n value: '123',\n },\n {\n relationTo: 'pages',\n value: '456',\n },\n ],\n })\n })\n\n it('should filter out empty polymorphic hasMany items', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphicMany',\n type: 'relationship',\n hasMany: true,\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n const data = {\n polymorphicMany_0_id: '123',\n polymorphicMany_0_relationTo: 'posts',\n polymorphicMany_1_id: null,\n polymorphicMany_1_relationTo: null,\n polymorphicMany_2_id: '456',\n polymorphicMany_2_relationTo: 'pages',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphicMany: [\n {\n relationTo: 'posts',\n value: '123',\n },\n {\n relationTo: 'pages',\n value: '456',\n },\n ],\n })\n })\n\n it('should handle all empty polymorphic hasMany items', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphicMany',\n type: 'relationship',\n hasMany: true,\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n const data = {\n polymorphicMany_0_id: null,\n polymorphicMany_0_relationTo: null,\n polymorphicMany_1_id: '',\n polymorphicMany_1_relationTo: '',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphicMany: [],\n })\n })\n })\n\n describe('edge cases', () => {\n it('should handle empty data', () => {\n const result = unflattenObject({ data: {}, fields: [], req: mockReq })\n expect(result).toEqual({})\n })\n\n it('should handle null data', () => {\n const result = unflattenObject({ data: null as any, fields: [], req: mockReq })\n expect(result).toEqual({})\n })\n\n it('should handle undefined values', () => {\n const data = {\n field1: undefined,\n field2: 'value',\n }\n\n const result = unflattenObject({ data, fields: [], req: mockReq })\n expect(result).toEqual({ field2: 'value' })\n })\n\n it('should preserve null values for validation', () => {\n const data = {\n field1: null,\n field2: 'value',\n }\n\n const result = unflattenObject({ data, fields: [], req: mockReq })\n // null values are preserved for validation\n expect(result).toEqual({ field1: null, field2: 'value' })\n })\n })\n})\n"],"names":["unflattenObject","describe","it","expect","vi","mockReq","payload","logger","error","fn","fields","name","type","hasMany","data","hasManyNumber","result","req","toEqual","hasManyNumber_0","hasManyNumber_1","hasManyNumber_2","hasManyNumber_3","undefined","relationTo","hasManyRelationship","hasManyRelationship_0","hasManyRelationship_1","hasManyRelationship_2","localized","title_en","title_es","title","en","es","blocks_0_hero_title","blocks_0_hero_subtitle","blocks_0_hero_blockType","blocks","blockType","subtitle","blocks_1_text_content","blocks_1_text_blockType","content","items_0_name","items_0_value","items_1_name","items_1_value","items","value","items_2_name","group_field1","group_field2","group","field1","field2","polymorphic_id","polymorphic_relationTo","polymorphic","polymorphicMany_0_id","polymorphicMany_0_relationTo","polymorphicMany_1_id","polymorphicMany_1_relationTo","polymorphicMany","polymorphicMany_2_id","polymorphicMany_2_relationTo"],"mappings":"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"}
|
|
1
|
+
{"version":3,"sources":["../../src/utilities/unflattenObject.spec.ts"],"sourcesContent":["import { FlattenedField, PayloadRequest } from 'payload'\n\nimport { unflattenObject } from './unflattenObject.js'\n\nimport { describe, it, expect, vi } from 'vitest'\n\ndescribe('unflattenObject', () => {\n const mockReq = {\n payload: {\n logger: {\n error: vi.fn(),\n },\n },\n } as unknown as PayloadRequest\n\n describe('hasMany number fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'hasManyNumber',\n type: 'number',\n hasMany: true,\n } as FlattenedField,\n ]\n\n it('should handle comma-separated number strings', () => {\n const data = {\n hasManyNumber: '1,2,3,5,8',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 2, 3, 5, 8],\n })\n })\n\n it('should handle comma-separated numbers with spaces', () => {\n const data = {\n hasManyNumber: ' 10 , 20 , 30 ',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [10, 20, 30],\n })\n })\n\n it('should filter out empty values in comma-separated strings', () => {\n const data = {\n hasManyNumber: '1,,3,,5',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 3, 5],\n })\n })\n\n it('should handle single number values', () => {\n const data = {\n hasManyNumber: 42,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [42],\n })\n })\n\n it('should handle single string number values', () => {\n const data = {\n hasManyNumber: '42',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [42],\n })\n })\n\n it('should handle indexed array format', () => {\n const data = {\n hasManyNumber_0: 1,\n hasManyNumber_1: 2,\n hasManyNumber_2: 3,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 2, 3],\n })\n })\n\n it('should filter out null and empty values from indexed arrays', () => {\n const data = {\n hasManyNumber_0: 1,\n hasManyNumber_1: null,\n hasManyNumber_2: '',\n hasManyNumber_3: 3,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyNumber: [1, 3],\n })\n })\n\n it('should handle empty, null, and undefined values', () => {\n // explicit null gets converted to empty array in postProcess for hasMany\n expect(unflattenObject({ data: { hasManyNumber: null }, fields, req: mockReq })).toEqual({\n hasManyNumber: [],\n })\n // undefined is skipped entirely (preserves existing data)\n expect(unflattenObject({ data: { hasManyNumber: undefined }, fields, req: mockReq })).toEqual(\n {},\n )\n })\n })\n\n describe('hasMany relationship fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'hasManyRelationship',\n type: 'relationship',\n hasMany: true,\n relationTo: 'posts',\n } as FlattenedField,\n ]\n\n it('should handle comma-separated ID strings', () => {\n const data = {\n hasManyRelationship: 'id1,id2,id3',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id2', 'id3'],\n })\n })\n\n it('should handle comma-separated IDs with spaces', () => {\n const data = {\n hasManyRelationship: ' id1 , id2 , id3 ',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id2', 'id3'],\n })\n })\n\n it('should filter out empty values in comma-separated IDs', () => {\n const data = {\n hasManyRelationship: 'id1,,id3,,id5',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id3', 'id5'],\n })\n })\n\n it('should handle single ID values', () => {\n const data = {\n hasManyRelationship: 'singleId',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['singleId'],\n })\n })\n\n it('should handle indexed array format', () => {\n const data = {\n hasManyRelationship_0: 'id1',\n hasManyRelationship_1: 'id2',\n hasManyRelationship_2: 'id3',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['id1', 'id2', 'id3'],\n })\n })\n\n it('should handle MongoDB ObjectIDs', () => {\n const data = {\n hasManyRelationship: '507f1f77bcf86cd799439011,507f191e810c19729de860ea',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n hasManyRelationship: ['507f1f77bcf86cd799439011', '507f191e810c19729de860ea'],\n })\n })\n })\n\n describe('localized fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'title',\n type: 'text',\n localized: true,\n } as FlattenedField,\n ]\n\n it('should transform locale-specific keys to nested structure', () => {\n const data = {\n title_en: 'English Title',\n title_es: 'Título en Español',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n title: {\n en: 'English Title',\n es: 'Título en Español',\n },\n })\n })\n\n it('should handle missing locales', () => {\n const data = {\n title_en: 'English Title',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n title: {\n en: 'English Title',\n },\n })\n })\n })\n\n describe('blocks fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'blocks',\n type: 'blocks',\n } as FlattenedField,\n ]\n\n it('should handle block fields with blockType', () => {\n const data = {\n blocks_0_hero_title: 'Hero Title',\n blocks_0_hero_subtitle: 'Hero Subtitle',\n blocks_0_hero_blockType: 'hero',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n blocks: [\n {\n blockType: 'hero',\n title: 'Hero Title',\n subtitle: 'Hero Subtitle',\n },\n ],\n })\n })\n\n it('should handle multiple blocks', () => {\n const data = {\n blocks_0_hero_title: 'Hero Title',\n blocks_0_hero_blockType: 'hero',\n blocks_1_text_content: 'Text Content',\n blocks_1_text_blockType: 'text',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n blocks: [\n {\n blockType: 'hero',\n title: 'Hero Title',\n },\n {\n blockType: 'text',\n content: 'Text Content',\n },\n ],\n })\n })\n })\n\n describe('array fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'items',\n type: 'array',\n } as FlattenedField,\n ]\n\n it('should handle indexed array objects', () => {\n const data = {\n items_0_name: 'Item 1',\n items_0_value: 10,\n items_1_name: 'Item 2',\n items_1_value: 20,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n items: [\n { name: 'Item 1', value: 10 },\n { name: 'Item 2', value: 20 },\n ],\n })\n })\n\n it('should handle sparse arrays', () => {\n const data = {\n items_0_name: 'Item 1',\n items_2_name: 'Item 3',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n items: [{ name: 'Item 1' }, null, { name: 'Item 3' }],\n })\n })\n })\n\n describe('group fields', () => {\n const fields: FlattenedField[] = [\n {\n name: 'group',\n type: 'group',\n } as FlattenedField,\n ]\n\n it('should handle nested group fields', () => {\n const data = {\n group_field1: 'Value 1',\n group_field2: 'Value 2',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n group: {\n field1: 'Value 1',\n field2: 'Value 2',\n },\n })\n })\n })\n\n describe('polymorphic relationships', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphic',\n type: 'relationship',\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n it('should handle polymorphic relationship with id and relationTo', () => {\n const data = {\n polymorphic_id: '123',\n polymorphic_relationTo: 'posts',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphic: {\n relationTo: 'posts',\n value: '123',\n },\n })\n })\n\n it('should handle explicitly null polymorphic relationships', () => {\n const data = {\n polymorphic_id: null,\n polymorphic_relationTo: null,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphic: null,\n })\n })\n\n it('should skip polymorphic relationships with undefined values', () => {\n const data = {\n polymorphic_id: undefined,\n polymorphic_relationTo: undefined,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n // Both undefined means field is not set (preserves existing data)\n expect(result).toEqual({})\n })\n\n it('should skip polymorphic relationship with undefined id', () => {\n const data = {\n polymorphic_id: undefined,\n polymorphic_relationTo: 'posts',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n // Undefined ID means don't update this field\n expect(result).toEqual({})\n })\n\n it('should skip polymorphic relationship with undefined relationTo', () => {\n const data = {\n polymorphic_id: '123',\n polymorphic_relationTo: undefined,\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n // Undefined relationTo means don't update this field\n expect(result).toEqual({})\n })\n\n it('should handle polymorphic hasMany relationships', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphicMany',\n type: 'relationship',\n hasMany: true,\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n const data = {\n polymorphicMany_0_id: '123',\n polymorphicMany_0_relationTo: 'posts',\n polymorphicMany_1_id: '456',\n polymorphicMany_1_relationTo: 'pages',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphicMany: [\n {\n relationTo: 'posts',\n value: '123',\n },\n {\n relationTo: 'pages',\n value: '456',\n },\n ],\n })\n })\n\n it('should filter out empty polymorphic hasMany items', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphicMany',\n type: 'relationship',\n hasMany: true,\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n const data = {\n polymorphicMany_0_id: '123',\n polymorphicMany_0_relationTo: 'posts',\n polymorphicMany_1_id: null,\n polymorphicMany_1_relationTo: null,\n polymorphicMany_2_id: '456',\n polymorphicMany_2_relationTo: 'pages',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphicMany: [\n {\n relationTo: 'posts',\n value: '123',\n },\n {\n relationTo: 'pages',\n value: '456',\n },\n ],\n })\n })\n\n it('should handle all empty polymorphic hasMany items', () => {\n const fields: FlattenedField[] = [\n {\n name: 'polymorphicMany',\n type: 'relationship',\n hasMany: true,\n relationTo: ['posts', 'pages'],\n } as FlattenedField,\n ]\n\n const data = {\n polymorphicMany_0_id: null,\n polymorphicMany_0_relationTo: null,\n polymorphicMany_1_id: '',\n polymorphicMany_1_relationTo: '',\n }\n\n const result = unflattenObject({ data, fields, req: mockReq })\n\n expect(result).toEqual({\n polymorphicMany: [],\n })\n })\n })\n\n describe('field-hook key resolution', () => {\n it('should fire an import hook on a digit-only field name nested inside an array', () => {\n const fields: FlattenedField[] = [\n {\n name: 'archives',\n type: 'array',\n flattenedFields: [{ name: '2024', type: 'text' }],\n } as unknown as FlattenedField,\n ]\n\n const importFieldHooks = {\n archives_2024: {\n type: 'beforeImport' as const,\n fn: ({ value }: { value: unknown }) => `transformed-${value as string}`,\n },\n }\n\n const data = { archives_0_2024: 'raw' }\n\n const result = unflattenObject({\n data,\n fields,\n importFieldHooks,\n req: mockReq,\n })\n\n const archives = result.archives as Array<Record<string, unknown>>\n expect(archives[0]!['2024']).toBe('transformed-raw')\n })\n })\n\n describe('edge cases', () => {\n it('should handle empty data', () => {\n const result = unflattenObject({ data: {}, fields: [], req: mockReq })\n expect(result).toEqual({})\n })\n\n it('should handle null data', () => {\n const result = unflattenObject({ data: null as any, fields: [], req: mockReq })\n expect(result).toEqual({})\n })\n\n it('should handle undefined values', () => {\n const data = {\n field1: undefined,\n field2: 'value',\n }\n\n const result = unflattenObject({ data, fields: [], req: mockReq })\n expect(result).toEqual({ field2: 'value' })\n })\n\n it('should preserve null values for validation', () => {\n const data = {\n field1: null,\n field2: 'value',\n }\n\n const result = unflattenObject({ data, fields: [], req: mockReq })\n // null values are preserved for validation\n expect(result).toEqual({ field1: null, field2: 'value' })\n })\n })\n})\n"],"names":["unflattenObject","describe","it","expect","vi","mockReq","payload","logger","error","fn","fields","name","type","hasMany","data","hasManyNumber","result","req","toEqual","hasManyNumber_0","hasManyNumber_1","hasManyNumber_2","hasManyNumber_3","undefined","relationTo","hasManyRelationship","hasManyRelationship_0","hasManyRelationship_1","hasManyRelationship_2","localized","title_en","title_es","title","en","es","blocks_0_hero_title","blocks_0_hero_subtitle","blocks_0_hero_blockType","blocks","blockType","subtitle","blocks_1_text_content","blocks_1_text_blockType","content","items_0_name","items_0_value","items_1_name","items_1_value","items","value","items_2_name","group_field1","group_field2","group","field1","field2","polymorphic_id","polymorphic_relationTo","polymorphic","polymorphicMany_0_id","polymorphicMany_0_relationTo","polymorphicMany_1_id","polymorphicMany_1_relationTo","polymorphicMany","polymorphicMany_2_id","polymorphicMany_2_relationTo","flattenedFields","importFieldHooks","archives_2024","archives_0_2024","archives","toBe"],"mappings":"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"}
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
import type { FlattenedField } from 'payload';
|
|
2
|
+
/**
|
|
3
|
+
* Post-processes the unflattened document to handle special field types:
|
|
4
|
+
* - Localized fields: transforms field_locale keys to nested { field: { locale: value } }
|
|
5
|
+
* - Number hasMany: converts comma-separated strings or arrays to number arrays
|
|
6
|
+
* - Relationship hasMany: converts comma-separated IDs to arrays
|
|
7
|
+
* - Polymorphic relationships: transforms flat {relationTo, id} to {relationTo, value}
|
|
8
|
+
* - Rich text fields: ensures proper data structure
|
|
9
|
+
*/
|
|
10
|
+
export declare const postProcessDocument: (doc: Record<string, unknown>, fields: FlattenedField[]) => void;
|
|
11
|
+
//# sourceMappingURL=unflattenPostProcess.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"unflattenPostProcess.d.ts","sourceRoot":"","sources":["../../src/utilities/unflattenPostProcess.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,SAAS,CAAA;AAuK7C;;;;;;;GAOG;AACH,eAAO,MAAM,mBAAmB,QACzB,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,UACpB,cAAc,EAAE,KACvB,IAMF,CAAA"}
|
|
@@ -0,0 +1,148 @@
|
|
|
1
|
+
import { isPlainObject } from './isPlainObject.js';
|
|
2
|
+
import { processRichTextField } from './processRichTextField.js';
|
|
3
|
+
const normalizeLocalizedFields = (doc, fields)=>{
|
|
4
|
+
const processed = new Set();
|
|
5
|
+
for (const field of fields.filter((f)=>f.localized)){
|
|
6
|
+
if (processed.has(field.name)) {
|
|
7
|
+
continue;
|
|
8
|
+
}
|
|
9
|
+
const localePattern = new RegExp(`^${field.name}_([a-z]{2}(?:_[A-Z]{2})?)$`);
|
|
10
|
+
const localeData = {};
|
|
11
|
+
const keysToDelete = [];
|
|
12
|
+
for (const [key, value] of Object.entries(doc)){
|
|
13
|
+
const match = key.match(localePattern);
|
|
14
|
+
if (match && match[1]) {
|
|
15
|
+
localeData[match[1]] = value;
|
|
16
|
+
keysToDelete.push(key);
|
|
17
|
+
}
|
|
18
|
+
}
|
|
19
|
+
if (Object.keys(localeData).length > 0) {
|
|
20
|
+
doc[field.name] = localeData;
|
|
21
|
+
keysToDelete.forEach((key)=>delete doc[key]);
|
|
22
|
+
processed.add(field.name);
|
|
23
|
+
}
|
|
24
|
+
}
|
|
25
|
+
};
|
|
26
|
+
const normalizeHasManyNumbers = (doc, fields)=>{
|
|
27
|
+
const numberFields = fields.filter((field)=>field.type === 'number' && field.hasMany);
|
|
28
|
+
for (const field of numberFields){
|
|
29
|
+
if (!(field.name in doc)) {
|
|
30
|
+
continue;
|
|
31
|
+
}
|
|
32
|
+
const value = doc[field.name];
|
|
33
|
+
if (typeof value === 'string' && value.includes(',')) {
|
|
34
|
+
doc[field.name] = value.split(',').map((v)=>v.trim()).filter((v)=>v !== '').map((v)=>{
|
|
35
|
+
const num = parseFloat(v);
|
|
36
|
+
return isNaN(num) ? 0 : num;
|
|
37
|
+
});
|
|
38
|
+
} else if (Array.isArray(value)) {
|
|
39
|
+
doc[field.name] = value.filter((v)=>v !== null && v !== undefined && v !== '').map((v)=>{
|
|
40
|
+
if (typeof v === 'string') {
|
|
41
|
+
const num = parseFloat(v);
|
|
42
|
+
return isNaN(num) ? 0 : num;
|
|
43
|
+
}
|
|
44
|
+
return v;
|
|
45
|
+
});
|
|
46
|
+
} else if (value !== null && value !== undefined && value !== '') {
|
|
47
|
+
const num = typeof value === 'string' ? parseFloat(value) : value;
|
|
48
|
+
doc[field.name] = isNaN(num) ? [] : [
|
|
49
|
+
num
|
|
50
|
+
];
|
|
51
|
+
} else {
|
|
52
|
+
doc[field.name] = [];
|
|
53
|
+
}
|
|
54
|
+
}
|
|
55
|
+
};
|
|
56
|
+
const normalizeHasManyRelationships = (doc, fields)=>{
|
|
57
|
+
const relationshipFields = fields.filter((field)=>(field.type === 'relationship' || field.type === 'upload') && field.hasMany === true && !Array.isArray(field.relationTo));
|
|
58
|
+
for (const field of relationshipFields){
|
|
59
|
+
const value = doc[field.name];
|
|
60
|
+
if (typeof value === 'string' && value.includes(',')) {
|
|
61
|
+
doc[field.name] = value.split(',').map((v)=>v.trim()).filter((v)=>v !== '');
|
|
62
|
+
} else if (Array.isArray(value)) {
|
|
63
|
+
doc[field.name] = value.filter((v)=>v !== null && v !== undefined && v !== '');
|
|
64
|
+
} else if (value !== null && value !== undefined && value !== '') {
|
|
65
|
+
doc[field.name] = [
|
|
66
|
+
value
|
|
67
|
+
];
|
|
68
|
+
}
|
|
69
|
+
}
|
|
70
|
+
};
|
|
71
|
+
const normalizePolymorphicRelationships = (doc, fields)=>{
|
|
72
|
+
for (const [key, value] of Object.entries(doc)){
|
|
73
|
+
if (Array.isArray(value)) {
|
|
74
|
+
const hasPolymorphicItems = value.some((item)=>isPlainObject(item) && 'relationTo' in item);
|
|
75
|
+
if (!hasPolymorphicItems) {
|
|
76
|
+
continue;
|
|
77
|
+
}
|
|
78
|
+
const processedArray = [];
|
|
79
|
+
for (const item of value){
|
|
80
|
+
if (isPlainObject(item) && 'relationTo' in item) {
|
|
81
|
+
if (!item.relationTo || !item.id && !item.value) {
|
|
82
|
+
continue;
|
|
83
|
+
}
|
|
84
|
+
if ('id' in item) {
|
|
85
|
+
item.value = item.id;
|
|
86
|
+
delete item.id;
|
|
87
|
+
}
|
|
88
|
+
processedArray.push(item);
|
|
89
|
+
} else if (item !== null && item !== undefined) {
|
|
90
|
+
processedArray.push(item);
|
|
91
|
+
}
|
|
92
|
+
}
|
|
93
|
+
if (value.length !== processedArray.length) {
|
|
94
|
+
doc[key] = processedArray.length > 0 ? processedArray : [];
|
|
95
|
+
}
|
|
96
|
+
} else if (isPlainObject(value)) {
|
|
97
|
+
if ('relationTo' in value && ('id' in value || 'value' in value)) {
|
|
98
|
+
if (!value.relationTo || !value.id && !value.value) {
|
|
99
|
+
doc[key] = null;
|
|
100
|
+
} else if ('id' in value && !('value' in value)) {
|
|
101
|
+
value.value = value.id;
|
|
102
|
+
delete value.id;
|
|
103
|
+
}
|
|
104
|
+
} else {
|
|
105
|
+
normalizePolymorphicRelationships(value, fields);
|
|
106
|
+
}
|
|
107
|
+
}
|
|
108
|
+
}
|
|
109
|
+
};
|
|
110
|
+
const normalizeRichText = (doc, fields)=>{
|
|
111
|
+
for (const field of fields.filter((f)=>f.type === 'richText')){
|
|
112
|
+
if (field.name in doc && doc[field.name]) {
|
|
113
|
+
doc[field.name] = processRichTextField(doc[field.name]);
|
|
114
|
+
}
|
|
115
|
+
}
|
|
116
|
+
for (const field of fields.filter((f)=>f.type === 'blocks')){
|
|
117
|
+
const blocks = doc[field.name];
|
|
118
|
+
if (!Array.isArray(blocks)) {
|
|
119
|
+
continue;
|
|
120
|
+
}
|
|
121
|
+
for (const block of blocks){
|
|
122
|
+
if (!isPlainObject(block)) {
|
|
123
|
+
continue;
|
|
124
|
+
}
|
|
125
|
+
for (const [key, value] of Object.entries(block)){
|
|
126
|
+
if (key === 'richText' || typeof key === 'string' && key.includes('richText')) {
|
|
127
|
+
block[key] = processRichTextField(value);
|
|
128
|
+
}
|
|
129
|
+
}
|
|
130
|
+
}
|
|
131
|
+
}
|
|
132
|
+
};
|
|
133
|
+
/**
|
|
134
|
+
* Post-processes the unflattened document to handle special field types:
|
|
135
|
+
* - Localized fields: transforms field_locale keys to nested { field: { locale: value } }
|
|
136
|
+
* - Number hasMany: converts comma-separated strings or arrays to number arrays
|
|
137
|
+
* - Relationship hasMany: converts comma-separated IDs to arrays
|
|
138
|
+
* - Polymorphic relationships: transforms flat {relationTo, id} to {relationTo, value}
|
|
139
|
+
* - Rich text fields: ensures proper data structure
|
|
140
|
+
*/ export const postProcessDocument = (doc, fields)=>{
|
|
141
|
+
normalizeLocalizedFields(doc, fields);
|
|
142
|
+
normalizeHasManyNumbers(doc, fields);
|
|
143
|
+
normalizeHasManyRelationships(doc, fields);
|
|
144
|
+
normalizePolymorphicRelationships(doc, fields);
|
|
145
|
+
normalizeRichText(doc, fields);
|
|
146
|
+
};
|
|
147
|
+
|
|
148
|
+
//# sourceMappingURL=unflattenPostProcess.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"sources":["../../src/utilities/unflattenPostProcess.ts"],"sourcesContent":["import type { FlattenedField } from 'payload'\n\nimport { isPlainObject } from './isPlainObject.js'\nimport { processRichTextField } from './processRichTextField.js'\n\nconst normalizeLocalizedFields = (doc: Record<string, unknown>, fields: FlattenedField[]): void => {\n const processed = new Set<string>()\n\n for (const field of fields.filter((f) => f.localized)) {\n if (processed.has(field.name)) {\n continue\n }\n\n const localePattern = new RegExp(`^${field.name}_([a-z]{2}(?:_[A-Z]{2})?)$`)\n const localeData: Record<string, unknown> = {}\n const keysToDelete: string[] = []\n\n for (const [key, value] of Object.entries(doc)) {\n const match = key.match(localePattern)\n if (match && match[1]) {\n localeData[match[1]] = value\n keysToDelete.push(key)\n }\n }\n\n if (Object.keys(localeData).length > 0) {\n doc[field.name] = localeData\n keysToDelete.forEach((key) => delete doc[key])\n processed.add(field.name)\n }\n }\n}\n\nconst normalizeHasManyNumbers = (doc: Record<string, unknown>, fields: FlattenedField[]): void => {\n const numberFields = fields.filter((field) => field.type === 'number' && field.hasMany)\n for (const field of numberFields) {\n if (!(field.name in doc)) {\n continue\n }\n const value = doc[field.name]\n\n if (typeof value === 'string' && value.includes(',')) {\n doc[field.name] = value\n .split(',')\n .map((v) => v.trim())\n .filter((v) => v !== '')\n .map((v) => {\n const num = parseFloat(v)\n return isNaN(num) ? 0 : num\n })\n } else if (Array.isArray(value)) {\n doc[field.name] = value\n .filter((v) => v !== null && v !== undefined && v !== '')\n .map((v) => {\n if (typeof v === 'string') {\n const num = parseFloat(v)\n return isNaN(num) ? 0 : num\n }\n return v\n })\n } else if (value !== null && value !== undefined && value !== '') {\n const num = typeof value === 'string' ? parseFloat(value) : value\n doc[field.name] = isNaN(num as number) ? [] : [num]\n } else {\n doc[field.name] = []\n }\n }\n}\n\nconst normalizeHasManyRelationships = (\n doc: Record<string, unknown>,\n fields: FlattenedField[],\n): void => {\n const relationshipFields = fields.filter(\n (field) =>\n (field.type === 'relationship' || field.type === 'upload') &&\n field.hasMany === true &&\n !Array.isArray(field.relationTo),\n )\n\n for (const field of relationshipFields) {\n const value = doc[field.name]\n\n if (typeof value === 'string' && value.includes(',')) {\n doc[field.name] = value\n .split(',')\n .map((v) => v.trim())\n .filter((v) => v !== '')\n } else if (Array.isArray(value)) {\n doc[field.name] = value.filter((v) => v !== null && v !== undefined && v !== '')\n } else if (value !== null && value !== undefined && value !== '') {\n doc[field.name] = [value]\n }\n }\n}\n\nconst normalizePolymorphicRelationships = (\n doc: Record<string, unknown>,\n fields: FlattenedField[],\n): void => {\n for (const [key, value] of Object.entries(doc)) {\n if (Array.isArray(value)) {\n const hasPolymorphicItems = value.some((item) => isPlainObject(item) && 'relationTo' in item)\n\n if (!hasPolymorphicItems) {\n continue\n }\n\n const processedArray: unknown[] = []\n for (const item of value) {\n if (isPlainObject(item) && 'relationTo' in item) {\n if (!item.relationTo || (!item.id && !item.value)) {\n continue\n }\n if ('id' in item) {\n item.value = item.id\n delete item.id\n }\n processedArray.push(item)\n } else if (item !== null && item !== undefined) {\n processedArray.push(item)\n }\n }\n\n if (value.length !== processedArray.length) {\n doc[key] = processedArray.length > 0 ? processedArray : []\n }\n } else if (isPlainObject(value)) {\n if ('relationTo' in value && ('id' in value || 'value' in value)) {\n if (!value.relationTo || (!value.id && !value.value)) {\n doc[key] = null\n } else if ('id' in value && !('value' in value)) {\n value.value = value.id\n delete value.id\n }\n } else {\n normalizePolymorphicRelationships(value, fields)\n }\n }\n }\n}\n\nconst normalizeRichText = (doc: Record<string, unknown>, fields: FlattenedField[]): void => {\n for (const field of fields.filter((f) => f.type === 'richText')) {\n if (field.name in doc && doc[field.name]) {\n doc[field.name] = processRichTextField(doc[field.name])\n }\n }\n\n for (const field of fields.filter((f) => f.type === 'blocks')) {\n const blocks = doc[field.name]\n if (!Array.isArray(blocks)) {\n continue\n }\n for (const block of blocks) {\n if (!isPlainObject(block)) {\n continue\n }\n for (const [key, value] of Object.entries(block)) {\n if (key === 'richText' || (typeof key === 'string' && key.includes('richText'))) {\n block[key] = processRichTextField(value)\n }\n }\n }\n }\n}\n\n/**\n * Post-processes the unflattened document to handle special field types:\n * - Localized fields: transforms field_locale keys to nested { field: { locale: value } }\n * - Number hasMany: converts comma-separated strings or arrays to number arrays\n * - Relationship hasMany: converts comma-separated IDs to arrays\n * - Polymorphic relationships: transforms flat {relationTo, id} to {relationTo, value}\n * - Rich text fields: ensures proper data structure\n */\nexport const postProcessDocument = (\n doc: Record<string, unknown>,\n fields: FlattenedField[],\n): void => {\n normalizeLocalizedFields(doc, fields)\n normalizeHasManyNumbers(doc, fields)\n normalizeHasManyRelationships(doc, fields)\n normalizePolymorphicRelationships(doc, fields)\n normalizeRichText(doc, fields)\n}\n"],"names":["isPlainObject","processRichTextField","normalizeLocalizedFields","doc","fields","processed","Set","field","filter","f","localized","has","name","localePattern","RegExp","localeData","keysToDelete","key","value","Object","entries","match","push","keys","length","forEach","add","normalizeHasManyNumbers","numberFields","type","hasMany","includes","split","map","v","trim","num","parseFloat","isNaN","Array","isArray","undefined","normalizeHasManyRelationships","relationshipFields","relationTo","normalizePolymorphicRelationships","hasPolymorphicItems","some","item","processedArray","id","normalizeRichText","blocks","block","postProcessDocument"],"mappings":"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"}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "@payloadcms/plugin-import-export",
|
|
3
|
-
"version": "3.
|
|
3
|
+
"version": "3.85.0",
|
|
4
4
|
"description": "Import-Export plugin for Payload",
|
|
5
5
|
"keywords": [
|
|
6
6
|
"payload",
|
|
@@ -64,17 +64,17 @@
|
|
|
64
64
|
"csv-parse": "5.6.0",
|
|
65
65
|
"csv-stringify": "6.5.2",
|
|
66
66
|
"qs-esm": "8.0.1",
|
|
67
|
-
"@payloadcms/translations": "3.
|
|
68
|
-
"@payloadcms/ui": "3.
|
|
67
|
+
"@payloadcms/translations": "3.85.0",
|
|
68
|
+
"@payloadcms/ui": "3.85.0"
|
|
69
69
|
},
|
|
70
70
|
"devDependencies": {
|
|
71
71
|
"@payloadcms/eslint-config": "3.28.0",
|
|
72
|
-
"@payloadcms/ui": "3.
|
|
73
|
-
"payload": "3.
|
|
72
|
+
"@payloadcms/ui": "3.85.0",
|
|
73
|
+
"payload": "3.85.0"
|
|
74
74
|
},
|
|
75
75
|
"peerDependencies": {
|
|
76
|
-
"@payloadcms/ui": "3.
|
|
77
|
-
"payload": "3.
|
|
76
|
+
"@payloadcms/ui": "3.85.0",
|
|
77
|
+
"payload": "3.85.0"
|
|
78
78
|
},
|
|
79
79
|
"homepage:": "https://payloadcms.com",
|
|
80
80
|
"scripts": {
|