UNPKG

@aws-cdk/cloudformation-diff

Version:

Utilities to diff CDK stacks against CloudFormation templates

497 lines 71.5 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.ResourceDifference = exports.ResourceImpact = exports.ParameterDifference = exports.OutputDifference = exports.MetadataDifference = exports.MappingDifference = exports.ConditionDifference = exports.DifferenceCollection = exports.PropertyDifference = exports.Difference = exports.TemplateDiff = void 0; exports.isPropertyDifference = isPropertyDifference; const assert_1 = require("assert"); const service_spec_types_1 = require("@aws-cdk/service-spec-types"); const util_1 = require("./util"); const iam_changes_1 = require("../iam/iam-changes"); const security_group_changes_1 = require("../network/security-group-changes"); /** Semantic differences between two CloudFormation templates. */ class TemplateDiff { constructor(args) { if (args.awsTemplateFormatVersion !== undefined) { this.awsTemplateFormatVersion = args.awsTemplateFormatVersion; } if (args.description !== undefined) { this.description = args.description; } if (args.transform !== undefined) { this.transform = args.transform; } this.conditions = args.conditions || new DifferenceCollection({}); this.mappings = args.mappings || new DifferenceCollection({}); this.metadata = args.metadata || new DifferenceCollection({}); this.outputs = args.outputs || new DifferenceCollection({}); this.parameters = args.parameters || new DifferenceCollection({}); this.resources = args.resources || new DifferenceCollection({}); this.unknown = args.unknown || new DifferenceCollection({}); this.iamChanges = new iam_changes_1.IamChanges({ propertyChanges: this.scrutinizablePropertyChanges(iam_changes_1.IamChanges.IamPropertyScrutinies), resourceChanges: this.scrutinizableResourceChanges(iam_changes_1.IamChanges.IamResourceScrutinies), }); this.securityGroupChanges = new security_group_changes_1.SecurityGroupChanges({ egressRulePropertyChanges: this.scrutinizablePropertyChanges([service_spec_types_1.PropertyScrutinyType.EgressRules]), ingressRulePropertyChanges: this.scrutinizablePropertyChanges([service_spec_types_1.PropertyScrutinyType.IngressRules]), egressRuleResourceChanges: this.scrutinizableResourceChanges([service_spec_types_1.ResourceScrutinyType.EgressRuleResource]), ingressRuleResourceChanges: this.scrutinizableResourceChanges([service_spec_types_1.ResourceScrutinyType.IngressRuleResource]), }); } get differenceCount() { let count = 0; if (this.awsTemplateFormatVersion !== undefined) { count += 1; } if (this.description !== undefined) { count += 1; } if (this.transform !== undefined) { count += 1; } count += this.conditions.differenceCount; count += this.mappings.differenceCount; count += this.metadata.differenceCount; count += this.outputs.differenceCount; count += this.parameters.differenceCount; count += this.resources.differenceCount; count += this.unknown.differenceCount; return count; } get isEmpty() { return this.differenceCount === 0; } /** * Return true if any of the permissions objects involve a broadening of permissions */ get permissionsBroadened() { return this.iamChanges.permissionsBroadened || this.securityGroupChanges.rulesAdded; } /** * Return true if any of the permissions objects have changed */ get permissionsAnyChanges() { return this.iamChanges.hasChanges || this.securityGroupChanges.hasChanges; } /** * Return all property changes of a given scrutiny type * * We don't just look at property updates; we also look at resource additions and deletions (in which * case there is no further detail on property values), and resource type changes. */ scrutinizablePropertyChanges(scrutinyTypes) { const ret = new Array(); for (const [resourceLogicalId, resourceChange] of Object.entries(this.resources.changes)) { if (resourceChange.resourceTypeChanged) { // we ignore resource type changes here, and handle them in scrutinizableResourceChanges() continue; } if (!resourceChange.resourceType) { // We use resourceChange.resourceType to loadResourceModel so that we can inspect the // properties of a resource even after the resource is removed from the template. continue; } const newTypeProps = (0, util_1.loadResourceModel)(resourceChange.resourceType)?.properties || {}; for (const [propertyName, prop] of Object.entries(newTypeProps)) { const propScrutinyType = prop.scrutinizable || service_spec_types_1.PropertyScrutinyType.None; if (scrutinyTypes.includes(propScrutinyType)) { ret.push({ resourceLogicalId, propertyName, resourceType: resourceChange.resourceType, scrutinyType: propScrutinyType, oldValue: resourceChange.oldProperties?.[propertyName], newValue: resourceChange.newProperties?.[propertyName], }); } } } return ret; } /** * Return all resource changes of a given scrutiny type * * We don't just look at resource updates; we also look at resource additions and deletions (in which * case there is no further detail on property values), and resource type changes. */ scrutinizableResourceChanges(scrutinyTypes) { const ret = new Array(); for (const [resourceLogicalId, resourceChange] of Object.entries(this.resources.changes)) { if (!resourceChange) { continue; } const commonProps = { oldProperties: resourceChange.oldProperties, newProperties: resourceChange.newProperties, resourceLogicalId, }; // changes to the Type of resources can happen when migrating from CFN templates that use Transforms if (resourceChange.resourceTypeChanged) { // Treat as DELETE+ADD if (resourceChange.oldResourceType) { const oldResourceModel = (0, util_1.loadResourceModel)(resourceChange.oldResourceType); if (oldResourceModel && this.resourceIsScrutinizable(oldResourceModel, scrutinyTypes)) { ret.push({ ...commonProps, newProperties: undefined, resourceType: resourceChange.oldResourceType, scrutinyType: oldResourceModel.scrutinizable, }); } } if (resourceChange.newResourceType) { const newResourceModel = (0, util_1.loadResourceModel)(resourceChange.newResourceType); if (newResourceModel && this.resourceIsScrutinizable(newResourceModel, scrutinyTypes)) { ret.push({ ...commonProps, oldProperties: undefined, resourceType: resourceChange.newResourceType, scrutinyType: newResourceModel.scrutinizable, }); } } } else { if (!resourceChange.resourceType) { continue; } const resourceModel = (0, util_1.loadResourceModel)(resourceChange.resourceType); if (resourceModel && this.resourceIsScrutinizable(resourceModel, scrutinyTypes)) { ret.push({ ...commonProps, resourceType: resourceChange.resourceType, scrutinyType: resourceModel.scrutinizable, }); } } } return ret; } resourceIsScrutinizable(res, scrutinyTypes) { return scrutinyTypes.includes(res.scrutinizable || service_spec_types_1.ResourceScrutinyType.None); } } exports.TemplateDiff = TemplateDiff; /** * Models an entity that changed between two versions of a CloudFormation template. */ class Difference { /** * @param oldValue - the old value, cannot be equal (to the sense of +deepEqual+) to +newValue+. * @param newValue - the new value, cannot be equal (to the sense of +deepEqual+) to +oldValue+. */ constructor(oldValue, newValue) { this.oldValue = oldValue; this.newValue = newValue; if (oldValue === undefined && newValue === undefined) { throw new assert_1.AssertionError({ message: 'oldValue and newValue are both undefined!' }); } this.isDifferent = !(0, util_1.deepEqual)(oldValue, newValue); } /** @returns +true+ if the element is new to the template. */ get isAddition() { return this.oldValue === undefined; } /** @returns +true+ if the element was removed from the template. */ get isRemoval() { return this.newValue === undefined; } /** @returns +true+ if the element was already in the template and is updated. */ get isUpdate() { return this.oldValue !== undefined && this.newValue !== undefined; } } exports.Difference = Difference; class PropertyDifference extends Difference { constructor(oldValue, newValue, args) { super(oldValue, newValue); this.changeImpact = args.changeImpact; } } exports.PropertyDifference = PropertyDifference; class DifferenceCollection { constructor(diffs) { this.diffs = diffs; } get changes() { return onlyChanges(this.diffs); } get differenceCount() { return Object.values(this.changes).length; } get(logicalId) { const ret = this.diffs[logicalId]; if (!ret) { throw new Error(`No object with logical ID '${logicalId}'`); } return ret; } remove(logicalId) { delete this.diffs[logicalId]; } get logicalIds() { return Object.keys(this.changes); } /** * Returns a new TemplateDiff which only contains changes for which `predicate` * returns `true`. */ filter(predicate) { const newChanges = {}; for (const id of Object.keys(this.changes)) { const diff = this.changes[id]; if (predicate(diff)) { newChanges[id] = diff; } } return new DifferenceCollection(newChanges); } /** * Invokes `cb` for all changes in this collection. * * Changes will be sorted as follows: * - Removed * - Added * - Updated * - Others * */ forEachDifference(cb) { const removed = new Array(); const added = new Array(); const updated = new Array(); const others = new Array(); for (const logicalId of this.logicalIds) { const change = this.changes[logicalId]; if (change.isAddition) { added.push({ logicalId, change }); } else if (change.isRemoval) { removed.push({ logicalId, change }); } else if (change.isUpdate) { updated.push({ logicalId, change }); } else if (change.isDifferent) { others.push({ logicalId, change }); } } removed.forEach(v => cb(v.logicalId, v.change)); added.forEach(v => cb(v.logicalId, v.change)); updated.forEach(v => cb(v.logicalId, v.change)); others.forEach(v => cb(v.logicalId, v.change)); } } exports.DifferenceCollection = DifferenceCollection; class ConditionDifference extends Difference { } exports.ConditionDifference = ConditionDifference; class MappingDifference extends Difference { } exports.MappingDifference = MappingDifference; class MetadataDifference extends Difference { } exports.MetadataDifference = MetadataDifference; class OutputDifference extends Difference { } exports.OutputDifference = OutputDifference; class ParameterDifference extends Difference { } exports.ParameterDifference = ParameterDifference; var ResourceImpact; (function (ResourceImpact) { /** The existing physical resource will be updated */ ResourceImpact["WILL_UPDATE"] = "WILL_UPDATE"; /** A new physical resource will be created */ ResourceImpact["WILL_CREATE"] = "WILL_CREATE"; /** The existing physical resource will be replaced */ ResourceImpact["WILL_REPLACE"] = "WILL_REPLACE"; /** The existing physical resource may be replaced */ ResourceImpact["MAY_REPLACE"] = "MAY_REPLACE"; /** The existing physical resource will be destroyed */ ResourceImpact["WILL_DESTROY"] = "WILL_DESTROY"; /** The existing physical resource will be removed from CloudFormation supervision */ ResourceImpact["WILL_ORPHAN"] = "WILL_ORPHAN"; /** The existing physical resource will be added to CloudFormation supervision */ ResourceImpact["WILL_IMPORT"] = "WILL_IMPORT"; /** There is no change in this resource */ ResourceImpact["NO_CHANGE"] = "NO_CHANGE"; })(ResourceImpact || (exports.ResourceImpact = ResourceImpact = {})); /** * This function can be used as a reducer to obtain the resource-level impact of a list * of property-level impacts. * @param one - the current worst impact so far. * @param two - the new impact being considered (can be undefined, as we may not always be * able to determine some properties impact). */ function worstImpact(one, two) { if (!two) { return one; } const badness = { [ResourceImpact.NO_CHANGE]: 0, [ResourceImpact.WILL_IMPORT]: 0, [ResourceImpact.WILL_UPDATE]: 1, [ResourceImpact.WILL_CREATE]: 2, [ResourceImpact.WILL_ORPHAN]: 3, [ResourceImpact.MAY_REPLACE]: 4, [ResourceImpact.WILL_REPLACE]: 5, [ResourceImpact.WILL_DESTROY]: 6, }; return badness[one] > badness[two] ? one : two; } /** * Change to a single resource between two CloudFormation templates * * This class can be mutated after construction. */ class ResourceDifference { constructor(oldValue, newValue, args) { this.oldValue = oldValue; this.newValue = newValue; this.resourceTypes = args.resourceType; this.propertyDiffs = args.propertyDiffs; this.otherDiffs = args.otherDiffs; this.isAddition = oldValue === undefined; this.isRemoval = newValue === undefined; this.isImport = undefined; } get oldProperties() { return this.oldValue && this.oldValue.Properties; } get newProperties() { return this.newValue && this.newValue.Properties; } /** * Whether this resource was modified at all */ get isDifferent() { return this.differenceCount > 0 || this.oldResourceType !== this.newResourceType; } /** * Whether the resource was updated in-place */ get isUpdate() { return this.isDifferent && !this.isAddition && !this.isRemoval; } get oldResourceType() { return this.resourceTypes.oldType; } get newResourceType() { return this.resourceTypes.newType; } /** * All actual property updates */ get propertyUpdates() { return onlyChanges(this.propertyDiffs); } /** * All actual "other" updates */ get otherChanges() { return onlyChanges(this.otherDiffs); } /** * Return whether the resource type was changed in this diff * * This is not a valid operation in CloudFormation but to be defensive we're going * to be aware of it anyway. */ get resourceTypeChanged() { return (this.resourceTypes.oldType !== undefined && this.resourceTypes.newType !== undefined && this.resourceTypes.oldType !== this.resourceTypes.newType); } /** * Return the resource type if it was unchanged * * If the resource type was changed, it's an error to call this. */ get resourceType() { if (this.resourceTypeChanged) { throw new Error('Cannot get .resourceType, because the type was changed'); } return this.resourceTypes.oldType || this.resourceTypes.newType; } /** * Replace a PropertyChange in this object * * This affects the property diff as it is summarized to users, but it DOES * NOT affect either the "oldValue" or "newValue" values; those still contain * the actual template values as provided by the user (they might still be * used for downstream processing). */ setPropertyChange(propertyName, change) { this.propertyDiffs[propertyName] = change; } /** * Replace a OtherChange in this object * * This affects the property diff as it is summarized to users, but it DOES * NOT affect either the "oldValue" or "newValue" values; those still contain * the actual template values as provided by the user (they might still be * used for downstream processing). */ setOtherChange(otherName, change) { this.otherDiffs[otherName] = change; } get changeImpact() { if (this.isImport) { return ResourceImpact.WILL_IMPORT; } // Check the Type first if (this.resourceTypes.oldType !== this.resourceTypes.newType) { if (this.resourceTypes.oldType === undefined) { return ResourceImpact.WILL_CREATE; } if (this.resourceTypes.newType === undefined) { return this.oldValue.DeletionPolicy === 'Retain' ? ResourceImpact.WILL_ORPHAN : ResourceImpact.WILL_DESTROY; } return ResourceImpact.WILL_REPLACE; } // Base impact (before we mix in the worst of the property impacts); // WILL_UPDATE if we have "other" changes, NO_CHANGE if there are no "other" changes. const baseImpact = Object.keys(this.otherChanges).length > 0 ? ResourceImpact.WILL_UPDATE : ResourceImpact.NO_CHANGE; return Object.values(this.propertyDiffs) .map(elt => elt.changeImpact) .reduce(worstImpact, baseImpact); } /** * Count of actual differences (not of elements) */ get differenceCount() { return Object.values(this.propertyUpdates).length + Object.values(this.otherChanges).length; } /** * Invoke a callback for each actual difference */ forEachDifference(cb) { for (const key of Object.keys(this.propertyUpdates).sort()) { cb('Property', key, this.propertyUpdates[key]); } for (const key of Object.keys(this.otherChanges).sort()) { cb('Other', key, this.otherDiffs[key]); } } } exports.ResourceDifference = ResourceDifference; function isPropertyDifference(diff) { return diff.changeImpact !== undefined; } /** * Filter a map of IDifferences down to only retain the actual changes */ function onlyChanges(xs) { const ret = {}; for (const [key, diff] of Object.entries(xs)) { if (diff.isDifferent) { ret[key] = diff; } } return ret; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"types.js","sourceRoot":"","sources":["types.ts"],"names":[],"mappings":"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","sourcesContent":["import { AssertionError } from 'assert';\nimport type { Resource as ResourceModel } from '@aws-cdk/service-spec-types';\nimport { PropertyScrutinyType, ResourceScrutinyType } from '@aws-cdk/service-spec-types';\nimport { deepEqual, loadResourceModel } from './util';\nimport { IamChanges } from '../iam/iam-changes';\nimport { SecurityGroupChanges } from '../network/security-group-changes';\n\nexport type PropertyMap = { [key: string]: any };\n\nexport type ChangeSetResources = { [logicalId: string]: ChangeSetResource };\n\n/**\n * @param beforeContext - is the BeforeContext field from the ChangeSet.ResourceChange.BeforeContext. This is the part of the CloudFormation template\n * that defines what the resource is before the change is applied; that is, BeforeContext is CloudFormationTemplate.Resources[LogicalId] before the ChangeSet is executed.\n *\n * @param afterContext - same as beforeContext but for after the change is made; that is, AfterContext is CloudFormationTemplate.Resources[LogicalId] after the ChangeSet is executed.\n *\n *  * Here is an example of what a beforeContext/afterContext looks like:\n *  '{\"Properties\":{\"Value\":\"sdflkja\",\"Type\":\"String\",\"Name\":\"mySsmParameterFromStack\"},\"Metadata\":{\"aws:cdk:path\":\"cdk/mySsmParameter/Resource\"}}'\n */\nexport interface ChangeSetResource {\n  resourceWasReplaced: boolean;\n  resourceType: string | undefined;\n  propertyReplacementModes: PropertyReplacementModeMap | undefined;\n}\n\nexport type PropertyReplacementModeMap = {\n  [propertyName: string]: {\n    replacementMode: ReplacementModes | undefined;\n  };\n};\n\n/**\n * 'Always' means that changing the corresponding property will always cause a resource replacement. Never means never. Conditionally means maybe.\n */\nexport type ReplacementModes = 'Always' | 'Never' | 'Conditionally';\n\n/** Semantic differences between two CloudFormation templates. */\nexport class TemplateDiff implements ITemplateDiff {\n  public awsTemplateFormatVersion?: Difference<string>;\n  public description?: Difference<string>;\n  public transform?: Difference<string>;\n  public conditions: DifferenceCollection<Condition, ConditionDifference>;\n  public mappings: DifferenceCollection<Mapping, MappingDifference>;\n  public metadata: DifferenceCollection<Metadata, MetadataDifference>;\n  public outputs: DifferenceCollection<Output, OutputDifference>;\n  public parameters: DifferenceCollection<Parameter, ParameterDifference>;\n  public resources: DifferenceCollection<Resource, ResourceDifference>;\n  /** The differences in unknown/unexpected parts of the template */\n  public unknown: DifferenceCollection<any, Difference<any>>;\n\n  /**\n   * Changes to IAM policies\n   */\n  public readonly iamChanges: IamChanges;\n\n  /**\n   * Changes to Security Group ingress and egress rules\n   */\n  public readonly securityGroupChanges: SecurityGroupChanges;\n\n  constructor(args: ITemplateDiff) {\n    if (args.awsTemplateFormatVersion !== undefined) {\n      this.awsTemplateFormatVersion = args.awsTemplateFormatVersion;\n    }\n    if (args.description !== undefined) {\n      this.description = args.description;\n    }\n    if (args.transform !== undefined) {\n      this.transform = args.transform;\n    }\n\n    this.conditions = args.conditions || new DifferenceCollection({});\n    this.mappings = args.mappings || new DifferenceCollection({});\n    this.metadata = args.metadata || new DifferenceCollection({});\n    this.outputs = args.outputs || new DifferenceCollection({});\n    this.parameters = args.parameters || new DifferenceCollection({});\n    this.resources = args.resources || new DifferenceCollection({});\n    this.unknown = args.unknown || new DifferenceCollection({});\n\n    this.iamChanges = new IamChanges({\n      propertyChanges: this.scrutinizablePropertyChanges(IamChanges.IamPropertyScrutinies),\n      resourceChanges: this.scrutinizableResourceChanges(IamChanges.IamResourceScrutinies),\n    });\n\n    this.securityGroupChanges = new SecurityGroupChanges({\n      egressRulePropertyChanges: this.scrutinizablePropertyChanges([PropertyScrutinyType.EgressRules]),\n      ingressRulePropertyChanges: this.scrutinizablePropertyChanges([PropertyScrutinyType.IngressRules]),\n      egressRuleResourceChanges: this.scrutinizableResourceChanges([ResourceScrutinyType.EgressRuleResource]),\n      ingressRuleResourceChanges: this.scrutinizableResourceChanges([ResourceScrutinyType.IngressRuleResource]),\n    });\n  }\n\n  public get differenceCount() {\n    let count = 0;\n\n    if (this.awsTemplateFormatVersion !== undefined) {\n      count += 1;\n    }\n    if (this.description !== undefined) {\n      count += 1;\n    }\n    if (this.transform !== undefined) {\n      count += 1;\n    }\n\n    count += this.conditions.differenceCount;\n    count += this.mappings.differenceCount;\n    count += this.metadata.differenceCount;\n    count += this.outputs.differenceCount;\n    count += this.parameters.differenceCount;\n    count += this.resources.differenceCount;\n    count += this.unknown.differenceCount;\n\n    return count;\n  }\n\n  public get isEmpty(): boolean {\n    return this.differenceCount === 0;\n  }\n\n  /**\n   * Return true if any of the permissions objects involve a broadening of permissions\n   */\n  public get permissionsBroadened(): boolean {\n    return this.iamChanges.permissionsBroadened || this.securityGroupChanges.rulesAdded;\n  }\n\n  /**\n   * Return true if any of the permissions objects have changed\n   */\n  public get permissionsAnyChanges(): boolean {\n    return this.iamChanges.hasChanges || this.securityGroupChanges.hasChanges;\n  }\n\n  /**\n   * Return all property changes of a given scrutiny type\n   *\n   * We don't just look at property updates; we also look at resource additions and deletions (in which\n   * case there is no further detail on property values), and resource type changes.\n   */\n  private scrutinizablePropertyChanges(scrutinyTypes: PropertyScrutinyType[]): PropertyChange[] {\n    const ret = new Array<PropertyChange>();\n\n    for (const [resourceLogicalId, resourceChange] of Object.entries(this.resources.changes)) {\n      if (resourceChange.resourceTypeChanged) {\n        // we ignore resource type changes here, and handle them in scrutinizableResourceChanges()\n        continue;\n      }\n\n      if (!resourceChange.resourceType) {\n        // We use resourceChange.resourceType to loadResourceModel so that we can inspect the\n        // properties of a resource even after the resource is removed from the template.\n        continue;\n      }\n\n      const newTypeProps = loadResourceModel(resourceChange.resourceType)?.properties || {};\n      for (const [propertyName, prop] of Object.entries(newTypeProps)) {\n        const propScrutinyType = prop.scrutinizable || PropertyScrutinyType.None;\n        if (scrutinyTypes.includes(propScrutinyType)) {\n          ret.push({\n            resourceLogicalId,\n            propertyName,\n            resourceType: resourceChange.resourceType,\n            scrutinyType: propScrutinyType,\n            oldValue: resourceChange.oldProperties?.[propertyName],\n            newValue: resourceChange.newProperties?.[propertyName],\n          });\n        }\n      }\n    }\n\n    return ret;\n  }\n\n  /**\n   * Return all resource changes of a given scrutiny type\n   *\n   * We don't just look at resource updates; we also look at resource additions and deletions (in which\n   * case there is no further detail on property values), and resource type changes.\n   */\n  private scrutinizableResourceChanges(scrutinyTypes: ResourceScrutinyType[]): ResourceChange[] {\n    const ret = new Array<ResourceChange>();\n\n    for (const [resourceLogicalId, resourceChange] of Object.entries(this.resources.changes)) {\n      if (!resourceChange) {\n        continue;\n      }\n\n      const commonProps = {\n        oldProperties: resourceChange.oldProperties,\n        newProperties: resourceChange.newProperties,\n        resourceLogicalId,\n      };\n\n      // changes to the Type of resources can happen when migrating from CFN templates that use Transforms\n      if (resourceChange.resourceTypeChanged) {\n        // Treat as DELETE+ADD\n        if (resourceChange.oldResourceType) {\n          const oldResourceModel = loadResourceModel(resourceChange.oldResourceType);\n          if (oldResourceModel && this.resourceIsScrutinizable(oldResourceModel, scrutinyTypes)) {\n            ret.push({\n              ...commonProps,\n              newProperties: undefined,\n              resourceType: resourceChange.oldResourceType!,\n              scrutinyType: oldResourceModel.scrutinizable!,\n            });\n          }\n        }\n\n        if (resourceChange.newResourceType) {\n          const newResourceModel = loadResourceModel(resourceChange.newResourceType);\n          if (newResourceModel && this.resourceIsScrutinizable(newResourceModel, scrutinyTypes)) {\n            ret.push({\n              ...commonProps,\n              oldProperties: undefined,\n              resourceType: resourceChange.newResourceType!,\n              scrutinyType: newResourceModel.scrutinizable!,\n            });\n          }\n        }\n      } else {\n        if (!resourceChange.resourceType) {\n          continue;\n        }\n\n        const resourceModel = loadResourceModel(resourceChange.resourceType);\n        if (resourceModel && this.resourceIsScrutinizable(resourceModel, scrutinyTypes)) {\n          ret.push({\n            ...commonProps,\n            resourceType: resourceChange.resourceType,\n            scrutinyType: resourceModel.scrutinizable!,\n          });\n        }\n      }\n    }\n\n    return ret;\n  }\n\n  private resourceIsScrutinizable(res: ResourceModel, scruti