From d8735176405f229f6cbe46bfd309db4bc3a27420 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 12:09:56 -0500 Subject: [PATCH 01/30] feat: customLabels => customLabel files --- .../convertContext/recompositionFinalizer.ts | 3 +- src/convert/streams.ts | 6 +- .../transformers/baseMetadataTransformer.ts | 5 +- .../decomposeLabelsTransformer.ts | 55 +++++++++++++++++++ .../decomposedMetadataTransformer.ts | 8 +-- .../defaultMetadataTransformer.ts | 8 ++- .../metadataTransformerFactory.ts | 14 +++-- .../nonDecomposedMetadataTransformer.ts | 5 +- .../staticResourceMetadataTransformer.ts | 4 +- src/convert/types.ts | 9 ++- src/index.ts | 15 ++--- src/registry/index.ts | 8 +-- .../presets/decomposeCustomLabelsBeta2.json | 28 ++++++++++ src/registry/types.ts | 6 +- test/convert/streams.test.ts | 11 ++-- .../decomposedMetadataTransformer.test.ts | 28 +++++----- .../defaultMetadataTransformer.test.ts | 20 +++---- .../nonDecomposedMetadataTransformer.test.ts | 4 +- .../staticResourceMetadataTransformer.test.ts | 34 ++++++------ test/registry/registryValidation.test.ts | 17 ++---- 20 files changed, 189 insertions(+), 99 deletions(-) create mode 100644 src/convert/transformers/decomposeLabelsTransformer.ts create mode 100644 src/registry/presets/decomposeCustomLabelsBeta2.json diff --git a/src/convert/convertContext/recompositionFinalizer.ts b/src/convert/convertContext/recompositionFinalizer.ts index 81e2cacc16..0b27140ca9 100644 --- a/src/convert/convertContext/recompositionFinalizer.ts +++ b/src/convert/convertContext/recompositionFinalizer.ts @@ -11,7 +11,6 @@ import { extractUniqueElementValue, getXmlElement, unwrapAndOmitNS } from '../.. import { MetadataComponent } from '../../resolve/types'; import { XML_NS_KEY, XML_NS_URL } from '../../common/constants'; import { ComponentSet } from '../../collections/componentSet'; -import { RecompositionStrategy } from '../../registry/types'; import { SourceComponent } from '../../resolve/sourceComponent'; import { JsToXml } from '../streams'; import { WriterFormat } from '../types'; @@ -127,7 +126,7 @@ const recompose = const getStartingXml = (cache: XmlCache) => async (parent: SourceComponent): Promise => - parent.type.strategies?.recomposition === RecompositionStrategy.StartEmpty + parent.type.strategies?.recomposition === 'startEmpty' ? {} : unwrapAndOmitNS(parent.type.name)(await getXmlFromCache(cache)(parent)) ?? {}; diff --git a/src/convert/streams.ts b/src/convert/streams.ts index 5fc18e1719..8fe41c92d0 100644 --- a/src/convert/streams.ts +++ b/src/convert/streams.ts @@ -72,11 +72,13 @@ export class ComponentConverter extends Transform { case 'source': if (mergeWith) { for (const mergeComponent of mergeWith) { - converts.push(transformer.toSourceFormat(chunk, mergeComponent)); + converts.push( + transformer.toSourceFormat({ component: chunk, mergeWith: mergeComponent, mergeSet: this.mergeSet }) + ); } } if (converts.length === 0) { - converts.push(transformer.toSourceFormat(chunk)); + converts.push(transformer.toSourceFormat({ component: chunk })); } break; case 'metadata': diff --git a/src/convert/transformers/baseMetadataTransformer.ts b/src/convert/transformers/baseMetadataTransformer.ts index f0c1666099..1b695269e9 100644 --- a/src/convert/transformers/baseMetadataTransformer.ts +++ b/src/convert/transformers/baseMetadataTransformer.ts @@ -20,5 +20,8 @@ export abstract class BaseMetadataTransformer implements MetadataTransformer { } public abstract toMetadataFormat(component: SourceComponent): Promise; - public abstract toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise; + public abstract toSourceFormat(input: { + component: SourceComponent; + mergeWith?: SourceComponent; + }): Promise; } diff --git a/src/convert/transformers/decomposeLabelsTransformer.ts b/src/convert/transformers/decomposeLabelsTransformer.ts new file mode 100644 index 0000000000..b870dd1f6e --- /dev/null +++ b/src/convert/transformers/decomposeLabelsTransformer.ts @@ -0,0 +1,55 @@ +/* + * Copyright (c) 2023, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ + +import type { CustomLabels, CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; +import { SfError } from '@salesforce/core/sfError'; +import { calculateRelativePath } from '../../utils/path'; +import { SourceComponent } from '../../resolve/sourceComponent'; +import { ToSourceFormatInput, WriteInfo } from '../types'; +import { JsToXml } from '../streams'; +import { DefaultMetadataTransformer } from './defaultMetadataTransformer'; + +/* Use for the metadata type CustomLabels */ +export class LabelsMetadataTransformer extends DefaultMetadataTransformer { + // CustomLabels file => CustomLabel + public async toSourceFormat({ component, mergeSet }: ToSourceFormatInput): Promise { + const labelType = this.registry.getTypeByName('CustomLabel'); + const partiallyAppliedPathCalculator = calculateRelativePath('source')({ + self: labelType, + }); + const xml = await component.parseXml(); + // split each label into a separate label file + return xml.labels.filter(customLabelHasFullName).map((l) => ({ + output: + // if present in the merge set, use that xml path, otherwise use the default path + mergeSet?.getComponentFilenamesByNameAndType({ fullName: l.fullName, type: labelType.name })?.[0] ?? + partiallyAppliedPathCalculator(l.fullName)(`${l.fullName}.label-meta.xml`), + source: new JsToXml({ CustomLabel: l }), + })); + } +} + +/* Use for the metadata type CustomLabel */ +export class LabelMetadataTransformer extends DefaultMetadataTransformer { + // eslint-disable-next-line @typescript-eslint/require-await, class-methods-use-this, @typescript-eslint/no-unused-vars + public async toMetadataFormat(component: SourceComponent): Promise { + // TODO: + // read all each label from the recomposition state, regardless of parents + // merge them all to a single CustomLabels file + // write the CustomLabels file + return []; + } + + // toSourceFormat uses the default (merge them with the existing label) +} + +const customLabelHasFullName = (label: CustomLabel): label is CustomLabel & { fullName: string } => { + if (label.fullName === undefined) { + throw SfError.create({ message: 'Label does not have a fullName', data: label }); + } + return true; +}; diff --git a/src/convert/transformers/decomposedMetadataTransformer.ts b/src/convert/transformers/decomposedMetadataTransformer.ts index fced068fe4..27eb82e629 100644 --- a/src/convert/transformers/decomposedMetadataTransformer.ts +++ b/src/convert/transformers/decomposedMetadataTransformer.ts @@ -14,10 +14,10 @@ import { calculateRelativePath } from '../../utils/path'; import { ForceIgnore } from '../../resolve/forceIgnore'; import { extractUniqueElementValue, objectHasSomeRealValues } from '../../utils/decomposed'; import type { MetadataComponent } from '../../resolve/types'; -import { DecompositionStrategy, type MetadataType } from '../../registry/types'; +import { type MetadataType } from '../../registry/types'; import { SourceComponent } from '../../resolve/sourceComponent'; import { JsToXml } from '../streams'; -import type { WriteInfo, XmlObj } from '../types'; +import type { ToSourceFormatInput, WriteInfo, XmlObj } from '../types'; import { META_XML_SUFFIX, XML_NS_KEY, XML_NS_URL } from '../../common/constants'; import type { SourcePath } from '../../common/types'; import { ComponentSet } from '../../collections/componentSet'; @@ -60,7 +60,7 @@ export class DecomposedMetadataTransformer extends BaseMetadataTransformer { return []; } - public async toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise { + public async toSourceFormat({ component, mergeWith }: ToSourceFormatInput): Promise { const forceIgnore = component.getForceIgnore(); // if the whole parent is ignored, we won't worry about decomposing things @@ -265,7 +265,7 @@ const getDefaultOutput = (component: MetadataComponent): SourcePath => { // there could be a '.' inside the child name (ex: PermissionSet.FieldPermissions.field uses Obj__c.Field__c) const childName = tail.length ? tail.join('.') : undefined; const output = join( - parent?.type.strategies?.decomposition === DecompositionStrategy.FolderPerType ? type.directoryName : '', + parent?.type.strategies?.decomposition === 'folderPerType' ? type.directoryName : '', `${childName ?? baseName}.${ensureString(component.type.suffix)}${META_XML_SUFFIX}` ); return join(calculateRelativePath('source')({ self: parent?.type ?? type })(fullName)(baseName), output); diff --git a/src/convert/transformers/defaultMetadataTransformer.ts b/src/convert/transformers/defaultMetadataTransformer.ts index 5865741596..8ca6b5a361 100644 --- a/src/convert/transformers/defaultMetadataTransformer.ts +++ b/src/convert/transformers/defaultMetadataTransformer.ts @@ -31,7 +31,13 @@ export class DefaultMetadataTransformer extends BaseMetadataTransformer { } // eslint-disable-next-line @typescript-eslint/require-await, class-methods-use-this - public async toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise { + public async toSourceFormat({ + component, + mergeWith, + }: { + component: SourceComponent; + mergeWith?: SourceComponent; + }): Promise { return getWriteInfos(component, 'source', mergeWith); } } diff --git a/src/convert/transformers/metadataTransformerFactory.ts b/src/convert/transformers/metadataTransformerFactory.ts index 311da4d478..d572a7d4e3 100644 --- a/src/convert/transformers/metadataTransformerFactory.ts +++ b/src/convert/transformers/metadataTransformerFactory.ts @@ -9,11 +9,11 @@ import { MetadataTransformer } from '../types'; import { SourceComponent } from '../../resolve/sourceComponent'; import { ConvertContext } from '../convertContext/convertContext'; import { RegistryAccess } from '../../registry/registryAccess'; -import { TransformerStrategy } from '../../registry/types'; import { DefaultMetadataTransformer } from './defaultMetadataTransformer'; import { DecomposedMetadataTransformer } from './decomposedMetadataTransformer'; import { StaticResourceMetadataTransformer } from './staticResourceMetadataTransformer'; import { NonDecomposedMetadataTransformer } from './nonDecomposedMetadataTransformer'; +import { LabelMetadataTransformer, LabelsMetadataTransformer } from './decomposeLabelsTransformer'; Messages.importMessagesDirectory(__dirname); const messages = Messages.loadMessages('@salesforce/source-deploy-retrieve', 'sdr'); @@ -32,15 +32,19 @@ export class MetadataTransformerFactory { const type = component.parent ? component.parent.type : component.type; const transformerId = type.strategies?.transformer; switch (transformerId) { - case TransformerStrategy.Standard: + case 'standard': case undefined: return new DefaultMetadataTransformer(this.registry, this.context); - case TransformerStrategy.Decomposed: + case 'decomposed': return new DecomposedMetadataTransformer(this.registry, this.context); - case TransformerStrategy.StaticResource: + case 'staticResource': return new StaticResourceMetadataTransformer(this.registry, this.context); - case TransformerStrategy.NonDecomposed: + case 'nonDecomposed': return new NonDecomposedMetadataTransformer(this.registry, this.context); + case 'decomposedLabels': + return component.type.name === 'CustomLabels' + ? new LabelsMetadataTransformer(this.registry, this.context) + : new LabelMetadataTransformer(this.registry, this.context); default: throw messages.createError('error_missing_transformer', [type.name, transformerId]); } diff --git a/src/convert/transformers/nonDecomposedMetadataTransformer.ts b/src/convert/transformers/nonDecomposedMetadataTransformer.ts index 94c53ef60e..6112399f7d 100644 --- a/src/convert/transformers/nonDecomposedMetadataTransformer.ts +++ b/src/convert/transformers/nonDecomposedMetadataTransformer.ts @@ -8,8 +8,7 @@ import { get, getString, JsonMap } from '@salesforce/ts-types'; import { ensureArray } from '@salesforce/kit'; import { Messages } from '@salesforce/core'; -import { WriteInfo } from '../types'; -import { SourceComponent } from '../../resolve/sourceComponent'; +import { ToSourceFormatInput, WriteInfo } from '../types'; import { DecomposedMetadataTransformer } from './decomposedMetadataTransformer'; Messages.importMessagesDirectory(__dirname); const messages = Messages.loadMessages('@salesforce/source-deploy-retrieve', 'sdr'); @@ -22,7 +21,7 @@ const messages = Messages.loadMessages('@salesforce/source-deploy-retrieve', 'sd export class NonDecomposedMetadataTransformer extends DecomposedMetadataTransformer { // streams uses mergeWith for all types. Removing it would break the interface // eslint-disable-next-line @typescript-eslint/no-unused-vars - public async toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise { + public async toSourceFormat({ component, mergeWith }: ToSourceFormatInput): Promise { // this will only include the incoming (retrieved) labels, not the local file const parentXml = await component.parseXml(); const xmlPathToChildren = `${component.type.name}.${component.type.directoryName}`; diff --git a/src/convert/transformers/staticResourceMetadataTransformer.ts b/src/convert/transformers/staticResourceMetadataTransformer.ts index 668a1eeb70..d5ca4b537e 100644 --- a/src/convert/transformers/staticResourceMetadataTransformer.ts +++ b/src/convert/transformers/staticResourceMetadataTransformer.ts @@ -13,7 +13,7 @@ import { createWriteStream } from 'graceful-fs'; import { Logger, Messages, SfError } from '@salesforce/core'; import { isEmpty } from '@salesforce/kit'; import { baseName } from '../../utils/path'; -import { WriteInfo } from '../types'; +import { ToSourceFormatInput, WriteInfo } from '../types'; import { SourceComponent } from '../../resolve/sourceComponent'; import { SourcePath } from '../../common/types'; import { ensureFileExists } from '../../utils/fileSystemHandler'; @@ -97,7 +97,7 @@ export class StaticResourceMetadataTransformer extends BaseMetadataTransformer { ]; } - public async toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise { + public async toSourceFormat({ component, mergeWith }: ToSourceFormatInput): Promise { const { xml, content } = component; if (!content) { diff --git a/src/convert/types.ts b/src/convert/types.ts index e0e790a5bf..36f30d69e1 100644 --- a/src/convert/types.ts +++ b/src/convert/types.ts @@ -6,6 +6,7 @@ */ import { Readable } from 'node:stream'; import { JsonMap } from '@salesforce/ts-types'; +import { ComponentSet } from '../collections/componentSet'; import { XML_NS_KEY, XML_NS_URL } from '../common/constants'; import { FileResponseSuccess } from '../client/types'; import { SourcePath } from '../common/types'; @@ -74,13 +75,19 @@ export type MergeConfig = { forceIgnoredPaths?: Set; }; +export type ToSourceFormatInput = { + component: SourceComponent; + mergeWith?: SourceComponent; + mergeSet?: ComponentSet; +}; +export type ToSourceFormat = (input: ToSourceFormatInput) => Promise; /** * Transforms metadata component files into different SFDX file formats */ export type MetadataTransformer = { defaultDirectory?: string; + toSourceFormat: ToSourceFormat; toMetadataFormat(component: SourceComponent): Promise; - toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise; }; // -------------- diff --git a/src/index.ts b/src/index.ts index 550b1ef50b..2e6e92ac42 100644 --- a/src/index.ts +++ b/src/index.ts @@ -90,15 +90,10 @@ export { FromSourceOptions, FromManifestOptions, } from './collections'; -export { - RegistryAccess, - registry, - getCurrentApiVersion, - MetadataRegistry, - MetadataType, - DecompositionStrategy, - RecompositionStrategy, - TransformerStrategy, -} from './registry'; + +export { RegistryAccess, registry, getCurrentApiVersion, MetadataRegistry, MetadataType } from './registry'; + +// TODO: don't export these strategies +export { DecompositionStrategy, TransformerStrategy, RecompositionStrategy } from './registry/types'; export { presetMap } from './registry/presets/presetMap'; diff --git a/src/registry/index.ts b/src/registry/index.ts index f214188211..d1f892d998 100644 --- a/src/registry/index.ts +++ b/src/registry/index.ts @@ -8,10 +8,4 @@ export { registry } from './registry'; export { standardValueSet } from './standardvalueset'; export { RegistryAccess } from './registryAccess'; export { getCurrentApiVersion } from './coverage'; -export { - MetadataRegistry, - MetadataType, - DecompositionStrategy, - RecompositionStrategy, - TransformerStrategy, -} from './types'; +export { MetadataRegistry, MetadataType } from './types'; diff --git a/src/registry/presets/decomposeCustomLabelsBeta2.json b/src/registry/presets/decomposeCustomLabelsBeta2.json new file mode 100644 index 0000000000..aee6b9bf9f --- /dev/null +++ b/src/registry/presets/decomposeCustomLabelsBeta2.json @@ -0,0 +1,28 @@ +{ + "suffixes": { + "label": "customlabel", + "labels": "customlabels" + }, + "types": { + "customlabel": { + "directoryName": "labels", + "id": "customlabel", + "name": "CustomLabel", + "strategies": { + "adapter": "default", + "transformer": "label" + }, + "suffix": "label" + }, + "customlabels": { + "directoryName": "labels", + "id": "customlabels", + "name": "CustomLabels", + "strategies": { + "adapter": "default", + "transformer": "labels" + }, + "suffix": "labels" + } + } +} diff --git a/src/registry/types.ts b/src/registry/types.ts index 034b7ce9ca..a0c4bb8710 100644 --- a/src/registry/types.ts +++ b/src/registry/types.ts @@ -140,7 +140,7 @@ export type MetadataType = { */ strategies?: { adapter: 'mixedContent' | 'matchingContentFile' | 'decomposed' | 'digitalExperience' | 'bundle' | 'default'; - transformer?: 'decomposed' | 'staticResource' | 'nonDecomposed' | 'standard'; + transformer?: 'decomposed' | 'staticResource' | 'nonDecomposed' | 'standard' | 'decomposedLabels'; decomposition?: 'topLevel' | 'folderPerType'; recomposition?: 'startEmpty'; }; @@ -168,6 +168,7 @@ type DirectoryIndex = { }; /** + * @deprecated. See the strategies union type on the registry types for the valid names * Strategy names for handling component decomposition. */ export const enum DecompositionStrategy { @@ -182,6 +183,7 @@ export const enum DecompositionStrategy { } /** + * @deprecated. See the strategies union type on the registry types for the valid names * Strategy names for handling component recomposition. */ export const enum RecompositionStrategy { @@ -192,6 +194,7 @@ export const enum RecompositionStrategy { } /** + * @deprecated. See the strategies union type on the registry types for the valid names * Strategy names for the type of transformation to use for metadata types. */ export const enum TransformerStrategy { @@ -199,6 +202,7 @@ export const enum TransformerStrategy { Decomposed = 'decomposed', StaticResource = 'staticResource', NonDecomposed = 'nonDecomposed', + DecomposedLabels = 'decomposedLabels', } type Channel = { diff --git a/test/convert/streams.test.ts b/test/convert/streams.test.ts index c93067b8c8..dc2d07e876 100644 --- a/test/convert/streams.test.ts +++ b/test/convert/streams.test.ts @@ -12,6 +12,7 @@ import { Logger, SfError, Messages } from '@salesforce/core'; import { expect, assert } from 'chai'; import { createSandbox, SinonStub } from 'sinon'; import JSZip from 'jszip'; +import { ToSourceFormatInput } from '../../src/convert/types'; import * as streams from '../../src/convert/streams'; import * as fsUtil from '../../src/utils/fileSystemHandler'; import { ComponentSet, MetadataResolver, RegistryAccess, SourceComponent, WriteInfo, WriterFormat } from '../../src'; @@ -35,7 +36,7 @@ class TestTransformer extends BaseMetadataTransformer { } // partial implementation only for tests // eslint-disable-next-line class-methods-use-this, @typescript-eslint/require-await - public async toSourceFormat(component: SourceComponent, mergeWith?: SourceComponent): Promise { + public async toSourceFormat({ mergeWith }: ToSourceFormatInput): Promise { const output = mergeWith ? mergeWith.content ?? mergeWith.xml : '/type/file.s'; assert(output); return [{ output, source: new Readable() }]; @@ -106,7 +107,7 @@ describe('Streams', () => { expect(err).to.be.undefined; expect(data).to.deep.equal({ component, - writeInfos: await transformer.toSourceFormat(component), + writeInfos: await transformer.toSourceFormat({ component }), }); done(); } catch (e) { @@ -133,7 +134,7 @@ describe('Streams', () => { expect(err).to.be.undefined; expect(data).to.deep.equal({ component: newComponent, - writeInfos: await transformer.toSourceFormat(newComponent, component), + writeInfos: await transformer.toSourceFormat({ component: newComponent, mergeWith: component }), }); done(); } catch (e) { @@ -162,8 +163,8 @@ describe('Streams', () => { expect(err).to.be.undefined; expect(data).to.deep.equal({ component: newComponent, - writeInfos: (await transformer.toSourceFormat(newComponent, component)).concat( - await transformer.toSourceFormat(newComponent, secondMergeComponent) + writeInfos: (await transformer.toSourceFormat({ component: newComponent, mergeWith: component })).concat( + await transformer.toSourceFormat({ component: newComponent, mergeWith: secondMergeComponent }) ), }); done(); diff --git a/test/convert/transformers/decomposedMetadataTransformer.test.ts b/test/convert/transformers/decomposedMetadataTransformer.test.ts index 1d48d4bfc1..980e2ca4e6 100644 --- a/test/convert/transformers/decomposedMetadataTransformer.test.ts +++ b/test/convert/transformers/decomposedMetadataTransformer.test.ts @@ -132,7 +132,7 @@ describe('DecomposedMetadataTransformer', () => { }, }); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(context.decomposition.transactionState.size).to.equal(0); expect(result).to.deep.equal([ @@ -193,7 +193,7 @@ describe('DecomposedMetadataTransformer', () => { ) .returns(false); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(context.decomposition.transactionState.size).to.equal(0); expect(result).to.deep.equal([ @@ -239,7 +239,7 @@ describe('DecomposedMetadataTransformer', () => { }, }); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(result).to.deep.equal([ { @@ -277,7 +277,7 @@ describe('DecomposedMetadataTransformer', () => { }, }); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(result).to.deep.equal([ { @@ -327,7 +327,7 @@ describe('DecomposedMetadataTransformer', () => { }, }); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(result).to.deep.equal([ { @@ -378,7 +378,7 @@ describe('DecomposedMetadataTransformer', () => { }, }); - const result = await transformer.toSourceFormat(cft, cot); + const result = await transformer.toSourceFormat({ component: cft, mergeWith: cot }); expect(result).to.deep.equal([ { @@ -417,7 +417,7 @@ describe('DecomposedMetadataTransformer', () => { const transformer = new DecomposedMetadataTransformer(); $$.SANDBOX.stub(component, 'parseXml').resolves({}); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(result).to.be.an('array').with.lengthOf(1); // there will be a file produced, with just the outer type (ex: CustomObject) and the xmlns declaration @@ -446,7 +446,7 @@ describe('DecomposedMetadataTransformer', () => { .withArgs(join('main', 'default', 'objects', 'customObject__c', 'customObject__c.object-meta.xml')) .returns(true); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(result).to.deep.equal([]); }); @@ -472,7 +472,7 @@ describe('DecomposedMetadataTransformer', () => { }); const transformer = new DecomposedMetadataTransformer(registryAccess); - const result = await transformer.toSourceFormat(componentToConvert, component); + const result = await transformer.toSourceFormat({ component: componentToConvert, mergeWith: component }); expect(result).to.deep.equal([ { @@ -509,7 +509,7 @@ describe('DecomposedMetadataTransformer', () => { }); const transformer = new DecomposedMetadataTransformer(); - const result = await transformer.toSourceFormat(componentToConvert, component); + const result = await transformer.toSourceFormat({ component: componentToConvert, mergeWith: component }); expect(result).to.deep.equal([ { @@ -556,7 +556,7 @@ describe('DecomposedMetadataTransformer', () => { try { // NOTE: it doesn't matter what the first component is for this test since it's all // about the child components of the parentComponent. - await transformer.toSourceFormat(component, parentComponent); + await transformer.toSourceFormat({ component, mergeWith: parentComponent }); assert(false, 'expected TypeInferenceError to be thrown'); } catch (err) { assert(err instanceof Error); @@ -590,7 +590,7 @@ describe('DecomposedMetadataTransformer', () => { const context = new ConvertContext(); const transformer = new DecomposedMetadataTransformer(registryAccess, context); - const result = await transformer.toSourceFormat(component, componentToMerge); + const result = await transformer.toSourceFormat({ component, mergeWith: componentToMerge }); expect(result).to.be.empty; expect( context.decomposition.transactionState.get(`${mergeComponentChild.type.name}#${mergeComponentChild.fullName}`) @@ -616,7 +616,7 @@ describe('DecomposedMetadataTransformer', () => { it('should defer write operation for parent xml that is not a member of merge component', async () => { const { fullName, type } = component; const root = join('main', 'default', type.directoryName, fullName); - const componentToMerge = SourceComponent.createVirtualComponent( + const mergeWith = SourceComponent.createVirtualComponent( { name: 'a', type, @@ -633,7 +633,7 @@ describe('DecomposedMetadataTransformer', () => { const context = new ConvertContext(); const transformer = new DecomposedMetadataTransformer(registryAccess, context); - const result = await transformer.toSourceFormat(component, componentToMerge); + const result = await transformer.toSourceFormat({ component, mergeWith }); expect(result).to.be.empty; expect(context.decomposition.transactionState.get(`${type.name}#${fullName}`)).to.deep.equal({ origin: component, diff --git a/test/convert/transformers/defaultMetadataTransformer.test.ts b/test/convert/transformers/defaultMetadataTransformer.test.ts index 67e7572462..fc5e25e702 100644 --- a/test/convert/transformers/defaultMetadataTransformer.test.ts +++ b/test/convert/transformers/defaultMetadataTransformer.test.ts @@ -197,7 +197,7 @@ describe('DefaultMetadataTransformer', () => { source: component.tree.stream(component.xml), }); - expect(await transformer.toSourceFormat(component)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equal(expectedInfos); }); it('should add in the -meta.xml suffix for components with no content', async () => { @@ -213,7 +213,7 @@ describe('DefaultMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equal(expectedInfos); }); it('should handle components in folders with no content', async () => { @@ -234,7 +234,7 @@ describe('DefaultMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equal(expectedInfos); }); it('should not remove file extension and preserve -meta.xml for DigitalExperienceBundle', async () => { @@ -263,7 +263,7 @@ describe('DefaultMetadataTransformer', () => { ), }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equal(expectedInfos); }); it('should handle folder components', async () => { @@ -284,7 +284,7 @@ describe('DefaultMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equal(expectedInfos); }); it('should merge output with merge component when content is a directory', async () => { @@ -322,7 +322,7 @@ describe('DefaultMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component, mergeWith)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component, mergeWith })).to.deep.equal(expectedInfos); }); it('should merge output with merge component when content is a file', async () => { @@ -356,7 +356,7 @@ describe('DefaultMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component, mergeWith)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component, mergeWith })).to.deep.equal(expectedInfos); }); it('should use merge component xml path', async () => { @@ -371,7 +371,7 @@ describe('DefaultMetadataTransformer', () => { [] ); assert(typeof component.xml === 'string'); - expect(await transformer.toSourceFormat(component, mergeWith)).to.deep.contain({ + expect(await transformer.toSourceFormat({ component, mergeWith })).to.deep.contain({ output: mergeWith.xml, source: component.tree.stream(component.xml), }); @@ -388,7 +388,7 @@ describe('DefaultMetadataTransformer', () => { ); assert(typeof component.xml === 'string'); - expect(await transformer.toSourceFormat(component, mergeWith)).to.deep.contain({ + expect(await transformer.toSourceFormat({ component, mergeWith })).to.deep.contain({ output: component.getPackageRelativePath(component.xml, 'source'), source: component.tree.stream(component.xml), }); @@ -411,7 +411,7 @@ describe('DefaultMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equal(expectedInfos); }); }); }); diff --git a/test/convert/transformers/nonDecomposedMetadataTransformer.test.ts b/test/convert/transformers/nonDecomposedMetadataTransformer.test.ts index 90b2219a29..faf7aa796b 100644 --- a/test/convert/transformers/nonDecomposedMetadataTransformer.test.ts +++ b/test/convert/transformers/nonDecomposedMetadataTransformer.test.ts @@ -40,7 +40,7 @@ describe('NonDecomposedMetadataTransformer', () => { const context = new ConvertContext(); const transformer = new NonDecomposedMetadataTransformer(registryAccess, context); - const result = await transformer.toSourceFormat(component); + const result = await transformer.toSourceFormat({ component }); expect(result).to.deep.equal([]); expect(context.decomposition.transactionState).to.deep.equal(new Map()); expect(context.recomposition.transactionState).to.deep.equal(new Map()); @@ -68,7 +68,7 @@ describe('NonDecomposedMetadataTransformer', () => { $$.SANDBOX.stub(componentToConvert, 'parseXml').resolves(nonDecomposed.FULL_XML_CONTENT); $$.SANDBOX.stub(componentToConvert, 'parseXmlSync').returns(nonDecomposed.FULL_XML_CONTENT); - const result = await transformer.toSourceFormat(componentToConvert, component); + const result = await transformer.toSourceFormat({ component: componentToConvert, mergeWith: component }); expect(result).to.deep.equal([]); expect(context.nonDecomposition.transactionState).to.deep.equal({ childrenByUniqueElement: new Map([ diff --git a/test/convert/transformers/staticResourceMetadataTransformer.test.ts b/test/convert/transformers/staticResourceMetadataTransformer.test.ts index f7f4043fb6..0e5b5d322e 100644 --- a/test/convert/transformers/staticResourceMetadataTransformer.test.ts +++ b/test/convert/transformers/staticResourceMetadataTransformer.test.ts @@ -222,7 +222,7 @@ describe('StaticResourceMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equalInAnyOrder(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equalInAnyOrder(expectedInfos); }); it('should rename extension from .resource for a fallback mime extension', async () => { @@ -247,7 +247,7 @@ describe('StaticResourceMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equalInAnyOrder(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equalInAnyOrder(expectedInfos); }); it('should rename extension from .resource for an unsupported mime extension', async () => { @@ -273,14 +273,14 @@ describe('StaticResourceMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equalInAnyOrder(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equalInAnyOrder(expectedInfos); }); it('should ignore components without content', async () => { const component = Object.assign({}, mixedContentSingleFile.COMPONENT); component.content = undefined; - expect(await transformer.toSourceFormat(component)).to.deep.equal([]); + expect(await transformer.toSourceFormat({ component })).to.deep.equal([]); }); it('should extract an archive', async () => { @@ -306,7 +306,7 @@ describe('StaticResourceMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equalInAnyOrder(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equalInAnyOrder(expectedInfos); expect(pipelineStub.callCount).to.equal(1); expect(pipelineStub.firstCall.args[1]).to.equal( join( @@ -340,7 +340,7 @@ describe('StaticResourceMetadataTransformer', () => { }, ]; - expect(await transformer.toSourceFormat(component)).to.deep.equalInAnyOrder(expectedInfos); + expect(await transformer.toSourceFormat({ component })).to.deep.equalInAnyOrder(expectedInfos); }); it('should merge output with merge component when content is archive', async () => { @@ -349,7 +349,7 @@ describe('StaticResourceMetadataTransformer', () => { assert(component.xml); assert(typeof transformer.defaultDirectory === 'string'); - const mergeComponent = SourceComponent.createVirtualComponent( + const mergeWith = SourceComponent.createVirtualComponent( { name: mixedContentSingleFile.COMPONENT.name, type: registry.types.staticresource, @@ -367,8 +367,8 @@ describe('StaticResourceMetadataTransformer', () => { }, ] ); - assert(mergeComponent.xml); - assert(mergeComponent.content); + assert(mergeWith.xml); + assert(mergeWith.content); env.stub(component, 'parseXml').resolves({ StaticResource: { contentType: 'application/zip', @@ -382,14 +382,14 @@ describe('StaticResourceMetadataTransformer', () => { const expectedInfos: WriteInfo[] = [ { source: component.tree.stream(component.xml), - output: mergeComponent.xml, + output: mergeWith.xml, }, ]; - expect(await transformer.toSourceFormat(component, mergeComponent)).to.deep.equal(expectedInfos); + expect(await transformer.toSourceFormat({ component, mergeWith })).to.deep.equal(expectedInfos); expect(pipelineStub.callCount).to.equal(1); expect(pipelineStub.firstCall.args[1]).to.deep.equal( - join(transformer.defaultDirectory, mergeComponent.content, 'b', 'c.css') + join(transformer.defaultDirectory, mergeWith.content, 'b', 'c.css') ); }); @@ -398,7 +398,7 @@ describe('StaticResourceMetadataTransformer', () => { const component = mixedContentSingleFile.COMPONENT; assert(component.content); assert(component.xml); - const mergeComponent = SourceComponent.createVirtualComponent( + const mergeWith = SourceComponent.createVirtualComponent( { name: mixedContentSingleFile.COMPONENT.name, type: registry.types.staticresource, @@ -416,7 +416,7 @@ describe('StaticResourceMetadataTransformer', () => { }, ] ); - assert(mergeComponent.xml); + assert(mergeWith.xml); env.stub(component, 'parseXml').resolves({ StaticResource: { @@ -426,15 +426,15 @@ describe('StaticResourceMetadataTransformer', () => { const expectedInfos: WriteInfo[] = [ { source: component.tree.stream(component.content), - output: `${mergeComponent.content}.txt`, + output: `${mergeWith.content}.txt`, }, { source: component.tree.stream(component.xml), - output: mergeComponent.xml, + output: mergeWith.xml, }, ]; - expect(await transformer.toSourceFormat(component, mergeComponent)).to.deep.equalInAnyOrder(expectedInfos); + expect(await transformer.toSourceFormat({ component, mergeWith })).to.deep.equalInAnyOrder(expectedInfos); }); }); }); diff --git a/test/registry/registryValidation.test.ts b/test/registry/registryValidation.test.ts index 304fdf18b3..230d28493a 100644 --- a/test/registry/registryValidation.test.ts +++ b/test/registry/registryValidation.test.ts @@ -5,7 +5,7 @@ * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause */ import { assert, expect } from 'chai'; -import { DecompositionStrategy, MetadataType, TransformerStrategy } from '../../src/registry/types'; +import { MetadataType } from '../../src/registry/types'; import { metadataTypes as UnsupportedTypes } from '../../src/registry/nonSupportedTypes'; import { presets } from './presetTesting'; @@ -324,20 +324,13 @@ for (const preset of presets) { .forEach((type) => { it(`${type.id} has expected properties`, () => { assert(typeof type.strategies?.decomposition === 'string'); - expect( - [DecompositionStrategy.FolderPerType.valueOf(), DecompositionStrategy.TopLevel.valueOf()].includes( - type.strategies.decomposition - ) - ).to.be.true; + expect(['folderPerType', 'topLevel'].includes(type.strategies.decomposition)).to.be.true; assert(typeof type.strategies?.transformer === 'string'); expect( - [ - TransformerStrategy.Standard.valueOf(), - TransformerStrategy.Decomposed.valueOf(), - TransformerStrategy.StaticResource.valueOf(), - TransformerStrategy.NonDecomposed.valueOf(), - ].includes(type.strategies.transformer) + ['decomposed', 'nondecomposed', 'standard', 'staticResource', 'decomposedLabels'].includes( + type.strategies.transformer + ) ).to.be.true; expect(type.strategies.recomposition).to.be.undefined; }); From 5065e62ba418ba0edcf472769cefe04149c09fc5 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 12:36:09 -0500 Subject: [PATCH 02/30] test: move existing snapshot project --- .../labels/CustomLabels.labels | 24 ++++++++ .../verify-md-files.expected/package.xml | 8 +++ .../CustomLabels/CustomLabels.labels-meta.xml | 2 + .../CustomLabels/DeleteMe.label-meta.xml | 8 +++ .../CustomLabels/KeepMe1.label-meta.xml | 8 +++ .../CustomLabels/KeepMe2.label-meta.xml | 8 +++ .../originalMdapi/labels/CustomLabels.labels | 24 ++++++++ .../originalMdapi/package.xml | 14 +++++ .../sfdx-project.json | 13 ++++ .../snapshots.test.ts | 59 +++++++++++++++++++ 10 files changed, 168 insertions(+) create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels new file mode 100644 index 0000000000..d57a2391aa --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels @@ -0,0 +1,24 @@ + + + + DeleteMe + en_US + true + DeleteMe + Test + + + KeepMe1 + en_US + true + KeepMe1 + Test + + + KeepMe2 + en_US + true + KeepMe2 + Test + + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml new file mode 100644 index 0000000000..51950dd57c --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml @@ -0,0 +1,8 @@ + + + + CustomLabels + CustomLabels + + 60.0 + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml new file mode 100644 index 0000000000..6d9520ca90 --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml @@ -0,0 +1,2 @@ + + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml new file mode 100644 index 0000000000..14bf9da669 --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml @@ -0,0 +1,8 @@ + + + DeleteMe + en_US + true + DeleteMe + Test + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml new file mode 100644 index 0000000000..9a78141d6e --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml @@ -0,0 +1,8 @@ + + + KeepMe1 + en_US + true + KeepMe1 + Test + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml new file mode 100644 index 0000000000..64ab9ef9b3 --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml @@ -0,0 +1,8 @@ + + + KeepMe2 + en_US + true + KeepMe2 + Test + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels new file mode 100644 index 0000000000..d57a2391aa --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels @@ -0,0 +1,24 @@ + + + + DeleteMe + en_US + true + DeleteMe + Test + + + KeepMe1 + en_US + true + KeepMe1 + Test + + + KeepMe2 + en_US + true + KeepMe2 + Test + + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml new file mode 100644 index 0000000000..76c38d0d62 --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml @@ -0,0 +1,14 @@ + + + + DeleteMe + KeepMe1 + KeepMe2 + CustomLabel + + + CustomLabels + CustomLabels + + 57.0 + diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json new file mode 100644 index 0000000000..97d16c7028 --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json @@ -0,0 +1,13 @@ +{ + "name": "variantTest", + "namespace": "", + "packageDirectories": [ + { + "default": true, + "path": "force-app" + } + ], + "sourceBehaviorOptions": ["decomposeCustomLabelsBeta"], + "sfdcLoginUrl": "https://login.salesforce.com", + "sourceApiVersion": "60.0" +} diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts new file mode 100644 index 0000000000..fbb625b584 --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts @@ -0,0 +1,59 @@ +/* + * Copyright (c) 2023, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ +import * as fs from 'node:fs'; +import * as path from 'node:path'; +import { + compareTwoXml, + fileSnap, + mdapiToSource, + sourceToMdapi, + MDAPI_OUT, + dirsAreIdentical, + FORCE_APP, +} from '../../helper/conversions'; + +// we don't want failing tests outputting over each other +/* eslint-disable no-await-in-loop */ + +describe('decomposed custom labels', () => { + const testDir = path.join('test', 'snapshot', 'sampleProjects', 'preset-decomposeLabels_Deprecated'); + let sourceFiles: string[]; + let mdFiles: string[]; + + before(async () => { + sourceFiles = await mdapiToSource(testDir); + mdFiles = await sourceToMdapi(testDir); + }); + it('verify source files', async () => { + for (const file of sourceFiles) { + await fileSnap(file, testDir); + } + dirsAreIdentical( + path.join(testDir, FORCE_APP), + path.join(testDir, '__snapshots__', 'verify-source-files.expected', FORCE_APP) + ); + }); + it('verify md files', async () => { + for (const file of mdFiles) { + await fileSnap(file, testDir); + } + }); + it('round trip of metadata format is equivalent', async () => { + const [old, updated] = await Promise.all([ + fs.promises.readFile(path.join(testDir, 'originalMdapi', 'labels', 'CustomLabels.labels'), 'utf8'), + fs.promises.readFile(path.join(testDir, MDAPI_OUT, 'labels', 'CustomLabels.labels'), 'utf8'), + ]); + compareTwoXml(old, updated); + }); + + after(async () => { + await Promise.all([ + fs.promises.rm(path.join(testDir, FORCE_APP), { recursive: true, force: true }), + fs.promises.rm(path.join(testDir, MDAPI_OUT), { recursive: true, force: true }), + ]); + }); +}); From 951f533dba3457e7c93bcf3b18a11b4c5f20061c Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 12:36:36 -0500 Subject: [PATCH 03/30] fix: things snapshots uncovered --- src/convert/transformers/decomposeLabelsTransformer.ts | 5 +++-- src/registry/presets/decomposeCustomLabelsBeta2.json | 8 ++++++-- src/registry/presets/presetMap.ts | 2 ++ 3 files changed, 11 insertions(+), 4 deletions(-) diff --git a/src/convert/transformers/decomposeLabelsTransformer.ts b/src/convert/transformers/decomposeLabelsTransformer.ts index b870dd1f6e..df9d5c57b7 100644 --- a/src/convert/transformers/decomposeLabelsTransformer.ts +++ b/src/convert/transformers/decomposeLabelsTransformer.ts @@ -5,12 +5,13 @@ * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause */ -import type { CustomLabels, CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; +import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; import { SfError } from '@salesforce/core/sfError'; import { calculateRelativePath } from '../../utils/path'; import { SourceComponent } from '../../resolve/sourceComponent'; import { ToSourceFormatInput, WriteInfo } from '../types'; import { JsToXml } from '../streams'; +import { unwrapAndOmitNS } from '../../utils/decomposed'; import { DefaultMetadataTransformer } from './defaultMetadataTransformer'; /* Use for the metadata type CustomLabels */ @@ -21,7 +22,7 @@ export class LabelsMetadataTransformer extends DefaultMetadataTransformer { const partiallyAppliedPathCalculator = calculateRelativePath('source')({ self: labelType, }); - const xml = await component.parseXml(); + const xml = unwrapAndOmitNS('CustomLabels')(await component.parseXml()) as { labels: CustomLabel[] }; // split each label into a separate label file return xml.labels.filter(customLabelHasFullName).map((l) => ({ output: diff --git a/src/registry/presets/decomposeCustomLabelsBeta2.json b/src/registry/presets/decomposeCustomLabelsBeta2.json index aee6b9bf9f..ea65af0b74 100644 --- a/src/registry/presets/decomposeCustomLabelsBeta2.json +++ b/src/registry/presets/decomposeCustomLabelsBeta2.json @@ -1,4 +1,8 @@ { + "childTypes": { + "customlabel": "" + }, + "strictDirectoryNames": {}, "suffixes": { "label": "customlabel", "labels": "customlabels" @@ -10,7 +14,7 @@ "name": "CustomLabel", "strategies": { "adapter": "default", - "transformer": "label" + "transformer": "decomposedLabels" }, "suffix": "label" }, @@ -20,7 +24,7 @@ "name": "CustomLabels", "strategies": { "adapter": "default", - "transformer": "labels" + "transformer": "decomposedLabels" }, "suffix": "labels" } diff --git a/src/registry/presets/presetMap.ts b/src/registry/presets/presetMap.ts index 2a0353bb68..9a195c52ea 100644 --- a/src/registry/presets/presetMap.ts +++ b/src/registry/presets/presetMap.ts @@ -8,12 +8,14 @@ import { MetadataRegistry } from '../types'; // we have to import all presets explicitly for VSCE's esbuild bundling process import * as decomposeCustomLabelsBeta from './decomposeCustomLabelsBeta.json'; +import * as decomposeCustomLabelsBeta2 from './decomposeCustomLabelsBeta2.json'; import * as decomposePermissionSetBeta from './decomposePermissionSetBeta.json'; import * as decomposeSharingRulesBeta from './decomposeSharingRulesBeta.json'; import * as decomposeWorkflowBeta from './decomposeWorkflowBeta.json'; export const presetMap = new Map([ ['decomposeCustomLabelsBeta', decomposeCustomLabelsBeta as MetadataRegistry], + ['decomposeCustomLabelsBeta2', decomposeCustomLabelsBeta2 as MetadataRegistry], ['decomposePermissionSetBeta', decomposePermissionSetBeta as MetadataRegistry], ['decomposeSharingRulesBeta', decomposeSharingRulesBeta as MetadataRegistry], ['decomposeWorkflowBeta', decomposeWorkflowBeta as MetadataRegistry], From 38ce600abdb7014fa8b74d41fbe290ae8c12422f Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 12:38:32 -0500 Subject: [PATCH 04/30] test: snapshot for mdapi => source (simple) --- .../labels/CustomLabels.labels | 24 ------------------- .../verify-md-files.expected/package.xml | 6 +++-- .../CustomLabels/CustomLabels.labels-meta.xml | 2 -- .../DeleteMe.label-meta.xml | 2 +- .../{CustomLabels => }/KeepMe1.label-meta.xml | 2 +- .../{CustomLabels => }/KeepMe2.label-meta.xml | 2 +- .../preset-decomposeLabels/sfdx-project.json | 4 ++-- 7 files changed, 9 insertions(+), 33 deletions(-) delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml rename test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/{CustomLabels => }/DeleteMe.label-meta.xml (78%) rename test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/{CustomLabels => }/KeepMe1.label-meta.xml (78%) rename test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/{CustomLabels => }/KeepMe2.label-meta.xml (78%) diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels deleted file mode 100644 index d57a2391aa..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels +++ /dev/null @@ -1,24 +0,0 @@ - - - - DeleteMe - en_US - true - DeleteMe - Test - - - KeepMe1 - en_US - true - KeepMe1 - Test - - - KeepMe2 - en_US - true - KeepMe2 - Test - - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/package.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/package.xml index 51950dd57c..eec5a245c9 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/package.xml +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/package.xml @@ -1,8 +1,10 @@ - CustomLabels - CustomLabels + DeleteMe + KeepMe1 + KeepMe2 + CustomLabel 60.0 diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml deleted file mode 100644 index 6d9520ca90..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml +++ /dev/null @@ -1,2 +0,0 @@ - - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/DeleteMe.label-meta.xml similarity index 78% rename from test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml rename to test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/DeleteMe.label-meta.xml index 14bf9da669..95a5f3c818 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/DeleteMe.label-meta.xml @@ -1,5 +1,5 @@ - + DeleteMe en_US true diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/KeepMe1.label-meta.xml similarity index 78% rename from test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml rename to test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/KeepMe1.label-meta.xml index 9a78141d6e..41a576cdc9 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/KeepMe1.label-meta.xml @@ -1,5 +1,5 @@ - + KeepMe1 en_US true diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/KeepMe2.label-meta.xml similarity index 78% rename from test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml rename to test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/KeepMe2.label-meta.xml index 64ab9ef9b3..4c559666cb 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-source-files.expected/force-app/main/default/labels/KeepMe2.label-meta.xml @@ -1,5 +1,5 @@ - + KeepMe2 en_US true diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json b/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json index 97d16c7028..ea1b12c4e1 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json @@ -7,7 +7,7 @@ "path": "force-app" } ], - "sourceBehaviorOptions": ["decomposeCustomLabelsBeta"], "sfdcLoginUrl": "https://login.salesforce.com", - "sourceApiVersion": "60.0" + "sourceApiVersion": "60.0", + "sourceBehaviorOptions": ["decomposeCustomLabelsBeta2"] } From 8b807384407d3f32c405450c6bc024dce5f6de6a Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 13:19:53 -0500 Subject: [PATCH 05/30] test: expected md snapshot --- .../labels/CustomLabels.labels-meta.xml | 24 +++++++++++++++++++ 1 file changed, 24 insertions(+) create mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels-meta.xml diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels-meta.xml new file mode 100644 index 0000000000..d57a2391aa --- /dev/null +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels-meta.xml @@ -0,0 +1,24 @@ + + + + DeleteMe + en_US + true + DeleteMe + Test + + + KeepMe1 + en_US + true + KeepMe1 + Test + + + KeepMe2 + en_US + true + KeepMe2 + Test + + From e8676fbb4ee50a15aa201f401210aebeac2771bf Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 13:46:05 -0500 Subject: [PATCH 06/30] feat: source to madapi finalizer --- src/convert/convertContext/convertContext.ts | 3 +- .../decomposedCustomLabelFinalizer.ts | 70 +++++++++++++++++++ .../decomposeLabelsTransformer.ts | 13 ++-- ...ls.labels-meta.xml => CustomLabels.labels} | 0 4 files changed, 79 insertions(+), 7 deletions(-) create mode 100644 src/convert/convertContext/decomposedCustomLabelFinalizer.ts rename test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/{CustomLabels.labels-meta.xml => CustomLabels.labels} (100%) diff --git a/src/convert/convertContext/convertContext.ts b/src/convert/convertContext/convertContext.ts index de219b4a89..689906b6c0 100644 --- a/src/convert/convertContext/convertContext.ts +++ b/src/convert/convertContext/convertContext.ts @@ -9,7 +9,7 @@ import { RecompositionFinalizer } from './recompositionFinalizer'; import { NonDecompositionFinalizer } from './nonDecompositionFinalizer'; import { DecompositionFinalizer } from './decompositionFinalizer'; import { ConvertTransactionFinalizer } from './transactionFinalizer'; - +import { DecomposedCustomLabelsFinalizer } from './decomposedCustomLabelFinalizer'; /** * A state manager over the course of a single metadata conversion call. */ @@ -17,6 +17,7 @@ export class ConvertContext { public readonly decomposition = new DecompositionFinalizer(); public readonly recomposition = new RecompositionFinalizer(); public readonly nonDecomposition = new NonDecompositionFinalizer(); + public readonly decomposedLabels = new DecomposedCustomLabelsFinalizer(); // eslint-disable-next-line @typescript-eslint/require-await public async *executeFinalizers(defaultDirectory?: string): AsyncIterable { diff --git a/src/convert/convertContext/decomposedCustomLabelFinalizer.ts b/src/convert/convertContext/decomposedCustomLabelFinalizer.ts new file mode 100644 index 0000000000..b157cb8f5e --- /dev/null +++ b/src/convert/convertContext/decomposedCustomLabelFinalizer.ts @@ -0,0 +1,70 @@ +/* + * Copyright (c) 2023, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ +import { join } from 'node:path'; +import { ensure, JsonMap } from '@salesforce/ts-types'; +import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; +import { MetadataType } from '../../registry'; +import { XML_NS_KEY, XML_NS_URL } from '../../common/constants'; +import { JsToXml } from '../streams'; +import { WriterFormat } from '../types'; +import { ConvertTransactionFinalizer } from './transactionFinalizer'; + +type CustomLabelState = { + /* + * Incoming child xml (CustomLabel) keyed by label fullname + */ + customLabelByFullName: Map; +}; + +/** + * Merges child components that share the same parent in the conversion pipeline + * into a single file. + * + * Inserts unclaimed child components into the parent that belongs to the default package + */ +export class DecomposedCustomLabelsFinalizer extends ConvertTransactionFinalizer { + public transactionState: CustomLabelState = { + customLabelByFullName: new Map(), + }; + + /** to support custom presets (the only way this code should get hit at all pass in the type from a transformer that has registry access */ + public customLabelsType?: MetadataType; + + // have to maintain the existing interface + // eslint-disable-next-line @typescript-eslint/require-await, @typescript-eslint/no-unused-vars + public async finalize(defaultDirectory?: string): Promise { + if (this.transactionState.customLabelByFullName.size === 0) { + return []; + } + return [ + { + component: { + type: ensure(this.customLabelsType, 'DecomposedCustomLabelsFinalizer should have set customLabelsType'), + fullName: 'CustomLabels', + }, + writeInfos: [ + { + output: join( + ensure(this.customLabelsType?.directoryName, 'directoryName missing from customLabels type'), + 'CustomLabels.labels' + ), + source: new JsToXml(generateXml(this.transactionState.customLabelByFullName)), + }, + ], + }, + ]; + } +} + +/** Return a json object that's built up from the mergeMap children */ +const generateXml = (children: Map): JsonMap => ({ + ['CustomLabels']: { + [XML_NS_KEY]: XML_NS_URL, + // for CustomLabels, that's `labels` + labels: Array.from(children.values()), + }, +}); diff --git a/src/convert/transformers/decomposeLabelsTransformer.ts b/src/convert/transformers/decomposeLabelsTransformer.ts index df9d5c57b7..68a6ff0607 100644 --- a/src/convert/transformers/decomposeLabelsTransformer.ts +++ b/src/convert/transformers/decomposeLabelsTransformer.ts @@ -16,7 +16,7 @@ import { DefaultMetadataTransformer } from './defaultMetadataTransformer'; /* Use for the metadata type CustomLabels */ export class LabelsMetadataTransformer extends DefaultMetadataTransformer { - // CustomLabels file => CustomLabel + /** CustomLabels file => Array of CustomLabel WriteInfo (one for each label) */ public async toSourceFormat({ component, mergeSet }: ToSourceFormatInput): Promise { const labelType = this.registry.getTypeByName('CustomLabel'); const partiallyAppliedPathCalculator = calculateRelativePath('source')({ @@ -36,12 +36,13 @@ export class LabelsMetadataTransformer extends DefaultMetadataTransformer { /* Use for the metadata type CustomLabel */ export class LabelMetadataTransformer extends DefaultMetadataTransformer { - // eslint-disable-next-line @typescript-eslint/require-await, class-methods-use-this, @typescript-eslint/no-unused-vars public async toMetadataFormat(component: SourceComponent): Promise { - // TODO: - // read all each label from the recomposition state, regardless of parents - // merge them all to a single CustomLabels file - // write the CustomLabels file + // only need to do this once + this.context.decomposedLabels.customLabelsType ??= this.registry.getTypeByName('CustomLabels'); + this.context.decomposedLabels.transactionState.customLabelByFullName.set( + component.fullName, + unwrapAndOmitNS('CustomLabel')(await component.parseXml()) as CustomLabel + ); return []; } diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels similarity index 100% rename from test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels-meta.xml rename to test/snapshot/sampleProjects/preset-decomposeLabels/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels From 99ceb9e88d83d7a51794857a2af7664ef46af4e0 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 13:54:35 -0500 Subject: [PATCH 07/30] feat: sorted labels when recomposed --- .../convertContext/decomposedCustomLabelFinalizer.ts | 8 +++++++- src/convert/transformers/decomposeLabelsTransformer.ts | 9 +-------- src/utils/metadata.ts | 9 +++++++++ 3 files changed, 17 insertions(+), 9 deletions(-) diff --git a/src/convert/convertContext/decomposedCustomLabelFinalizer.ts b/src/convert/convertContext/decomposedCustomLabelFinalizer.ts index b157cb8f5e..5383cfc654 100644 --- a/src/convert/convertContext/decomposedCustomLabelFinalizer.ts +++ b/src/convert/convertContext/decomposedCustomLabelFinalizer.ts @@ -7,6 +7,7 @@ import { join } from 'node:path'; import { ensure, JsonMap } from '@salesforce/ts-types'; import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; +import { customLabelHasFullName } from '../../utils/metadata'; import { MetadataType } from '../../registry'; import { XML_NS_KEY, XML_NS_URL } from '../../common/constants'; import { JsToXml } from '../streams'; @@ -65,6 +66,11 @@ const generateXml = (children: Map): JsonMap => ({ ['CustomLabels']: { [XML_NS_KEY]: XML_NS_URL, // for CustomLabels, that's `labels` - labels: Array.from(children.values()), + labels: Array.from(children.values()).filter(customLabelHasFullName).sort(sortLabelsByFullName), }, }); + +type CustomLabelWithFullName = CustomLabel & { fullName: string }; + +const sortLabelsByFullName = (a: CustomLabelWithFullName, b: CustomLabelWithFullName): number => + a.fullName.localeCompare(b.fullName); diff --git a/src/convert/transformers/decomposeLabelsTransformer.ts b/src/convert/transformers/decomposeLabelsTransformer.ts index 68a6ff0607..552a4c3a83 100644 --- a/src/convert/transformers/decomposeLabelsTransformer.ts +++ b/src/convert/transformers/decomposeLabelsTransformer.ts @@ -6,7 +6,7 @@ */ import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; -import { SfError } from '@salesforce/core/sfError'; +import { customLabelHasFullName } from '../../utils/metadata'; import { calculateRelativePath } from '../../utils/path'; import { SourceComponent } from '../../resolve/sourceComponent'; import { ToSourceFormatInput, WriteInfo } from '../types'; @@ -48,10 +48,3 @@ export class LabelMetadataTransformer extends DefaultMetadataTransformer { // toSourceFormat uses the default (merge them with the existing label) } - -const customLabelHasFullName = (label: CustomLabel): label is CustomLabel & { fullName: string } => { - if (label.fullName === undefined) { - throw SfError.create({ message: 'Label does not have a fullName', data: label }); - } - return true; -}; diff --git a/src/utils/metadata.ts b/src/utils/metadata.ts index ad7cd4f92b..4fc6983dc2 100644 --- a/src/utils/metadata.ts +++ b/src/utils/metadata.ts @@ -5,6 +5,8 @@ * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause */ +import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; +import { SfError } from '@salesforce/core'; import { META_XML_SUFFIX } from '../common'; export function generateMetaXML(typeName: string, apiVersion: string, status: string): string { @@ -25,3 +27,10 @@ export function generateMetaXMLPath(sourcePath: string): string { export function trimMetaXmlSuffix(sourcePath: string): string { return sourcePath.endsWith(META_XML_SUFFIX) ? sourcePath.replace(META_XML_SUFFIX, '') : sourcePath; } + +export const customLabelHasFullName = (label: CustomLabel): label is CustomLabel & { fullName: string } => { + if (label.fullName === undefined) { + throw SfError.create({ message: 'Label does not have a fullName', data: label }); + } + return true; +}; From d78be22917282430837b7e8e8e2614479dc7d890 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 14:38:05 -0500 Subject: [PATCH 08/30] fix: remove original customLabelsBeta --- .../presets/decomposeCustomLabelsBeta.json | 44 +++++--------- .../presets/decomposeCustomLabelsBeta2.json | 32 ---------- src/registry/presets/presetMap.ts | 2 - test/registry/registryValidation.test.ts | 12 ++-- .../preset-decomposeLabels/sfdx-project.json | 2 +- .../labels/CustomLabels.labels | 24 -------- .../verify-md-files.expected/package.xml | 8 --- .../CustomLabels/CustomLabels.labels-meta.xml | 2 - .../CustomLabels/DeleteMe.label-meta.xml | 8 --- .../CustomLabels/KeepMe1.label-meta.xml | 8 --- .../CustomLabels/KeepMe2.label-meta.xml | 8 --- .../originalMdapi/labels/CustomLabels.labels | 24 -------- .../originalMdapi/package.xml | 14 ----- .../sfdx-project.json | 13 ---- .../snapshots.test.ts | 59 ------------------- 15 files changed, 24 insertions(+), 236 deletions(-) delete mode 100644 src/registry/presets/decomposeCustomLabelsBeta2.json delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json delete mode 100644 test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts diff --git a/src/registry/presets/decomposeCustomLabelsBeta.json b/src/registry/presets/decomposeCustomLabelsBeta.json index 4d121d4e63..ea65af0b74 100644 --- a/src/registry/presets/decomposeCustomLabelsBeta.json +++ b/src/registry/presets/decomposeCustomLabelsBeta.json @@ -1,44 +1,32 @@ { "childTypes": { - "customlabel": "customlabels" + "customlabel": "" }, + "strictDirectoryNames": {}, "suffixes": { - "label": "customlabel" - }, - "strictDirectoryNames": { + "label": "customlabel", "labels": "customlabels" }, "types": { - "customlabels": { - "children": { - "suffixes": { - "label": "customlabel" - }, - "types": { - "customlabel": { - "directoryName": "test", - "id": "customlabel", - "name": "CustomLabel", - "suffix": "label", - "isAddressable": false, - "supportsWildcardAndName": true, - "uniqueIdElement": "fullName", - "xmlElementName": "labels" - } - } + "customlabel": { + "directoryName": "labels", + "id": "customlabel", + "name": "CustomLabel", + "strategies": { + "adapter": "default", + "transformer": "decomposedLabels" }, - "strictDirectoryName": true, + "suffix": "label" + }, + "customlabels": { "directoryName": "labels", "id": "customlabels", - "ignoreParsedFullName": false, "name": "CustomLabels", "strategies": { - "adapter": "decomposed", - "decomposition": "topLevel", - "transformer": "decomposed" + "adapter": "default", + "transformer": "decomposedLabels" }, - "suffix": "labels", - "supportsPartialDelete": true + "suffix": "labels" } } } diff --git a/src/registry/presets/decomposeCustomLabelsBeta2.json b/src/registry/presets/decomposeCustomLabelsBeta2.json deleted file mode 100644 index ea65af0b74..0000000000 --- a/src/registry/presets/decomposeCustomLabelsBeta2.json +++ /dev/null @@ -1,32 +0,0 @@ -{ - "childTypes": { - "customlabel": "" - }, - "strictDirectoryNames": {}, - "suffixes": { - "label": "customlabel", - "labels": "customlabels" - }, - "types": { - "customlabel": { - "directoryName": "labels", - "id": "customlabel", - "name": "CustomLabel", - "strategies": { - "adapter": "default", - "transformer": "decomposedLabels" - }, - "suffix": "label" - }, - "customlabels": { - "directoryName": "labels", - "id": "customlabels", - "name": "CustomLabels", - "strategies": { - "adapter": "default", - "transformer": "decomposedLabels" - }, - "suffix": "labels" - } - } -} diff --git a/src/registry/presets/presetMap.ts b/src/registry/presets/presetMap.ts index 9a195c52ea..2a0353bb68 100644 --- a/src/registry/presets/presetMap.ts +++ b/src/registry/presets/presetMap.ts @@ -8,14 +8,12 @@ import { MetadataRegistry } from '../types'; // we have to import all presets explicitly for VSCE's esbuild bundling process import * as decomposeCustomLabelsBeta from './decomposeCustomLabelsBeta.json'; -import * as decomposeCustomLabelsBeta2 from './decomposeCustomLabelsBeta2.json'; import * as decomposePermissionSetBeta from './decomposePermissionSetBeta.json'; import * as decomposeSharingRulesBeta from './decomposeSharingRulesBeta.json'; import * as decomposeWorkflowBeta from './decomposeWorkflowBeta.json'; export const presetMap = new Map([ ['decomposeCustomLabelsBeta', decomposeCustomLabelsBeta as MetadataRegistry], - ['decomposeCustomLabelsBeta2', decomposeCustomLabelsBeta2 as MetadataRegistry], ['decomposePermissionSetBeta', decomposePermissionSetBeta as MetadataRegistry], ['decomposeSharingRulesBeta', decomposeSharingRulesBeta as MetadataRegistry], ['decomposeWorkflowBeta', decomposeWorkflowBeta as MetadataRegistry], diff --git a/test/registry/registryValidation.test.ts b/test/registry/registryValidation.test.ts index 230d28493a..2ec410c5b5 100644 --- a/test/registry/registryValidation.test.ts +++ b/test/registry/registryValidation.test.ts @@ -79,12 +79,14 @@ for (const preset of presets) { }); describe('every childTypes top-level property maps to a top-level type that has it in its childTypes', () => { - Object.entries(registry.childTypes).forEach(([childId, parentId]) => { - it(`childTypes member ${childId} matches a parent type ${parentId}`, () => { - expect(registry.types[parentId]).to.have.property('children'); - expect(registry.types[parentId]?.children?.types).to.have.property(childId); + Object.entries(registry.childTypes) + .filter(([, parentId]) => parentId) + .forEach(([childId, parentId]) => { + it(`childTypes member ${childId} matches a parent type ${parentId}`, () => { + expect(registry.types[parentId]).to.have.property('children'); + expect(registry.types[parentId]?.children?.types).to.have.property(childId); + }); }); - }); }); }); diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json b/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json index ea1b12c4e1..2e2c204f80 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json @@ -9,5 +9,5 @@ ], "sfdcLoginUrl": "https://login.salesforce.com", "sourceApiVersion": "60.0", - "sourceBehaviorOptions": ["decomposeCustomLabelsBeta2"] + "sourceBehaviorOptions": ["decomposeCustomLabelsBeta"] } diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels deleted file mode 100644 index d57a2391aa..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/labels/CustomLabels.labels +++ /dev/null @@ -1,24 +0,0 @@ - - - - DeleteMe - en_US - true - DeleteMe - Test - - - KeepMe1 - en_US - true - KeepMe1 - Test - - - KeepMe2 - en_US - true - KeepMe2 - Test - - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml deleted file mode 100644 index 51950dd57c..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-md-files.expected/package.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - - CustomLabels - CustomLabels - - 60.0 - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml deleted file mode 100644 index 6d9520ca90..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/CustomLabels.labels-meta.xml +++ /dev/null @@ -1,2 +0,0 @@ - - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml deleted file mode 100644 index 14bf9da669..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/DeleteMe.label-meta.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - DeleteMe - en_US - true - DeleteMe - Test - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml deleted file mode 100644 index 9a78141d6e..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe1.label-meta.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - KeepMe1 - en_US - true - KeepMe1 - Test - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml deleted file mode 100644 index 64ab9ef9b3..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/__snapshots__/verify-source-files.expected/force-app/main/default/labels/CustomLabels/KeepMe2.label-meta.xml +++ /dev/null @@ -1,8 +0,0 @@ - - - KeepMe2 - en_US - true - KeepMe2 - Test - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels deleted file mode 100644 index d57a2391aa..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/labels/CustomLabels.labels +++ /dev/null @@ -1,24 +0,0 @@ - - - - DeleteMe - en_US - true - DeleteMe - Test - - - KeepMe1 - en_US - true - KeepMe1 - Test - - - KeepMe2 - en_US - true - KeepMe2 - Test - - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml deleted file mode 100644 index 76c38d0d62..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/originalMdapi/package.xml +++ /dev/null @@ -1,14 +0,0 @@ - - - - DeleteMe - KeepMe1 - KeepMe2 - CustomLabel - - - CustomLabels - CustomLabels - - 57.0 - diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json deleted file mode 100644 index 97d16c7028..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/sfdx-project.json +++ /dev/null @@ -1,13 +0,0 @@ -{ - "name": "variantTest", - "namespace": "", - "packageDirectories": [ - { - "default": true, - "path": "force-app" - } - ], - "sourceBehaviorOptions": ["decomposeCustomLabelsBeta"], - "sfdcLoginUrl": "https://login.salesforce.com", - "sourceApiVersion": "60.0" -} diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts b/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts deleted file mode 100644 index fbb625b584..0000000000 --- a/test/snapshot/sampleProjects/preset-decomposeLabels_Deprecated/snapshots.test.ts +++ /dev/null @@ -1,59 +0,0 @@ -/* - * Copyright (c) 2023, salesforce.com, inc. - * All rights reserved. - * Licensed under the BSD 3-Clause license. - * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause - */ -import * as fs from 'node:fs'; -import * as path from 'node:path'; -import { - compareTwoXml, - fileSnap, - mdapiToSource, - sourceToMdapi, - MDAPI_OUT, - dirsAreIdentical, - FORCE_APP, -} from '../../helper/conversions'; - -// we don't want failing tests outputting over each other -/* eslint-disable no-await-in-loop */ - -describe('decomposed custom labels', () => { - const testDir = path.join('test', 'snapshot', 'sampleProjects', 'preset-decomposeLabels_Deprecated'); - let sourceFiles: string[]; - let mdFiles: string[]; - - before(async () => { - sourceFiles = await mdapiToSource(testDir); - mdFiles = await sourceToMdapi(testDir); - }); - it('verify source files', async () => { - for (const file of sourceFiles) { - await fileSnap(file, testDir); - } - dirsAreIdentical( - path.join(testDir, FORCE_APP), - path.join(testDir, '__snapshots__', 'verify-source-files.expected', FORCE_APP) - ); - }); - it('verify md files', async () => { - for (const file of mdFiles) { - await fileSnap(file, testDir); - } - }); - it('round trip of metadata format is equivalent', async () => { - const [old, updated] = await Promise.all([ - fs.promises.readFile(path.join(testDir, 'originalMdapi', 'labels', 'CustomLabels.labels'), 'utf8'), - fs.promises.readFile(path.join(testDir, MDAPI_OUT, 'labels', 'CustomLabels.labels'), 'utf8'), - ]); - compareTwoXml(old, updated); - }); - - after(async () => { - await Promise.all([ - fs.promises.rm(path.join(testDir, FORCE_APP), { recursive: true, force: true }), - fs.promises.rm(path.join(testDir, MDAPI_OUT), { recursive: true, force: true }), - ]); - }); -}); From 92461b8a960b042821a9de0b405bc40868057bef Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 15:19:40 -0500 Subject: [PATCH 09/30] test: remove wrong snapshot path --- .../labels/CustomLabels.labels-meta.xml | 24 ------------------- 1 file changed, 24 deletions(-) delete mode 100644 test/snapshot/sampleProjects/customLabels-simple/__snapshots__/verify-source-files.expected/force-app/test/snapshot/sampleProjects/customLabels-simple/force-app/main/default/labels/CustomLabels.labels-meta.xml diff --git a/test/snapshot/sampleProjects/customLabels-simple/__snapshots__/verify-source-files.expected/force-app/test/snapshot/sampleProjects/customLabels-simple/force-app/main/default/labels/CustomLabels.labels-meta.xml b/test/snapshot/sampleProjects/customLabels-simple/__snapshots__/verify-source-files.expected/force-app/test/snapshot/sampleProjects/customLabels-simple/force-app/main/default/labels/CustomLabels.labels-meta.xml deleted file mode 100644 index d57a2391aa..0000000000 --- a/test/snapshot/sampleProjects/customLabels-simple/__snapshots__/verify-source-files.expected/force-app/test/snapshot/sampleProjects/customLabels-simple/force-app/main/default/labels/CustomLabels.labels-meta.xml +++ /dev/null @@ -1,24 +0,0 @@ - - - - DeleteMe - en_US - true - DeleteMe - Test - - - KeepMe1 - en_US - true - KeepMe1 - Test - - - KeepMe2 - en_US - true - KeepMe2 - Test - - From 3245b046cd7087e28b61feb402811862cb5c87e5 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 20:42:16 -0500 Subject: [PATCH 10/30] feat: show all decomposed labels in FileResponses --- src/convert/streams.ts | 8 +++++++- 1 file changed, 7 insertions(+), 1 deletion(-) diff --git a/src/convert/streams.ts b/src/convert/streams.ts index 8fe41c92d0..4b0a03bec5 100644 --- a/src/convert/streams.ts +++ b/src/convert/streams.ts @@ -160,7 +160,13 @@ export class StandardWriter extends ComponentWriter { } // if there are children, resolve each file. o/w just pick one of the files to resolve - if (toResolve.size === 0 || chunk.component.type.children) { + // "resolve" means "make these show up in the FileResponses" + if ( + toResolve.size === 0 || + chunk.component.type.children !== undefined || + // make each decomposed label show up in the fileResponses + chunk.component.type.strategies?.transformer === 'decomposedLabels' + ) { // This is a workaround for a server side ListViews bug where // duplicate components are sent. W-9614275 if (toResolve.has(info.output)) { From 26cf2cd6c50d2ba00b3f5dc776769ce61a4f008f Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 20:42:54 -0500 Subject: [PATCH 11/30] fix: handling single label --- .../decomposeLabelsTransformer.ts | 21 +++++++++++-------- 1 file changed, 12 insertions(+), 9 deletions(-) diff --git a/src/convert/transformers/decomposeLabelsTransformer.ts b/src/convert/transformers/decomposeLabelsTransformer.ts index 552a4c3a83..ed6079280e 100644 --- a/src/convert/transformers/decomposeLabelsTransformer.ts +++ b/src/convert/transformers/decomposeLabelsTransformer.ts @@ -6,6 +6,7 @@ */ import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; +import { ensureArray } from '@salesforce/kit'; import { customLabelHasFullName } from '../../utils/metadata'; import { calculateRelativePath } from '../../utils/path'; import { SourceComponent } from '../../resolve/sourceComponent'; @@ -22,15 +23,17 @@ export class LabelsMetadataTransformer extends DefaultMetadataTransformer { const partiallyAppliedPathCalculator = calculateRelativePath('source')({ self: labelType, }); - const xml = unwrapAndOmitNS('CustomLabels')(await component.parseXml()) as { labels: CustomLabel[] }; - // split each label into a separate label file - return xml.labels.filter(customLabelHasFullName).map((l) => ({ - output: - // if present in the merge set, use that xml path, otherwise use the default path - mergeSet?.getComponentFilenamesByNameAndType({ fullName: l.fullName, type: labelType.name })?.[0] ?? - partiallyAppliedPathCalculator(l.fullName)(`${l.fullName}.label-meta.xml`), - source: new JsToXml({ CustomLabel: l }), - })); + const xml = unwrapAndOmitNS('CustomLabels')(await component.parseXml()) as { labels: CustomLabel | CustomLabel[] }; + return ensureArray(xml.labels) // labels could parse to a single object and not an array if there's only 1 label + .filter(customLabelHasFullName) + .map((l) => ({ + // split each label into a separate label file + output: + // if present in the merge set, use that xml path, otherwise use the default path + mergeSet?.getComponentFilenamesByNameAndType({ fullName: l.fullName, type: labelType.name })?.[0] ?? + partiallyAppliedPathCalculator(l.fullName)(`${l.fullName}.label-meta.xml`), + source: new JsToXml({ CustomLabel: l }), + })); } } From 378c61f5cbb78f235ca6e958bc39b310fe4ad2b1 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Thu, 8 Aug 2024 20:43:26 -0500 Subject: [PATCH 12/30] fix: empty string to remove props from normal registry --- HANDBOOK.md | 10 ++++++++++ src/registry/variants.ts | 18 ++++++++++++++++-- src/utils/filePathGenerator.ts | 5 ++--- 3 files changed, 28 insertions(+), 5 deletions(-) diff --git a/HANDBOOK.md b/HANDBOOK.md index 20c662365a..e80562e841 100644 --- a/HANDBOOK.md +++ b/HANDBOOK.md @@ -172,6 +172,16 @@ Be careful when instantiating classes (ex: ComponentSet) that will default a Reg **Updating presets** If you do need to update a preset to make a breaking change, it's better to copy it to a new preset and give it a unique name (ex: `decomposeFooV2`). This preserves the existing behavior for existing projects with the old preset. +Presets **can** remove strings from the default metadataRegistry by setting values to empty string ex: + +```json +{ + "childTypes": { + "somethingThatIsUsuallyAChild": "" + } +} +``` + ### Querying registry data While it’s perfectly fine to reference the registry export directly, the `RegistryAccess` class was created to make accessing the object a bit more streamlined. Querying types and searching the registry is oftentimes easier and cleaner this way and contains built-in checking for whether or not a metadata type exists. Here’s a comparison of using each: diff --git a/src/registry/variants.ts b/src/registry/variants.ts index 78532d9b4a..59d4174ba7 100644 --- a/src/registry/variants.ts +++ b/src/registry/variants.ts @@ -20,7 +20,7 @@ export type RegistryLoadInput = { /** combine the standard registration with any overrides specific in the sfdx-project.json */ export const getEffectiveRegistry = (input?: RegistryLoadInput): MetadataRegistry => - deepFreeze(firstLevelMerge(registryData as MetadataRegistry, loadVariants(input))); + deepFreeze(removeEmptyStrings(firstLevelMerge(registryData as MetadataRegistry, loadVariants(input)))); /** read the project to get additional registry customizations and sourceBehaviorOptions */ const loadVariants = ({ projectDir }: RegistryLoadInput = {}): MetadataRegistry => { @@ -101,5 +101,19 @@ export const firstLevelMerge = (original: MetadataRegistry, overrides: MetadataR types: { ...original.types, ...(overrides.types ?? {}) }, childTypes: { ...original.childTypes, ...(overrides.childTypes ?? {}) }, suffixes: { ...original.suffixes, ...(overrides.suffixes ?? {}) }, - strictDirectoryNames: { ...original.strictDirectoryNames, ...(overrides.strictDirectoryNames ?? {}) }, + strictDirectoryNames: { + ...original.strictDirectoryNames, + ...(overrides.strictDirectoryNames ?? {}), + }, }); + +const removeEmptyStrings = (reg: MetadataRegistry): MetadataRegistry => ({ + types: reg.types, + childTypes: removeEmptyString(reg.childTypes), + suffixes: removeEmptyString(reg.suffixes), + strictDirectoryNames: removeEmptyString(reg.strictDirectoryNames), +}); + +// presets can remove an entry by setting it to an empty string ex: { "childTypes": { "foo": "" } } +const removeEmptyString = (obj: Record): Record => + Object.fromEntries(Object.entries(obj).filter(([, v]) => v !== '')); diff --git a/src/utils/filePathGenerator.ts b/src/utils/filePathGenerator.ts index b133a0a73a..273720fdb0 100644 --- a/src/utils/filePathGenerator.ts +++ b/src/utils/filePathGenerator.ts @@ -10,7 +10,6 @@ import { isPlainObject } from '@salesforce/ts-types'; import { MetadataComponent } from '../resolve/types'; import { META_XML_SUFFIX } from '../common/constants'; import { RegistryAccess } from '../registry/registryAccess'; -import { registry } from '../registry/registry'; Messages.importMessagesDirectory(__dirname); const messages = Messages.loadMessages('@salesforce/source-deploy-retrieve', 'sdr'); @@ -65,7 +64,7 @@ export const filePathsFromMetadataComponent = ( } // this needs to be done before the other types because of potential overlaps - if (!type.children && Object.keys(registry.childTypes).includes(type.id)) { + if (!type.children && Object.keys(registryAccess.getRegistry().childTypes).includes(type.id)) { return getDecomposedChildType({ fullName, type }, packageDir); } @@ -75,7 +74,7 @@ export const filePathsFromMetadataComponent = ( } // basic metadata (with or without folders) - if (!type.children && !type.strategies && type.suffix) { + if (!type.children && type.suffix && (!type.strategies || type.strategies.transformer === 'decomposedLabels')) { return (type.inFolder ?? type.folderType ? generateFolders({ fullName, type }, packageDirWithTypeDir) : []).concat([ join(packageDirWithTypeDir, `${fullName}.${type.suffix}${META_XML_SUFFIX}`), ]); From 49ee212c889b41e873b72b6252691cb10950bee5 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 08:51:59 -0500 Subject: [PATCH 13/30] fix: provide mergeSet for toSourceFormat --- src/convert/streams.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/convert/streams.ts b/src/convert/streams.ts index 4b0a03bec5..b1c2fa7e1c 100644 --- a/src/convert/streams.ts +++ b/src/convert/streams.ts @@ -78,7 +78,7 @@ export class ComponentConverter extends Transform { } } if (converts.length === 0) { - converts.push(transformer.toSourceFormat({ component: chunk })); + converts.push(transformer.toSourceFormat({ component: chunk, mergeSet: this.mergeSet })); } break; case 'metadata': From 6046634bfe2a786a254e87a9ddcdde638a0d8e38 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 10:51:48 -0500 Subject: [PATCH 14/30] fix: error for using -m CustomLabels with the preset --- src/collections/componentSetBuilder.ts | 3 +++ 1 file changed, 3 insertions(+) diff --git a/src/collections/componentSetBuilder.ts b/src/collections/componentSetBuilder.ts index 205f190c64..d2c6a71ef1 100644 --- a/src/collections/componentSetBuilder.ts +++ b/src/collections/componentSetBuilder.ts @@ -291,6 +291,9 @@ export const entryToTypeAndName = if (type.isAddressable === false && parent !== undefined && !type.unaddressableWithoutParent) { throw new Error(`Cannot use this type, instead use ${parent.name}`); } + if (type.name === 'CustomLabels' && type.strategies?.transformer === 'decomposedLabels') { + throw new Error('Use CustomLabel instead of CustomLabels for decomposed labels'); + } return { type, metadataName: name.length ? name.join(':').trim() : '*' }; }; From 5acd1899e0870967ba6b225445e4bdfd80117f3f Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 14:29:47 -0500 Subject: [PATCH 15/30] chore: rename file to match class name --- src/convert/convertContext/convertContext.ts | 4 ++-- ...edCustomLabelFinalizer.ts => decomposedLabelsFinalizer.ts} | 2 +- 2 files changed, 3 insertions(+), 3 deletions(-) rename src/convert/convertContext/{decomposedCustomLabelFinalizer.ts => decomposedLabelsFinalizer.ts} (96%) diff --git a/src/convert/convertContext/convertContext.ts b/src/convert/convertContext/convertContext.ts index 689906b6c0..4793e09071 100644 --- a/src/convert/convertContext/convertContext.ts +++ b/src/convert/convertContext/convertContext.ts @@ -9,7 +9,7 @@ import { RecompositionFinalizer } from './recompositionFinalizer'; import { NonDecompositionFinalizer } from './nonDecompositionFinalizer'; import { DecompositionFinalizer } from './decompositionFinalizer'; import { ConvertTransactionFinalizer } from './transactionFinalizer'; -import { DecomposedCustomLabelsFinalizer } from './decomposedCustomLabelFinalizer'; +import { DecomposedLabelsFinalizer } from './decomposedLabelsFinalizer'; /** * A state manager over the course of a single metadata conversion call. */ @@ -17,7 +17,7 @@ export class ConvertContext { public readonly decomposition = new DecompositionFinalizer(); public readonly recomposition = new RecompositionFinalizer(); public readonly nonDecomposition = new NonDecompositionFinalizer(); - public readonly decomposedLabels = new DecomposedCustomLabelsFinalizer(); + public readonly decomposedLabels = new DecomposedLabelsFinalizer(); // eslint-disable-next-line @typescript-eslint/require-await public async *executeFinalizers(defaultDirectory?: string): AsyncIterable { diff --git a/src/convert/convertContext/decomposedCustomLabelFinalizer.ts b/src/convert/convertContext/decomposedLabelsFinalizer.ts similarity index 96% rename from src/convert/convertContext/decomposedCustomLabelFinalizer.ts rename to src/convert/convertContext/decomposedLabelsFinalizer.ts index 5383cfc654..5ebc998a28 100644 --- a/src/convert/convertContext/decomposedCustomLabelFinalizer.ts +++ b/src/convert/convertContext/decomposedLabelsFinalizer.ts @@ -27,7 +27,7 @@ type CustomLabelState = { * * Inserts unclaimed child components into the parent that belongs to the default package */ -export class DecomposedCustomLabelsFinalizer extends ConvertTransactionFinalizer { +export class DecomposedLabelsFinalizer extends ConvertTransactionFinalizer { public transactionState: CustomLabelState = { customLabelByFullName: new Map(), }; From 12ec0cda1a946b1413fe1ce134bcd9b51d707e6e Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 14:30:44 -0500 Subject: [PATCH 16/30] test: injectable presets into reg loader --- src/registry/variants.ts | 79 +++++++++++++++++++++++++++------------- 1 file changed, 54 insertions(+), 25 deletions(-) diff --git a/src/registry/variants.ts b/src/registry/variants.ts index 59d4174ba7..f383cb9331 100644 --- a/src/registry/variants.ts +++ b/src/registry/variants.ts @@ -10,60 +10,89 @@ import { MetadataRegistry } from './types'; import * as registryData from './metadataRegistry.json'; import { presetMap } from './presets/presetMap'; -export type RegistryLoadInput = { - /** The project directory to look at sfdx-project.json file - * will default to the current working directory - * if no project file is found, the standard registry will be returned without modifications - */ - projectDir?: string; +type ProjectVariants = { + registryCustomizations?: MetadataRegistry; + presets?: MetadataRegistry[]; + projectDir?: never; }; +export type RegistryLoadInput = + | { + /** The project directory to look at sfdx-project.json file + * will default to the current working directory + * if no project file is found, the standard registry will be returned without modifications + */ + projectDir?: string; + registryCustomizations?: never; + presets?: never; + } + | ProjectVariants; + /** combine the standard registration with any overrides specific in the sfdx-project.json */ export const getEffectiveRegistry = (input?: RegistryLoadInput): MetadataRegistry => - deepFreeze(removeEmptyStrings(firstLevelMerge(registryData as MetadataRegistry, loadVariants(input)))); + deepFreeze( + removeEmptyStrings( + firstLevelMerge( + registryData as MetadataRegistry, + mergeVariants( + input?.presets?.length ?? input?.registryCustomizations ? input : getProjectVariants(input?.projectDir) + ) + ) + ) + ); /** read the project to get additional registry customizations and sourceBehaviorOptions */ -const loadVariants = ({ projectDir }: RegistryLoadInput = {}): MetadataRegistry => { +const getProjectVariants = (projectDir?: string): ProjectVariants => { const logger = Logger.childFromRoot('variants'); const projJson = maybeGetProject(projectDir); if (!projJson) { logger.debug('no project found, using standard registry'); // there might not be a project at all and that's ok - return emptyRegistry; + return {}; } // there might not be any customizations in a project, so we default to the emptyRegistry - const customizations = projJson.get('registryCustomizations') ?? emptyRegistry; - const sourceBehaviorOptions = [ + const registryCustomizations = projJson.get('registryCustomizations') ?? emptyRegistry; + const presets = [ ...new Set([ // TODO: deprecated, remove this ...(projJson.get('registryPresets') ?? []), ...(projJson.get('sourceBehaviorOptions') ?? []), ]), ]; - if (Object.keys(customizations.types).length > 0) { + if (Object.keys(registryCustomizations.types).length > 0) { logger.debug( - `found registryCustomizations for types [${Object.keys(customizations.types).join(',')}] in ${projJson.getPath()}` + `found registryCustomizations for types [${Object.keys(registryCustomizations.types).join( + ',' + )}] in ${projJson.getPath()}` ); } - if (sourceBehaviorOptions.length > 0) { - logger.debug(`using sourceBehaviorOptions [${sourceBehaviorOptions.join(',')}] in ${projJson.getPath()}`); + if (presets.length > 0) { + logger.debug(`using sourceBehaviorOptions [${presets.join(',')}] in ${projJson.getPath()}`); } - const registryFromPresets = sourceBehaviorOptions.reduce( - (prev, curr) => firstLevelMerge(prev, loadPreset(curr)), - emptyRegistry - ); - if (sourceBehaviorOptions.length > 0 || Object.keys(customizations.types).length > 0) { + if (presets.length > 0 || Object.keys(registryCustomizations.types).length > 0) { void Lifecycle.getInstance().emitTelemetry({ library: 'SDR', eventName: 'RegistryVariants', - presetCount: sourceBehaviorOptions.length, - presets: sourceBehaviorOptions.join(','), - customizationsCount: Object.keys(customizations.types).length, - customizationsTypes: Object.keys(customizations.types).join(','), + presetCount: presets.length, + presets: presets.join(','), + customizationsCount: Object.keys(registryCustomizations.types).length, + customizationsTypes: Object.keys(registryCustomizations.types).join(','), }); } - return firstLevelMerge(registryFromPresets, customizations); + return { + registryCustomizations, + presets: presets.map(loadPreset), + }; +}; + +const mergeVariants = ({ registryCustomizations, presets }: ProjectVariants): MetadataRegistry => { + const registryFromPresets = [...(presets ?? []), registryCustomizations ?? emptyRegistry].reduce( + (prev, curr) => firstLevelMerge(prev, curr), + emptyRegistry + ); + + return firstLevelMerge(registryFromPresets, registryCustomizations ?? emptyRegistry); }; const maybeGetProject = (projectDir?: string): SfProjectJson | undefined => { From aec5d4e934964ebff5d6ca7c8b4fb75b73129953 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 15:18:13 -0500 Subject: [PATCH 17/30] test: ut for label transformer --- .../decomposedLabelsTransformer.test.ts | 133 ++++++++++++++ .../decomposedCustomLabelsConstant.ts | 173 ++++++++++++++++++ 2 files changed, 306 insertions(+) create mode 100644 test/convert/transformers/decomposedLabelsTransformer.test.ts create mode 100644 test/mock/type-constants/decomposedCustomLabelsConstant.ts diff --git a/test/convert/transformers/decomposedLabelsTransformer.test.ts b/test/convert/transformers/decomposedLabelsTransformer.test.ts new file mode 100644 index 0000000000..924eab931e --- /dev/null +++ b/test/convert/transformers/decomposedLabelsTransformer.test.ts @@ -0,0 +1,133 @@ +/* + * Copyright (c) 2023, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ +import { join } from 'node:path'; +import { expect } from 'chai'; +import { ComponentSet } from '../../../src/collections/componentSet'; +import { RegistryAccess } from '../../../src/registry/registryAccess'; +import { + EMPTY_CUSTOM_LABELS_CMP, + ONE_CUSTOM_LABELS_CMP, + ONLY_LABEL_CMP_IN_ANOTHER_DIR_CMP, + ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP, + ONLY_LABEL_NO_DIR_CMP, + THREE_CUSTOM_LABELS_CMP, +} from '../../mock/type-constants/decomposedCustomLabelsConstant'; +import { + LabelMetadataTransformer, + LabelsMetadataTransformer, +} from '../../../src/convert/transformers/decomposeLabelsTransformer'; +import { getEffectiveRegistry } from '../../../src/registry/variants'; +import { presetMap } from '../../../src/registry/presets/presetMap'; + +describe('DecomposedCustomLabelTransformer', () => { + const regAcc = new RegistryAccess(getEffectiveRegistry({ presets: [presetMap.get('decomposeCustomLabelsBeta')!] })); + + describe('LabelsMetadataTransformer', () => { + describe('toSourceFormat', () => { + describe('WriteInfo output (where the file will write to)', () => { + describe('default dir', () => { + it('multiple labels in a single customLabels', async () => { + const component = THREE_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ component }); + expect(result).to.have.length(3); + result.map((l) => { + expect(l.output).to.include(join('main', 'default', 'labels')); + }); + expect(result[0].output).to.match(/DeleteMe.label-meta.xml$/); + expect(result[1].output).to.match(/KeepMe1.label-meta.xml$/); + expect(result[2].output).to.match(/KeepMe2.label-meta.xml$/); + }); + it('single label in customLabels', async () => { + const component = ONE_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ component }); + expect(result).to.have.length(1); + expect(result[0].output).to.equal(join('main', 'default', 'labels', 'OnlyLabel.label-meta.xml')); + }); + it('empty customLabels ', async () => { + const component = EMPTY_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ component }); + expect(result).to.deep.equal([]); + }); + it('merge component in defaultDir', async () => { + const component = ONE_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ + component, + mergeSet: new ComponentSet([ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP], regAcc), + }); + expect(result).to.have.length(1); + expect(result[0].output).to.equal(ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP.xml!); + }); + }); + }); + describe('no labels dir', () => { + it('merge matches label not in a labels dir', async () => { + const component = ONE_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ + component, + mergeSet: new ComponentSet([ONLY_LABEL_NO_DIR_CMP], regAcc), + }); + expect(result).to.have.length(1); + expect(result[0].output).to.equal(ONLY_LABEL_NO_DIR_CMP.xml!); + }); + }); + describe('non-default dir', () => { + it('merge component in nonDefault dir => matches the original location', async () => { + const component = ONE_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ + component, + mergeSet: new ComponentSet([ONLY_LABEL_CMP_IN_ANOTHER_DIR_CMP], regAcc), + }); + expect(result).to.have.length(1); + expect(result[0].output).to.equal(ONLY_LABEL_CMP_IN_ANOTHER_DIR_CMP.xml!); + }); + it('merge component in nonDefault dir, but mdapi does not match existing source ', async () => { + const component = THREE_CUSTOM_LABELS_CMP; + const xf = new LabelsMetadataTransformer(regAcc); + const result = await xf.toSourceFormat({ + component, + mergeSet: new ComponentSet([ONLY_LABEL_CMP_IN_ANOTHER_DIR_CMP], regAcc), + }); + expect(result).to.have.length(3); + result.map((l) => { + expect(l.output).to.include(join('main', 'default', 'labels')); + }); + }); + }); + }); + }); + + describe('LabelMetadataTransformer', () => { + describe('toMetadataFormat', () => { + it('should set the customLabelsType in the context', async () => { + const component = ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP; + const xf = new LabelMetadataTransformer(regAcc); + const result = await xf.toMetadataFormat(component); + expect(result).to.deep.equal([]); + expect(xf.context.decomposedLabels.customLabelsType).to.equal(regAcc.getTypeByName('CustomLabels')); + }); + it('should put the entire CustomLabel xml content into the transactionState', async () => { + const component = ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP; + const xf = new LabelMetadataTransformer(regAcc); + const result = await xf.toMetadataFormat(component); + expect(result).to.deep.equal([]); + expect(xf.context.decomposedLabels.transactionState.customLabelByFullName.size).to.equal(1); + const stateEntry = xf.context.decomposedLabels.transactionState.customLabelByFullName.get( + ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP.fullName + ); + // component properties are in the state. We could say the same about the rest of CustomLabel properties + // but would have to the xml => js and omitNS stuff here to compare + expect(stateEntry?.fullName).to.deep.equal(component.fullName); + }); + }); + }); +}); diff --git a/test/mock/type-constants/decomposedCustomLabelsConstant.ts b/test/mock/type-constants/decomposedCustomLabelsConstant.ts new file mode 100644 index 0000000000..2c475cf5dc --- /dev/null +++ b/test/mock/type-constants/decomposedCustomLabelsConstant.ts @@ -0,0 +1,173 @@ +/* + * Copyright (c) 2023, salesforce.com, inc. + * All rights reserved. + * Licensed under the BSD 3-Clause license. + * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause + */ +import { join } from 'node:path'; + +import { SourceComponent, VirtualTreeContainer, presetMap, RegistryAccess } from '../../../src'; +import { getEffectiveRegistry } from '../../../src/registry/variants'; + +// Constants for a matching content file type +const regAcc = new RegistryAccess(getEffectiveRegistry({ presets: [presetMap.get('decomposeCustomLabelsBeta')!] })); + +const customLabelsType = regAcc.getTypeByName('CustomLabels'); +const customLabelType = regAcc.getTypeByName('CustomLabel'); + +const MDAPI_XML_NAME = 'CustomLabels.labels'; + +export const EMPTY_CUSTOM_LABELS_CMP = new SourceComponent( + { + name: 'CustomLabels', + type: customLabelsType, + xml: join('labels', MDAPI_XML_NAME), + }, + new VirtualTreeContainer([ + { + dirPath: 'labels', + children: [ + { + name: MDAPI_XML_NAME, + data: Buffer.from(` +`), + }, + ], + }, + ]) +); +export const ONE_CUSTOM_LABELS_CMP = new SourceComponent( + { + name: 'CustomLabels', + type: customLabelsType, + xml: join('labels', MDAPI_XML_NAME), + }, + new VirtualTreeContainer([ + { + dirPath: 'labels', + children: [ + { + name: MDAPI_XML_NAME, + data: Buffer.from(` + + + OnlyLabel + en_US + true + OnlyLabel + OnlyLabel + +`), + }, + ], + }, + ]) +); + +export const THREE_CUSTOM_LABELS_CMP = new SourceComponent( + { + name: 'CustomLabels', + type: customLabelsType, + xml: join('labels', MDAPI_XML_NAME), + }, + new VirtualTreeContainer([ + { + dirPath: 'labels', + children: [ + { + name: MDAPI_XML_NAME, + data: Buffer.from(` + + + DeleteMe + en_US + true + DeleteMe + Test + + + KeepMe1 + en_US + true + KeepMe1 + Test + + + KeepMe2 + en_US + true + KeepMe2 + Test + +`), + }, + ], + }, + ]) +); + +const ONLY_LABEL_CONTENTS = ` + + OnlyLabel + en_US + true + OnlyLabel + OnlyLabel +`; + +export const ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP = new SourceComponent( + { + name: 'OnlyLabel', + type: customLabelType, + xml: join('main', 'default', 'labels', 'OnlyLabel.label-meta.xml'), + }, + new VirtualTreeContainer([ + { + dirPath: join('main', 'default', 'labels'), + children: [ + { + name: 'OnlyLabel.label-meta.xml', + data: Buffer.from(ONLY_LABEL_CONTENTS), + }, + ], + }, + ]) +); + +export const ONLY_LABEL_CMP_IN_ANOTHER_DIR_CMP = new SourceComponent( + { + name: 'OnlyLabel', + type: customLabelType, + xml: join('other', 'dir', 'labels', 'OnlyLabel.label-meta.xml'), + }, + new VirtualTreeContainer([ + { + dirPath: join('other', 'dir', 'labels'), + children: [ + { + name: 'OnlyLabel.label-meta.xml', + data: Buffer.from(ONLY_LABEL_CONTENTS), + }, + ], + }, + ]) +); + +export const ONLY_LABEL_NO_DIR_CMP = new SourceComponent( + { + name: 'OnlyLabel', + type: customLabelType, + xml: 'OnlyLabel.label-meta.xml', + }, + new VirtualTreeContainer([ + { + dirPath: '', + children: [ + { + name: 'OnlyLabel.label-meta.xml', + data: Buffer.from(ONLY_LABEL_CONTENTS), + }, + ], + }, + ]) +); From 0cbd4f8ee16770aba3beff2c5c5bde9dce4c3f2b Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 15:52:36 -0500 Subject: [PATCH 18/30] refactor: move xml parsing to shared const --- src/resolve/sourceComponent.ts | 14 ++------------ src/utils/metadata.ts | 14 +++++++++++++- test/snapshot/helper/conversions.ts | 16 +++------------- 3 files changed, 18 insertions(+), 26 deletions(-) diff --git a/src/resolve/sourceComponent.ts b/src/resolve/sourceComponent.ts index 791b15a94a..57f68c432a 100644 --- a/src/resolve/sourceComponent.ts +++ b/src/resolve/sourceComponent.ts @@ -9,10 +9,10 @@ import { SfError } from '@salesforce/core/sfError'; import { Messages } from '@salesforce/core/messages'; import { Lifecycle } from '@salesforce/core/lifecycle'; -import { XMLParser, XMLValidator } from 'fast-xml-parser'; +import { XMLValidator } from 'fast-xml-parser'; import { get, getString, JsonMap } from '@salesforce/ts-types'; import { ensureArray } from '@salesforce/kit'; -import { XML_COMMENT_PROP_NAME } from '../common/constants'; +import { parser } from '../utils/metadata'; import { getXmlElement } from '../utils/decomposed'; import { baseName, baseWithoutSuffixes, parseMetadataXml, calculateRelativePath } from '../utils/path'; import { replacementIterations } from '../convert/replacements'; @@ -269,16 +269,6 @@ export class SourceComponent implements MetadataComponent { } private parse(contents: string): T { - // include tag attributes and don't parse text node as number - const parser = new XMLParser({ - ignoreAttributes: false, - parseTagValue: false, - parseAttributeValue: false, - cdataPropName: '__cdata', - ignoreDeclaration: true, - numberParseOptions: { leadingZeros: false, hex: false }, - commentPropName: XML_COMMENT_PROP_NAME, - }); const parsed = parser.parse(String(contents)) as T; const [firstElement] = Object.keys(parsed); if (firstElement === this.type.name) { diff --git a/src/utils/metadata.ts b/src/utils/metadata.ts index 4fc6983dc2..d09c72aadc 100644 --- a/src/utils/metadata.ts +++ b/src/utils/metadata.ts @@ -7,7 +7,19 @@ import type { CustomLabel } from '@jsforce/jsforce-node/lib/api/metadata'; import { SfError } from '@salesforce/core'; -import { META_XML_SUFFIX } from '../common'; +import { XMLParser } from 'fast-xml-parser'; +import { META_XML_SUFFIX, XML_COMMENT_PROP_NAME } from '../common/constants'; + +export const parser = new XMLParser({ + // include tag attributes and don't parse text node as number + ignoreAttributes: false, + parseTagValue: false, + parseAttributeValue: false, + cdataPropName: '__cdata', + ignoreDeclaration: true, + numberParseOptions: { leadingZeros: false, hex: false }, + commentPropName: XML_COMMENT_PROP_NAME, +}); export function generateMetaXML(typeName: string, apiVersion: string, status: string): string { let templateResult = '\n'; diff --git a/test/snapshot/helper/conversions.ts b/test/snapshot/helper/conversions.ts index 81616fc7f7..e8781096af 100644 --- a/test/snapshot/helper/conversions.ts +++ b/test/snapshot/helper/conversions.ts @@ -9,8 +9,8 @@ import * as fs from 'node:fs'; import snap from 'mocha-snap'; import { expect, config, use } from 'chai'; import deepEqualInAnyOrder from 'deep-equal-in-any-order'; -import { XMLParser } from 'fast-xml-parser'; +import { parser } from '../../../src/utils/metadata'; import { RegistryAccess } from '../../../src/registry/registryAccess'; import { MetadataConverter } from '../../../src/convert/metadataConverter'; import { ComponentSetBuilder } from '../../../src/collections/componentSetBuilder'; @@ -92,18 +92,8 @@ export const sourceToMdapi = async (testDir: string): Promise => { }; /** checks that the two xml bodies have the same equivalent json (handles out-of-order things, etc) */ -export const compareTwoXml = (file1: string, file2: string): Chai.Assertion => { - const parser = new XMLParser({ - ignoreAttributes: false, - parseTagValue: false, - parseAttributeValue: false, - cdataPropName: '__cdata', - ignoreDeclaration: true, - numberParseOptions: { leadingZeros: false, hex: false }, - }); - - return expect(parser.parse(file1)).to.deep.equalInAnyOrder(parser.parse(file2)); -}; +export const compareTwoXml = (file1: string, file2: string): Chai.Assertion => + expect(parser.parse(file1)).to.deep.equalInAnyOrder(parser.parse(file2)); /** * catches missing files by asserting that two directories have the exact same children From b61b0a11541293ddfbb3985379086c4c1ac6e156 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Fri, 9 Aug 2024 15:53:16 -0500 Subject: [PATCH 19/30] test: ut for label (source) => labels (mdapi) --- .../decomposedLabelsTransformer.test.ts | 44 ++++++++++++++++++- .../decomposedCustomLabelsConstant.ts | 26 +++++++++++ 2 files changed, 69 insertions(+), 1 deletion(-) diff --git a/test/convert/transformers/decomposedLabelsTransformer.test.ts b/test/convert/transformers/decomposedLabelsTransformer.test.ts index 924eab931e..22b42b68c3 100644 --- a/test/convert/transformers/decomposedLabelsTransformer.test.ts +++ b/test/convert/transformers/decomposedLabelsTransformer.test.ts @@ -5,7 +5,10 @@ * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause */ import { join } from 'node:path'; -import { expect } from 'chai'; +import { expect, assert } from 'chai'; +import { parser } from '../../../src/utils/metadata'; +import { stream2buffer } from '../../../src/convert/streams'; +import { DecomposedLabelsFinalizer } from '../../../src/convert/convertContext/decomposedLabelsFinalizer'; import { ComponentSet } from '../../../src/collections/componentSet'; import { RegistryAccess } from '../../../src/registry/registryAccess'; import { @@ -14,6 +17,7 @@ import { ONLY_LABEL_CMP_IN_ANOTHER_DIR_CMP, ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP, ONLY_LABEL_NO_DIR_CMP, + OTHER_LABEL_CMP, THREE_CUSTOM_LABELS_CMP, } from '../../mock/type-constants/decomposedCustomLabelsConstant'; import { @@ -129,5 +133,43 @@ describe('DecomposedCustomLabelTransformer', () => { expect(stateEntry?.fullName).to.deep.equal(component.fullName); }); }); + describe('finalizer', () => { + it('single label from source to mdapi', async () => { + const component = ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP; + const xf = new LabelMetadataTransformer(regAcc); + await xf.toMetadataFormat(component); + const finalizer = new DecomposedLabelsFinalizer(); + finalizer.customLabelsType = regAcc.getTypeByName('CustomLabels'); + finalizer.transactionState = xf.context.decomposedLabels.transactionState; + const result = await finalizer.finalize(); + expect(result).to.have.length(1); + expect(result[0].component.fullName).to.equal('CustomLabels'); + expect(result[0].component.type.name).to.equal('CustomLabels'); + expect(result[0].writeInfos).to.have.length(1); + assert(result[0].writeInfos[0].source); + const contents = (await stream2buffer(result[0].writeInfos[0].source)).toString(); + expect(parser.parse(contents)).to.deep.equal(await ONE_CUSTOM_LABELS_CMP.parseXml()); + }); + it('2 labels from source to mdapi', async () => { + const component1 = ONLY_LABEL_CMP_IN_DEFAULT_DIR_CMP; + const component2 = OTHER_LABEL_CMP; + const xf = new LabelMetadataTransformer(regAcc); + await xf.toMetadataFormat(component1); + await xf.toMetadataFormat(component2); + const finalizer = new DecomposedLabelsFinalizer(); + finalizer.customLabelsType = regAcc.getTypeByName('CustomLabels'); + finalizer.transactionState = xf.context.decomposedLabels.transactionState; + const result = await finalizer.finalize(); + expect(result).to.have.length(1); + expect(result[0].component.fullName).to.equal('CustomLabels'); + expect(result[0].component.type.name).to.equal('CustomLabels'); + // still produces only 1 writeInfo + expect(result[0].writeInfos).to.have.length(1); + assert(result[0].writeInfos[0].source); + const contents = (await stream2buffer(result[0].writeInfos[0].source)).toString(); + // with 2 labels in it + expect(parser.parse(contents).CustomLabels.labels).to.have.length(2); + }); + }); }); }); diff --git a/test/mock/type-constants/decomposedCustomLabelsConstant.ts b/test/mock/type-constants/decomposedCustomLabelsConstant.ts index 2c475cf5dc..cf891da923 100644 --- a/test/mock/type-constants/decomposedCustomLabelsConstant.ts +++ b/test/mock/type-constants/decomposedCustomLabelsConstant.ts @@ -171,3 +171,29 @@ export const ONLY_LABEL_NO_DIR_CMP = new SourceComponent( }, ]) ); + +export const OTHER_LABEL_CMP = new SourceComponent( + { + name: 'OtherLabel', + type: customLabelType, + xml: join('labels', 'OtherLabel.label-meta.xml'), + }, + new VirtualTreeContainer([ + { + dirPath: 'labels', + children: [ + { + name: 'OtherLabel.label-meta.xml', + data: Buffer.from(` + + OtherLabel + en_US + true + OtherLabel + OtherLabel +`), + }, + ], + }, + ]) +); From 9327a45d2abdab835fe62d34798335d0869bd013 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Mon, 12 Aug 2024 16:35:02 -0500 Subject: [PATCH 20/30] refactor: use a new name for the updated beta --- src/Presets.md | 19 ++++++++++- .../presets/decomposeCustomLabelsBeta2.json | 32 +++++++++++++++++++ src/registry/presets/presetMap.ts | 2 ++ .../decomposedLabelsTransformer.test.ts | 2 +- .../decomposedCustomLabelsConstant.ts | 2 +- .../preset-decomposeLabels/sfdx-project.json | 2 +- 6 files changed, 55 insertions(+), 4 deletions(-) create mode 100644 src/registry/presets/decomposeCustomLabelsBeta2.json diff --git a/src/Presets.md b/src/Presets.md index 793f8701ef..4f5101300f 100644 --- a/src/Presets.md +++ b/src/Presets.md @@ -64,10 +64,12 @@ Simple fields (ex: `fullName`) can remain in the top-level `Account.workflow-met ## `decomposeCustomLabelsBeta` +> This will definitely not become GA. Based on user feedback, we replaced it with `decomposeCustomLabelsBeta2` + CustomLabels are decomposed to a folder named `CustomLabels` the labels are then placed into individual files metadata format -`/labels/CustomLabels.customlabes-meta.xml` +`/labels/CustomLabels.customlabels-meta.xml` source format @@ -77,3 +79,18 @@ source format /labels/CustomLabels/b.label-meta.xml /labels/CustomLabels/c.label-meta.xml ``` + +## `decomposeCustomLabelsBeta2` + +CustomLabels are decomposed to a folder named `labels`; the labels are then placed into individual files. There is no top-level file. + +metadata format +`/labels/CustomLabels.customlabels-meta.xml` + +source format + +```txt +/labels/a.label-meta.xml +/labels/b.label-meta.xml +/labels/c.label-meta.xml +``` diff --git a/src/registry/presets/decomposeCustomLabelsBeta2.json b/src/registry/presets/decomposeCustomLabelsBeta2.json new file mode 100644 index 0000000000..ea65af0b74 --- /dev/null +++ b/src/registry/presets/decomposeCustomLabelsBeta2.json @@ -0,0 +1,32 @@ +{ + "childTypes": { + "customlabel": "" + }, + "strictDirectoryNames": {}, + "suffixes": { + "label": "customlabel", + "labels": "customlabels" + }, + "types": { + "customlabel": { + "directoryName": "labels", + "id": "customlabel", + "name": "CustomLabel", + "strategies": { + "adapter": "default", + "transformer": "decomposedLabels" + }, + "suffix": "label" + }, + "customlabels": { + "directoryName": "labels", + "id": "customlabels", + "name": "CustomLabels", + "strategies": { + "adapter": "default", + "transformer": "decomposedLabels" + }, + "suffix": "labels" + } + } +} diff --git a/src/registry/presets/presetMap.ts b/src/registry/presets/presetMap.ts index 2a0353bb68..1cb0ea73b6 100644 --- a/src/registry/presets/presetMap.ts +++ b/src/registry/presets/presetMap.ts @@ -8,11 +8,13 @@ import { MetadataRegistry } from '../types'; // we have to import all presets explicitly for VSCE's esbuild bundling process import * as decomposeCustomLabelsBeta from './decomposeCustomLabelsBeta.json'; +import * as decomposeCustomLabelsBeta2 from './decomposeCustomLabelsBeta2.json'; import * as decomposePermissionSetBeta from './decomposePermissionSetBeta.json'; import * as decomposeSharingRulesBeta from './decomposeSharingRulesBeta.json'; import * as decomposeWorkflowBeta from './decomposeWorkflowBeta.json'; export const presetMap = new Map([ + ['decomposeCustomLabelsBeta2', decomposeCustomLabelsBeta2 as MetadataRegistry], ['decomposeCustomLabelsBeta', decomposeCustomLabelsBeta as MetadataRegistry], ['decomposePermissionSetBeta', decomposePermissionSetBeta as MetadataRegistry], ['decomposeSharingRulesBeta', decomposeSharingRulesBeta as MetadataRegistry], diff --git a/test/convert/transformers/decomposedLabelsTransformer.test.ts b/test/convert/transformers/decomposedLabelsTransformer.test.ts index 22b42b68c3..0590e11240 100644 --- a/test/convert/transformers/decomposedLabelsTransformer.test.ts +++ b/test/convert/transformers/decomposedLabelsTransformer.test.ts @@ -28,7 +28,7 @@ import { getEffectiveRegistry } from '../../../src/registry/variants'; import { presetMap } from '../../../src/registry/presets/presetMap'; describe('DecomposedCustomLabelTransformer', () => { - const regAcc = new RegistryAccess(getEffectiveRegistry({ presets: [presetMap.get('decomposeCustomLabelsBeta')!] })); + const regAcc = new RegistryAccess(getEffectiveRegistry({ presets: [presetMap.get('decomposeCustomLabelsBeta2')!] })); describe('LabelsMetadataTransformer', () => { describe('toSourceFormat', () => { diff --git a/test/mock/type-constants/decomposedCustomLabelsConstant.ts b/test/mock/type-constants/decomposedCustomLabelsConstant.ts index cf891da923..488f00f1bd 100644 --- a/test/mock/type-constants/decomposedCustomLabelsConstant.ts +++ b/test/mock/type-constants/decomposedCustomLabelsConstant.ts @@ -10,7 +10,7 @@ import { SourceComponent, VirtualTreeContainer, presetMap, RegistryAccess } from import { getEffectiveRegistry } from '../../../src/registry/variants'; // Constants for a matching content file type -const regAcc = new RegistryAccess(getEffectiveRegistry({ presets: [presetMap.get('decomposeCustomLabelsBeta')!] })); +const regAcc = new RegistryAccess(getEffectiveRegistry({ presets: [presetMap.get('decomposeCustomLabelsBeta2')!] })); const customLabelsType = regAcc.getTypeByName('CustomLabels'); const customLabelType = regAcc.getTypeByName('CustomLabel'); diff --git a/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json b/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json index 2e2c204f80..ea1b12c4e1 100644 --- a/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json +++ b/test/snapshot/sampleProjects/preset-decomposeLabels/sfdx-project.json @@ -9,5 +9,5 @@ ], "sfdcLoginUrl": "https://login.salesforce.com", "sourceApiVersion": "60.0", - "sourceBehaviorOptions": ["decomposeCustomLabelsBeta"] + "sourceBehaviorOptions": ["decomposeCustomLabelsBeta2"] } From 837c5177d60dd1e7e85eaef7a0417912995aa334 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Mon, 12 Aug 2024 16:35:36 -0500 Subject: [PATCH 21/30] refactor: restore the original beta --- .../presets/decomposeCustomLabelsBeta.json | 46 ++++++++++++------- 1 file changed, 29 insertions(+), 17 deletions(-) diff --git a/src/registry/presets/decomposeCustomLabelsBeta.json b/src/registry/presets/decomposeCustomLabelsBeta.json index ea65af0b74..591d87c7d4 100644 --- a/src/registry/presets/decomposeCustomLabelsBeta.json +++ b/src/registry/presets/decomposeCustomLabelsBeta.json @@ -1,32 +1,44 @@ { "childTypes": { - "customlabel": "" + "customlabel": "customlabels" }, - "strictDirectoryNames": {}, - "suffixes": { - "label": "customlabel", + "strictDirectoryNames": { "labels": "customlabels" }, + "suffixes": { + "label": "customlabel" + }, "types": { - "customlabel": { - "directoryName": "labels", - "id": "customlabel", - "name": "CustomLabel", - "strategies": { - "adapter": "default", - "transformer": "decomposedLabels" - }, - "suffix": "label" - }, "customlabels": { + "children": { + "suffixes": { + "label": "customlabel" + }, + "types": { + "customlabel": { + "directoryName": "test", + "id": "customlabel", + "isAddressable": false, + "name": "CustomLabel", + "suffix": "label", + "supportsWildcardAndName": true, + "uniqueIdElement": "fullName", + "xmlElementName": "labels" + } + } + }, "directoryName": "labels", "id": "customlabels", + "ignoreParsedFullName": false, "name": "CustomLabels", "strategies": { - "adapter": "default", - "transformer": "decomposedLabels" + "adapter": "decomposed", + "decomposition": "topLevel", + "transformer": "decomposed" }, - "suffix": "labels" + "strictDirectoryName": true, + "suffix": "labels", + "supportsPartialDelete": true } } } From 63ceb3cb433573c98270b2e1a619a1ac61953bf0 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 13 Aug 2024 15:57:57 -0500 Subject: [PATCH 22/30] refactor: separate logging fn --- src/registry/variants.ts | 66 ++++++++++++++++++++++------------------ 1 file changed, 37 insertions(+), 29 deletions(-) diff --git a/src/registry/variants.ts b/src/registry/variants.ts index f383cb9331..fcfb393f01 100644 --- a/src/registry/variants.ts +++ b/src/registry/variants.ts @@ -43,7 +43,7 @@ export const getEffectiveRegistry = (input?: RegistryLoadInput): MetadataRegistr /** read the project to get additional registry customizations and sourceBehaviorOptions */ const getProjectVariants = (projectDir?: string): ProjectVariants => { - const logger = Logger.childFromRoot('variants'); + const logger = Logger.childFromRoot('variants:getProjectVariants'); const projJson = maybeGetProject(projectDir); if (!projJson) { logger.debug('no project found, using standard registry'); @@ -60,39 +60,22 @@ const getProjectVariants = (projectDir?: string): ProjectVariants => { ...(projJson.get('sourceBehaviorOptions') ?? []), ]), ]; - if (Object.keys(registryCustomizations.types).length > 0) { - logger.debug( - `found registryCustomizations for types [${Object.keys(registryCustomizations.types).join( - ',' - )}] in ${projJson.getPath()}` - ); - } - if (presets.length > 0) { - logger.debug(`using sourceBehaviorOptions [${presets.join(',')}] in ${projJson.getPath()}`); - } - if (presets.length > 0 || Object.keys(registryCustomizations.types).length > 0) { - void Lifecycle.getInstance().emitTelemetry({ - library: 'SDR', - eventName: 'RegistryVariants', - presetCount: presets.length, - presets: presets.join(','), - customizationsCount: Object.keys(registryCustomizations.types).length, - customizationsTypes: Object.keys(registryCustomizations.types).join(','), - }); - } - return { - registryCustomizations, - presets: presets.map(loadPreset), - }; + return logProjectVariants( + { + registryCustomizations, + presets: presets.map(loadPreset), + }, + projJson.getPath() + ); }; -const mergeVariants = ({ registryCustomizations, presets }: ProjectVariants): MetadataRegistry => { - const registryFromPresets = [...(presets ?? []), registryCustomizations ?? emptyRegistry].reduce( +const mergeVariants = ({ registryCustomizations = emptyRegistry, presets }: ProjectVariants): MetadataRegistry => { + const registryFromPresets = [...(presets ?? []), registryCustomizations].reduce( (prev, curr) => firstLevelMerge(prev, curr), emptyRegistry ); - return firstLevelMerge(registryFromPresets, registryCustomizations ?? emptyRegistry); + return firstLevelMerge(registryFromPresets, registryCustomizations); }; const maybeGetProject = (projectDir?: string): SfProjectJson | undefined => { @@ -123,7 +106,7 @@ const emptyRegistry = { childTypes: {}, suffixes: {}, strictDirectoryNames: {}, -} satisfies MetadataRegistry; +} as const satisfies MetadataRegistry; /** merge the children of the top-level properties (ex: types, suffixes, etc) on 2 registries */ export const firstLevelMerge = (original: MetadataRegistry, overrides: MetadataRegistry): MetadataRegistry => ({ @@ -146,3 +129,28 @@ const removeEmptyStrings = (reg: MetadataRegistry): MetadataRegistry => ({ // presets can remove an entry by setting it to an empty string ex: { "childTypes": { "foo": "" } } const removeEmptyString = (obj: Record): Record => Object.fromEntries(Object.entries(obj).filter(([, v]) => v !== '')); + +// returns the projectVariants passed in. Side effects: logger and telemetry only +const logProjectVariants = (variants: ProjectVariants, projectDir: string): ProjectVariants => { + const customizationTypes = Object.keys(variants.registryCustomizations?.types ?? {}); + const logger = Logger.childFromRoot('variants:logProjectVariants'); + if (customizationTypes.length) { + logger.debug(`found registryCustomizations for types [${customizationTypes.join(',')}] in ${projectDir}`); + } + if (variants.presets?.length) { + logger.debug(`using sourceBehaviorOptions [${variants.presets.join(',')}] in ${projectDir}`); + } + if (variants?.presets ?? Object.keys(variants.registryCustomizations?.types ?? {}).length > 0) { + void Lifecycle.getInstance().emitTelemetry({ + library: 'SDR', + eventName: 'RegistryVariants', + presetCount: variants.presets?.length ?? 0, + presets: variants.presets?.join(','), + customizationsCount: customizationTypes.length, + customizationsTypes: customizationTypes.join(','), + }); + } else { + logger.debug(`no registryCustomizations or sourceBehaviorOptions found in ${projectDir}`); + } + return variants; +}; From 715e8803fb80abe0ccdc5c739ac7659a5f154490 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 13 Aug 2024 15:58:30 -0500 Subject: [PATCH 23/30] chore: bump core --- package.json | 4 ++-- yarn.lock | 41 ++++++++--------------------------------- 2 files changed, 10 insertions(+), 35 deletions(-) diff --git a/package.json b/package.json index d46922b15f..9415d1b825 100644 --- a/package.json +++ b/package.json @@ -25,8 +25,8 @@ "node": ">=18.0.0" }, "dependencies": { - "@salesforce/core": "^8.3.0", - "@salesforce/kit": "^3.1.6", + "@salesforce/core": "^8.4.0", + "@salesforce/kit": "^3.2.1", "@salesforce/ts-types": "^2.0.12", "fast-levenshtein": "^3.0.0", "fast-xml-parser": "^4.4.1", diff --git a/yarn.lock b/yarn.lock index b285cfebd8..2a137e25b9 100644 --- a/yarn.lock +++ b/yarn.lock @@ -523,10 +523,10 @@ strip-ansi "6.0.1" ts-retry-promise "^0.8.1" -"@salesforce/core@^8.2.3", "@salesforce/core@^8.2.8", "@salesforce/core@^8.3.0": - version "8.3.0" - resolved "https://registry.yarnpkg.com/@salesforce/core/-/core-8.3.0.tgz#b61fb6c0c0dec5664ce12ba62ebe35136ae33878" - integrity sha512-HZchC42oGJ5RQsG9HpAb1bT7ohjB31ATDz46ryMvLngMmrfHnyzv2mlIi6UdYkJ/2meH2BJkibHi8paPrtF+/A== +"@salesforce/core@^8.2.3", "@salesforce/core@^8.2.8", "@salesforce/core@^8.4.0": + version "8.4.0" + resolved "https://registry.yarnpkg.com/@salesforce/core/-/core-8.4.0.tgz#d2ddfe07994c42b1e917e581e9cf47ad27b97a93" + integrity sha512-P+n0+Sp+v6voLTShW2E5sdF7gCUxEXJjNcc9Jtto0ZMyQesmQJ6WGpWmAuRoi+BVYc8OPSlEffndaYDAo/u73g== dependencies: "@jsforce/jsforce-node" "^3.4.0" "@salesforce/kit" "^3.1.6" @@ -584,7 +584,7 @@ typescript "^5.5.4" wireit "^0.14.5" -"@salesforce/kit@^3.1.6", "@salesforce/kit@^3.2.0": +"@salesforce/kit@^3.1.6", "@salesforce/kit@^3.2.0", "@salesforce/kit@^3.2.1": version "3.2.1" resolved "https://registry.yarnpkg.com/@salesforce/kit/-/kit-3.2.1.tgz#3de2c9ff52710a169fc887716d97c00d26065c56" integrity sha512-LrZH2F06XPLUTMXC3av6A0VDAJykUqRtYB6tTjAKzwS1gCnp6BEn6VyjZNg0Fg/Kfp6OTrMxiIgbUFsNehEV7A== @@ -5071,16 +5071,7 @@ srcset@^5.0.0: resolved "https://registry.yarnpkg.com/srcset/-/srcset-5.0.0.tgz#9df6c3961b5b44a02532ce6ae4544832609e2e3f" integrity sha512-SqEZaAEhe0A6ETEa9O1IhSPC7MdvehZtCnTR0AftXk3QhY2UNgb+NApFOUPZILXk/YTDfFxMTNJOBpzrJsEdIA== -"string-width-cjs@npm:string-width@^4.2.0": - version "4.2.3" - resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" - integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== - dependencies: - emoji-regex "^8.0.0" - is-fullwidth-code-point "^3.0.0" - strip-ansi "^6.0.1" - -string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: +"string-width-cjs@npm:string-width@^4.2.0", string-width@^4.1.0, string-width@^4.2.0, string-width@^4.2.3: version "4.2.3" resolved "https://registry.yarnpkg.com/string-width/-/string-width-4.2.3.tgz#269c7117d27b05ad2e536830a8ec895ef9c6d010" integrity sha512-wKyQRQpjJ0sIp62ErSZdGsjMJWsap5oRNihHhu6G7JVO/9jIB6UyevL+tXuOqrng8j/cxKTWyWUwvSTriiZz/g== @@ -5139,14 +5130,7 @@ string_decoder@~1.1.1: dependencies: safe-buffer "~5.1.0" -"strip-ansi-cjs@npm:strip-ansi@^6.0.1": - version "6.0.1" - resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" - integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== - dependencies: - ansi-regex "^5.0.1" - -strip-ansi@6.0.1, strip-ansi@^6.0.0, strip-ansi@^6.0.1: +"strip-ansi-cjs@npm:strip-ansi@^6.0.1", strip-ansi@6.0.1, strip-ansi@^6.0.0, strip-ansi@^6.0.1: version "6.0.1" resolved "https://registry.yarnpkg.com/strip-ansi/-/strip-ansi-6.0.1.tgz#9e26c63d30f53443e9489495b2105d37b67a85d9" integrity sha512-Y38VPSHcqkFrCpFnQ9vuSXmquuv5oXOKpGeT6aGrr3o3Gc9AlVa6JBfUSOCnbxGGZF+/0ooI7KrPuUSztUdU5A== @@ -5636,7 +5620,7 @@ workerpool@^6.5.1: resolved "https://registry.yarnpkg.com/workerpool/-/workerpool-6.5.1.tgz#060f73b39d0caf97c6db64da004cd01b4c099544" integrity sha512-Fs4dNYcsdpYSAfVxhnl1L5zTksjvOJxtC5hzMNl+1t9B8hTJTdKDyZ5ju7ztgPy+ft9tBFXoOlDNiOT9WUXZlA== -"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0": +"wrap-ansi-cjs@npm:wrap-ansi@^7.0.0", wrap-ansi@^7.0.0: version "7.0.0" resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== @@ -5654,15 +5638,6 @@ wrap-ansi@^6.2.0: string-width "^4.1.0" strip-ansi "^6.0.0" -wrap-ansi@^7.0.0: - version "7.0.0" - resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-7.0.0.tgz#67e145cff510a6a6984bdf1152911d69d2eb9e43" - integrity sha512-YVGIj2kamLSTxw6NsZjoBxfSwsn0ycdesmc4p+Q21c5zPuZ1pl+NfxVdxPtdHvmNVOQ6XSYG4AUtyt/Fi7D16Q== - dependencies: - ansi-styles "^4.0.0" - string-width "^4.1.0" - strip-ansi "^6.0.0" - wrap-ansi@^8.1.0: version "8.1.0" resolved "https://registry.yarnpkg.com/wrap-ansi/-/wrap-ansi-8.1.0.tgz#56dc22368ee570face1b49819975d9b9a5ead214" From f4d786a10218e9694a3e01db2e3557c076e3de49 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 13 Aug 2024 15:59:36 -0500 Subject: [PATCH 24/30] chore: allow v1 and v2 of CustomLabelsBeta --- src/collections/componentSetBuilder.ts | 7 ------- 1 file changed, 7 deletions(-) diff --git a/src/collections/componentSetBuilder.ts b/src/collections/componentSetBuilder.ts index d2c6a71ef1..578d5f06f6 100644 --- a/src/collections/componentSetBuilder.ts +++ b/src/collections/componentSetBuilder.ts @@ -284,13 +284,6 @@ export const entryToTypeAndName = // split on the first colon, and then join the rest back together to support names that include colons const [typeName, ...name] = rawEntry.split(':'); const type = reg.getTypeByName(typeName.trim()); - const parent = reg.getParentType(type.name); - // If a user is requesting a child type that is unaddressable (more common with custom registries to create proper behavior) - // throw an error letting them know to use the entire parent instead - // or if they're requesting a COFT, unadressable without parent, don't throw because the parent could be requested - we don't know at this point - if (type.isAddressable === false && parent !== undefined && !type.unaddressableWithoutParent) { - throw new Error(`Cannot use this type, instead use ${parent.name}`); - } if (type.name === 'CustomLabels' && type.strategies?.transformer === 'decomposedLabels') { throw new Error('Use CustomLabel instead of CustomLabels for decomposed labels'); } From e3554fcd30a5f542cdc4cec774e3fb9c99b5bcb7 Mon Sep 17 00:00:00 2001 From: svc-cli-bot Date: Tue, 13 Aug 2024 21:27:35 +0000 Subject: [PATCH 25/30] chore(release): 12.1.13-qa.0 [skip ci] --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 620b154cb0..4fed26e1d6 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@salesforce/source-deploy-retrieve", - "version": "12.1.12", + "version": "12.1.13-qa.0", "description": "JavaScript library to run Salesforce metadata deploys and retrieves", "main": "lib/src/index.js", "author": "Salesforce", From aeca46019a55ab71798f202a145cdda6749a235f Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 13 Aug 2024 16:46:03 -0500 Subject: [PATCH 26/30] test: don't test with both CL presets together --- test/registry/presetTesting.ts | 2 ++ 1 file changed, 2 insertions(+) diff --git a/test/registry/presetTesting.ts b/test/registry/presetTesting.ts index 2ac10e97c0..7c9ce16eb5 100644 --- a/test/registry/presetTesting.ts +++ b/test/registry/presetTesting.ts @@ -19,6 +19,8 @@ type RegistryIterator = { const registriesFromPresets = fs .readdirSync(presetFolder, { withFileTypes: true }) .filter((file) => file.name.endsWith('.json')) + // we don't want to test the original preset since it conflicts with the CustomLabelsBeta2 + .filter((file) => !file.name.endsWith('CustomLabelsBeta.json')) .map((file) => ({ name: file.name, registry: JSON.parse(fs.readFileSync(path.join(file.path, file.name), 'utf-8')) as MetadataRegistry, From 641e4581585d620e0a685b07539f537ce596f1cd Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 13 Aug 2024 16:51:27 -0500 Subject: [PATCH 27/30] fix: only emit variant telemetry when there are presets/variants --- src/registry/variants.ts | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/src/registry/variants.ts b/src/registry/variants.ts index fcfb393f01..7846cc81b7 100644 --- a/src/registry/variants.ts +++ b/src/registry/variants.ts @@ -140,7 +140,7 @@ const logProjectVariants = (variants: ProjectVariants, projectDir: string): Proj if (variants.presets?.length) { logger.debug(`using sourceBehaviorOptions [${variants.presets.join(',')}] in ${projectDir}`); } - if (variants?.presets ?? Object.keys(variants.registryCustomizations?.types ?? {}).length > 0) { + if (variants?.presets?.length ?? customizationTypes.length) { void Lifecycle.getInstance().emitTelemetry({ library: 'SDR', eventName: 'RegistryVariants', From 7cc14598dd41f4062509bf9941681233405f419f Mon Sep 17 00:00:00 2001 From: svc-cli-bot Date: Tue, 13 Aug 2024 21:52:32 +0000 Subject: [PATCH 28/30] chore(release): 12.1.13-qa.1 [skip ci] --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index d169b461e5..ac0a61adf8 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@salesforce/source-deploy-retrieve", - "version": "12.1.13-qa.0", + "version": "12.1.13-qa.1", "description": "JavaScript library to run Salesforce metadata deploys and retrieves", "main": "lib/src/index.js", "author": "Salesforce", From db9b23c7efea30b3bf0e13a5f747e5ba14c246a2 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 13 Aug 2024 17:14:51 -0500 Subject: [PATCH 29/30] chore: manually bump pjson version --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index ac0a61adf8..3a343ce699 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@salesforce/source-deploy-retrieve", - "version": "12.1.13-qa.1", + "version": "12.3.0-qa.0", "description": "JavaScript library to run Salesforce metadata deploys and retrieves", "main": "lib/src/index.js", "author": "Salesforce", From 15162a9d3094d7c7b91b5c22697df7c82aff8a4c Mon Sep 17 00:00:00 2001 From: svc-cli-bot Date: Tue, 13 Aug 2024 22:15:46 +0000 Subject: [PATCH 30/30] chore(release): 12.3.0-qa.1 [skip ci] --- package.json | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/package.json b/package.json index 3a343ce699..ac06f110c2 100644 --- a/package.json +++ b/package.json @@ -1,6 +1,6 @@ { "name": "@salesforce/source-deploy-retrieve", - "version": "12.3.0-qa.0", + "version": "12.3.0-qa.1", "description": "JavaScript library to run Salesforce metadata deploys and retrieves", "main": "lib/src/index.js", "author": "Salesforce",