diff --git a/packages/contentstack-export-to-csv/src/commands/cm/export-to-csv.js b/packages/contentstack-export-to-csv/src/commands/cm/export-to-csv.js index 26cbe7fe2e..1898bbeacc 100644 --- a/packages/contentstack-export-to-csv/src/commands/cm/export-to-csv.js +++ b/packages/contentstack-export-to-csv/src/commands/cm/export-to-csv.js @@ -277,7 +277,7 @@ class ExportToCsvCommand extends Command { throw new Error(branchExists.errorMessage); } stack.branch_uid = branchUid; - stackAPIClient = getStackClient(managementAPIClient, stack); + stackAPIClient = this.getStackClient(managementAPIClient, stack); } catch (error) { if (error?.message || error?.errorMessage) { cliux.error(util.formatError(error)); diff --git a/packages/contentstack-export/src/export/modules/taxonomies.ts b/packages/contentstack-export/src/export/modules/taxonomies.ts index fa463a880f..09aa0db139 100644 --- a/packages/contentstack-export/src/export/modules/taxonomies.ts +++ b/packages/contentstack-export/src/export/modules/taxonomies.ts @@ -1,7 +1,7 @@ import omit from 'lodash/omit'; +import keys from 'lodash/keys'; import isEmpty from 'lodash/isEmpty'; import flatten from 'lodash/flatten'; -import keys from 'lodash/keys'; import { resolve as pResolve } from 'node:path'; import { cliux, configHandler, HttpClient } from '@contentstack/cli-utilities'; @@ -120,7 +120,7 @@ export default class ExportTaxonomies extends BaseClass { log(this.exportConfig, `No terms found for taxonomy - '${taxonomyUID}'`, 'info'); } else { fsUtil.writeFile(pResolve(this.termsFolderPath, `${taxonomyUID}-${this.termsConfig.fileName}`), this.terms); - log(this.exportConfig, `Terms from taxonomy '${taxonomyUID}' were successfully exported.`, 'success'); + log(this.exportConfig, `Terms from taxonomy '${taxonomyUID}' were exported successfully.`, 'success'); } } log(this.exportConfig, `All the terms have been exported successfully!`, 'success'); @@ -172,6 +172,7 @@ export default class ExportTaxonomies extends BaseClass { include_count: true, skip: 0, limit: this.taxonomiesConfig.limit || 100, + depth: 0 // include all the terms if set to 0 }; if (skip >= 0) params['skip'] = skip; diff --git a/packages/contentstack-import/src/config/index.ts b/packages/contentstack-import/src/config/index.ts index c9003f3622..315565ab28 100644 --- a/packages/contentstack-import/src/config/index.ts +++ b/packages/contentstack-import/src/config/index.ts @@ -26,6 +26,7 @@ const config: DefaultConfig = { 'locales', 'environments', 'assets', + 'taxonomies', 'extensions', 'marketplace-apps', 'global-fields', @@ -142,6 +143,14 @@ const config: DefaultConfig = { dirName: 'marketplace_apps', fileName: 'marketplace_apps.json', }, + taxonomies: { + dirName: 'taxonomies', + fileName: 'taxonomies.json' + }, + terms: { + dirName: 'terms', + fileName: 'terms.json' + }, }, languagesCode: [ 'af-za', diff --git a/packages/contentstack-import/src/import/modules/base-class.ts b/packages/contentstack-import/src/import/modules/base-class.ts index fa99462ead..e47f8790ab 100644 --- a/packages/contentstack-import/src/import/modules/base-class.ts +++ b/packages/contentstack-import/src/import/modules/base-class.ts @@ -18,6 +18,7 @@ import { LabelData } from '@contentstack/management/types/stack/label'; import { WebhookData } from '@contentstack/management/types/stack/webhook'; import { WorkflowData } from '@contentstack/management/types/stack/workflow'; import { RoleData } from '@contentstack/management/types/stack/role'; +import { HttpClient } from '@contentstack/cli-utilities'; import { log } from '../../utils'; import { ImportConfig, ModuleClassParams } from '../../types'; @@ -47,7 +48,9 @@ export type ApiModuleType = | 'create-entries' | 'update-entries' | 'publish-entries' - | 'delete-entries'; + | 'delete-entries' + | 'create-taxonomies' + | 'create-terms'; export type ApiOptions = { uid?: string; @@ -381,6 +384,19 @@ export default abstract class BaseClass { .delete({ locale: additionalInfo.locale }) .then(onSuccess) .catch(onReject); + case 'create-taxonomies': + return new HttpClient() + .headers(additionalInfo.headers) + .post(additionalInfo.url, { taxonomy: apiData }) + .then(onSuccess) + .catch(onReject); + case 'create-terms': + const url = `${additionalInfo.baseUrl}/${apiData.taxonomy_uid}/terms`; + return new HttpClient() + .headers(additionalInfo.headers) + .post(url, { term: apiData }) + .then(onSuccess) + .catch(onReject); default: return Promise.resolve(); } diff --git a/packages/contentstack-import/src/import/modules/taxonomies.ts b/packages/contentstack-import/src/import/modules/taxonomies.ts new file mode 100644 index 0000000000..37cade6c80 --- /dev/null +++ b/packages/contentstack-import/src/import/modules/taxonomies.ts @@ -0,0 +1,308 @@ +import keys from 'lodash/keys'; +import { join } from 'node:path'; +import values from 'lodash/values'; +import isEmpty from 'lodash/isEmpty'; +import flatten from 'lodash/flatten'; +import { configHandler } from '@contentstack/cli-utilities'; + +import BaseClass, { ApiOptions } from './base-class'; +import { log, formatError, fsUtil, fileHelper } from '../../utils'; +import { ModuleClassParams, TaxonomiesConfig, TermsConfig } from '../../types'; + +//NOTE: Temp types need to remove once sdk available +type TaxonomyPayload = { + baseUrl: string; + url: string; + mgToken: string; + reqPayload: Record; + headers: Record; +}; + +export default class ImportTaxonomies extends BaseClass { + private taxonomiesMapperDirPath: string; + private taxonomiesFolderPath: string; + private taxSuccessPath: string; + private taxFailsPath: string; + private taxonomiesConfig: TaxonomiesConfig; + private taxonomies: Record; + private termsFolderPath: string; + private termsMapperDirPath: string; + private termsConfig: TermsConfig; + private termsSuccessPath: string; + private termsFailsPath: string; + private taxonomyPayload: TaxonomyPayload; + public taxonomiesSuccess: Record = {}; + public taxonomiesFailed: Record = {}; + public termsSuccess: Record> = {}; + public termsFailed: Record> = {}; + public terms: Record = {}; + public taxonomyUIDs: string[] = []; + + constructor({ importConfig, stackAPIClient }: ModuleClassParams) { + super({ importConfig, stackAPIClient }); + this.taxonomiesConfig = importConfig.modules.taxonomies; + this.termsConfig = importConfig.modules.terms; + this.taxonomiesMapperDirPath = join(importConfig.backupDir, 'mapper', 'taxonomies'); + this.termsMapperDirPath = join(this.taxonomiesMapperDirPath, 'terms'); + this.taxonomiesFolderPath = join(importConfig.backupDir, this.taxonomiesConfig.dirName); + this.termsFolderPath = join(this.taxonomiesFolderPath, this.termsConfig.dirName); + this.taxSuccessPath = join(this.taxonomiesMapperDirPath, 'success.json'); + this.taxFailsPath = join(this.taxonomiesMapperDirPath, 'fails.json'); + this.termsSuccessPath = join(this.termsMapperDirPath, 'success.json'); + this.termsFailsPath = join(this.termsMapperDirPath, 'fails.json'); + this.taxonomyPayload = { + baseUrl: '', + url: '', + mgToken: importConfig.management_token, + reqPayload: {}, + headers: { + 'Content-Type': 'application/json', + api_key: importConfig.target_stack, + }, + }; + } + + /** + * @method start + * @returns {Promise} Promise + */ + async start(): Promise { + log(this.importConfig, 'Migrating taxonomies', 'info'); + + //Step1 check folder exists or not + if (fileHelper.fileExistsSync(this.taxonomiesFolderPath)) { + this.taxonomies = fsUtil.readFile(join(this.taxonomiesFolderPath, 'taxonomies.json'), true) as Record< + string, + unknown + >; + } else { + log(this.importConfig, `No such file or directory - '${this.taxonomiesFolderPath}'`, 'error'); + return; + } + + //NOTE - Temp code for api request + const { cma } = configHandler.get('region') || {}; + this.taxonomyPayload.baseUrl = `${cma}/v3/taxonomies`; + this.taxonomyPayload.url = this.taxonomyPayload.baseUrl; + if (this.taxonomyPayload?.mgToken) this.taxonomyPayload.headers['authorization'] = this.taxonomyPayload.mgToken; + else this.taxonomyPayload.headers['authtoken'] = configHandler.get('authtoken'); + + //Step 2 create taxonomies & terms mapper directory + await fsUtil.makeDirectory(this.taxonomiesMapperDirPath); + await fsUtil.makeDirectory(this.termsMapperDirPath); + + //Step 3 import taxonomy and create success & failure file + await this.importTaxonomies(); + this.createTaxonomySuccessAndFailedFile(); + + if (!fileHelper.fileExistsSync(this.termsFolderPath)) { + log(this.importConfig, `No such file or directory - '${this.termsFolderPath}'`, 'error'); + return; + } + //Step 4 import terms and create success & failure file + await this.importTerms(); + this.createTermSuccessAndFailedFile(); + + log(this.importConfig, 'Taxonomies have been imported successfully!', 'success'); + } + + /** + * create taxonomy and enter success & failure related data into taxonomies mapper file + * @method importTaxonomies + * @async + * @returns {Promise} Promise + */ + async importTaxonomies(): Promise { + if (this.taxonomies === undefined || isEmpty(this.taxonomies)) { + log(this.importConfig, 'No Taxonomies Found', 'info'); + return; + } + + const apiContent = values(this.taxonomies); + this.taxonomyUIDs = keys(this.taxonomies); + + const onSuccess = ({ + response: { data, status } = { data: null, status: null }, + apiData: { uid, name } = { uid: null, name: '' }, + }: any) => { + //NOTE - Temp code to handle error thru API. Will remove this once sdk is ready + if ([200, 201, 202].includes(status)) { + const { taxonomy } = data; + this.taxonomiesSuccess[taxonomy.uid] = taxonomy; + log(this.importConfig, `Taxonomy '${name}' imported successfully`, 'success'); + } else { + let errorMsg; + if ([500, 503, 502].includes(status)) errorMsg = data?.message || data; + else errorMsg = data?.error_message; + if (errorMsg === undefined) { + errorMsg = Object.values(data?.errors) && flatten(Object.values(data.errors)); + } + this.taxonomiesFailed[uid] = `Taxonomy '${name}' failed to be import. ${JSON.stringify(errorMsg)}`; + log(this.importConfig, `Taxonomy '${name}' failed to be import. ${JSON.stringify(errorMsg)}`, 'error'); + } + }; + + const onReject = ({ error, apiData }: any) => { + const err = error?.message ? JSON.parse(error.message) : error; + const { name } = apiData; + if (err?.errors?.name) { + log(this.importConfig, `Taxonomy '${name}' already exists`, 'info'); + } else { + this.taxonomiesFailed[apiData.uid] = apiData; + log(this.importConfig, `Taxonomy '${name}' failed to be import ${formatError(error)}`, 'error'); + log(this.importConfig, error, 'error'); + } + }; + + await this.makeConcurrentCall( + { + apiContent, + processName: 'import taxonomies', + apiParams: { + serializeData: this.serializeTaxonomy.bind(this), + reject: onReject, + resolve: onSuccess, + entity: 'create-taxonomies', + includeParamOnCompletion: true, + additionalInfo: this.taxonomyPayload, + }, + concurrencyLimit: this.importConfig.concurrency || this.importConfig.fetchConcurrency || 1, + }, + undefined, + false, + ); + } + + /** + * @method serializeTaxonomy + * @param {ApiOptions} apiOptions ApiOptions + * @returns {ApiOptions} ApiOptions + */ + serializeTaxonomy(apiOptions: ApiOptions): ApiOptions { + const { apiData: taxonomy } = apiOptions; + if (this.taxonomiesSuccess.hasOwnProperty(taxonomy.uid)) { + log(this.importConfig, `Taxonomy '${taxonomy.name}' already exists. Skipping it to avoid duplicates!`, 'info'); + apiOptions.entity = undefined; + } else { + apiOptions.apiData = taxonomy; + } + apiOptions.apiData = taxonomy; + return apiOptions; + } + + /** + * create taxonomies success and fail in (mapper/taxonomies) + * @method createTaxonomySuccessAndFailedFile + */ + createTaxonomySuccessAndFailedFile() { + if (this.taxonomiesSuccess !== undefined && !isEmpty(this.taxonomiesSuccess)) { + fsUtil.writeFile(this.taxSuccessPath, this.taxonomiesSuccess); + } + + if (this.taxonomiesFailed !== undefined && !isEmpty(this.taxonomiesFailed)) { + fsUtil.writeFile(this.taxFailsPath, this.taxonomiesFailed); + } + } + + /** + * create terms and enter success & failure related data into terms mapper file + * @method importTerms + * @async + * @returns {Promise} Promise + */ + async importTerms(): Promise { + if (!this.taxonomyUIDs?.length) { + return; + } + + const onSuccess = ({ + response: { data, status } = { data: null, status: null }, + apiData: { uid, name, taxonomy_uid } = { uid: null, name: '', taxonomy_uid: null }, + }: any) => { + //NOTE - Temp code to handle error thru API. Will remove this once sdk is ready + if ([200, 201, 202].includes(status)) { + if (!this.termsSuccess[taxonomy_uid]) this.termsSuccess[taxonomy_uid] = {}; + const { term } = data; + this.termsSuccess[taxonomy_uid][term.uid] = term; + log(this.importConfig, `Term '${name}' imported successfully`, 'success'); + } else { + if (!this.termsFailed[taxonomy_uid]) this.termsFailed[taxonomy_uid] = {}; + let errorMsg; + if ([500, 503, 502].includes(status)) errorMsg = data?.message || data; + else errorMsg = data?.error_message; + if (errorMsg === undefined) { + errorMsg = Object.values(data?.errors) && flatten(Object.values(data.errors)); + } + this.termsFailed[taxonomy_uid][uid] = `Terms '${name}' failed to be import. ${JSON.stringify(errorMsg)}`; + log(this.importConfig, `Terms '${name}' failed to be import. ${JSON.stringify(errorMsg)}`, 'error'); + } + }; + + const onReject = ({ error, apiData }: any) => { + const { uid, taxonomy_uid, name } = apiData; + if (!this.termsFailed[taxonomy_uid]) this.termsFailed[taxonomy_uid] = {}; + const err = error?.message ? JSON.parse(error.message) : error; + if (err?.errors?.name) { + log(this.importConfig, `Term '${name}' already exists`, 'info'); + } else { + this.termsFailed[taxonomy_uid][apiData.uid] = apiData; + log(this.importConfig, `Term '${name}' failed to be import ${formatError(error)}`, 'error'); + log(this.importConfig, error, 'error'); + } + }; + + for (const taxUID of this.taxonomyUIDs) { + //read terms from respective taxonomy + this.terms = fsUtil.readFile( + join(this.termsFolderPath, `${taxUID}-${this.termsConfig.fileName}`), + true, + ) as Record; + + if (this.terms !== undefined && !isEmpty(this.terms)) { + const apiContent = values(this.terms); + await this.makeConcurrentCall( + { + apiContent, + processName: 'import terms', + apiParams: { + serializeData: this.serializeTerms.bind(this), + reject: onReject, + resolve: onSuccess, + entity: 'create-terms', + includeParamOnCompletion: true, + additionalInfo: this.taxonomyPayload, + }, + concurrencyLimit: this.importConfig.concurrency || this.importConfig.fetchConcurrency || 1, + }, + undefined, + false, + ); + } + } + } + + /** + * @method serializeTerms + * @param {ApiOptions} apiOptions ApiOptions + * @returns {ApiOptions} ApiOptions + */ + serializeTerms(apiOptions: ApiOptions): ApiOptions { + const { apiData: term } = apiOptions; + apiOptions.apiData = term; + return apiOptions; + } + + /** + * create terms success and fail in (mapper/taxonomies/terms) + * @method createTermSuccessAndFailedFile + */ + createTermSuccessAndFailedFile() { + if (this.termsSuccess !== undefined && !isEmpty(this.termsSuccess)) { + fsUtil.writeFile(this.termsSuccessPath, this.termsSuccess); + } + + if (this.termsFailed !== undefined && !isEmpty(this.termsFailed)) { + fsUtil.writeFile(this.termsFailsPath, this.termsFailed); + } + } +} diff --git a/packages/contentstack-import/src/types/default-config.ts b/packages/contentstack-import/src/types/default-config.ts index c41b949fdb..27b0af0859 100644 --- a/packages/contentstack-import/src/types/default-config.ts +++ b/packages/contentstack-import/src/types/default-config.ts @@ -113,6 +113,16 @@ export default interface DefaultConfig { fileName: string; requiredKeys: string[]; }; + taxonomies: { + dirName: string; + fileName: string; + dependencies?: Modules[]; + }; + terms: { + dirName: string; + fileName: string; + dependencies?: Modules[]; + }; }; languagesCode: string[]; apis: { diff --git a/packages/contentstack-import/src/types/index.ts b/packages/contentstack-import/src/types/index.ts index 104328215f..b9d5d5111a 100644 --- a/packages/contentstack-import/src/types/index.ts +++ b/packages/contentstack-import/src/types/index.ts @@ -40,7 +40,8 @@ export type Modules = | 'custom-roles' | 'workflows' | 'labels' - | 'marketplace-apps'; + | 'marketplace-apps' + | 'taxonomies'; export type ModuleClassParams = { stackAPIClient: ReturnType; @@ -88,5 +89,17 @@ export interface CustomRoleConfig{ customRolesLocalesFileName: string; } +export interface TaxonomiesConfig{ + dirName: string; + fileName: string; + dependencies?: Modules[]; +} + +export interface TermsConfig{ + dirName: string; + fileName: string; + dependencies?: Modules[]; +} + export { default as DefaultConfig } from './default-config'; export { default as ImportConfig } from './import-config'; diff --git a/packages/contentstack-import/src/utils/asset-helper.ts b/packages/contentstack-import/src/utils/asset-helper.ts index 7e7cc9e257..ceaead0bed 100644 --- a/packages/contentstack-import/src/utils/asset-helper.ts +++ b/packages/contentstack-import/src/utils/asset-helper.ts @@ -2,7 +2,7 @@ import Bluebird from 'bluebird'; import * as url from 'url'; import * as path from 'path'; import { ContentstackClient, managementSDKClient } from '@contentstack/cli-utilities'; -import { ImportConfig } from 'src/types'; +import { ImportConfig } from '../types'; const debug = require('debug')('util:requests'); let _ = require('lodash'); let { marked } = require('marked');