sf-decomposer
Version:
Split large Salesforce metadata files into version-control-friendly pieces and rebuild deployment-ready files.
67 lines • 3.62 kB
JavaScript
;
import { getRegistryValuesBySuffix } from '../metadata/getRegistryValuesBySuffix.js';
import { parseManifest } from '../metadata/parseManifest.js';
import { decomposeFileHandler } from '../service/decompose/decomposeFileHandler.js';
import { CONCURRENCY_LIMITS } from '../helpers/constants.js';
import { pLimit } from '../helpers/pLimit.js';
import { resolveDecomposeOptionsForType } from '../helpers/configOverrides.js';
export async function decomposeMetadataTypes(options) {
const { metadataTypes, prepurge, postpurge, format, ignoreDirs, strategy, decomposeNestedPerms, manifest, overrides, log, repoRoot, } = options;
let manifestFilter;
let effectiveTypes;
if (manifest) {
manifestFilter = await parseManifest(manifest, ignoreDirs, repoRoot);
for (const { type, member } of manifestFilter.unresolvedComponents) {
log(`Warning: manifest component ${type}:${member} not found in local source; skipping.`);
}
// Stryker disable next-line ConditionalExpression, EqualityOperator
if (metadataTypes && metadataTypes.length > 0) {
const manifestTypes = new Set(manifestFilter.suffixes);
effectiveTypes = metadataTypes.filter((type) => manifestTypes.has(type));
}
else {
effectiveTypes = manifestFilter.suffixes;
}
}
else {
if (!metadataTypes || metadataTypes.length === 0) {
throw Error('Either --metadata-type or --manifest must be provided.');
}
effectiveTypes = metadataTypes;
}
if (effectiveTypes.length === 0) {
log('No metadata types to decompose after applying the manifest filter.');
return { metadata: [] };
}
// Limit concurrent metadata type processing to prevent file system overload
const limit = pLimit(CONCURRENCY_LIMITS.METADATA_TYPES);
const processed = [];
const tasks = effectiveTypes.map((metadataType) => limit(async () => {
const manifestXmlPaths = manifestFilter?.parentXmlsBySuffix.get(metadataType);
// Type-scope resolved options serve as the base for component-scope resolution further
// down the call stack. Hard strategy rules (labels / loyaltyProgramSetup) are applied per
// file inside the disassembler so they remain in force even when a component-scope override
// tries to flip the strategy.
const typeResolved = resolveDecomposeOptionsForType(metadataType, { format, strategy, decomposeNestedPerms, prepurge, postpurge }, overrides);
let metaAttributes;
let ignorePath;
try {
({ metaAttributes, ignorePath } = await getRegistryValuesBySuffix(metadataType, 'decompose', ignoreDirs, repoRoot, typeResolved.uniqueIdElements));
}
catch (err) {
/* istanbul ignore if -- @preserve: preserves non-manifest behavior; unreachable via known CLI types */
if (!manifestFilter)
throw err;
/* istanbul ignore next -- @preserve: getRegistryValuesBySuffix always throws Error instances */
const message = err instanceof Error ? err.message : String(err);
log(`Skipping ${metadataType}: ${message}`);
return;
}
await decomposeFileHandler(metaAttributes, typeResolved, ignorePath, overrides, manifestXmlPaths);
processed.push(metadataType);
log(`All metadata files have been decomposed for the metadata type: ${metadataType}`);
}));
await Promise.all(tasks);
return { metadata: processed };
}
//# sourceMappingURL=decomposeMetadataTypes.js.map