diff --git a/lib/core/logs/loggers.ts b/lib/core/logs/loggers.ts index 0f39f5e..771a45f 100644 --- a/lib/core/logs/loggers.ts +++ b/lib/core/logs/loggers.ts @@ -1,7 +1,7 @@ import chalk, { ChalkInstance } from 'chalk'; +import { match, P } from 'ts-pattern'; import { Logger, LogSpinnerMessage } from '../models/log.models.js'; import { getCurrentEnvironment } from '../utils/global.utils.js'; -import { match, P } from 'ts-pattern'; const originalWarn = console.warn; @@ -55,6 +55,10 @@ const defaultBrowserLogger: Logger = { }; function getLogDataMessage(data: LogSpinnerMessage): string { + if (!data.type) { + return data.message; + } + const color = match(data.type) .returnType() .with('info', () => chalk.cyan) diff --git a/lib/core/models/core.models.ts b/lib/core/models/core.models.ts index 4293a1a..ffc9f59 100644 --- a/lib/core/models/core.models.ts +++ b/lib/core/models/core.models.ts @@ -145,3 +145,19 @@ export interface OriginalManagementError { }; }; } + +export type ItemProcessingResult = + | { + readonly state: 'valid'; + readonly inputItem: InputItem; + readonly outputItem: OutputItem; + } + | { + readonly state: 'error'; + readonly inputItem: InputItem; + readonly error: unknown; + } + | { + readonly state: '404'; + readonly inputItem: InputItem; + }; diff --git a/lib/core/models/log.models.ts b/lib/core/models/log.models.ts index fdc7c31..3d79840 100644 --- a/lib/core/models/log.models.ts +++ b/lib/core/models/log.models.ts @@ -1,4 +1,4 @@ -import { MapiAction, MapiType, MigrationItemType } from '../index.js'; +import { LiteralUnion, MapiAction, MapiType, MigrationItemType } from '../index.js'; export type DebugType = | 'error' @@ -17,7 +17,7 @@ export type DebugType = | MapiAction; export interface LogMessage { - readonly type: DebugType; + readonly type?: LiteralUnion; readonly message: string; } diff --git a/lib/core/utils/http.utils.ts b/lib/core/utils/http.utils.ts index b7ce5b7..37f5369 100644 --- a/lib/core/utils/http.utils.ts +++ b/lib/core/utils/http.utils.ts @@ -1,9 +1,8 @@ import { HttpService, IRetryStrategyOptions } from '@kontent-ai/core-sdk'; -import { OriginalManagementError } from '../models/core.models.js'; import { match } from 'ts-pattern'; +import { OriginalManagementError } from '../models/core.models.js'; const rateExceededErrorCode: number = 10000; -const notFoundErrorCode: number = 10000; export const defaultHttpService: Readonly = new HttpService({ logErrorsToConsole: false @@ -19,8 +18,6 @@ export const defaultRetryStrategy: Readonly = { match(errorCode) // retry rate exceeded error .with(rateExceededErrorCode, () => true) - // do not retry errors indicating resource does not exist - .with(notFoundErrorCode, () => false) // if error code is set, do not retry the request .when( (errorCode) => errorCode >= 0, diff --git a/lib/core/utils/processing-utils.ts b/lib/core/utils/processing-utils.ts index 85f1dd3..a6fda3a 100644 --- a/lib/core/utils/processing-utils.ts +++ b/lib/core/utils/processing-utils.ts @@ -1,6 +1,6 @@ import chalk from 'chalk'; import pLimit from 'p-limit'; -import { ItemInfo } from '../models/core.models.js'; +import { ItemInfo, ItemProcessingResult } from '../models/core.models.js'; import { LogSpinnerData, Logger } from '../models/log.models.js'; type ProcessSetAction = @@ -19,9 +19,9 @@ export async function processItemsAsync(data: { readonly logger: Logger; readonly items: Readonly; readonly parallelLimit: number; - readonly processAsync: (item: Readonly, logSpinner: LogSpinnerData) => Promise>; + readonly processAsync: (item: Readonly, logSpinner: LogSpinnerData) => Promise | '404'>; readonly itemInfo: (item: Readonly) => ItemInfo; -}): Promise { +}): Promise[]> { if (!data.items.length) { return []; } @@ -32,21 +32,43 @@ export async function processItemsAsync(data: { const limit = pLimit(data.parallelLimit); let processedItemsCount: number = 1; - const requests: Promise[] = data.items.map((item) => + const requests: Promise>[] = data.items.map((item) => limit(() => { - return data.processAsync(item, logSpinner).then((output) => { - const itemInfo = data.itemInfo(item); - const prefix = getPercentagePrefix(processedItemsCount, data.items.length); + return data + .processAsync(item, logSpinner) + .then((output) => { + const itemInfo = data.itemInfo(item); + const prefix = getPercentagePrefix(processedItemsCount, data.items.length); - logSpinner({ - prefix: prefix, - message: itemInfo.title, - type: itemInfo.itemType - }); + logSpinner({ + prefix: prefix, + message: itemInfo.title, + type: itemInfo.itemType + }); - processedItemsCount++; - return output; - }); + processedItemsCount++; + return output; + }) + .then>((outputItem) => { + if (outputItem === '404') { + return { + state: '404', + inputItem: item + }; + } + return { + inputItem: item, + outputItem: outputItem, + state: 'valid' + }; + }) + .catch>((error) => { + return { + state: 'error', + inputItem: item, + error: error + }; + }); }) ); @@ -58,11 +80,19 @@ export async function processItemsAsync(data: { }); // Only '' promises at a time - const outputItems = await Promise.all(requests); + const resultItems = await Promise.all(requests); + + const failedItemsCount = resultItems.filter((m) => m.state === 'error').length; + const failedText = failedItemsCount ? ` Failed (${chalk.red(failedItemsCount)}) items` : ``; - logSpinner({ type: 'info', message: `Completed '${chalk.yellow(data.action)}' (${outputItems.length})` }); + logSpinner({ + type: 'info', + message: `Completed '${chalk.yellow(data.action)}'. Successfully processed (${chalk.green( + resultItems.filter((m) => m.state === 'valid').length + )}) items.${failedText}` + }); - return outputItems; + return resultItems; }); } diff --git a/lib/export/context/export-context-fetcher.ts b/lib/export/context/export-context-fetcher.ts index f702123..f1ccc56 100644 --- a/lib/export/context/export-context-fetcher.ts +++ b/lib/export/context/export-context-fetcher.ts @@ -1,23 +1,22 @@ import { - WorkflowModels, - LanguageVariantModels, - ContentItemModels, AssetModels, CollectionModels, - LanguageModels + ContentItemModels, + LanguageModels, + LanguageVariantModels, + WorkflowModels } from '@kontent-ai/management-sdk'; import chalk from 'chalk'; import { - processItemsAsync, - runMapiRequestAsync, - is404Error, - ItemStateInSourceEnvironmentById, AssetStateInSourceEnvironmentById, + findRequired, FlattenedContentType, - isNotUndefined, - managementClientUtils, + is404Error, + ItemStateInSourceEnvironmentById, LogSpinnerData, - findRequired, + managementClientUtils, + processItemsAsync, + runMapiRequestAsync, workflowHelper } from '../../core/index.js'; import { itemsExtractionProcessor } from '../../translation/index.js'; @@ -25,10 +24,10 @@ import { DefaultExportContextConfig, ExportContext, ExportContextEnvironmentData, - SourceExportItem, ExportItem, + ExportItemVersion, GetFlattenedElementByIds, - ExportItemVersion + SourceExportItem } from '../export.models.js'; import { throwErrorForItemRequest } from '../utils/export.utils.js'; @@ -173,11 +172,11 @@ export async function exportContextFetcherAsync(config: DefaultExportContextConf readonly contentItem: Readonly; readonly languageVariant: Readonly; }): { - collection: Readonly; - language: Readonly; - workflow: Readonly; - contentType: Readonly; - workflowStepCodename: string; + readonly collection: Readonly; + readonly language: Readonly; + readonly workflow: Readonly; + readonly contentType: Readonly; + readonly workflowStepCodename: string; } => { const collection = findRequired( environmentData.collections, @@ -234,49 +233,53 @@ export async function exportContextFetcherAsync(config: DefaultExportContextConf message: `Preparing '${chalk.yellow(config.exportItems.length.toString())}' items for export` }); - return await processItemsAsync({ - logger: config.logger, - action: 'Preparing content items & language variants', - parallelLimit: 1, - itemInfo: (input) => { - return { - title: `${input.itemCodename} (${input.languageCodename})`, - itemType: 'exportItem' - }; - }, - items: exportItems, - processAsync: async (requestItem, logSpinner) => { - const contentItem = await getContentItemAsync(requestItem, logSpinner); - const versions = await getExportItemVersionsAsync(requestItem, contentItem, logSpinner); - - // get shared attributes from any version - const anyVersion = versions[0]; - if (!anyVersion) { - throwErrorForItemRequest(requestItem, `Expected at least 1 version of the content item`); - } + return ( + await processItemsAsync({ + logger: config.logger, + action: 'Preparing content items & language variants', + parallelLimit: 1, + itemInfo: (input) => { + return { + title: `${input.itemCodename} (${input.languageCodename})`, + itemType: 'exportItem' + }; + }, + items: exportItems, + processAsync: async (requestItem, logSpinner) => { + const contentItem = await getContentItemAsync(requestItem, logSpinner); + const versions = await getExportItemVersionsAsync(requestItem, contentItem, logSpinner); + + // get shared attributes from any version + const anyVersion = versions[0]; + if (!anyVersion) { + throwErrorForItemRequest(requestItem, `Expected at least 1 version of the content item`); + } - const { collection, contentType, language, workflow } = validateExportItem({ - sourceItem: requestItem, - contentItem: contentItem, - languageVariant: anyVersion.languageVariant - }); + const { collection, contentType, language, workflow } = validateExportItem({ + sourceItem: requestItem, + contentItem: contentItem, + languageVariant: anyVersion.languageVariant + }); - return { - contentItem, - versions, - contentType, - requestItem, - workflow, - collection, - language - }; - } - }); + return { + contentItem, + versions, + contentType, + requestItem, + workflow, + collection, + language + }; + } + }) + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; const getContentItemsByIdsAsync = async (itemIds: ReadonlySet): Promise[]> => { return ( - await processItemsAsync | undefined>({ + await processItemsAsync>({ logger: config.logger, action: 'Fetching content items', parallelLimit: 1, @@ -302,16 +305,18 @@ export async function exportContextFetcherAsync(config: DefaultExportContextConf throw error; } - return undefined; + return '404'; } } }) - ).filter(isNotUndefined); + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; const getAssetsByIdsAsync = async (itemIds: ReadonlySet): Promise[]> => { return ( - await processItemsAsync | undefined>({ + await processItemsAsync>({ logger: config.logger, action: 'Fetching assets', parallelLimit: 1, @@ -337,11 +342,13 @@ export async function exportContextFetcherAsync(config: DefaultExportContextConf throw error; } - return undefined; + return '404'; } } }) - ).filter(isNotUndefined); + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; const getItemStatesAsync = async (itemIds: ReadonlySet): Promise => { diff --git a/lib/export/export-manager.ts b/lib/export/export-manager.ts index 04d8c12..9a117c2 100644 --- a/lib/export/export-manager.ts +++ b/lib/export/export-manager.ts @@ -124,10 +124,10 @@ export function exportManager(config: ExportConfig) { }; const getMigrationElementToStore = (data: { - context: ExportContext; - contentType: FlattenedContentType; - typeElement: FlattenedContentTypeElement; - exportElement: ElementModels.ContentItemElement; + readonly context: ExportContext; + readonly contentType: FlattenedContentType; + readonly typeElement: FlattenedContentTypeElement; + readonly exportElement: ElementModels.ContentItemElement; }): MigrationElementTransformData => { try { return exportTransforms[data.typeElement.type]({ @@ -175,58 +175,62 @@ export function exportManager(config: ExportConfig) { message: `Preparing to download '${chalk.yellow(assets.length.toString())}' assets` }); - return await processItemsAsync, MigrationAsset>({ - action: 'Downloading assets', - logger: logger, - parallelLimit: 5, - itemInfo: (input) => { - return { - title: input.codename, - itemType: 'asset' - }; - }, - items: assets, - processAsync: async (asset, logSpinner) => { - const assetCollection: Readonly | undefined = context.environmentData.collections.find( - (m) => m.id === asset.collection?.reference?.id - ); - const assetFolder: Readonly | undefined = context.environmentData.assetFolders.find( - (m) => m.id === asset.folder?.id - ); + return ( + await processItemsAsync, MigrationAsset>({ + action: 'Downloading assets', + logger: logger, + parallelLimit: 5, + itemInfo: (input) => { + return { + title: input.codename, + itemType: 'asset' + }; + }, + items: assets, + processAsync: async (asset, logSpinner) => { + const assetCollection: Readonly | undefined = context.environmentData.collections.find( + (m) => m.id === asset.collection?.reference?.id + ); + const assetFolder: Readonly | undefined = context.environmentData.assetFolders.find( + (m) => m.id === asset.folder?.id + ); - logSpinner({ - type: 'download', - message: `${asset.url}` - }); - - const migrationAsset: MigrationAsset = { - filename: asset.fileName, - title: asset.title ?? '', - codename: asset.codename, - binary_data: (await getBinaryDataFromUrlAsync(asset.url)).data, - collection: assetCollection ? { codename: assetCollection.codename } : undefined, - folder: assetFolder ? { codename: assetFolder.codename } : undefined, - descriptions: asset.descriptions.map((description) => { - const language = findRequired( - context.environmentData.languages, - (language) => language.id === description.language.id, - `Could not find language with id '${chalk.red(description.language.id)}' requested by asset '${chalk.red( - asset.codename - )}'` - ); - - return { - description: description.description ?? undefined, - language: { - codename: language.codename - } - }; - }) - }; + logSpinner({ + type: 'download', + message: `${asset.url}` + }); - return migrationAsset; - } - }); + const migrationAsset: MigrationAsset = { + filename: asset.fileName, + title: asset.title ?? '', + codename: asset.codename, + binary_data: (await getBinaryDataFromUrlAsync(asset.url)).data, + collection: assetCollection ? { codename: assetCollection.codename } : undefined, + folder: assetFolder ? { codename: assetFolder.codename } : undefined, + descriptions: asset.descriptions.map((description) => { + const language = findRequired( + context.environmentData.languages, + (language) => language.id === description.language.id, + `Could not find language with id '${chalk.red(description.language.id)}' requested by asset '${chalk.red( + asset.codename + )}'` + ); + + return { + description: description.description ?? undefined, + language: { + codename: language.codename + } + }; + }) + }; + + return migrationAsset; + } + }) + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; return { diff --git a/lib/import/context/import-context-fetcher.ts b/lib/import/context/import-context-fetcher.ts index b91bf4a..dc98c01 100644 --- a/lib/import/context/import-context-fetcher.ts +++ b/lib/import/context/import-context-fetcher.ts @@ -1,4 +1,6 @@ import { AssetModels, ContentItemModels, LanguageVariantModels } from '@kontent-ai/management-sdk'; +import chalk from 'chalk'; +import { match } from 'ts-pattern'; import { AssetStateInTargetEnvironmentByCodename, ItemStateInTargetEnvironmentByCodename, @@ -12,16 +14,13 @@ import { WorkflowStep, findRequired, is404Error, - isNotUndefined, managementClientUtils, processItemsAsync, runMapiRequestAsync, workflowHelper as workflowHelperInit } from '../../core/index.js'; -import { GetFlattenedElementByCodenames, ImportContext, ImportContextConfig, ImportContextEnvironmentData } from '../import.models.js'; import { ExtractItemByCodename, itemsExtractionProcessor } from '../../translation/index.js'; -import chalk from 'chalk'; -import { match } from 'ts-pattern'; +import { GetFlattenedElementByCodenames, ImportContext, ImportContextConfig, ImportContextEnvironmentData } from '../import.models.js'; interface LanguageVariantWrapper { readonly draftLanguageVariant: Readonly | undefined; @@ -150,7 +149,7 @@ export async function importContextFetcherAsync(config: ImportContextConfig) { migrationItems: readonly MigrationItem[] ): Promise => { return ( - await processItemsAsync({ + await processItemsAsync({ action: 'Fetching language variants', logger: config.logger, parallelLimit: 1, @@ -166,7 +165,7 @@ export async function importContextFetcherAsync(config: ImportContextConfig) { if (!latestLanguageVariant) { // there is neither published or draft version as latest version does not exist at all - return undefined; + return '404'; } if (workflowHelper.isPublishedStepById(latestLanguageVariant.workflow.stepIdentifier.id ?? '')) { @@ -186,14 +185,16 @@ export async function importContextFetcherAsync(config: ImportContextConfig) { }; } }) - ).filter(isNotUndefined); + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; const getContentItemsByCodenamesAsync = async ( itemCodenames: ReadonlySet ): Promise => { return ( - await processItemsAsync | undefined>({ + await processItemsAsync>({ action: 'Fetching content items', logger: config.logger, parallelLimit: 1, @@ -219,16 +220,18 @@ export async function importContextFetcherAsync(config: ImportContextConfig) { throw error; } - return undefined; + return '404'; } } }) - ).filter(isNotUndefined); + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; const getAssetsByCodenamesAsync = async (assetCodenames: ReadonlySet): Promise => { return ( - await processItemsAsync | undefined>({ + await processItemsAsync>({ action: 'Fetching assets', logger: config.logger, parallelLimit: 1, @@ -254,11 +257,13 @@ export async function importContextFetcherAsync(config: ImportContextConfig) { throw error; } - return undefined; + return '404'; } } }) - ).filter(isNotUndefined); + ) + .filter((m) => m.state === 'valid') + .map((m) => m.outputItem); }; const getVariantState = (languageVariant: Readonly): LanguageVariantStateData => { diff --git a/lib/import/import-manager.ts b/lib/import/import-manager.ts index fa89151..a449b86 100644 --- a/lib/import/import-manager.ts +++ b/lib/import/import-manager.ts @@ -1,12 +1,42 @@ -import { ContentItemModels, LanguageVariantModels, ManagementClient } from '@kontent-ai/management-sdk'; - -import { defaultExternalIdGenerator, getDefaultLogger, getMigrationManagementClient, Logger } from '../core/index.js'; +import { ManagementClient } from '@kontent-ai/management-sdk'; +import chalk from 'chalk'; +import { defaultExternalIdGenerator, extractErrorData, getDefaultLogger, getMigrationManagementClient, Logger } from '../core/index.js'; import { importContextFetcherAsync } from './context/import-context-fetcher.js'; -import { ImportConfig, ImportContext, ImportResult } from './import.models.js'; +import { ImportConfig, ImportContext, ImportedItem, ImportedLanguageVariant, ImportResult } from './import.models.js'; import { assetsImporter } from './importers/assets-importer.js'; import { contentItemsImporter } from './importers/content-items-importer.js'; import { languageVariantImporter } from './importers/language-variant-importer.js'; +const reportFilename: string = `import-report.json`; + +type ReportResult = { + readonly errorsCount: number; + readonly assets: { + readonly count: number; + readonly successful: Array<{ readonly codename: string }>; + readonly failed: Array<{ readonly codename: string; readonly error: string }>; + }; + readonly languageVariants: { + readonly count: number; + readonly successful: Array<{ + readonly codename: string; + readonly language: { readonly codename: string }; + readonly type: { readonly codename: string }; + }>; + readonly failed: Array<{ + readonly codename: string; + readonly language: { readonly codename: string }; + readonly type: { readonly codename: string }; + readonly error: string; + }>; + }; + readonly contentItems: { + readonly count: number; + readonly successful: Array<{ readonly codename: string }>; + readonly failed: Array<{ readonly codename: string; readonly type: { readonly codename: string }; readonly error: string }>; + }; +}; + export function importManager(config: ImportConfig) { const logger: Logger = config.logger ?? getDefaultLogger(); const targetEnvironmentClient: Readonly = getMigrationManagementClient(config); @@ -29,7 +59,8 @@ export function importManager(config: ImportConfig) { logger: logger }).importAsync(); }; - const importContentItemsAsync = async (importContext: ImportContext): Promise[]> => { + + const importContentItemsAsync = async (importContext: ImportContext): Promise => { if (!importContext.categorizedImportData.contentItems.length) { logger.log({ type: 'info', @@ -46,8 +77,8 @@ export function importManager(config: ImportConfig) { const importLanguageVariantsAsync = async ( importContext: ImportContext, - contentItems: readonly Readonly[] - ): Promise[]> => { + contentItems: readonly ImportedItem[] + ): Promise => { if (!importContext.categorizedImportData.contentItems.length) { logger.log({ type: 'info', @@ -63,6 +94,116 @@ export function importManager(config: ImportConfig) { }).importAsync(); }; + const getReportResult = (importResult: ImportResult): ReportResult => { + return { + errorsCount: + importResult.editedAssets.filter((m) => m.state === 'error').length + + importResult.uploadedAssets.filter((m) => m.state === 'error').length + + importResult.contentItems.filter((m) => m.state === 'error').length + + importResult.languageVariants.filter((m) => m.state === 'error').length, + assets: { + count: importResult.uploadedAssets.length + importResult.editedAssets.length, + successful: [ + ...importResult.uploadedAssets.filter((m) => m.state === 'valid').map((m) => m.inputItem.codename), + ...importResult.editedAssets.filter((m) => m.state === 'valid').map((m) => m.inputItem.migrationAsset.codename) + ].map((m) => { + return { + codename: m + }; + }), + failed: [ + ...importResult.uploadedAssets + .filter((m) => m.state === 'error') + .map((m) => { + return { + codename: m.inputItem.codename, + error: extractErrorData(m.state === 'error').message + }; + }), + ...importResult.editedAssets + .filter((m) => m.state === 'error') + .map((m) => { + return { + codename: m.inputItem.migrationAsset.codename, + error: extractErrorData(m.state === 'error').message + }; + }) + ] + }, + contentItems: { + count: importResult.contentItems.length, + successful: importResult.contentItems + .filter((m) => m.state === 'valid') + .map((m) => { + return { + codename: m.inputItem.system.codename + }; + }), + failed: importResult.contentItems + .filter((m) => m.state === 'error') + .map((m) => { + return { + codename: m.inputItem.system.codename, + type: m.inputItem.system.type, + error: extractErrorData(m.state === 'error').message + }; + }) + }, + languageVariants: { + count: importResult.languageVariants.length, + successful: importResult.languageVariants + .filter((m) => m.state === 'valid') + .map((m) => { + return { + codename: m.inputItem.system.codename, + language: m.inputItem.system.language, + type: m.inputItem.system.type + }; + }), + failed: importResult.languageVariants + .filter((m) => m.state === 'error') + .map((m) => { + return { + codename: m.inputItem.system.codename, + language: m.inputItem.system.language, + type: m.inputItem.system.type, + error: extractErrorData(m.state === 'error').message + }; + }) + } + }; + }; + + const printReportToConsole = (reportResult: ReportResult, logger: Logger): void => { + const errors = [ + ...reportResult.assets.failed.map((m) => [ + `Object type: ${chalk.yellow('Asset')}`, + `Codename: ${chalk.yellow(m.codename)}`, + `${chalk.red(m.error)}` + ]), + ...reportResult.contentItems.failed.map((m) => [ + `Object type: ${chalk.yellow('Content item')}`, + `Codename:${chalk.yellow(m.codename)}`, + `Content Type: ${chalk.yellow(m.type.codename)}`, + `${chalk.red(m.error)}` + ]), + ...reportResult.languageVariants.failed.map((m) => [ + `Object type: ${chalk.yellow('Language variant')}`, + `Codename: ${chalk.yellow(m.codename)}`, + `Language: ${chalk.yellow(m.language.codename)}`, + `Content Type: ${chalk.yellow(m.type.codename)}`, + `${chalk.red(m.error)}` + ]) + ]; + + errors.forEach((error, index) => { + logger.log({ message: `${chalk.red(`\nError #${index + 1}`)}` }); + error.forEach((m) => { + logger.log({ message: m }); + }); + }); + }; + return { async importAsync(): Promise { const importContext = await ( @@ -83,17 +224,40 @@ export function importManager(config: ImportConfig) { // #3 Language variants const languageVariants = await importLanguageVariantsAsync(importContext, contentItems); - logger.log({ - type: 'completed', - message: `Finished import` + const reportResult = getReportResult({ + contentItems, + editedAssets, + languageVariants, + uploadedAssets }); + if (reportResult.errorsCount) { + printReportToConsole(reportResult, logger); + logger.log({ + type: 'completed', + message: `Finished import with '${chalk.red(reportResult.errorsCount)}' ${ + reportResult.errorsCount === 1 ? 'error' : 'errors' + }` + }); + } else { + logger.log({ + type: 'completed', + message: `Finished import` + }); + } + return { editedAssets, uploadedAssets, contentItems, languageVariants }; + }, + getReportFile(importResult: ImportResult): { readonly filename: string; readonly content: string } { + return { + filename: reportFilename, + content: JSON.stringify(getReportResult(importResult)) + }; } }; } diff --git a/lib/import/import.models.ts b/lib/import/import.models.ts index f4d7912..7119031 100644 --- a/lib/import/import.models.ts +++ b/lib/import/import.models.ts @@ -14,6 +14,7 @@ import { ExternalIdGenerator, FlattenedContentType, FlattenedContentTypeElement, + ItemProcessingResult, ItemStateInTargetEnvironmentByCodename, LanguageVariantStateInTargetEnvironmentByCodename, Logger, @@ -75,12 +76,27 @@ export type ImportTransformFunc = (data: { export interface ImportConfig extends ManagementClientConfig { readonly data: MigrationData; readonly externalIdGenerator?: ExternalIdGenerator; + readonly createReportFile?: boolean; readonly logger?: Logger; } +export interface AssetToEdit { + readonly migrationAsset: MigrationAsset; + readonly targetAsset: Readonly; + readonly replaceBinaryFile: boolean; +} + +export type ImportedItem = ItemProcessingResult>; +export type ImportedLanguageVariant = ItemProcessingResult< + MigrationItem, + readonly Readonly[] +>; +export type EditedAsset = ItemProcessingResult>; +export type ImportedAsset = ItemProcessingResult>; + export interface ImportResult { - readonly uploadedAssets: readonly Readonly[]; - readonly editedAssets: readonly Readonly[]; - readonly contentItems: readonly Readonly[]; - readonly languageVariants: readonly Readonly[]; + readonly uploadedAssets: readonly ImportedAsset[]; + readonly editedAssets: readonly EditedAsset[]; + readonly contentItems: readonly ImportedItem[]; + readonly languageVariants: readonly ImportedLanguageVariant[]; } diff --git a/lib/import/importers/assets-importer.ts b/lib/import/importers/assets-importer.ts index b6c5ff7..51ed484 100644 --- a/lib/import/importers/assets-importer.ts +++ b/lib/import/importers/assets-importer.ts @@ -13,13 +13,7 @@ import { runMapiRequestAsync } from '../../core/index.js'; import { shouldReplaceBinaryFile, shouldUpdateAsset } from '../comparers/asset-comparer.js'; -import { ImportContext, ImportResult } from '../import.models.js'; - -interface AssetToEdit { - migrationAsset: MigrationAsset; - targetAsset: Readonly; - replaceBinaryFile: boolean; -} +import { AssetToEdit, EditedAsset, ImportContext, ImportedAsset, ImportResult } from '../import.models.js'; export function assetsImporter(data: { readonly logger: Logger; @@ -70,7 +64,7 @@ export function assetsImporter(data: { .filter(isNotUndefined); }; - const editAssets = async (assetsToEdit: readonly AssetToEdit[]): Promise[]> => { + const editAssets = async (assetsToEdit: readonly AssetToEdit[]): Promise => { data.logger.log({ type: 'upsert', message: `Upserting '${chalk.yellow(assetsToEdit.length.toString())}' assets` @@ -189,7 +183,7 @@ export function assetsImporter(data: { }); }; - const uploadAssetsAsync = async (assetsToUpload: readonly MigrationAsset[]): Promise[]> => { + const uploadAssetsAsync = async (assetsToUpload: readonly MigrationAsset[]): Promise => { data.logger.log({ type: 'upload', message: `Uploading '${chalk.yellow(assetsToUpload.length.toString())}' assets` diff --git a/lib/import/importers/content-items-importer.ts b/lib/import/importers/content-items-importer.ts index 345c57d..eca6d52 100644 --- a/lib/import/importers/content-items-importer.ts +++ b/lib/import/importers/content-items-importer.ts @@ -1,14 +1,7 @@ import { ContentItemModels, ManagementClient } from '@kontent-ai/management-sdk'; -import { - Logger, - processItemsAsync, - runMapiRequestAsync, - MigrationItem, - LogSpinnerData, - findRequired -} from '../../core/index.js'; import chalk from 'chalk'; -import { ImportContext } from '../import.models.js'; +import { LogSpinnerData, Logger, MigrationItem, findRequired, processItemsAsync, runMapiRequestAsync } from '../../core/index.js'; +import { ImportContext, ImportedItem } from '../import.models.js'; export function contentItemsImporter(data: { readonly logger: Logger; @@ -26,8 +19,7 @@ export function contentItemsImporter(data: { ); return ( - migrationContentItem.system.name !== contentItem.name || - migrationContentItem.system.collection.codename !== collection.codename + migrationContentItem.system.name !== contentItem.name || migrationContentItem.system.collection.codename !== collection.codename ); }; @@ -35,9 +27,7 @@ export function contentItemsImporter(data: { logSpinner: LogSpinnerData, migrationContentItem: MigrationItem ): Promise<{ contentItem: Readonly; status: 'created' | 'itemAlreadyExists' }> => { - const itemStateInTargetEnv = data.importContext.getItemStateInTargetEnvironment( - migrationContentItem.system.codename - ); + const itemStateInTargetEnv = data.importContext.getItemStateInTargetEnvironment(migrationContentItem.system.codename); if (itemStateInTargetEnv.state === 'exists' && itemStateInTargetEnv.item) { return { @@ -111,8 +101,19 @@ export function contentItemsImporter(data: { return preparedContentItemResult.contentItem; }; - const importAsync = async (): Promise[]> => { - const contentItemsToImport = data.importContext.categorizedImportData.contentItems; + const importAsync = async (): Promise => { + // Only import unique content items based on their codename. The input may contain items in various language variants which share + // the same underlying content item. + const contentItemsToImport = data.importContext.categorizedImportData.contentItems.reduce>( + (filteredItems, item) => { + if (filteredItems.some((m) => m.system.codename === item.system.codename)) { + return filteredItems; + } + + return [...filteredItems, item]; + }, + [] + ); data.logger.log({ type: 'info', @@ -127,7 +128,7 @@ export function contentItemsImporter(data: { itemInfo: (item) => { return { itemType: 'contentItem', - title: `${item.system.codename} -> ${item.system.language.codename}` + title: `${item.system.codename} -> ${item.system.type.codename}` }; }, processAsync: async (item, logSpinner) => { diff --git a/lib/import/importers/language-variant-importer.ts b/lib/import/importers/language-variant-importer.ts index b74293a..656e978 100644 --- a/lib/import/importers/language-variant-importer.ts +++ b/lib/import/importers/language-variant-importer.ts @@ -1,4 +1,4 @@ -import { ContentItemModels, ElementContracts, LanguageVariantModels, ManagementClient, WorkflowModels } from '@kontent-ai/management-sdk'; +import { ElementContracts, LanguageVariantModels, ManagementClient, WorkflowModels } from '@kontent-ai/management-sdk'; import chalk from 'chalk'; import { match } from 'ts-pattern'; import { @@ -16,13 +16,13 @@ import { workflowHelper } from '../../core/index.js'; import { importTransforms } from '../../translation/index.js'; -import { ImportContext } from '../import.models.js'; +import { ImportContext, ImportedItem, ImportedLanguageVariant } from '../import.models.js'; import { throwErrorForMigrationItem } from '../utils/import.utils.js'; import { workflowImporter as workflowImporterInit } from './workflow-importer.js'; export function languageVariantImporter(config: { readonly logger: Logger; - readonly preparedContentItems: readonly ContentItemModels.ContentItem[]; + readonly preparedContentItems: readonly ImportedItem[]; readonly importContext: ImportContext; readonly client: Readonly; }) { @@ -37,7 +37,7 @@ export function languageVariantImporter(config: { readonly logSpinner: LogSpinnerData; readonly migrationItem: MigrationItem; readonly migrationItemVersion: MigrationItemVersion; - readonly preparedContentItem: Readonly; + readonly preparedContentItem: ImportedItem; }): Promise> => { return await runMapiRequestAsync({ logger: config.logger, @@ -45,7 +45,7 @@ export function languageVariantImporter(config: { return ( await config.client .upsertLanguageVariant() - .byItemCodename(data.preparedContentItem.codename) + .byItemCodename(data.preparedContentItem.inputItem.system.codename) .byLanguageCodename(data.migrationItem.system.language.codename) .withData(() => { return { @@ -110,7 +110,7 @@ export function languageVariantImporter(config: { readonly logSpinner: LogSpinnerData; readonly migrationItem: MigrationItem; readonly migrationItemVersion: MigrationItemVersion; - readonly preparedContentItem: Readonly; + readonly preparedContentItem: ImportedItem; readonly createNewVersion?: boolean; }): Promise> => { // validate workflow @@ -158,7 +158,7 @@ export function languageVariantImporter(config: { const importLanguageVariantAsync = async ( logSpinner: LogSpinnerData, migrationItem: MigrationItem, - preparedContentItem: Readonly + preparedContentItem: ImportedItem ): Promise => { const { draftVersion, publishedVersion } = categorizeVersions(migrationItem); @@ -297,7 +297,7 @@ export function languageVariantImporter(config: { return importTransformResult; }; - const importAsync = async (): Promise[]> => { + const importAsync = async (): Promise => { config.logger.log({ type: 'info', message: `Importing '${chalk.yellow( @@ -305,32 +305,30 @@ export function languageVariantImporter(config: { )}' language variants` }); - return ( - await processItemsAsync[]>({ - action: 'Importing language variants', - logger: config.logger, - parallelLimit: 1, - items: config.importContext.categorizedImportData.contentItems, - itemInfo: (input) => { - return { - itemType: 'languageVariant', - title: input.system.name, - partA: input.system.language.codename - }; - }, - processAsync: async (migrationItem, logSpinner) => { - const contentItem = findRequired( - config.preparedContentItems, - (item) => item.codename === migrationItem.system.codename, - `Missing content item with codename '${chalk.red( - migrationItem.system.codename - )}'. Content item should have been prepepared.` - ); - - return await importLanguageVariantAsync(logSpinner, migrationItem, contentItem); - } - }) - ).flatMap((m) => m.map((s) => s)); + return await processItemsAsync[]>({ + action: 'Importing language variants', + logger: config.logger, + parallelLimit: 1, + items: config.importContext.categorizedImportData.contentItems, + itemInfo: (input) => { + return { + itemType: 'languageVariant', + title: input.system.name, + partA: input.system.language.codename + }; + }, + processAsync: async (migrationItem, logSpinner) => { + const contentItem = findRequired( + config.preparedContentItems, + (item) => item.inputItem.system.codename === migrationItem.system.codename, + `Missing content item with codename '${chalk.red( + migrationItem.system.codename + )}'. Content item should have been prepepared.` + ); + + return await importLanguageVariantAsync(logSpinner, migrationItem, contentItem); + } + }); }; return { diff --git a/lib/toolkit/import.ts b/lib/toolkit/import.ts index 35d590d..e428f9b 100644 --- a/lib/toolkit/import.ts +++ b/lib/toolkit/import.ts @@ -1,5 +1,7 @@ +import chalk from 'chalk'; +import { writeFile } from 'fs/promises'; import { executeWithTrackingAsync } from '../core/index.js'; -import { ImportConfig, ImportResult, importManager } from '../import/index.js'; +import { ImportConfig, ImportResult, importManager as _importManager } from '../import/index.js'; import { libMetadata } from '../metadata.js'; export async function importAsync(config: ImportConfig): Promise { @@ -15,7 +17,19 @@ export async function importAsync(config: ImportConfig): Promise { details: {} }, func: async () => { - return await importManager(config).importAsync(); + const importManager = _importManager(config); + const importResult = await importManager.importAsync(); + + if (config.createReportFile) { + const reportFile = importManager.getReportFile(importResult); + await writeFile(reportFile.filename, reportFile.content); + config.logger?.log({ + type: 'writeFs', + message: `Report '${chalk.yellow(reportFile.filename)}' was created` + }); + } + + return importResult; }, logger: config.logger }); diff --git a/lib/translation/extraction/items-extraction.processor.ts b/lib/translation/extraction/items-extraction.processor.ts index b4b2ebb..43ef75e 100644 --- a/lib/translation/extraction/items-extraction.processor.ts +++ b/lib/translation/extraction/items-extraction.processor.ts @@ -12,7 +12,7 @@ import { GetFlattenedElementByCodenames } from '../../import/index.js'; import { richTextProcessor } from '../index.js'; export interface ExtractItemById { - readonly elements: Readonly[]; + readonly elements: readonly Readonly[]; readonly contentTypeId: string; } @@ -95,7 +95,7 @@ export function itemsExtractionProcessor() { { itemIds: new Set(), assetIds: new Set() } ); - return { + return { itemIds: extractedIds.itemIds, assetIds: extractedIds.assetIds }; @@ -175,7 +175,7 @@ export function itemsExtractionProcessor() { } ); - return { + return { itemCodenames: extractedCodenames.itemCodenames, assetCodenames: extractedCodenames.assetCodenames }; diff --git a/scripts/test/import-report.json b/scripts/test/import-report.json new file mode 100644 index 0000000..ee8ecf5 --- /dev/null +++ b/scripts/test/import-report.json @@ -0,0 +1 @@ +{"errorsCount":0,"assets":{"count":0,"successful":[],"failed":[]},"contentItems":{"count":1,"successful":[{"codename":"warrior"}],"failed":[]},"languageVariants":{"count":2,"successful":[{"codename":"warrior","language":{"codename":"en"},"type":{"codename":"movie"}},{"codename":"warrior","language":{"codename":"es"},"type":{"codename":"movie"}}],"failed":[]}} \ No newline at end of file diff --git a/scripts/test/test-export.ts b/scripts/test/test-export.ts index a130162..c35bf39 100644 --- a/scripts/test/test-export.ts +++ b/scripts/test/test-export.ts @@ -14,6 +14,10 @@ const run = async () => { { itemCodename: getEnvironmentRequiredValue('item'), languageCodename: getEnvironmentRequiredValue('language') + }, + { + itemCodename: getEnvironmentRequiredValue('item'), + languageCodename: getEnvironmentRequiredValue('languageSecondary') } ]; diff --git a/scripts/test/test-import.ts b/scripts/test/test-import.ts index e2d632b..fa1365e 100644 --- a/scripts/test/test-import.ts +++ b/scripts/test/test-import.ts @@ -1,5 +1,5 @@ import * as dotenv from 'dotenv'; -import { confirmImportAsync, extractAsync, importAsync, getDefaultLogger, handleError } from '../../lib/index.js'; +import { confirmImportAsync, extractAsync, getDefaultLogger, handleError, importAsync } from '../../lib/index.js'; import { getEnvironmentRequiredValue } from './utils/test.utils.js'; const run = async () => { @@ -26,7 +26,8 @@ const run = async () => { logger: log, data: data, environmentId: environmentId, - apiKey: apiKey + apiKey: apiKey, + createReportFile: true }); };