From da3448a9c019eb80c5b610ad1d5b137d12ddf658 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 4 Jun 2024 13:23:17 -0500 Subject: [PATCH 1/5] chore: oclif perf import from top-level --- package.json | 7 +- src/sourceTracking.ts | 2 +- yarn.lock | 263 ++++++++++++------------------------------ 3 files changed, 81 insertions(+), 191 deletions(-) diff --git a/package.json b/package.json index a0322f47..28cef31b 100644 --- a/package.json +++ b/package.json @@ -49,10 +49,10 @@ "node": ">=18.0.0" }, "dependencies": { - "@oclif/core": "^3.26.6", - "@salesforce/core": "^7.3.9", + "@oclif/core": "^4.0.0", + "@salesforce/core": "^7.3.10", "@salesforce/kit": "^3.1.2", - "@salesforce/source-deploy-retrieve": "^11.6.3", + "@salesforce/source-deploy-retrieve": "^11.6.5", "@salesforce/ts-types": "^2.0.9", "fast-xml-parser": "^4.3.6", "graceful-fs": "^4.2.11", @@ -62,6 +62,7 @@ "devDependencies": { "@salesforce/cli-plugins-testkit": "^5.3.4", "@salesforce/dev-scripts": "^9.1.2", + "@salesforce/schemas": "^1.9.0", "@types/graceful-fs": "^4.1.9", "eslint-plugin-sf-plugin": "^1.18.4", "ts-node": "^10.9.2", diff --git a/src/sourceTracking.ts b/src/sourceTracking.ts index 30484a05..c85b4f38 100644 --- a/src/sourceTracking.ts +++ b/src/sourceTracking.ts @@ -28,7 +28,7 @@ import { } from '@salesforce/source-deploy-retrieve'; // this is not exported by SDR (see the comments in SDR regarding its limitations) import { filePathsFromMetadataComponent } from '@salesforce/source-deploy-retrieve/lib/src/utils/filePathGenerator'; -import { Performance } from '@oclif/core'; +import { Performance } from '@oclif/core/performance'; import { RemoteSourceTrackingService, remoteChangeElementToChangeResult } from './shared/remoteSourceTrackingService'; import { ShadowRepo } from './shared/local/localShadowRepo'; import { throwIfConflicts, findConflictsInComponentSet, getDedupedConflictsFromChanges } from './shared/conflicts'; diff --git a/yarn.lock b/yarn.lock index dfca3dfb..0c5cb6da 100644 --- a/yarn.lock +++ b/yarn.lock @@ -543,36 +543,25 @@ "@nodelib/fs.scandir" "2.1.5" fastq "^1.6.0" -"@oclif/core@^3.26.6": - version "3.26.6" - resolved "https://registry.yarnpkg.com/@oclif/core/-/core-3.26.6.tgz#f371868cfa0fe150a6547e6af98b359065d2f971" - integrity sha512-+FiTw1IPuJTF9tSAlTsY8bGK4sgthehjz7c2SvYdgQncTkxI2xvUch/8QpjNYGLEmUneNygvYMRBax2KJcLccA== +"@oclif/core@^4.0.0": + version "4.0.0" + resolved "https://registry.yarnpkg.com/@oclif/core/-/core-4.0.0.tgz#c47783c9803cb440e3fd325157437400765eae3f" + integrity sha512-BMWGvJrzn5PnG60gTNFEvaBT0jvGNiJCKN4aJBYP6E7Bq/Y5XPnxPrkj7ZZs/Jsd1oVn6K/JRmF6gWpv72DOew== dependencies: - "@types/cli-progress" "^3.11.5" ansi-escapes "^4.3.2" - ansi-styles "^4.3.0" - cardinal "^2.1.1" - chalk "^4.1.2" + ansis "^3.1.1" clean-stack "^3.0.1" - cli-progress "^3.12.0" - color "^4.2.3" - debug "^4.3.4" + cli-spinners "^2.9.2" + cosmiconfig "^9.0.0" + debug "^4.3.5" ejs "^3.1.10" get-package-type "^0.1.0" globby "^11.1.0" - hyperlinker "^1.0.0" indent-string "^4.0.0" is-wsl "^2.2.0" - js-yaml "^3.14.1" minimatch "^9.0.4" - natural-orderby "^2.0.3" - object-treeify "^1.1.33" - password-prompt "^1.1.3" - slice-ansi "^4.0.0" string-width "^4.2.3" - strip-ansi "^6.0.1" - supports-color "^8.1.1" - supports-hyperlinks "^2.2.0" + supports-color "^8" widest-line "^3.1.0" wordwrap "^1.0.0" wrap-ansi "^7.0.0" @@ -598,16 +587,16 @@ strip-ansi "6.0.1" ts-retry-promise "^0.8.0" -"@salesforce/core@^7.3.5", "@salesforce/core@^7.3.8", "@salesforce/core@^7.3.9": - version "7.3.9" - resolved "https://registry.yarnpkg.com/@salesforce/core/-/core-7.3.9.tgz#8abe2b3e2393989d11e92b7a6b96043fc9d5b9c8" - integrity sha512-eJqDiA5b7wU50Ee/xjmGzSnHrNVJ8S77B7enfX30gm7gxU3i3M3QeBdiV6XAOPLSIL96DseofP6Tv6c+rljlKA== +"@salesforce/core@^7.3.10", "@salesforce/core@^7.3.5", "@salesforce/core@^7.3.8", "@salesforce/core@^7.3.9": + version "7.3.10" + resolved "https://registry.yarnpkg.com/@salesforce/core/-/core-7.3.10.tgz#83da85c4e93ca625e2c13118aad9c1df2931bc0f" + integrity sha512-kEKoqkmhWNoiucAE3Ylv6FpC4iVgk4aE0dmcwSmNrMjxSbtjQJGUybprfO/itrLJv+56eM7/4FARQQ2gDbRzQQ== dependencies: "@jsforce/jsforce-node" "^3.2.0" "@salesforce/kit" "^3.1.1" "@salesforce/schemas" "^1.9.0" "@salesforce/ts-types" "^2.0.9" - ajv "^8.13.0" + ajv "^8.15.0" change-case "^4.1.2" faye "^1.4.0" form-data "^4.0.0" @@ -615,11 +604,11 @@ jsonwebtoken "9.0.2" jszip "3.10.1" pino "^8.21.0" - pino-abstract-transport "^1.1.0" + pino-abstract-transport "^1.2.0" pino-pretty "^10.3.1" proper-lockfile "^4.1.2" semver "^7.6.2" - ts-retry-promise "^0.7.1" + ts-retry-promise "^0.8.1" "@salesforce/dev-config@^4.1.0": version "4.1.0" @@ -676,10 +665,10 @@ resolved "https://registry.yarnpkg.com/@salesforce/schemas/-/schemas-1.9.0.tgz#ba477a112653a20b4edcf989c61c57bdff9aa3ca" integrity sha512-LiN37zG5ODT6z70sL1fxF7BQwtCX9JOWofSU8iliSNIM+WDEeinnoFtVqPInRSNt8I0RiJxIKCrqstsmQRBNvA== -"@salesforce/source-deploy-retrieve@^11.6.3": - version "11.6.3" - resolved "https://registry.yarnpkg.com/@salesforce/source-deploy-retrieve/-/source-deploy-retrieve-11.6.3.tgz#24ff55a17f8d7862b1467e40968707fedaa3e624" - integrity sha512-6h/KJV8cRfzLZ3aE6PP76oP1bz28zRxL9wtCx6yD+jz4GRdIxHNjNQhiKJJI9m2Z+npWl/sjth8aHB7rrSgSrg== +"@salesforce/source-deploy-retrieve@^11.6.5": + version "11.6.5" + resolved "https://registry.yarnpkg.com/@salesforce/source-deploy-retrieve/-/source-deploy-retrieve-11.6.5.tgz#5a35edc42b9f0fc32384ce2515cecea6ccf3c459" + integrity sha512-JwYhLOLAkyIBsZXMt9OnDUqI8BJRu6sr5cs6EXfOD4rpIrEqpNN3+m9Gnkoe3onmTBUg4AuxFTuBT8SaBorw2g== dependencies: "@salesforce/core" "^7.3.9" "@salesforce/kit" "^3.1.1" @@ -811,13 +800,6 @@ resolved "https://registry.yarnpkg.com/@types/chai/-/chai-4.3.14.tgz#ae3055ea2be43c91c9fd700a36d67820026d96e6" integrity sha512-Wj71sXE4Q4AkGdG9Tvq1u/fquNz9EdG4LIJMwVVII7ashjD/8cf8fyIfJAjRr6YcsXnSE8cOGQPq1gqeR8z+3w== -"@types/cli-progress@^3.11.5": - version "3.11.5" - resolved "https://registry.yarnpkg.com/@types/cli-progress/-/cli-progress-3.11.5.tgz#9518c745e78557efda057e3f96a5990c717268c3" - integrity sha512-D4PbNRbviKyppS5ivBGyFO29POlySLmA2HyUFE4p5QGazAMM3CwkKWcvTl8gvElSuxRh6FPKL8XmidX873ou4g== - dependencies: - "@types/node" "*" - "@types/glob@~7.2.0": version "7.2.0" resolved "https://registry.yarnpkg.com/@types/glob/-/glob-7.2.0.tgz#bc1b5bf3aa92f25bd5dd39f35c57361bdce5b2eb" @@ -1067,15 +1049,15 @@ ajv@^6.12.4: json-schema-traverse "^0.4.1" uri-js "^4.2.2" -ajv@^8.11.0, ajv@^8.13.0: - version "8.13.0" - resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.13.0.tgz#a3939eaec9fb80d217ddf0c3376948c023f28c91" - integrity sha512-PRA911Blj99jR5RMeTunVbNXMF6Lp4vZXnk5GQjcnUWUTsrXtekg/pnmFFI2u/I36Y/2bITGS30GZCXei6uNkA== +ajv@^8.11.0, ajv@^8.15.0: + version "8.15.0" + resolved "https://registry.yarnpkg.com/ajv/-/ajv-8.15.0.tgz#d918c661e3e820bbbc65a320e182ee56a1aa978a" + integrity sha512-15BTtQUOsSrmHCy+B4VnAiJAJxJ8IFgu6fcjFQF3jQYZ78nLSQthlFg4ehp+NLIyfvFgOlxNsjKIEhydtFPVHQ== dependencies: fast-deep-equal "^3.1.3" + fast-uri "^2.3.0" json-schema-traverse "^1.0.0" require-from-string "^2.0.2" - uri-js "^4.4.1" ansi-colors@4.1.1: version "4.1.1" @@ -1111,7 +1093,7 @@ ansi-styles@^3.2.1: dependencies: color-convert "^1.9.0" -ansi-styles@^4.0.0, ansi-styles@^4.1.0, ansi-styles@^4.3.0: +ansi-styles@^4.0.0, ansi-styles@^4.1.0: version "4.3.0" resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-4.3.0.tgz#edd803628ae71c04c85ae7a0906edad34b648937" integrity sha512-zbB9rCJAT1rbjiVDb2hqKFHNYLxgtk8NURxZ3IZwD3F6NtxbXZQCnnSi1Lkx+IDohdPlFp222wVALIheZJQSEg== @@ -1123,10 +1105,10 @@ ansi-styles@^6.1.0: resolved "https://registry.yarnpkg.com/ansi-styles/-/ansi-styles-6.2.1.tgz#0e62320cf99c21afff3b3012192546aacbfb05c5" integrity sha512-bN798gFfQX+viw3R7yrGWRqnrN2oRkEkUjjl4JNn4E8GxxbjtG3FbrEIIY3l8/hrwUwIeCZvi4QuOTP4MErVug== -ansicolors@~0.3.2: - version "0.3.2" - resolved "https://registry.yarnpkg.com/ansicolors/-/ansicolors-0.3.2.tgz#665597de86a9ffe3aa9bfbe6cae5c6ea426b4979" - integrity sha512-QXu7BPrP29VllRxH8GwB7x5iX5qWKAAMLqKQGWTeLWVlNHNOpVMJ91dsxQAIWXpjuW5wqvxu3Jd/nRjrJ+0pqg== +ansis@^3.1.1: + version "3.2.0" + resolved "https://registry.yarnpkg.com/ansis/-/ansis-3.2.0.tgz#0e050c5be94784f32ffdac4b84fccba064aeae4b" + integrity sha512-Yk3BkHH9U7oPyCN3gL5Tc7CpahG/+UFv/6UG03C311Vy9lzRmA5uoxDTpU9CO3rGHL6KzJz/pdDeXZCZ5Mu/Sg== anymatch@~3.1.2: version "3.1.3" @@ -1264,11 +1246,6 @@ ast-types@^0.13.4: dependencies: tslib "^2.0.1" -astral-regex@^2.0.0: - version "2.0.0" - resolved "https://registry.yarnpkg.com/astral-regex/-/astral-regex-2.0.0.tgz#483143c567aeed4785759c0865786dc77d7d2e31" - integrity sha512-Z7tMw1ytTXt5jqMcOP+OQteU1VuNK9Y02uuJtKQ1Sv69jXQKKg5cibLwGJow8yzZP+eAc18EmLGPal0bp36rvQ== - async-lock@^1.4.1: version "1.4.1" resolved "https://registry.yarnpkg.com/async-lock/-/async-lock-1.4.1.tgz#56b8718915a9b68b10fce2f2a9a3dddf765ef53f" @@ -1462,14 +1439,6 @@ capital-case@^1.0.4: tslib "^2.0.3" upper-case-first "^2.0.2" -cardinal@^2.1.1: - version "2.1.1" - resolved "https://registry.yarnpkg.com/cardinal/-/cardinal-2.1.1.tgz#7cc1055d822d212954d07b085dea251cc7bc5505" - integrity sha512-JSr5eOgoEymtYHBjNWyjrMqet9Am2miJhlfKNdqLp6zoeAh0KN5dRAcxlecj5mAJrmQomgiOBj35xHLrFjqBpw== - dependencies: - ansicolors "~0.3.2" - redeyed "~2.1.0" - chai@^4.3.10: version "4.3.10" resolved "https://registry.yarnpkg.com/chai/-/chai-4.3.10.tgz#d784cec635e3b7e2ffb66446a63b4e33bd390384" @@ -1574,12 +1543,10 @@ clean-stack@^3.0.1: dependencies: escape-string-regexp "4.0.0" -cli-progress@^3.12.0: - version "3.12.0" - resolved "https://registry.yarnpkg.com/cli-progress/-/cli-progress-3.12.0.tgz#807ee14b66bcc086258e444ad0f19e7d42577942" - integrity sha512-tRkV3HJ1ASwm19THiiLIXLO7Im7wlTuKnvkYaTkyoAPefqjNg7W7DHKUlGRxy9vxDvbyCYQkQozvptuMkGCg8A== - dependencies: - string-width "^4.2.3" +cli-spinners@^2.9.2: + version "2.9.2" + resolved "https://registry.yarnpkg.com/cli-spinners/-/cli-spinners-2.9.2.tgz#1773a8f4b9c4d6ac31563df53b3fc1d79462fe41" + integrity sha512-ywqV+5MmyL4E7ybXgKys4DugZbX0FC6LnwrhjuykIjnK9k8OQacQ7axGKnjDXWNhns0xot3bZI5h55H8yo9cJg== cliui@^6.0.0: version "6.0.0" @@ -1634,27 +1601,11 @@ color-name@1.1.3: resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.3.tgz#a7d0558bd89c42f795dd42328f740831ca53bc25" integrity sha512-72fSenhMw2HZMTVHeCA9KCmpEIbzWiQsjN+BHcBbS9vr1mtt+vJjPdksIBNUmKAW8TFUDPJK5SUU3QhE9NEXDw== -color-name@^1.0.0, color-name@~1.1.4: +color-name@~1.1.4: version "1.1.4" resolved "https://registry.yarnpkg.com/color-name/-/color-name-1.1.4.tgz#c2a09a87acbde69543de6f63fa3995c826c536a2" integrity sha512-dOy+3AuW3a2wNbZHIuMZpTcgjGuLU/uBL/ubcZF9OXbDo8ff4O8yVp5Bf0efS8uEoYo5q4Fx7dY9OgQGXgAsQA== -color-string@^1.9.0: - version "1.9.1" - resolved "https://registry.yarnpkg.com/color-string/-/color-string-1.9.1.tgz#4467f9146f036f855b764dfb5bf8582bf342c7a4" - integrity sha512-shrVawQFojnZv6xM40anx4CkoDP+fZsw/ZerEMsW/pyzsRbElpsL/DBVW7q3ExxwusdNXI3lXpuhEZkzs8p5Eg== - dependencies: - color-name "^1.0.0" - simple-swizzle "^0.2.2" - -color@^4.2.3: - version "4.2.3" - resolved "https://registry.yarnpkg.com/color/-/color-4.2.3.tgz#d781ecb5e57224ee43ea9627560107c0e0c6463a" - integrity sha512-1rXeuUUiGGrykh+CeBdu5Ie7OJwinCgQY0bc7GCRxy5xVHy+moaqkpL/jqQq0MtQOeYcrqEz4abc5f0KtU7W4A== - dependencies: - color-convert "^2.0.1" - color-string "^1.9.0" - colorette@^2.0.7: version "2.0.20" resolved "https://registry.yarnpkg.com/colorette/-/colorette-2.0.20.tgz#9eb793e6833067f7235902fcd3b09917a000a95a" @@ -1758,6 +1709,16 @@ cosmiconfig@^8.0.0, cosmiconfig@^8.3.6: parse-json "^5.2.0" path-type "^4.0.0" +cosmiconfig@^9.0.0: + version "9.0.0" + resolved "https://registry.yarnpkg.com/cosmiconfig/-/cosmiconfig-9.0.0.tgz#34c3fc58287b915f3ae905ab6dc3de258b55ad9d" + integrity sha512-itvL5h8RETACmOTFc4UfIyB2RfEHi71Ax6E/PivVxq9NseKbOWpeyHEOIbmAw1rs8Ak0VursQNww7lf7YtUwzg== + dependencies: + env-paths "^2.2.1" + import-fresh "^3.3.0" + js-yaml "^4.1.0" + parse-json "^5.2.0" + crc-32@^1.2.0: version "1.2.2" resolved "https://registry.yarnpkg.com/crc-32/-/crc-32-1.2.2.tgz#3cad35a934b8bf71f25ca524b6da51fb7eace2ff" @@ -1809,7 +1770,14 @@ dateformat@^4.6.3: resolved "https://registry.yarnpkg.com/dateformat/-/dateformat-4.6.3.tgz#556fa6497e5217fedb78821424f8a1c22fa3f4b5" integrity sha512-2P0p0pFGzHS5EMnhdxQi7aJN+iMheud0UhG4dlE1DLAlvL8JHjJJTX/CSm4JXwV0Ka5nGk3zC5mcb5bUQUxxMA== -debug@4, debug@4.3.4, debug@^4.1.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.4: +debug@4, debug@^4.1.0, debug@^4.1.1, debug@^4.3.1, debug@^4.3.2, debug@^4.3.4, debug@^4.3.5: + version "4.3.5" + resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.5.tgz#e83444eceb9fedd4a1da56d671ae2446a01a6e1e" + integrity sha512-pt0bNEmneDIvdL1Xsd9oDQ/wrQRkXDT4AUWlNZNPKvW5x/jyO9VFXkJUP07vQ2upmw5PlaITaPKc31jK13V+jg== + dependencies: + ms "2.1.2" + +debug@4.3.4: version "4.3.4" resolved "https://registry.yarnpkg.com/debug/-/debug-4.3.4.tgz#1319f6579357f2338d3337d2cdd4914bb5dcc865" integrity sha512-PRWFHuSU3eDtQJPvnNY7Jcket1j0t5OuOsFzPPzsekD52Zl8qUfFIPEiswXqIvHWGVHOgX+7G/vCNNhehwxfkQ== @@ -2035,6 +2003,11 @@ entities@^4.2.0, entities@^4.5.0: resolved "https://registry.yarnpkg.com/entities/-/entities-4.5.0.tgz#5d268ea5e7113ec74c4d033b79ea5a35a488fb48" integrity sha512-V0hjH4dGPh9Ao5p0MoRY6BVqtwCjhz6vI5LT8AJ55H+4g9/4vbHx1I54fS0XuclLhDHArPQCiMjDxjaL8fPxhw== +env-paths@^2.2.1: + version "2.2.1" + resolved "https://registry.yarnpkg.com/env-paths/-/env-paths-2.2.1.tgz#420399d416ce1fbe9bc0a07c62fa68d67fd0f8f2" + integrity sha512-+h1lkLKhZMTYjog1VEpJNG7NZJWcuc2DDk/qsqSTRRCOXiLjeQ1d1/udrUGhqMxUgAlwKNZ0cf2uqan5GLuS2A== + error-ex@^1.3.1: version "1.3.2" resolved "https://registry.yarnpkg.com/error-ex/-/error-ex-1.3.2.tgz#b4ac40648107fdcdcfae242f428bea8a14d4f1bf" @@ -2334,7 +2307,7 @@ espree@^9.6.0, espree@^9.6.1: acorn-jsx "^5.3.2" eslint-visitor-keys "^3.4.1" -esprima@^4.0.0, esprima@^4.0.1, esprima@~4.0.0: +esprima@^4.0.0, esprima@^4.0.1: version "4.0.1" resolved "https://registry.yarnpkg.com/esprima/-/esprima-4.0.1.tgz#13b04cdb3e6c5d19df91ab6987a8695619b0aa71" integrity sha512-eGuFFw7Upda+g4p+QHvnW0RyTX/SVeJBDM/gCtMARO0cLuT2HcEKnTPvhjV6aGeqrCB/sbNop0Kszm0jsaWU4A== @@ -2456,6 +2429,11 @@ fast-safe-stringify@^2.1.1: resolved "https://registry.yarnpkg.com/fast-safe-stringify/-/fast-safe-stringify-2.1.1.tgz#c406a83b6e70d9e35ce3b30a81141df30aeba884" integrity sha512-W+KJc2dmILlPplD/H4K9l9LcAHAfPtP6BY84uVLXQ6Evcz9Lcg33Y2z1IVblT6xdY54PXYVHEv+0Wpq8Io6zkA== +fast-uri@^2.3.0: + version "2.3.0" + resolved "https://registry.yarnpkg.com/fast-uri/-/fast-uri-2.3.0.tgz#bdae493942483d299e7285dcb4627767d42e2793" + integrity sha512-eel5UKGn369gGEWOqBShmFJWfq/xSJvsgDzgLYC845GneayWvXBf0lJCBn5qTABfewy1ZDPoaR5OZCP+kssfuw== + fast-xml-parser@^4.3.6: version "4.3.6" resolved "https://registry.yarnpkg.com/fast-xml-parser/-/fast-xml-parser-4.3.6.tgz#190f9d99097f0c8f2d3a0e681a10404afca052ff" @@ -3021,11 +2999,6 @@ husky@^7.0.4: resolved "https://registry.yarnpkg.com/husky/-/husky-7.0.4.tgz#242048245dc49c8fb1bf0cc7cfb98dd722531535" integrity sha512-vbaCKN2QLtP/vD4yvs6iz6hBEo6wkSzs8HpRah1Z6aGmF2KW5PdYuAd7uX5a+OyBZHBhd+TFLqgjUgytQr4RvQ== -hyperlinker@^1.0.0: - version "1.0.0" - resolved "https://registry.yarnpkg.com/hyperlinker/-/hyperlinker-1.0.0.tgz#23dc9e38a206b208ee49bc2d6c8ef47027df0c0e" - integrity sha512-Ty8UblRWFEcfSuIaajM34LdPXIhbs1ajEX/BBPv24J+enSVaEVY63xQ6lTO9VRYS5LAoghIG0IDJ+p+IPzKUQQ== - ieee754@^1.2.1: version "1.2.1" resolved "https://registry.yarnpkg.com/ieee754/-/ieee754-1.2.1.tgz#8eb7a10a63fff25d15a57b001586d177d1b0d352" @@ -3115,11 +3088,6 @@ is-arrayish@^0.2.1: resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.2.1.tgz#77c99840527aa8ecb1a8ba697b80645a7a926a9d" integrity sha512-zz06S8t0ozoDXMG+ube26zeCTNXcKIPJZJi8hBrF4idCLms4CG9QtK7qBl1boi5ODzFpjswb5JPmHCbMpjaYzg== -is-arrayish@^0.3.1: - version "0.3.2" - resolved "https://registry.yarnpkg.com/is-arrayish/-/is-arrayish-0.3.2.tgz#4574a2ae56f7ab206896fb431eaeed066fdf8f03" - integrity sha512-eVRqCvVlZbuw3GrM63ovNSNAeA1K16kaR/LRY/92w0zxQ5/1YzwblUX652i4Xs9RwAGjW9d9y6X88t8OaAJfWQ== - is-bigint@^1.0.1: version "1.0.4" resolved "https://registry.yarnpkg.com/is-bigint/-/is-bigint-1.0.4.tgz#08147a1875bc2b32005d41ccd8291dffc6691df3" @@ -3437,7 +3405,7 @@ js-yaml@4.1.0, js-yaml@^4.1.0: dependencies: argparse "^2.0.1" -js-yaml@^3.13.1, js-yaml@^3.14.1: +js-yaml@^3.13.1: version "3.14.1" resolved "https://registry.yarnpkg.com/js-yaml/-/js-yaml-3.14.1.tgz#dae812fdb3825fa306609a8717383c50c36a0537" integrity sha512-okMH7OXXJ7YrN9Ok3/SXrnu4iX9yOk+25nqX4imS2npuvTYDmo/QEZoqwZkYaIDk3jVvBOTOIEgEhaLOynBS9g== @@ -4030,11 +3998,6 @@ natural-compare@^1.4.0: resolved "https://registry.yarnpkg.com/natural-compare/-/natural-compare-1.4.0.tgz#4abebfeed7541f2c27acfb29bdbbd15c8d5ba4f7" integrity sha512-OWND8ei3VtNC9h7V60qff3SVobHr996CTwgxubgyQYEpg290h9J0buyECNNJexkFm5sOajh5G116RYA1c8ZMSw== -natural-orderby@^2.0.3: - version "2.0.3" - resolved "https://registry.yarnpkg.com/natural-orderby/-/natural-orderby-2.0.3.tgz#8623bc518ba162f8ff1cdb8941d74deb0fdcc016" - integrity sha512-p7KTHxU0CUrcOXe62Zfrb5Z13nLvPhSWR/so3kFulUQU0sgUll2Z0LwpsLN351eOOD+hRGu/F1g+6xDfPeD++Q== - netmask@^2.0.2: version "2.0.2" resolved "https://registry.yarnpkg.com/netmask/-/netmask-2.0.2.tgz#8b01a07644065d536383835823bc52004ebac5e7" @@ -4169,11 +4132,6 @@ object-keys@^1.1.1: resolved "https://registry.yarnpkg.com/object-keys/-/object-keys-1.1.1.tgz#1c47f272df277f3b1daf061677d9c82e2322c60e" integrity sha512-NuAESUOUMrlIXOfHKzD6bpPu3tYt3xvjNdRIQ+FeT0lNb4K8WR70CaDxhuNguS2XG+GjkyMwOzsN5ZktImfhLA== -object-treeify@^1.1.33: - version "1.1.33" - resolved "https://registry.yarnpkg.com/object-treeify/-/object-treeify-1.1.33.tgz#f06fece986830a3cba78ddd32d4c11d1f76cdf40" - integrity sha512-EFVjAYfzWqWsBMRHPMAXLCDIJnpMhdWAqR7xG6M6a2cs6PMFpl/+Z20w9zDW4vkxOFfddegBKq9Rehd0bxWE7A== - object.assign@^4.1.4: version "4.1.4" resolved "https://registry.yarnpkg.com/object.assign/-/object.assign-4.1.4.tgz#9673c7c7c351ab8c4d0b516f4343ebf4dfb7799f" @@ -4359,14 +4317,6 @@ pascal-case@^3.1.2: no-case "^3.0.4" tslib "^2.0.3" -password-prompt@^1.1.3: - version "1.1.3" - resolved "https://registry.yarnpkg.com/password-prompt/-/password-prompt-1.1.3.tgz#05e539f4e7ca4d6c865d479313f10eb9db63ee5f" - integrity sha512-HkrjG2aJlvF0t2BMH0e2LB/EHf3Lcq3fNMzy4GYHcQblAvOl+QQji1Lx7WRBMqpVK8p+KR7bCg7oqAMXtdgqyw== - dependencies: - ansi-escapes "^4.3.2" - cross-spawn "^7.0.3" - path-case@^3.0.4: version "3.0.4" resolved "https://registry.yarnpkg.com/path-case/-/path-case-3.0.4.tgz#9168645334eb942658375c56f80b4c0cb5f82c6f" @@ -4445,7 +4395,7 @@ pify@^4.0.1: resolved "https://registry.yarnpkg.com/pify/-/pify-4.0.1.tgz#4b2cd25c50d598735c50292224fd8c6df41e3231" integrity sha512-uB80kBFb/tfd68bVleG9T5GGsGPjJrLAUpR5PZIrhBnIaRTQRjqdJSsIKkOP6OAIFbj7GOrcudc5pNjZ+geV2g== -pino-abstract-transport@^1.0.0, pino-abstract-transport@^1.1.0, pino-abstract-transport@^1.2.0: +pino-abstract-transport@^1.0.0, pino-abstract-transport@^1.2.0: version "1.2.0" resolved "https://registry.yarnpkg.com/pino-abstract-transport/-/pino-abstract-transport-1.2.0.tgz#97f9f2631931e242da531b5c66d3079c12c9d1b5" integrity sha512-Guhh8EZfPCfH+PMXAb6rKOjGQEoy0xlAIn+irODG5kgfYV+BQ0rGYYWTIel3P5mmyXqkYkPmdIkywsn6QKUR1Q== @@ -4714,13 +4664,6 @@ redent@^3.0.0: indent-string "^4.0.0" strip-indent "^3.0.0" -redeyed@~2.1.0: - version "2.1.1" - resolved "https://registry.yarnpkg.com/redeyed/-/redeyed-2.1.1.tgz#8984b5815d99cb220469c99eeeffe38913e6cc0b" - integrity sha512-FNpGGo1DycYAdnrKFxCMmKYgo/mILAqtRYbkdQD8Ep/Hk2PQ5+aEAEx+IU713RTDmuBaH0c8P5ZozurNu5ObRQ== - dependencies: - esprima "~4.0.0" - regexp-tree@^0.1.27: version "0.1.27" resolved "https://registry.yarnpkg.com/regexp-tree/-/regexp-tree-0.1.27.tgz#2198f0ef54518ffa743fe74d983b56ffd631b6cd" @@ -5015,13 +4958,6 @@ simple-get@^4.0.1: once "^1.3.1" simple-concat "^1.0.0" -simple-swizzle@^0.2.2: - version "0.2.2" - resolved "https://registry.yarnpkg.com/simple-swizzle/-/simple-swizzle-0.2.2.tgz#a4da6b635ffcccca33f70d17cb92592de95e557a" - integrity sha512-JA//kQgZtbuY83m+xT+tXJkmJncGMTFT+C+g2h2R9uxkYIrE2yy9sgmcLhCnw57/WSD+Eh3J97FPEDFnbXnDUg== - dependencies: - is-arrayish "^0.3.1" - sinon@10.0.0: version "10.0.0" resolved "https://registry.yarnpkg.com/sinon/-/sinon-10.0.0.tgz#52279f97e35646ff73d23207d0307977c9b81430" @@ -5051,15 +4987,6 @@ slash@^3.0.0: resolved "https://registry.yarnpkg.com/slash/-/slash-3.0.0.tgz#6539be870c165adbd5240220dbe361f1bc4d4634" integrity sha512-g9Q1haeby36OSStwb4ntCGGGaKsaVSjQ68fBxoQcutl5fS1vuY18H3wSt3jFyFtrkx+Kz0V1G85A4MyAdDMi2Q== -slice-ansi@^4.0.0: - version "4.0.0" - resolved "https://registry.yarnpkg.com/slice-ansi/-/slice-ansi-4.0.0.tgz#500e8dd0fd55b05815086255b3195adf2a45fe6b" - integrity sha512-qMCMfhY040cVHT43K9BFygqYbUPFZKHOg7K73mtTWJRb8pyP3fzf4Ixd5SzdEJQ6MRUg/WBnOLxghZtKKurENQ== - dependencies: - ansi-styles "^4.0.0" - astral-regex "^2.0.0" - is-fullwidth-code-point "^3.0.0" - smart-buffer@^4.2.0: version "4.2.0" resolved "https://registry.yarnpkg.com/smart-buffer/-/smart-buffer-4.2.0.tgz#6e1d71fa4f18c05f7d0ff216dd16a481d0e8d9ae" @@ -5178,16 +5105,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.0.0, 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.0.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== @@ -5246,14 +5164,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== @@ -5299,7 +5210,7 @@ strnum@^1.0.5: resolved "https://registry.yarnpkg.com/strnum/-/strnum-1.0.5.tgz#5c4e829fe15ad4ff0d20c3db5ac97b73c9b072db" integrity sha512-J8bbNyKKXl5qYcR36TIO8W3mVGVHrmmxsd5PAItGkmyzwJvybiw2IVq5nqd0i4LSNSkB/sx9VHllbfFdr9k1JA== -supports-color@8.1.1, supports-color@^8.1.1: +supports-color@8.1.1, supports-color@^8: version "8.1.1" resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-8.1.1.tgz#cd6fc17e28500cff56c1b86c0a7fd4a54a73005c" integrity sha512-MpUEN2OodtUzxvKQl72cUF7RQ5EiHsGvSsVG0ia9c5RbWGL2CI4C7EpPS8UTBIplnlzZiNuV56w+FuNxy3ty2Q== @@ -5313,21 +5224,13 @@ supports-color@^5.3.0: dependencies: has-flag "^3.0.0" -supports-color@^7, supports-color@^7.0.0, supports-color@^7.1.0: +supports-color@^7, supports-color@^7.1.0: version "7.2.0" resolved "https://registry.yarnpkg.com/supports-color/-/supports-color-7.2.0.tgz#1b7dcdcb32b8138801b3e478ba6a51caa89648da" integrity sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw== dependencies: has-flag "^4.0.0" -supports-hyperlinks@^2.2.0: - version "2.3.0" - resolved "https://registry.yarnpkg.com/supports-hyperlinks/-/supports-hyperlinks-2.3.0.tgz#3943544347c1ff90b15effb03fc14ae45ec10624" - integrity sha512-RpsAZlpWcDwOPQA22aCH4J0t7L8JmAvsCxfOSEwm7cQs3LshN36QaTkwd70DnBOXDWGssw2eUoc8CaRWT0XunA== - dependencies: - has-flag "^4.0.0" - supports-color "^7.0.0" - supports-preserve-symlinks-flag@^1.0.0: version "1.0.0" resolved "https://registry.yarnpkg.com/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz#6eda4bd344a3c94aea376d4cc31bc77311039e09" @@ -5439,15 +5342,10 @@ ts-patch@^3.1.2: semver "^7.5.4" strip-ansi "^6.0.1" -ts-retry-promise@^0.7.1: - version "0.7.1" - resolved "https://registry.yarnpkg.com/ts-retry-promise/-/ts-retry-promise-0.7.1.tgz#176d6eee6415f07b6c7c286d3657355e284a6906" - integrity sha512-NhHOCZ2AQORvH42hOPO5UZxShlcuiRtm7P2jIq2L2RY3PBxw2mLnUsEdHrIslVBFya1v5aZmrR55lWkzo13LrQ== - -ts-retry-promise@^0.8.0: - version "0.8.0" - resolved "https://registry.yarnpkg.com/ts-retry-promise/-/ts-retry-promise-0.8.0.tgz#0a0c57b510827d5630da4b0c47f36b55b83a49e3" - integrity sha512-elI/GkojPANBikPaMWQnk4T/bOJ6tq/hqXyQRmhfC9PAD6MoHmXIXK7KilJrlpx47VAKCGcmBrTeK5dHk6YAYg== +ts-retry-promise@^0.8.0, ts-retry-promise@^0.8.1: + version "0.8.1" + resolved "https://registry.yarnpkg.com/ts-retry-promise/-/ts-retry-promise-0.8.1.tgz#ba90eb07cb03677fcbf78fe38e94c9183927e154" + integrity sha512-+AHPUmAhr5bSRRK5CurE9kNH8gZlEHnCgusZ0zy2bjfatUBDX0h6vGQjiT0YrGwSDwRZmU+bapeX6mj55FOPvg== tsconfig-paths@^3.15.0: version "3.15.0" @@ -5626,7 +5524,7 @@ upper-case@^2.0.2: dependencies: tslib "^2.0.3" -uri-js@^4.2.2, uri-js@^4.4.1: +uri-js@^4.2.2: version "4.4.1" resolved "https://registry.yarnpkg.com/uri-js/-/uri-js-4.4.1.tgz#9b1a52595225859e55f669d928f88c6c57f2a77e" integrity sha512-7rKUyy33Q1yc98pQ1DAmLtwX109F7TIfWlW1Ydo8Wl1ii1SeHieeh0HHfPeL2fMXK6z0s8ecKs9frCuLJvndBg== @@ -5770,7 +5668,7 @@ workerpool@6.2.1: resolved "https://registry.yarnpkg.com/workerpool/-/workerpool-6.2.1.tgz#46fc150c17d826b86a008e5a4508656777e9c343" integrity sha512-ILEIE97kDZvF9Wb9f6h5aXK4swSlKGUcOEGiIYb2OOu/IrDU9iwj0fD//SsA6E5ibwJxpEvhullJY4Sl4GcpAw== -"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== @@ -5788,15 +5686,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 7ba53ae0c1dd8f3e2e7b6e7b9e7bfda7c2b64fc1 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 4 Jun 2024 14:21:27 -0500 Subject: [PATCH 2/5] feat: handle the move-then-modify scenario --- src/shared/local/localShadowRepo.ts | 20 ++- src/shared/local/moveDetection.ts | 138 ++++++++++++------ .../localTrackingFileMoveThenEdit.nut.ts | 91 ++++++++++++ 3 files changed, 190 insertions(+), 59 deletions(-) create mode 100644 test/nuts/local/localTrackingFileMoveThenEdit.nut.ts diff --git a/src/shared/local/localShadowRepo.ts b/src/shared/local/localShadowRepo.ts index 41f6efdb..019ae776 100644 --- a/src/shared/local/localShadowRepo.ts +++ b/src/shared/local/localShadowRepo.ts @@ -15,7 +15,7 @@ import git from 'isomorphic-git'; import { Performance } from '@oclif/core'; import { RegistryAccess } from '@salesforce/source-deploy-retrieve'; import { chunkArray, excludeLwcLocalOnlyTest, folderContainsPath } from '../functions'; -import { filenameMatchesToMap, getMatches } from './moveDetection'; +import { filenameMatchesToMap, getLogMessage, getMatches } from './moveDetection'; import { StatusRow } from './types'; import { isDeleted, isAdded, toFilenames } from './functions'; @@ -348,21 +348,19 @@ export class ShadowRepo { const movedFilesMarker = Performance.mark('@salesforce/source-tracking', 'localShadowRepo.detectMovedFiles'); const matches = await filenameMatchesToMap(IS_WINDOWS)(this.registry)(this.projectPath)(this.gitDir)(matchingFiles); - if (matches.size === 0) return movedFilesMarker?.stop(); + if (matches.deleteOnly.size === 0 && matches.fullMatches.size === 0) return movedFilesMarker?.stop(); - this.logger.debug( - [ - 'Files have moved. Committing moved files:', - [...matches.entries()].map(([add, del]) => `- File ${del} was moved to ${add}`).join(os.EOL), - ].join(os.EOL) - ); + this.logger.debug(getLogMessage(matches)); - movedFilesMarker?.addDetails({ filesMoved: matches.size }); + movedFilesMarker?.addDetails({ + filesMoved: matches.fullMatches.size, + filesMovedAndEdited: matches.deleteOnly.size, + }); // Commit the moved files and refresh the status await this.commitChanges({ - deletedFiles: [...matches.values()], - deployedFiles: [...matches.keys()], + deletedFiles: [...matches.fullMatches.values(), ...matches.deleteOnly.values()], + deployedFiles: [...matches.fullMatches.keys()], message: 'Committing moved files', }); diff --git a/src/shared/local/moveDetection.ts b/src/shared/local/moveDetection.ts index a2a58d15..be84331f 100644 --- a/src/shared/local/moveDetection.ts +++ b/src/shared/local/moveDetection.ts @@ -5,6 +5,7 @@ * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause */ import path from 'node:path'; +import { EOL } from 'node:os'; import { Logger, Lifecycle } from '@salesforce/core'; import { MetadataResolver, @@ -15,13 +16,20 @@ import { // @ts-expect-error isogit has both ESM and CJS exports but node16 module/resolution identifies it as ESM import git from 'isomorphic-git'; import * as fs from 'graceful-fs'; -import { Performance } from '@oclif/core'; +import { Performance } from '@oclif/core/performance'; import { sourceComponentGuard } from '../guards'; import { isDeleted, isAdded, ensureWindows, toFilenames } from './functions'; import { AddAndDeleteMaps, FilenameBasenameHash, StatusRow, StringMap } from './types'; +const JOIN_CHAR = '#__#'; // the __ makes it unlikely to be used in metadata names type AddAndDeleteFileInfos = { addedInfo: FilenameBasenameHash[]; deletedInfo: FilenameBasenameHash[] }; type AddedAndDeletedFilenames = { added: Set; deleted: Set }; +type StringMapsForMatches = { + /** these matches filename=>basename, metadata type/name, and git object hash */ + fullMatches: StringMap; + /** these did not match the hash. They *probably* are matches where the "add" is also modified */ + deleteOnly: StringMap; +}; /** composed functions to simplified use by the shadowRepo class */ export const filenameMatchesToMap = @@ -29,8 +37,8 @@ export const filenameMatchesToMap = (registry: RegistryAccess) => (projectPath: string) => (gitDir: string) => - async ({ added, deleted }: AddedAndDeletedFilenames): Promise => - removeNonMatches(isWindows)(registry)( + async ({ added, deleted }: AddedAndDeletedFilenames): Promise => + excludeNonMatchingTypes(isWindows)(registry)( compareHashes( await buildMaps( await toFileInfo({ @@ -73,7 +81,14 @@ export const getMatches = (status: StatusRow[]): AddedAndDeletedFilenames => { return { added: addedFilenamesWithMatches, deleted: deletedFilenamesWithMatches }; }; -/** build maps of the add/deletes with filenames, returning the matches Logs if non-matches */ +export const getLogMessage = (matches: StringMapsForMatches): string => + [ + 'Files have moved. Committing moved files:', + ...[...matches.fullMatches.entries()].map(([add, del]) => `- File ${del} was moved to ${add}`), + ...[...matches.deleteOnly.entries()].map(([add, del]) => `- File ${del} was moved to ${add} and modified`), + ].join(EOL); + +/** build maps of the add/deletes with filenames, returning the matches Logs if we can't make a match because buildMap puts them in the ignored bucket */ const buildMaps = async ({ addedInfo, deletedInfo }: AddAndDeleteFileInfos): Promise => { const [addedMap, addedIgnoredMap] = buildMap(addedInfo); const [deletedMap, deletedIgnoredMap] = buildMap(deletedInfo); @@ -96,51 +111,70 @@ const buildMaps = async ({ addedInfo, deletedInfo }: AddAndDeleteFileInfos): Pro return { addedMap, deletedMap }; }; -/** builds a map of the values from both maps */ -const compareHashes = ({ addedMap, deletedMap }: AddAndDeleteMaps): StringMap => { +/** + * builds a map of the values from both maps + * side effect: mutates the passed-in maps! + */ +const compareHashes = ({ addedMap, deletedMap }: AddAndDeleteMaps): StringMapsForMatches => { const matches: StringMap = new Map(); - for (const [addedKey, addedValue] of addedMap) { + [...addedMap.entries()].map(([addedKey, addedValue]) => { const deletedValue = deletedMap.get(addedKey); if (deletedValue) { + // these are an exact basename and hash match matches.set(addedValue, deletedValue); + deletedMap.delete(addedKey); + addedMap.delete(addedKey); } - } + }); - return matches; + if (addedMap.size && deletedMap.size) { + // the remaining deletes didn't match the basename+hash of an add, and vice versa. + // They *might* match the basename of an add, in which case we *could* have the "move, then edit" case. + const addedBasenameMap = new Map([...addedMap.entries()].map(hashEntryToBasenameEntry)); + const deletedBasenameMap = new Map([...deletedMap.entries()].map(hashEntryToBasenameEntry)); + const deleteOnly = new Map( + Array.from(deletedBasenameMap.entries()) + .filter(([k]) => addedBasenameMap.has(k)) + .map(([k, v]) => [addedBasenameMap.get(k) as string, v]) + ); + return { fullMatches: matches, deleteOnly }; + } + return { fullMatches: matches, deleteOnly: new Map() }; }; /** given a StringMap, resolve the metadata types and return things that having matching type/parent */ -const removeNonMatches = +const excludeNonMatchingTypes = (isWindows: boolean) => (registry: RegistryAccess) => - (matches: StringMap): StringMap => { - if (!matches.size) return matches; - const addedFiles = isWindows ? [...matches.keys()].map(ensureWindows) : [...matches.keys()]; - const deletedFiles = isWindows ? [...matches.values()].map(ensureWindows) : [...matches.values()]; - const resolverAdded = new MetadataResolver(registry, VirtualTreeContainer.fromFilePaths(addedFiles)); - const resolverDeleted = new MetadataResolver(registry, VirtualTreeContainer.fromFilePaths(deletedFiles)); - - return new Map( - [...matches.entries()].filter(([addedFile, deletedFile]) => { - // we're only ever using the first element of the arrays - const [resolvedAdded] = resolveType(resolverAdded, isWindows ? [ensureWindows(addedFile)] : [addedFile]); - const [resolvedDeleted] = resolveType( - resolverDeleted, - isWindows ? [ensureWindows(deletedFile)] : [deletedFile] - ); - return ( - // they could match, or could both be undefined (because unresolved by SDR) - resolvedAdded?.type.name === resolvedDeleted?.type.name && - // parent names match, if resolved and there are parents - resolvedAdded?.parent?.name === resolvedDeleted?.parent?.name && - // parent types match, if resolved and there are parents - resolvedAdded?.parent?.type.name === resolvedDeleted?.parent?.type.name - ); - }) - ); + ({ fullMatches: matches, deleteOnly }: StringMapsForMatches): StringMapsForMatches => { + if (!matches.size && !deleteOnly.size) return { fullMatches: matches, deleteOnly }; + const [resolvedAdded, resolvedDeleted] = [ + [...matches.keys(), ...deleteOnly.keys()], // the keys/values are only used for the resolver, so we use 1 for both add and delete + [...matches.values(), ...deleteOnly.values()], + ] + .map((filenames) => filenames.map(isWindows ? ensureWindows : stringNoOp)) + .map((filenames) => new MetadataResolver(registry, VirtualTreeContainer.fromFilePaths(filenames))) + .map(resolveType); + + return { + fullMatches: new Map([...matches.entries()].filter(typeFilter(isWindows)(resolvedAdded, resolvedDeleted))), + deleteOnly: new Map([...deleteOnly.entries()].filter(typeFilter(isWindows)(resolvedAdded, resolvedDeleted))), + }; }; +const typeFilter = + (isWindows: boolean) => + (resolveAdd: ReturnType, resolveDelete: ReturnType) => + ([added, deleted]: [string, string]): boolean => { + const [resolvedAdded] = resolveAdd(isWindows ? [ensureWindows(added)] : [added]); + const [resolvedDeleted] = resolveDelete(isWindows ? [ensureWindows(deleted)] : [deleted]); + return ( + resolvedAdded?.type.name === resolvedDeleted?.type.name && + resolvedAdded?.parent?.name === resolvedDeleted?.parent?.name && + resolvedAdded?.parent?.type.name === resolvedDeleted?.parent?.type.name + ); + }; /** enrich the filenames with basename and oid (hash) */ const toFileInfo = async ({ projectPath, @@ -170,11 +204,12 @@ const toFileInfo = async ({ return { addedInfo, deletedInfo }; }; +/** returns a map of . If two items result in the same hash+basename, return that in the ignore bucket */ const buildMap = (info: FilenameBasenameHash[]): StringMap[] => { const map: StringMap = new Map(); const ignore: StringMap = new Map(); info.map((i) => { - const key = `${i.hash}#${i.basename}`; + const key = `${i.hash}${JOIN_CHAR}${i.basename}`; // If we find a duplicate key, we need to remove it and ignore it in the future. // Finding duplicate hash#basename means that we cannot accurately determine where it was moved to or from if (map.has(key) || ignore.has(key)) { @@ -195,18 +230,20 @@ const getHashForAddedFile = hash: (await git.hashBlob({ object: await fs.promises.readFile(path.join(projectPath, filepath)) })).oid, }); -const resolveType = (resolver: MetadataResolver, filenames: string[]): SourceComponent[] => - filenames - .flatMap((filename) => { - try { - return resolver.getComponentsFromPath(filename); - } catch (e) { - const logger = Logger.childFromRoot('ShadowRepo.compareTypes'); - logger.warn(`unable to resolve ${filename}`); - return undefined; - } - }) - .filter(sourceComponentGuard); +const resolveType = + (resolver: MetadataResolver) => + (filenames: string[]): SourceComponent[] => + filenames + .flatMap((filename) => { + try { + return resolver.getComponentsFromPath(filename); + } catch (e) { + const logger = Logger.childFromRoot('ShadowRepo.compareTypes'); + logger.warn(`unable to resolve ${filename}`); + return undefined; + } + }) + .filter(sourceComponentGuard); /** where we don't have git objects to use, read the file contents to generate the hash */ const getHashFromActualFileContents = @@ -218,3 +255,8 @@ const getHashFromActualFileContents = basename: path.basename(filepath), hash: (await git.readBlob({ fs, dir: projectPath, gitdir, filepath, oid })).oid, }); + +const hashEntryToBasenameEntry = ([k, v]: [string, string]): [string, string] => [hashToBasename(k), v]; +const hashToBasename = (hash: string): string => hash.split(JOIN_CHAR)[1]; + +const stringNoOp = (s: string): string => s; diff --git a/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts b/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts new file mode 100644 index 00000000..db86f86a --- /dev/null +++ b/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts @@ -0,0 +1,91 @@ +/* + * Copyright (c) 2020, 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 path from 'node:path'; +import { TestSession } from '@salesforce/cli-plugins-testkit'; +import { expect } from 'chai'; +import * as fs from 'graceful-fs'; +import { RegistryAccess } from '@salesforce/source-deploy-retrieve'; +import { ProjectJson } from '@salesforce/schemas'; +import { NamedPackageDir, PackageDir } from '@salesforce/core'; +import { ShadowRepo } from '../../../src/shared/local/localShadowRepo'; + +describe('handles local files moves that also change the file', () => { + let session: TestSession; + let repo: ShadowRepo; + let modifiedPackageDirs: PackageDir[]; + const NOT_PROJECT_DIR = 'not-project-dir'; + before(async () => { + session = await TestSession.create({ + project: { + sourceDir: path.join('test', 'nuts', 'ebikes-lwc'), + }, + devhubAuthStrategy: 'NONE', + }); + // create the other dir + const notProjectDir = path.join(session.project.dir, NOT_PROJECT_DIR, 'main', 'default', 'classes'); + await fs.promises.mkdir(notProjectDir, { recursive: true }); + + // modify the project json to include the new dir + const sfdxProjectJsonPath = path.join(session.project.dir, 'sfdx-project.json'); + const originalProject = JSON.parse(await fs.promises.readFile(sfdxProjectJsonPath, 'utf8')) as ProjectJson; + modifiedPackageDirs = [...originalProject.packageDirectories, { path: NOT_PROJECT_DIR }]; + await fs.promises.writeFile( + sfdxProjectJsonPath, + JSON.stringify({ + ...originalProject, + packageDirectories: modifiedPackageDirs, + } satisfies ProjectJson) + ); + }); + + after(async () => { + // await session?.clean(); + }); + + it('initialize the local tracking', async () => { + expect(typeof process.env.SF_BETA_TRACK_FILE_MOVES).to.equal('undefined'); + process.env.SF_BETA_TRACK_FILE_MOVES = 'true'; + + repo = await ShadowRepo.getInstance({ + orgId: 'fakeOrgId', + projectPath: session.project.dir, + packageDirs: modifiedPackageDirs.map( + (pd): NamedPackageDir => ({ ...pd, name: NOT_PROJECT_DIR, fullPath: path.join(session.project.dir, pd.path) }) + ), + registry: new RegistryAccess(), + }); + + // Commit the existing status + const filesToSync = await repo.getChangedFilenames(); + await repo.commitChanges({ deployedFiles: filesToSync }); + + expect(await repo.getChangedFilenames()).to.have.lengthOf(0); + }); + + it('move a file and edit it. Only the delete is committed', async () => { + // move all two classes to the new folder + const classFolder = path.join('main', 'default', 'classes'); + ['OrderController.cls', 'OrderController.cls-meta.xml', 'PagedResult.cls', 'PagedResult.cls-meta.xml'].map((f) => + fs.renameSync( + path.join(session.project.dir, 'force-app', classFolder, f), + path.join(session.project.dir, NOT_PROJECT_DIR, classFolder, f) + ) + ); + const editedFilePath = path.join(NOT_PROJECT_DIR, classFolder, 'OrderController.cls'); + // edit the contents of OrderController.cls + fs.appendFileSync(path.join(session.project.dir, editedFilePath), '//comment'); + await repo.getStatus(true); + + // all the deletes were committed + expect(await repo.getDeleteFilenames()).to.deep.equal([]); + // this is still considered an "add" because the moved file was changed + expect(await repo.getAddFilenames()).to.deep.equal([editedFilePath]); + + delete process.env.SF_BETA_TRACK_FILE_MOVES; + }); +}); From 3e2ea7620c8933bbef57a1997f59b8b4d4d0691b Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 4 Jun 2024 14:53:41 -0500 Subject: [PATCH 3/5] test: update ut for behavior change --- test/unit/localDetectMovedFiles.test.ts | 6 +++--- 1 file changed, 3 insertions(+), 3 deletions(-) diff --git a/test/unit/localDetectMovedFiles.test.ts b/test/unit/localDetectMovedFiles.test.ts index 20098813..1dc71ba2 100644 --- a/test/unit/localDetectMovedFiles.test.ts +++ b/test/unit/localDetectMovedFiles.test.ts @@ -207,7 +207,7 @@ describe('local detect moved files', () => { } }); - it('ignores moved files if the contents have also changed', async () => { + it('ignores moved files (add) if the contents have also changed, but notices deletes match', async () => { expect(process.env.SF_BETA_TRACK_FILE_MOVES).to.be.undefined; process.env.SF_BETA_TRACK_FILE_MOVES = 'true'; let projectDir!: string; @@ -247,9 +247,9 @@ describe('local detect moved files', () => { // Refresh the status await shadowRepo.getStatus(true); - // Moved file should NOT have been detected and committed + // delete is detected and committed, but the add is still considered a change expect(gitAdd.calledOnce).to.be.true; - expect(await shadowRepo.getDeletes()).to.have.lengthOf(1); + expect(await shadowRepo.getDeletes()).to.have.lengthOf(0); expect(await shadowRepo.getAdds()).to.have.lengthOf(1); } finally { delete process.env.SF_BETA_TRACK_FILE_MOVES; From ef0b1bdc51b3ec4467465dacf61701bd6cb4c3a6 Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 4 Jun 2024 15:08:59 -0500 Subject: [PATCH 4/5] refactor: reduce instanceof in favor of union narrowing --- src/shared/guards.ts | 6 ++---- src/shared/local/moveDetection.ts | 29 ++++++++++++++-------------- src/shared/localComponentSetArray.ts | 8 ++++---- src/shared/populateTypesAndNames.ts | 4 ++-- src/sourceTracking.ts | 4 ++-- 5 files changed, 25 insertions(+), 26 deletions(-) diff --git a/src/shared/guards.ts b/src/shared/guards.ts index acf371ac..7efb4243 100644 --- a/src/shared/guards.ts +++ b/src/shared/guards.ts @@ -5,7 +5,6 @@ * For full license text, see LICENSE.txt file in the repo root or https://opensource.org/licenses/BSD-3-Clause */ import { - SourceComponent, MetadataMember, FileResponse, ComponentStatus, @@ -15,9 +14,6 @@ import { import { ChangeResult } from './types'; import { ChangeResultWithNameAndType } from './types'; -export const sourceComponentGuard = (input: SourceComponent | undefined): input is SourceComponent => - input instanceof SourceComponent; - export const metadataMemberGuard = ( input: MetadataMember | undefined | Partial ): input is MetadataMember => @@ -42,3 +38,5 @@ export const FileResponseHasPath = ( export const isChangeResultWithNameAndType = (cr?: ChangeResult): cr is ChangeResultWithNameAndType => typeof cr === 'object' && typeof cr.name === 'string' && typeof cr.type === 'string'; + +export const isDefined = (x: T | undefined): x is T => x !== undefined; diff --git a/src/shared/local/moveDetection.ts b/src/shared/local/moveDetection.ts index be84331f..76514e0a 100644 --- a/src/shared/local/moveDetection.ts +++ b/src/shared/local/moveDetection.ts @@ -17,7 +17,7 @@ import { import git from 'isomorphic-git'; import * as fs from 'graceful-fs'; import { Performance } from '@oclif/core/performance'; -import { sourceComponentGuard } from '../guards'; +import { isDefined } from '../guards'; import { isDeleted, isAdded, ensureWindows, toFilenames } from './functions'; import { AddAndDeleteMaps, FilenameBasenameHash, StatusRow, StringMap } from './types'; @@ -116,17 +116,19 @@ const buildMaps = async ({ addedInfo, deletedInfo }: AddAndDeleteFileInfos): Pro * side effect: mutates the passed-in maps! */ const compareHashes = ({ addedMap, deletedMap }: AddAndDeleteMaps): StringMapsForMatches => { - const matches: StringMap = new Map(); - - [...addedMap.entries()].map(([addedKey, addedValue]) => { - const deletedValue = deletedMap.get(addedKey); - if (deletedValue) { - // these are an exact basename and hash match - matches.set(addedValue, deletedValue); - deletedMap.delete(addedKey); - addedMap.delete(addedKey); - } - }); + const matches = new Map( + [...addedMap.entries()] + .map(([addedKey, addedValue]) => { + const deletedValue = deletedMap.get(addedKey); + if (deletedValue) { + // these are an exact basename and hash match + deletedMap.delete(addedKey); + addedMap.delete(addedKey); + return [addedValue, deletedValue] as const; + } + }) + .filter(isDefined) + ); if (addedMap.size && deletedMap.size) { // the remaining deletes didn't match the basename+hash of an add, and vice versa. @@ -243,7 +245,7 @@ const resolveType = return undefined; } }) - .filter(sourceComponentGuard); + .filter(isDefined); /** where we don't have git objects to use, read the file contents to generate the hash */ const getHashFromActualFileContents = @@ -258,5 +260,4 @@ const getHashFromActualFileContents = const hashEntryToBasenameEntry = ([k, v]: [string, string]): [string, string] => [hashToBasename(k), v]; const hashToBasename = (hash: string): string => hash.split(JOIN_CHAR)[1]; - const stringNoOp = (s: string): string => s; diff --git a/src/shared/localComponentSetArray.ts b/src/shared/localComponentSetArray.ts index 7c672aab..bc4c8784 100644 --- a/src/shared/localComponentSetArray.ts +++ b/src/shared/localComponentSetArray.ts @@ -14,7 +14,7 @@ import { DestructiveChangesType, RegistryAccess, } from '@salesforce/source-deploy-retrieve'; -import { sourceComponentGuard } from './guards'; +import { isDefined } from './guards'; import { supportsPartialDelete, pathIsInFolder } from './functions'; type GroupedFileInput = { @@ -83,7 +83,7 @@ export const getComponentSets = ({ grouping.deletes .flatMap((filename) => resolverForDeletes.getComponentsFromPath(filename)) - .filter(sourceComponentGuard) + .filter(isDefined) .map((component) => { // if the component supports partial delete AND there are files that are not deleted, // set the component for deploy, not for delete. @@ -92,7 +92,7 @@ export const getComponentSets = ({ try { resolverForNonDeletes .getComponentsFromPath(resolve(component.content)) - .filter(sourceComponentGuard) + .filter(isDefined) .map((nonDeletedComponent) => componentSet.add(nonDeletedComponent)); } catch (e) { logger.warn( @@ -113,7 +113,7 @@ export const getComponentSets = ({ return undefined; } }) - .filter(sourceComponentGuard) + .filter(isDefined) .map((component) => componentSet.add(component)); // there may have been ignored files, but componentSet.add doesn't automatically track them. // We'll manually set the ignored paths from what the resolver has been tracking diff --git a/src/shared/populateTypesAndNames.ts b/src/shared/populateTypesAndNames.ts index 811cc92b..b429ebbb 100644 --- a/src/shared/populateTypesAndNames.ts +++ b/src/shared/populateTypesAndNames.ts @@ -13,7 +13,7 @@ import { RegistryAccess, } from '@salesforce/source-deploy-retrieve'; import { ChangeResult } from './types'; -import { isChangeResultWithNameAndType, sourceComponentGuard } from './guards'; +import { isChangeResultWithNameAndType, isDefined } from './guards'; import { ensureRelative, excludeLwcLocalOnlyTest, @@ -68,7 +68,7 @@ export const populateTypesAndNames = return undefined; } }) - .filter(sourceComponentGuard); + .filter(isDefined); logger.debug(` matching SourceComponents have ${sourceComponents.length} items from local`); diff --git a/src/sourceTracking.ts b/src/sourceTracking.ts index c85b4f38..769c6791 100644 --- a/src/sourceTracking.ts +++ b/src/sourceTracking.ts @@ -46,8 +46,8 @@ import { FileResponseIsDeleted, FileResponseIsNotDeleted, isChangeResultWithNameAndType, + isDefined, isSdrSuccess, - sourceComponentGuard, } from './shared/guards'; import { removeIgnored } from './shared/remoteChangeIgnoring'; import { @@ -309,7 +309,7 @@ export class SourceTracking extends AsyncCreatable { return undefined; } }) - .filter(sourceComponentGuard); + .filter(isDefined); } } From be3a67ac9cfcbf4aedbc225a2b6fe53f349cd01b Mon Sep 17 00:00:00 2001 From: mshanemc Date: Tue, 4 Jun 2024 15:17:25 -0500 Subject: [PATCH 5/5] test: nut unique test names, do session cleanup --- test/nuts/local/localTrackingFileMoveThenEdit.nut.ts | 2 +- test/nuts/local/localTrackingFileMovesScale.nut.ts | 2 +- 2 files changed, 2 insertions(+), 2 deletions(-) diff --git a/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts b/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts index db86f86a..04ad869c 100644 --- a/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts +++ b/test/nuts/local/localTrackingFileMoveThenEdit.nut.ts @@ -44,7 +44,7 @@ describe('handles local files moves that also change the file', () => { }); after(async () => { - // await session?.clean(); + await session?.clean(); }); it('initialize the local tracking', async () => { diff --git a/test/nuts/local/localTrackingFileMovesScale.nut.ts b/test/nuts/local/localTrackingFileMovesScale.nut.ts index 5a5f1458..85b1ae8e 100644 --- a/test/nuts/local/localTrackingFileMovesScale.nut.ts +++ b/test/nuts/local/localTrackingFileMovesScale.nut.ts @@ -30,7 +30,7 @@ describe(`handles local files moves of ${classCount.toLocaleString()} classes ($ before(async () => { session = await TestSession.create({ project: { - name: 'large-repo', + name: 'large-repo-moves', }, devhubAuthStrategy: 'NONE', });