Skip to content

Commit

Permalink
Merge pull request #1305 from contentstack/feat/taxonomy-export-impor…
Browse files Browse the repository at this point in the history
…t-api

Feat/taxonomy export import api
  • Loading branch information
aman19K authored Feb 21, 2024
2 parents 7adff74 + 5b7963a commit 26e06a9
Show file tree
Hide file tree
Showing 33 changed files with 1,213 additions and 655 deletions.
604 changes: 531 additions & 73 deletions package-lock.json

Large diffs are not rendered by default.

2 changes: 1 addition & 1 deletion packages/contentstack-bootstrap/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -15,7 +15,7 @@ $ npm install -g @contentstack/cli-cm-bootstrap
$ csdx COMMAND
running command...
$ csdx (--version)
@contentstack/cli-cm-bootstrap/1.7.1 darwin-arm64 node-v20.8.0
@contentstack/cli-cm-bootstrap/1.8.0 darwin-arm64 node-v18.19.0
$ csdx --help [COMMAND]
USAGE
$ csdx COMMAND
Expand Down
2 changes: 1 addition & 1 deletion packages/contentstack-clone/README.md
Original file line number Diff line number Diff line change
Expand Up @@ -16,7 +16,7 @@ $ npm install -g @contentstack/cli-cm-clone
$ csdx COMMAND
running command...
$ csdx (--version)
@contentstack/cli-cm-clone/1.10.0 darwin-arm64 node-v20.8.0
@contentstack/cli-cm-clone/1.10.1 darwin-arm64 node-v20.8.0
$ csdx --help [COMMAND]
USAGE
$ csdx COMMAND
Expand Down
6 changes: 3 additions & 3 deletions packages/contentstack-clone/package.json
Original file line number Diff line number Diff line change
@@ -1,12 +1,12 @@
{
"name": "@contentstack/cli-cm-clone",
"description": "Contentstack stack clone plugin",
"version": "1.10.0",
"version": "1.10.1",
"author": "Contentstack",
"bugs": "https://github.com/rohitmishra209/cli-cm-clone/issues",
"dependencies": {
"@contentstack/cli-cm-export": "~1.10.5",
"@contentstack/cli-cm-import": "~1.13.5",
"@contentstack/cli-cm-export": "~1.11.0",
"@contentstack/cli-cm-import": "~1.14.0",
"@contentstack/cli-command": "~1.2.16",
"@contentstack/cli-utilities": "~1.5.13",
"@colors/colors": "^1.5.0",
Expand Down
2 changes: 1 addition & 1 deletion packages/contentstack-export-to-csv/package.json
Original file line number Diff line number Diff line change
@@ -1,7 +1,7 @@
{
"name": "@contentstack/cli-cm-export-to-csv",
"description": "Export entities to csv",
"version": "1.6.2",
"version": "1.7.0",
"author": "Abhinav Gupta @abhinav-from-contentstack",
"bugs": "https://github.com/contentstack/cli/issues",
"dependencies": {
Expand Down
167 changes: 82 additions & 85 deletions packages/contentstack-export-to-csv/src/util/index.js
Original file line number Diff line number Diff line change
Expand Up @@ -460,7 +460,7 @@ function write(command, entries, fileName, message, delimiter, headers) {
process.chdir(directory);
}
// eslint-disable-next-line no-undef
cliux.print(`Writing ${message} to file: ${process.cwd()}${delimeter}${fileName}`);
cliux.print(`Writing ${message} to file: "${process.cwd()}${delimeter}${fileName}"`);
if (headers?.length) fastcsv.writeToPath(fileName, entries, { headers, delimiter });
else fastcsv.writeToPath(fileName, entries, { headers: true, delimiter });
}
Expand Down Expand Up @@ -700,10 +700,10 @@ function handleErrorMsg(err) {

/**
* This function does the sdk calls to get all the teams in org
* @param {object} managementAPIClient
* @param {object} org
* @param {object} queryParam
* @returns
* @param {object} managementAPIClient
* @param {object} org
* @param {object} queryParam
* @returns
*/
async function getAllTeams(managementAPIClient, org, queryParam = {}) {
try {
Expand All @@ -715,8 +715,8 @@ async function getAllTeams(managementAPIClient, org, queryParam = {}) {

/**
* This function is used to handle the pagination and call the sdk
* @param {object} managementAPIClient
* @param {object} org
* @param {object} managementAPIClient
* @param {object} org
*/
async function exportOrgTeams(managementAPIClient, org) {
let allTeamsInOrg = [];
Expand All @@ -737,9 +737,9 @@ async function exportOrgTeams(managementAPIClient, org) {
}

/**
* This function will get all the org level roles
* @param {object} managementAPIClient
* @param {object} org
* This function will get all the org level roles
* @param {object} managementAPIClient
* @param {object} org
*/
async function getOrgRolesForTeams(managementAPIClient, org) {
let roleMap = {}; // for org level there are two roles only admin and member
Expand All @@ -763,9 +763,9 @@ async function getOrgRolesForTeams(managementAPIClient, org) {

/**
* Removes the unnecessary fields from the objects in the data and assign org level roles to the team based on role uid
* @param {array} data
* @param {object} managementAPIClient
* @param {object} org
* @param {array} data
* @param {object} managementAPIClient
* @param {object} org
*/
async function cleanTeamsData(data, managementAPIClient, org) {
const roleMap = await getOrgRolesForTeams(managementAPIClient, org);
Expand All @@ -784,7 +784,7 @@ async function cleanTeamsData(data, managementAPIClient, org) {
'delete',
'fetch',
'stackRoleMappings',
'teamUsers'
'teamUsers',
];
if (data?.length) {
return data.map((team) => {
Expand All @@ -806,10 +806,10 @@ async function cleanTeamsData(data, managementAPIClient, org) {

/**
* This function is used to call all the other teams function to export the required files
* @param {object} managementAPIClient
* @param {object} organization
* @param {string} teamUid
* @param {character} delimiter
* @param {object} managementAPIClient
* @param {object} organization
* @param {string} teamUid
* @param {character} delimiter
*/
async function exportTeams(managementAPIClient, organization, teamUid, delimiter) {
cliux.print(
Expand Down Expand Up @@ -852,10 +852,10 @@ async function exportTeams(managementAPIClient, organization, teamUid, delimiter

/**
* This function is used to get individual team user details and write to file
* @param {array} allTeamsData
* @param {object} organization
* @param {array} allTeamsData
* @param {object} organization
* @param {string} teamUid optional
* @param {character} delimiter
* @param {character} delimiter
*/
async function getTeamsDetail(allTeamsData, organization, teamUid, delimiter) {
if (!teamUid) {
Expand Down Expand Up @@ -883,10 +883,10 @@ async function getTeamsDetail(allTeamsData, organization, teamUid, delimiter) {

/**
* This will export the role mappings of the team, for which stack the team has which role
* @param {object} managementAPIClient
* @param {array} allTeamsData Data for all the teams in the stack
* @param {object} managementAPIClient
* @param {array} allTeamsData Data for all the teams in the stack
* @param {string} teamUid for a particular team who's data we want
* @param {character} delimiter
* @param {character} delimiter
*/
async function exportRoleMappings(managementAPIClient, allTeamsData, teamUid, delimiter) {
let stackRoleWithTeamData = [];
Expand Down Expand Up @@ -935,7 +935,7 @@ async function exportRoleMappings(managementAPIClient, allTeamsData, teamUid, de
];
try {
const exportStackRole = await inquirer.prompt(export_stack_role);
if(exportStackRole.chooseExport==='no') {
if (exportStackRole.chooseExport === 'no') {
process.exit(1);
}
} catch (error) {
Expand All @@ -953,10 +953,10 @@ async function exportRoleMappings(managementAPIClient, allTeamsData, teamUid, de

/**
* Mapping the team stacks with the stack role and returning and array of object
* @param {object} managementAPIClient
* @param {array} stackRoleMapping
* @param {string} teamName
* @param {string} teamUid
* @param {object} managementAPIClient
* @param {array} stackRoleMapping
* @param {string} teamName
* @param {string} teamUid
*/
async function mapRoleWithTeams(managementAPIClient, stackRoleMapping, teamName, teamUid) {
const roles = await getRoleData(managementAPIClient, stackRoleMapping.stackApiKey);
Expand All @@ -982,8 +982,8 @@ async function mapRoleWithTeams(managementAPIClient, stackRoleMapping, teamName,

/**
* Making sdk call to get all the roles in the given stack
* @param {object} managementAPIClient
* @param {string} stackApiKey
* @param {object} managementAPIClient
* @param {string} stackApiKey
*/
async function getRoleData(managementAPIClient, stackApiKey) {
try {
Expand All @@ -995,7 +995,7 @@ async function getRoleData(managementAPIClient, stackApiKey) {

/**
* Here in the users array we are adding the team-name and team-uid to individual users and returning an array of object of user details only
* @param {array} teams
* @param {array} teams
*/
async function getTeamsUserDetails(teams) {
const allTeamUsers = [];
Expand Down Expand Up @@ -1080,7 +1080,7 @@ async function getTaxonomy(payload) {
* @returns {*} Promise<any>
*/
async function taxonomySDKHandler(payload, skip) {
const { stackAPIClient, taxonomyUID, type } = payload;
const { stackAPIClient, taxonomyUID, type, format } = payload;

const queryParams = { include_count: true, limit: payload.limit };
if (skip >= 0) queryParams['skip'] = skip || 0;
Expand All @@ -1092,13 +1092,13 @@ async function taxonomySDKHandler(payload, skip) {
.query(queryParams)
.find()
.then((data) => data)
.catch((err) => handleErrorMsg(err));
.catch((err) => handleTaxonomyErrorMsg(err));
case 'taxonomy':
return await stackAPIClient
.taxonomy(taxonomyUID)
.fetch()
.then((data) => data)
.catch((err) => handleErrorMsg(err));
.catch((err) => handleTaxonomyErrorMsg(err));
case 'terms':
queryParams['depth'] = 0;
return await stackAPIClient
Expand All @@ -1107,9 +1107,15 @@ async function taxonomySDKHandler(payload, skip) {
.query(queryParams)
.find()
.then((data) => data)
.catch((err) => handleErrorMsg(err));
.catch((err) => handleTaxonomyErrorMsg(err));
case 'export-taxonomies':
return await stackAPIClient
.taxonomy(taxonomyUID)
.export({ format })
.then((data) => data)
.catch((err) => handleTaxonomyErrorMsg(err));
default:
handleErrorMsg({ errorMessage: 'Invalid module!' });
handleTaxonomyErrorMsg({ errorMessage: 'Invalid module!' });
}
}

Expand Down Expand Up @@ -1152,11 +1158,9 @@ function formatTermsOfTaxonomyData(terms, taxonomyUID) {
}
}

function handleErrorMsg(err) {
if (err?.errorMessage) {
cliux.print(`Error: ${err.errorMessage}`, { color: 'red' });
} else if (err?.message) {
const errorMsg = err?.errors?.taxonomy || err?.errors?.term || err?.message;
function handleTaxonomyErrorMsg(err) {
if (err?.errorMessage || err?.message) {
const errorMsg = err?.errorMessage || err?.errors?.taxonomy || err?.errors?.term || err?.message;
cliux.print(`Error: ${errorMsg}`, { color: 'red' });
} else {
console.log(err);
Expand All @@ -1166,60 +1170,53 @@ function handleErrorMsg(err) {
}

/**
* create an importable CSV file, to utilize with the migration script.
* Generate a CSV file that can be imported for use with the migration script.
* @param {*} payload api request payload
* @param {*} taxonomies taxonomies data
* @returns
*/
async function createImportableCSV(payload, taxonomies) {
let taxonomiesData = [];
let headers = ['Taxonomy Name', 'Taxonomy UID', 'Taxonomy Description'];
for (let index = 0; index < taxonomies?.length; index++) {
const taxonomy = taxonomies[index];
const taxonomyUID = taxonomy?.uid;
if (taxonomyUID) {
const sanitizedTaxonomy = sanitizeData({
'Taxonomy Name': taxonomy?.name,
'Taxonomy UID': taxonomyUID,
'Taxonomy Description': taxonomy?.description,
});
taxonomiesData.push(sanitizedTaxonomy);
payload['taxonomyUID'] = taxonomyUID;
const terms = await getAllTermsOfTaxonomy(payload);
//fetch all parent terms
const parentTerms = terms.filter((term) => term?.parent_uid === null);
const termsData = getParentAndChildTerms(parentTerms, terms, headers);
taxonomiesData.push(...termsData);
let taxonomiesData = [];
let headers = [];
payload['type'] = 'export-taxonomies';
payload['format'] = 'csv';
for (const taxonomy of taxonomies) {
if (taxonomy?.uid) {
payload['taxonomyUID'] = taxonomy?.uid;
const data = await taxonomySDKHandler(payload);
const taxonomies = await csvParse(data, headers);
taxonomiesData.push(...taxonomies);
}
}
}

return { taxonomiesData, headers };
return { taxonomiesData, headers };
}

/**
* Get the parent and child terms, then arrange them hierarchically in a CSV file.
* @param {*} parentTerms list of parent terms
* @param {*} terms respective terms of taxonomies
* @param {*} headers list of csv headers include taxonomy and terms column
* @param {*} termsData parent and child terms
* Parse the CSV data and segregate the headers from the actual data.
* @param {*} data taxonomy csv data with headers
* @param {*} headers list of csv headers
* @returns taxonomy data without headers
*/
function getParentAndChildTerms(parentTerms, terms, headers, termsData = []) {
for (let i = 0; i < parentTerms?.length; i++) {
const parentTerm = parentTerms[i];
const levelUID = `Term Level${parentTerm.depth} UID`;
const levelName = `Term Level${parentTerm.depth} Name`;
if (headers.indexOf(levelName) === -1) headers.push(levelName);
if (headers.indexOf(levelUID) === -1) headers.push(levelUID);
const sanitizedTermData = sanitizeData({ [levelName]: parentTerm.name, [levelUID]: parentTerm.uid });
termsData.push(sanitizedTermData);
//fetch all sibling terms
const newParents = terms.filter((term) => term.parent_uid === parentTerm.uid);
if (newParents?.length) {
getParentAndChildTerms(newParents, terms, headers, termsData);
}
}
return termsData;
}
const csvParse = (data, headers) => {
return new Promise((resolve, reject) => {
const taxonomies = [];
const stream = fastcsv.parseStream(fastcsv.parse());
stream.write(data);
stream.end();
stream
.on('data', (data) => {
taxonomies.push(data);
})
.on('error', (err) => reject(err))
.on('end', () => {
taxonomies[0]?.forEach((header) => {
if (!headers.includes(header)) headers.push(header);
});
resolve(taxonomies.splice(1));
});
});
};

module.exports = {
chooseOrganization: chooseOrganization,
Expand Down
Original file line number Diff line number Diff line change
Expand Up @@ -395,5 +395,6 @@
"users": ["user_1_uid_member", "user_2_uid_member", "user_3_uid_member", "user_4_uid_member"]
}
]
}
},
"taxonomyCSVData": "`taxonomy1,taxonomy1,,,,,,,\n,,,term1,term1,,,,\n,,,,,term1_2,term1_2,,\n,,,term2,term2,,,,\n,,,,,term2_2,term2_2,,\n,,,,,,,term2_2_1,term2_2_1\n,,,,,term2_1,term2_1,,`"
}
Loading

0 comments on commit 26e06a9

Please sign in to comment.