From facffb548d80ed17dc777c81b8705c4ac608e9b0 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Mon, 13 Jan 2025 17:07:18 -0500
Subject: [PATCH 01/22] feat: fully onboard to create
---
.DS_Store | Bin 0 -> 6148 bytes
.github/DEVELOPMENT.md | 163 ----
README.md | 5 +-
bin/index.js | 16 +-
docs/{Tooling.md => Blocks.md} | 92 +-
docs/Creation.md | 8 +-
docs/FAQs.md | 29 +-
docs/Initialization.md | 17 +-
docs/Migration.md | 16 +-
docs/Options.md | 161 +---
package.json | 18 +-
pnpm-lock.yaml | 179 +---
script/__snapshots__/migrate-test-e2e.ts.snap | 273 ------
script/create-test-e2e.ts | 44 -
script/initialize-test-e2e.ts | 61 --
script/migrate-test-e2e.ts | 151 ----
script/vitest.config.ts | 3 -
src/base.test.ts | 53 ++
src/{next => }/base.ts | 22 +-
src/bin/help.test.ts | 354 --------
src/bin/help.ts | 148 ---
src/bin/index.test.ts | 205 -----
src/bin/index.ts | 108 ---
src/bin/packageJson.test.ts | 31 -
src/bin/packageJson.ts | 16 -
src/bin/promptForMode.test.ts | 129 ---
src/bin/promptForMode.ts | 127 ---
.../blocks/blockAllContributors.test.ts | 0
src/{next => }/blocks/blockAllContributors.ts | 2 +-
.../blocks/blockAreTheTypesWrong.test.ts | 0
.../blocks/blockAreTheTypesWrong.ts | 2 +-
src/{next => }/blocks/blockCSpell.test.ts | 0
src/{next => }/blocks/blockCSpell.ts | 2 +-
.../blocks/blockContributingDocs.test.ts | 0
.../blocks/blockContributingDocs.ts | 0
.../blocks/blockContributorCovenant.test.ts | 0
.../blocks/blockContributorCovenant.ts | 0
.../blocks/blockDevelopmentDocs.test.ts | 0
src/{next => }/blocks/blockDevelopmentDocs.ts | 0
src/{next => }/blocks/blockESLint.test.ts | 6 +-
src/{next => }/blocks/blockESLint.ts | 2 +-
src/{next => }/blocks/blockESLintComments.ts | 0
src/{next => }/blocks/blockESLintJSDoc.ts | 0
src/{next => }/blocks/blockESLintJSONC.ts | 0
src/{next => }/blocks/blockESLintMarkdown.ts | 0
.../blocks/blockESLintMoreStyling.ts | 0
src/{next => }/blocks/blockESLintNode.ts | 0
.../blocks/blockESLintPackageJson.ts | 0
.../blocks/blockESLintPerfectionist.ts | 0
src/{next => }/blocks/blockESLintRegexp.ts | 0
src/{next => }/blocks/blockESLintYML.ts | 0
.../blocks/blockExampleFiles.test.ts | 0
src/{next => }/blocks/blockExampleFiles.ts | 0
src/{next => }/blocks/blockFunding.ts | 2 +-
.../blocks/blockGitHubActionsCI.test.ts | 10 +-
src/{next => }/blocks/blockGitHubActionsCI.ts | 4 +-
src/{next => }/blocks/blockGitHubApps.ts | 0
.../blocks/blockGitHubIssueTemplates.ts | 2 +-
.../blocks/blockGitHubPRTemplate.ts | 0
src/{next => }/blocks/blockGitignore.test.ts | 0
src/{next => }/blocks/blockGitignore.ts | 2 +-
src/{next => }/blocks/blockKnip.ts | 5 +-
src/{next => }/blocks/blockMITLicense.ts | 0
.../blocks/blockMarkdownlint.test.ts | 0
src/{next => }/blocks/blockMarkdownlint.ts | 2 +-
src/{next => }/blocks/blockNvmrc.test.ts | 0
src/{next => }/blocks/blockNvmrc.ts | 0
src/{next => }/blocks/blockPRCompliance.ts | 2 +-
.../blocks/blockPackageJson.test.ts | 0
src/{next => }/blocks/blockPackageJson.ts | 0
src/{next => }/blocks/blockPnpmDedupe.ts | 0
src/{next => }/blocks/blockPrettier.test.ts | 6 +-
src/{next => }/blocks/blockPrettier.ts | 2 +-
.../blocks/blockPrettierPluginCurly.ts | 0
.../blocks/blockPrettierPluginPackageJson.ts | 0
.../blocks/blockPrettierPluginSh.ts | 0
src/{next => }/blocks/blockREADME.test.ts | 0
src/{next => }/blocks/blockREADME.ts | 0
src/{next => }/blocks/blockReleaseIt.ts | 4 +-
src/{next => }/blocks/blockRenovate.ts | 0
.../blockRepositoryBranchRuleset.test.ts | 0
.../blocks/blockRepositoryBranchRuleset.ts | 0
.../blocks/blockRepositoryLabels.ts | 2 +-
.../blocks/blockRepositorySecrets.ts | 0
.../blocks/blockRepositorySettings.ts | 0
src/{next => }/blocks/blockSecurityDocs.ts | 0
src/{next => }/blocks/blockTSup.test.ts | 0
src/{next => }/blocks/blockTSup.ts | 2 +-
.../blockTemplatedWith.ts} | 2 +-
src/{next => }/blocks/blockTypeScript.test.ts | 0
src/{next => }/blocks/blockTypeScript.ts | 2 +-
src/{next => }/blocks/blockVSCode.test.ts | 0
src/{next => }/blocks/blockVSCode.ts | 0
src/{next => }/blocks/blockVitest.test.ts | 0
src/{next => }/blocks/blockVitest.ts | 2 +-
.../files}/createMultiWorkflowFile.ts | 0
.../files}/createSoloWorkflowFile.ts | 0
.../files}/formatIgnoreFile.ts | 0
.../files}/formatWorkflowYaml.ts | 2 +-
.../formatters => blocks/files}/formatYaml.ts | 4 +-
src/blocks/index.ts | 130 +++
src/{next => }/blocks/options.fakes.ts | 0
.../outcomeLabels.ts | 2 +-
src/{next => }/blocks/phases.ts | 0
src/create/createAndEnterGitDirectory.test.ts | 82 --
src/create/createAndEnterGitDirectory.ts | 26 -
.../createRerunDirectorySuggestion.test.ts | 48 -
src/create/createRerunDirectorySuggestion.ts | 7 -
src/create/createRerunSuggestion.test.ts | 185 ----
src/create/createRerunSuggestion.ts | 81 --
src/create/createWithOptions.test.ts | 195 ----
src/create/createWithOptions.ts | 81 --
src/create/index.test.ts | 72 --
src/create/index.ts | 74 --
src/{next/blocks => data}/packageData.test.ts | 2 +-
src/{next/blocks => data}/packageData.ts | 10 +-
src/docs.test.ts | 74 ++
src/greet.test.ts | 44 -
src/greet.ts | 13 -
src/index.test.ts | 16 +
src/index.ts | 13 +-
src/initialize/index.test.ts | 108 ---
src/initialize/index.ts | 44 -
src/initialize/initializeWithOptions.test.ts | 188 ----
src/initialize/initializeWithOptions.ts | 68 --
src/migrate/index.test.ts | 107 ---
src/migrate/index.ts | 44 -
src/migrate/migrateWithOptions.test.ts | 199 -----
src/migrate/migrateWithOptions.ts | 74 --
src/next/blocks/index.ts | 42 -
src/next/blocks/sourcePackageJson.ts | 7 -
src/next/presets/index.ts | 3 -
src/next/runCreateEnginePreset.ts | 23 -
.../getUsageFromReadme.test.ts | 0
.../getUsageFromReadme.ts | 0
.../parsePackageAuthor.test.ts | 10 +-
.../parsePackageAuthor.ts | 2 +-
.../readDefaultsFromReadme.test.ts | 4 +-
.../readDefaultsFromReadme.ts | 9 +-
src/{next => options}/readDescription.test.ts | 16 +-
src/{next => options}/readDescription.ts | 6 +-
.../readDescriptionFromReadme.test.ts | 0
.../readDescriptionFromReadme.ts | 0
.../readDocumentation.test.ts | 0
src/{next => options}/readDocumentation.ts | 2 +-
.../readEmails.ts | 2 +-
.../readFileAsJson.test.ts | 0
src/{shared => options}/readFileAsJson.ts | 0
src/{shared => options}/readFileSafe.test.ts | 0
src/{shared => options}/readFileSafe.ts | 0
.../readFunding.ts | 2 +-
.../readGitHubEmail.test.ts | 2 +-
.../readGitHubEmail.ts | 2 +-
.../readGuide.test.ts | 2 +-
.../readGuide.ts | 2 +-
.../options/readLogoSizing.test.ts | 0
src/{shared => }/options/readLogoSizing.ts | 0
src/options/readPackageData.ts | 10 +
.../presetCommon.ts => presets/common.ts} | 2 +-
.../everything.ts} | 2 +-
src/presets/index.ts | 13 +
.../presetMinimal.ts => presets/minimal.ts} | 4 +-
.../generateNextSteps.test.ts.snap | 183 ----
src/shared/cli/lines.ts | 14 -
src/shared/cli/lowerFirst.ts | 3 -
src/shared/cli/outro.test.ts | 61 --
src/shared/cli/outro.ts | 28 -
src/shared/cli/spinners.ts | 72 --
src/shared/cli/startLineWithDots.ts | 38 -
src/shared/codes.ts | 7 -
src/shared/createCleanupCommands.test.ts | 32 -
src/shared/createCleanupCommands.ts | 12 -
src/shared/doesRepositoryExist.test.ts | 47 -
src/shared/doesRepositoryExist.ts | 27 -
src/shared/ensureGitRepository.test.ts | 29 -
src/shared/ensureGitRepository.ts | 9 -
src/shared/generateNextSteps.test.ts | 50 --
src/shared/generateNextSteps.ts | 69 --
.../getGitHubUserAsAllContributor.test.ts | 103 ---
src/shared/getGitHubUserAsAllContributor.ts | 50 --
src/shared/isUsingCreateEngine.ts | 5 -
src/shared/options/args.ts | 332 -------
.../augmentOptionsWithExcludes.test.ts | 189 ----
.../options/augmentOptionsWithExcludes.ts | 108 ---
.../createOptionDefaults/index.test.ts | 218 -----
.../options/createOptionDefaults/index.ts | 61 --
.../options/createRepositoryWithApi.test.ts | 71 --
src/shared/options/createRepositoryWithApi.ts | 41 -
.../options/detectEmailRedundancy.test.ts | 38 -
src/shared/options/detectEmailRedundancy.ts | 23 -
.../options/ensureRepositoryExists.test.ts | 170 ----
src/shared/options/ensureRepositoryExists.ts | 95 --
src/shared/options/exclusionKeys.ts | 158 ----
src/shared/options/getBase.test.ts | 59 --
src/shared/options/getBase.ts | 34 -
.../getPrefillOrPromptedOption.test.ts | 106 ---
.../options/getPrefillOrPromptedOption.ts | 48 -
src/shared/options/logInferredOptions.test.ts | 118 ---
src/shared/options/logInferredOptions.ts | 39 -
src/shared/options/optionsSchema.ts | 66 --
src/shared/options/readOptions.test.ts | 843 ------------------
src/shared/options/readOptions.ts | 305 -------
src/shared/packages.test.ts | 66 --
src/shared/packages.ts | 26 -
src/shared/prompts.ts | 5 -
src/shared/readFileSafeAsJson.ts | 5 -
src/shared/runOrRestore.test.ts | 95 --
src/shared/runOrRestore.ts | 43 -
src/shared/types.ts | 127 ---
src/steps/addOwnerAsAllContributor.test.ts | 104 ---
src/steps/addOwnerAsAllContributor.ts | 61 --
src/steps/addToolAllContributors.test.ts | 39 -
src/steps/addToolAllContributors.ts | 16 -
src/steps/clearChangelog.ts | 9 -
src/steps/clearLocalGitTags.ts | 7 -
src/steps/clearUnnecessaryFiles.ts | 37 -
.../createJoshuaKGoldbergReplacement.test.ts | 26 -
src/steps/createJoshuaKGoldbergReplacement.ts | 21 -
src/steps/finalizeDependencies.test.ts | 103 ---
src/steps/finalizeDependencies.ts | 75 --
...initializeBranchProtectionSettings.test.ts | 222 -----
src/steps/initializeGitHubRepository/index.ts | 22 -
.../initializeBranchProtectionSettings.ts | 76 --
src/steps/initializeGitRemote.test.ts | 68 --
src/steps/initializeGitRemote.ts | 17 -
src/steps/initializeRepositorySettings.ts | 35 -
src/steps/populateCSpellDictionary.test.ts | 121 ---
src/steps/populateCSpellDictionary.ts | 42 -
src/steps/removeSetupScripts.ts | 29 -
src/steps/resetGitTags.ts | 13 -
src/steps/runCleanup.test.ts | 71 --
src/steps/runCleanup.ts | 43 -
src/steps/uninstallPackages.test.ts | 50 --
src/steps/uninstallPackages.ts | 65 --
src/steps/updateAllContributorsTable.ts | 24 -
src/steps/updateLocalFiles.test.ts | 550 ------------
src/steps/updateLocalFiles.ts | 94 --
src/steps/updateReadme.test.ts | 138 ---
src/steps/updateReadme.ts | 34 -
.../writeReadme/findExistingBadges.test.ts | 111 ---
src/steps/writeReadme/findExistingBadges.ts | 30 -
.../writeReadme/findIntroSectionClose.test.ts | 65 --
.../writeReadme/findIntroSectionClose.ts | 29 -
.../writeReadme/generateTopContent.test.ts | 147 ---
src/steps/writeReadme/generateTopContent.ts | 84 --
src/steps/writeReadme/index.test.ts | 319 -------
src/steps/writeReadme/index.ts | 73 --
.../creation/createCSpellConfig.test.ts | 58 --
.../writing/creation/createCSpellConfig.ts | 38 -
.../creation/createDotGitignore.test.ts | 26 -
.../writing/creation/createDotGitignore.ts | 12 -
.../creation/createESLintConfig.test.ts | 172 ----
.../writing/creation/createESLintConfig.ts | 138 ---
.../writing/creation/createKnipConfig.test.ts | 66 --
.../writing/creation/createKnipConfig.ts | 19 -
.../writing/creation/createTsupConfig.test.ts | 43 -
.../writing/creation/createTsupConfig.ts | 17 -
.../creation/dotGitHub/actions.test.ts | 31 -
.../writing/creation/dotGitHub/actions.ts | 31 -
.../dotGitHub/createDevelopment/index.test.ts | 403 ---------
.../dotGitHub/createDevelopment/index.ts | 177 ----
.../splitIntoSections.test.ts | 40 -
.../createDevelopment/splitIntoSections.ts | 27 -
.../dotGitHub/createDotGitHubFiles.test.ts | 600 -------------
.../dotGitHub/createDotGitHubFiles.ts | 292 ------
.../dotGitHub/createMultiWorkflowFile.test.ts | 48 -
.../dotGitHub/createSoloWorkflowFile.test.ts | 25 -
.../dotGitHub/createWorkflows.test.ts | 378 --------
.../creation/dotGitHub/createWorkflows.ts | 247 -----
src/steps/writing/creation/dotGitHub/index.ts | 14 -
.../creation/dotGitHub/issueTemplate.ts | 206 -----
src/steps/writing/creation/dotHusky.ts | 8 -
src/steps/writing/creation/dotVSCode.test.ts | 290 ------
src/steps/writing/creation/dotVSCode.ts | 85 --
.../creation/formatters/formatJson.test.ts | 14 -
.../writing/creation/formatters/formatJson.ts | 15 -
.../creation/formatters/formatTypeScript.ts | 5 -
src/steps/writing/creation/index.test.ts | 311 -------
src/steps/writing/creation/index.ts | 99 --
src/steps/writing/creation/rootFiles.ts | 149 ----
src/steps/writing/creation/src.ts | 88 --
.../creation/writeAllContributorsRC.ts | 18 -
.../writing/creation/writePackageJson.test.ts | 184 ----
.../writing/creation/writePackageJson.ts | 137 ---
src/steps/writing/types.ts | 3 -
src/steps/writing/writeStructure.test.ts | 46 -
src/steps/writing/writeStructure.ts | 26 -
.../writing/writeStructureWorker.test.ts | 55 --
src/steps/writing/writeStructureWorker.ts | 55 --
src/{next => }/template.ts | 6 +-
src/types.ts | 28 +-
src/{next => }/utils/resolveBin.ts | 0
src/{next => }/utils/sortObject.ts | 0
src/{next => }/utils/swallowError.ts | 0
src/{shared => utils}/tryCatchAsync.ts | 0
.../tryCatchLazyValueAsync.test.ts | 0
.../tryCatchLazyValueAsync.ts | 0
tsconfig.json | 2 +-
298 files changed, 537 insertions(+), 16247 deletions(-)
create mode 100644 .DS_Store
rename docs/{Tooling.md => Blocks.md} (66%)
delete mode 100644 script/__snapshots__/migrate-test-e2e.ts.snap
delete mode 100644 script/create-test-e2e.ts
delete mode 100644 script/initialize-test-e2e.ts
delete mode 100644 script/migrate-test-e2e.ts
delete mode 100644 script/vitest.config.ts
create mode 100644 src/base.test.ts
rename src/{next => }/base.ts (88%)
delete mode 100644 src/bin/help.test.ts
delete mode 100644 src/bin/help.ts
delete mode 100644 src/bin/index.test.ts
delete mode 100644 src/bin/index.ts
delete mode 100644 src/bin/packageJson.test.ts
delete mode 100644 src/bin/packageJson.ts
delete mode 100644 src/bin/promptForMode.test.ts
delete mode 100644 src/bin/promptForMode.ts
rename src/{next => }/blocks/blockAllContributors.test.ts (100%)
rename src/{next => }/blocks/blockAllContributors.ts (94%)
rename src/{next => }/blocks/blockAreTheTypesWrong.test.ts (100%)
rename src/{next => }/blocks/blockAreTheTypesWrong.ts (93%)
rename src/{next => }/blocks/blockCSpell.test.ts (100%)
rename src/{next => }/blocks/blockCSpell.ts (97%)
rename src/{next => }/blocks/blockContributingDocs.test.ts (100%)
rename src/{next => }/blocks/blockContributingDocs.ts (100%)
rename src/{next => }/blocks/blockContributorCovenant.test.ts (100%)
rename src/{next => }/blocks/blockContributorCovenant.ts (100%)
rename src/{next => }/blocks/blockDevelopmentDocs.test.ts (100%)
rename src/{next => }/blocks/blockDevelopmentDocs.ts (100%)
rename src/{next => }/blocks/blockESLint.test.ts (98%)
rename src/{next => }/blocks/blockESLint.ts (99%)
rename src/{next => }/blocks/blockESLintComments.ts (100%)
rename src/{next => }/blocks/blockESLintJSDoc.ts (100%)
rename src/{next => }/blocks/blockESLintJSONC.ts (100%)
rename src/{next => }/blocks/blockESLintMarkdown.ts (100%)
rename src/{next => }/blocks/blockESLintMoreStyling.ts (100%)
rename src/{next => }/blocks/blockESLintNode.ts (100%)
rename src/{next => }/blocks/blockESLintPackageJson.ts (100%)
rename src/{next => }/blocks/blockESLintPerfectionist.ts (100%)
rename src/{next => }/blocks/blockESLintRegexp.ts (100%)
rename src/{next => }/blocks/blockESLintYML.ts (100%)
rename src/{next => }/blocks/blockExampleFiles.test.ts (100%)
rename src/{next => }/blocks/blockExampleFiles.ts (100%)
rename src/{next => }/blocks/blockFunding.ts (77%)
rename src/{next => }/blocks/blockGitHubActionsCI.test.ts (96%)
rename src/{next => }/blocks/blockGitHubActionsCI.ts (93%)
rename src/{next => }/blocks/blockGitHubApps.ts (100%)
rename src/{next => }/blocks/blockGitHubIssueTemplates.ts (98%)
rename src/{next => }/blocks/blockGitHubPRTemplate.ts (100%)
rename src/{next => }/blocks/blockGitignore.test.ts (100%)
rename src/{next => }/blocks/blockGitignore.ts (79%)
rename src/{next => }/blocks/blockKnip.ts (93%)
rename src/{next => }/blocks/blockMITLicense.ts (100%)
rename src/{next => }/blocks/blockMarkdownlint.test.ts (100%)
rename src/{next => }/blocks/blockMarkdownlint.ts (96%)
rename src/{next => }/blocks/blockNvmrc.test.ts (100%)
rename src/{next => }/blocks/blockNvmrc.ts (100%)
rename src/{next => }/blocks/blockPRCompliance.ts (89%)
rename src/{next => }/blocks/blockPackageJson.test.ts (100%)
rename src/{next => }/blocks/blockPackageJson.ts (100%)
rename src/{next => }/blocks/blockPnpmDedupe.ts (100%)
rename src/{next => }/blocks/blockPrettier.test.ts (98%)
rename src/{next => }/blocks/blockPrettier.ts (97%)
rename src/{next => }/blocks/blockPrettierPluginCurly.ts (100%)
rename src/{next => }/blocks/blockPrettierPluginPackageJson.ts (100%)
rename src/{next => }/blocks/blockPrettierPluginSh.ts (100%)
rename src/{next => }/blocks/blockREADME.test.ts (100%)
rename src/{next => }/blocks/blockREADME.ts (100%)
rename src/{next => }/blocks/blockReleaseIt.ts (95%)
rename src/{next => }/blocks/blockRenovate.ts (100%)
rename src/{next => }/blocks/blockRepositoryBranchRuleset.test.ts (100%)
rename src/{next => }/blocks/blockRepositoryBranchRuleset.ts (100%)
rename src/{next => }/blocks/blockRepositoryLabels.ts (84%)
rename src/{next => }/blocks/blockRepositorySecrets.ts (100%)
rename src/{next => }/blocks/blockRepositorySettings.ts (100%)
rename src/{next => }/blocks/blockSecurityDocs.ts (100%)
rename src/{next => }/blocks/blockTSup.test.ts (100%)
rename src/{next => }/blocks/blockTSup.ts (96%)
rename src/{next/blocks/blockTemplatedBy.ts => blocks/blockTemplatedWith.ts} (91%)
rename src/{next => }/blocks/blockTypeScript.test.ts (100%)
rename src/{next => }/blocks/blockTypeScript.ts (98%)
rename src/{next => }/blocks/blockVSCode.test.ts (100%)
rename src/{next => }/blocks/blockVSCode.ts (100%)
rename src/{next => }/blocks/blockVitest.test.ts (100%)
rename src/{next => }/blocks/blockVitest.ts (98%)
rename src/{steps/writing/creation/dotGitHub => blocks/files}/createMultiWorkflowFile.ts (100%)
rename src/{steps/writing/creation/dotGitHub => blocks/files}/createSoloWorkflowFile.ts (100%)
rename src/{steps/writing/creation/formatters => blocks/files}/formatIgnoreFile.ts (100%)
rename src/{steps/writing/creation/dotGitHub => blocks/files}/formatWorkflowYaml.ts (82%)
rename src/{steps/writing/creation/formatters => blocks/files}/formatYaml.ts (78%)
create mode 100644 src/blocks/index.ts
rename src/{next => }/blocks/options.fakes.ts (100%)
rename src/{steps/initializeGitHubRepository => blocks}/outcomeLabels.ts (98%)
rename src/{next => }/blocks/phases.ts (100%)
delete mode 100644 src/create/createAndEnterGitDirectory.test.ts
delete mode 100644 src/create/createAndEnterGitDirectory.ts
delete mode 100644 src/create/createRerunDirectorySuggestion.test.ts
delete mode 100644 src/create/createRerunDirectorySuggestion.ts
delete mode 100644 src/create/createRerunSuggestion.test.ts
delete mode 100644 src/create/createRerunSuggestion.ts
delete mode 100644 src/create/createWithOptions.test.ts
delete mode 100644 src/create/createWithOptions.ts
delete mode 100644 src/create/index.test.ts
delete mode 100644 src/create/index.ts
rename src/{next/blocks => data}/packageData.test.ts (95%)
rename src/{next/blocks => data}/packageData.ts (68%)
create mode 100644 src/docs.test.ts
delete mode 100644 src/greet.test.ts
delete mode 100644 src/greet.ts
create mode 100644 src/index.test.ts
delete mode 100644 src/initialize/index.test.ts
delete mode 100644 src/initialize/index.ts
delete mode 100644 src/initialize/initializeWithOptions.test.ts
delete mode 100644 src/initialize/initializeWithOptions.ts
delete mode 100644 src/migrate/index.test.ts
delete mode 100644 src/migrate/index.ts
delete mode 100644 src/migrate/migrateWithOptions.test.ts
delete mode 100644 src/migrate/migrateWithOptions.ts
delete mode 100644 src/next/blocks/index.ts
delete mode 100644 src/next/blocks/sourcePackageJson.ts
delete mode 100644 src/next/presets/index.ts
delete mode 100644 src/next/runCreateEnginePreset.ts
rename src/{shared/options/createOptionDefaults => options}/getUsageFromReadme.test.ts (100%)
rename src/{shared/options/createOptionDefaults => options}/getUsageFromReadme.ts (100%)
rename src/{shared/options/createOptionDefaults => options}/parsePackageAuthor.test.ts (75%)
rename src/{shared/options/createOptionDefaults => options}/parsePackageAuthor.ts (91%)
rename src/{shared/options/createOptionDefaults => options}/readDefaultsFromReadme.test.ts (98%)
rename src/{shared/options/createOptionDefaults => options}/readDefaultsFromReadme.ts (85%)
rename src/{next => options}/readDescription.test.ts (79%)
rename src/{next => options}/readDescription.ts (74%)
rename src/{next => options}/readDescriptionFromReadme.test.ts (100%)
rename src/{next => options}/readDescriptionFromReadme.ts (100%)
rename src/{next => options}/readDocumentation.test.ts (100%)
rename src/{next => options}/readDocumentation.ts (92%)
rename src/{shared/options/createOptionDefaults => options}/readEmails.ts (91%)
rename src/{shared => options}/readFileAsJson.test.ts (100%)
rename src/{shared => options}/readFileAsJson.ts (100%)
rename src/{shared => options}/readFileSafe.test.ts (100%)
rename src/{shared => options}/readFileSafe.ts (100%)
rename src/{shared/options/createOptionDefaults => options}/readFunding.ts (77%)
rename src/{shared/options/createOptionDefaults => options}/readGitHubEmail.test.ts (96%)
rename src/{shared/options/createOptionDefaults => options}/readGitHubEmail.ts (88%)
rename src/{shared/options/createOptionDefaults => options}/readGuide.test.ts (96%)
rename src/{shared/options/createOptionDefaults => options}/readGuide.ts (84%)
rename src/{shared => }/options/readLogoSizing.test.ts (100%)
rename src/{shared => }/options/readLogoSizing.ts (100%)
create mode 100644 src/options/readPackageData.ts
rename src/{next/presets/presetCommon.ts => presets/common.ts} (90%)
rename src/{next/presets/presetEverything.ts => presets/everything.ts} (97%)
create mode 100644 src/presets/index.ts
rename src/{next/presets/presetMinimal.ts => presets/minimal.ts} (95%)
delete mode 100644 src/shared/__snapshots__/generateNextSteps.test.ts.snap
delete mode 100644 src/shared/cli/lines.ts
delete mode 100644 src/shared/cli/lowerFirst.ts
delete mode 100644 src/shared/cli/outro.test.ts
delete mode 100644 src/shared/cli/outro.ts
delete mode 100644 src/shared/cli/spinners.ts
delete mode 100644 src/shared/cli/startLineWithDots.ts
delete mode 100644 src/shared/codes.ts
delete mode 100644 src/shared/createCleanupCommands.test.ts
delete mode 100644 src/shared/createCleanupCommands.ts
delete mode 100644 src/shared/doesRepositoryExist.test.ts
delete mode 100644 src/shared/doesRepositoryExist.ts
delete mode 100644 src/shared/ensureGitRepository.test.ts
delete mode 100644 src/shared/ensureGitRepository.ts
delete mode 100644 src/shared/generateNextSteps.test.ts
delete mode 100644 src/shared/generateNextSteps.ts
delete mode 100644 src/shared/getGitHubUserAsAllContributor.test.ts
delete mode 100644 src/shared/getGitHubUserAsAllContributor.ts
delete mode 100644 src/shared/isUsingCreateEngine.ts
delete mode 100644 src/shared/options/args.ts
delete mode 100644 src/shared/options/augmentOptionsWithExcludes.test.ts
delete mode 100644 src/shared/options/augmentOptionsWithExcludes.ts
delete mode 100644 src/shared/options/createOptionDefaults/index.test.ts
delete mode 100644 src/shared/options/createOptionDefaults/index.ts
delete mode 100644 src/shared/options/createRepositoryWithApi.test.ts
delete mode 100644 src/shared/options/createRepositoryWithApi.ts
delete mode 100644 src/shared/options/detectEmailRedundancy.test.ts
delete mode 100644 src/shared/options/detectEmailRedundancy.ts
delete mode 100644 src/shared/options/ensureRepositoryExists.test.ts
delete mode 100644 src/shared/options/ensureRepositoryExists.ts
delete mode 100644 src/shared/options/exclusionKeys.ts
delete mode 100644 src/shared/options/getBase.test.ts
delete mode 100644 src/shared/options/getBase.ts
delete mode 100644 src/shared/options/getPrefillOrPromptedOption.test.ts
delete mode 100644 src/shared/options/getPrefillOrPromptedOption.ts
delete mode 100644 src/shared/options/logInferredOptions.test.ts
delete mode 100644 src/shared/options/logInferredOptions.ts
delete mode 100644 src/shared/options/optionsSchema.ts
delete mode 100644 src/shared/options/readOptions.test.ts
delete mode 100644 src/shared/options/readOptions.ts
delete mode 100644 src/shared/packages.test.ts
delete mode 100644 src/shared/packages.ts
delete mode 100644 src/shared/prompts.ts
delete mode 100644 src/shared/readFileSafeAsJson.ts
delete mode 100644 src/shared/runOrRestore.test.ts
delete mode 100644 src/shared/runOrRestore.ts
delete mode 100644 src/shared/types.ts
delete mode 100644 src/steps/addOwnerAsAllContributor.test.ts
delete mode 100644 src/steps/addOwnerAsAllContributor.ts
delete mode 100644 src/steps/addToolAllContributors.test.ts
delete mode 100644 src/steps/addToolAllContributors.ts
delete mode 100644 src/steps/clearChangelog.ts
delete mode 100644 src/steps/clearLocalGitTags.ts
delete mode 100644 src/steps/clearUnnecessaryFiles.ts
delete mode 100644 src/steps/createJoshuaKGoldbergReplacement.test.ts
delete mode 100644 src/steps/createJoshuaKGoldbergReplacement.ts
delete mode 100644 src/steps/finalizeDependencies.test.ts
delete mode 100644 src/steps/finalizeDependencies.ts
delete mode 100644 src/steps/initializeBranchProtectionSettings.test.ts
delete mode 100644 src/steps/initializeGitHubRepository/index.ts
delete mode 100644 src/steps/initializeGitHubRepository/initializeBranchProtectionSettings.ts
delete mode 100644 src/steps/initializeGitRemote.test.ts
delete mode 100644 src/steps/initializeGitRemote.ts
delete mode 100644 src/steps/initializeRepositorySettings.ts
delete mode 100644 src/steps/populateCSpellDictionary.test.ts
delete mode 100644 src/steps/populateCSpellDictionary.ts
delete mode 100644 src/steps/removeSetupScripts.ts
delete mode 100644 src/steps/resetGitTags.ts
delete mode 100644 src/steps/runCleanup.test.ts
delete mode 100644 src/steps/runCleanup.ts
delete mode 100644 src/steps/uninstallPackages.test.ts
delete mode 100644 src/steps/uninstallPackages.ts
delete mode 100644 src/steps/updateAllContributorsTable.ts
delete mode 100644 src/steps/updateLocalFiles.test.ts
delete mode 100644 src/steps/updateLocalFiles.ts
delete mode 100644 src/steps/updateReadme.test.ts
delete mode 100644 src/steps/updateReadme.ts
delete mode 100644 src/steps/writeReadme/findExistingBadges.test.ts
delete mode 100644 src/steps/writeReadme/findExistingBadges.ts
delete mode 100644 src/steps/writeReadme/findIntroSectionClose.test.ts
delete mode 100644 src/steps/writeReadme/findIntroSectionClose.ts
delete mode 100644 src/steps/writeReadme/generateTopContent.test.ts
delete mode 100644 src/steps/writeReadme/generateTopContent.ts
delete mode 100644 src/steps/writeReadme/index.test.ts
delete mode 100644 src/steps/writeReadme/index.ts
delete mode 100644 src/steps/writing/creation/createCSpellConfig.test.ts
delete mode 100644 src/steps/writing/creation/createCSpellConfig.ts
delete mode 100644 src/steps/writing/creation/createDotGitignore.test.ts
delete mode 100644 src/steps/writing/creation/createDotGitignore.ts
delete mode 100644 src/steps/writing/creation/createESLintConfig.test.ts
delete mode 100644 src/steps/writing/creation/createESLintConfig.ts
delete mode 100644 src/steps/writing/creation/createKnipConfig.test.ts
delete mode 100644 src/steps/writing/creation/createKnipConfig.ts
delete mode 100644 src/steps/writing/creation/createTsupConfig.test.ts
delete mode 100644 src/steps/writing/creation/createTsupConfig.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/actions.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/actions.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createDevelopment/index.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createDevelopment/index.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createDevelopment/splitIntoSections.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createDevelopment/splitIntoSections.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createDotGitHubFiles.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createDotGitHubFiles.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createMultiWorkflowFile.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createSoloWorkflowFile.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createWorkflows.test.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/createWorkflows.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/index.ts
delete mode 100644 src/steps/writing/creation/dotGitHub/issueTemplate.ts
delete mode 100644 src/steps/writing/creation/dotHusky.ts
delete mode 100644 src/steps/writing/creation/dotVSCode.test.ts
delete mode 100644 src/steps/writing/creation/dotVSCode.ts
delete mode 100644 src/steps/writing/creation/formatters/formatJson.test.ts
delete mode 100644 src/steps/writing/creation/formatters/formatJson.ts
delete mode 100644 src/steps/writing/creation/formatters/formatTypeScript.ts
delete mode 100644 src/steps/writing/creation/index.test.ts
delete mode 100644 src/steps/writing/creation/index.ts
delete mode 100644 src/steps/writing/creation/rootFiles.ts
delete mode 100644 src/steps/writing/creation/src.ts
delete mode 100644 src/steps/writing/creation/writeAllContributorsRC.ts
delete mode 100644 src/steps/writing/creation/writePackageJson.test.ts
delete mode 100644 src/steps/writing/creation/writePackageJson.ts
delete mode 100644 src/steps/writing/types.ts
delete mode 100644 src/steps/writing/writeStructure.test.ts
delete mode 100644 src/steps/writing/writeStructure.ts
delete mode 100644 src/steps/writing/writeStructureWorker.test.ts
delete mode 100644 src/steps/writing/writeStructureWorker.ts
rename src/{next => }/template.ts (56%)
rename src/{next => }/utils/resolveBin.ts (100%)
rename src/{next => }/utils/sortObject.ts (100%)
rename src/{next => }/utils/swallowError.ts (100%)
rename src/{shared => utils}/tryCatchAsync.ts (100%)
rename src/{shared => utils}/tryCatchLazyValueAsync.test.ts (100%)
rename src/{shared => utils}/tryCatchLazyValueAsync.ts (100%)
diff --git a/.DS_Store b/.DS_Store
new file mode 100644
index 0000000000000000000000000000000000000000..69944ba1a5be222e2807859e027cd7d2fb74ba4c
GIT binary patch
literal 6148
zcmeHK%}T>S5Z<*_6N-?7LXQhx3)ZTI;w8lT0!H+pQWH}&7_%iw&7l->))(?gd>&_Z
zH()Vm5jz9B-~8@oKgj+t#<(|+4jFS8V-_?-j!J`|yD_w3k`XzMkrDf#o){no8W_O+K|n)v4OSY})&U(}pE2G>L;)S&5{SZ}Yp~J?5fH9R0d*-iPYkZh
z!7ognYp~L&%NbWQ!#HN<`ticm?BEwFopDzq^~3-%u*yJ7n+~4;=kUukKJr&f$RY-a
zfq%vTZ;$=42a7Ui>yPE(Su3F3Lqoy53KbC0S1tiy;6Bn-K^+&UL!N7}(ukv=UzG#W
OML-ck9Wn3=4158H$4Q9*
literal 0
HcmV?d00001
diff --git a/.github/DEVELOPMENT.md b/.github/DEVELOPMENT.md
index 39161bc07..c7187f848 100644
--- a/.github/DEVELOPMENT.md
+++ b/.github/DEVELOPMENT.md
@@ -104,166 +104,3 @@ Add `--watch` to keep the type checker running in a watch mode that updates the
```shell
pnpm tsc --watch
```
-
-## Setup Scripts
-
-As described in the `README.md` file and `docs/`, this template repository comes with three scripts that can set up an existing or new repository.
-
-Each follows roughly the same general flow:
-
-1. `bin/index.ts` uses `bin/mode.ts` to determine which of the three setup scripts to run
-2. `readOptions` parses in options from local files, Git commands, npm APIs, and/or files on disk
-3. `runOrRestore` wraps the setup script's main logic in a friendly prompt wrapper
-4. The setup script wraps each portion of its main logic with `withSpinner`
- - Each step of setup logic is generally imported from within `src/steps`
-5. A call to `outro` summarizes the results for the user
-
-> **Warning**
-> Each setup script overrides many files in the directory they're run in.
-> Make sure to save any changes you want to preserve before running them.
-
-### The Creation Script
-
-> 📝 See [`docs/Creation.md`](../docs/Creation.md) for user documentation on the creation script.
-
-This template's "creation" script is located in `src/create/`.
-You can run it locally with `node bin/index.js --mode create`.
-Note that files need to be built with `pnpm run build` beforehand.
-
-#### Testing the Creation Script
-
-You can run the end-to-end test for creation locally on the command-line.
-Note that the files need to be built with `pnpm run build` beforehand.
-
-```shell
-pnpm run test:create
-```
-
-That end-to-end test executes `script/create-test-e2e.ts`, which:
-
-1. Runs the creation script to create a new `test-repository` child directory and repository, capturing code coverage
-2. Asserts that commands such as `build` and `lint` each pass
-
-The `pnpm run test:create` script is run in CI to ensure that templating changes are in sync with the template's actual files.
-See `.github/workflows/ci.yml`'s `test_creation_script` job.
-
-### The Initialization Script
-
-> 📝 See [`docs/Initialization.md`](../docs/Initialization.md) for user documentation on the initialization script.
-
-This template's "initialization" script is located in `src/initialize/`.
-You can run it locally with `pnpm run initialize`.
-It uses [`tsx`](https://github.com/esbuild-kit/tsx) so you don't need to build files before running.
-
-```shell
-pnpm run initialize
-```
-
-#### Testing the Initialization Script
-
-You can run the end-to-end test for initializing locally on the command-line.
-Note that files need to be built with `pnpm run build` beforehand.
-
-```shell
-pnpm run test:initialize
-```
-
-That end-to-end test executes `script/initialize-test-e2e.ts`, which:
-
-1. Runs the initialization script using `--skip-github-api` and other skip flags
-2. Checks that the local repository's files were changed correctly (e.g. removed initialization-only files)
-3. Runs `pnpm run lint:knip` to make sure no excess dependencies or files were left over
-4. Resets everything
-5. Runs initialization a second time, capturing test coverage
-
-The `pnpm run test:initialize` script is run in CI to ensure that templating changes are in sync with the template's actual files.
-See `.github/workflows/ci.yml`'s `test_initialization_script` job.
-
-### The Migration Script
-
-> 📝 See [`docs/Migration.md`](../docs/Migration.md) for user documentation on the migration script.
-
-This template's "migration" script is located in `src/migrate/`.
-Note that files need to be built with `pnpm run build` beforehand.
-
-To test out the script locally, run it from a different repository's directory:
-
-```shell
-cd ../other-repo
-node ../create-typescript-app/bin/migrate.js
-```
-
-The migration script will work on any directory.
-You can try it out in a blank directory with scripts like:
-
-```shell
-cd ..
-mkdir temp
-cd temp
-node ../create-typescript-app/bin/migrate.js
-```
-
-#### Testing the Migration Script
-
-> 💡 Seeing `Oh no! Running the migrate script unexpectedly modified:` errors?
-> _[Unexpected File Modifications](#unexpected-file-modifications)_ covers that below.
-
-You can run the end-to-end test for migrating locally on the command-line:
-
-```shell
-pnpm run test:migrate
-```
-
-That end-to-end test executes `script/migrate-test-e2e.ts`, which:
-
-1. Runs the migration script using `--skip-github-api` and other skip flags, capturing code coverage
-2. Checks that only a small list of allowed files were changed
-3. Checks that the local repository's files were changed correctly (e.g. removed initialization-only files)
-
-The `pnpm run test:migrate` script is run in CI to ensure that templating changes are in sync with the template's actual files.
-See `.github/workflows/ci.yml`'s `test_migration_script` job.
-
-> Tip: if the migration test is failing in CI and you don't see any errors, try [downloading the full logs](https://docs.github.com/en/actions/monitoring-and-troubleshooting-workflows/using-workflow-run-logs#downloading-logs).
-
-##### Migration Snapshot Failures
-
-The migration test uses the [Vitest file snapshot](https://vitest.dev/guide/snapshot#file-snapshots) in `script/__snapshots__/migrate-test-e2e.ts.snap` to store expected differences to this repository after running the migration script.
-The end-to-end migration test will fail any changes that don't keep the same differences in that snapshot.
-
-You can update the snapshot file by:
-
-1. Committing any changes to your local repository
-2. Running `pnpm i` and `pnpm build` if any updates have been made to the `package.json` or `src/` files, respectively
-3. Running `pnpm run test:migrate -u` to update the snapshot
-
-At this point there will be some files changed:
-
-- `script/__snapshots__/migrate-test-e2e.ts.snap` will have updates if any files mismatched templates
-- The actual updated files on disk will be there too
-
-If the snapshot file changes are what you expected, then you can commit them.
-The rest of the file changes can be reverted.
-
-> [🚀 Feature: Add a way to apply known file changes after migration #1184](https://github.com/JoshuaKGoldberg/create-typescript-app/issues/1184) tracks turning the test snapshot into a feature.
-
-##### Unexpected File Modifications
-
-The migration test also asserts that no files were unexpectedly changed.
-If you see a failure like:
-
-```plaintext
-Oh no! Running the migrate script unexpectedly modified:
- - ...
-```
-
-...then that means the file generated from templates differs from what's checked into the repository.
-This is most often caused by changes to templates not being applied to checked-in files too.
-
-Templates for files are generally stored in [`src/steps/writing/creation`] under a path roughly corresponding to the file they describe.
-For example, the template for `tsup.config.ts` is stored in [`src/steps/writing/creation/createTsupConfig.ts`](../src/steps/writing/creation/createTsupConfig.ts).
-If the `createTsupConfig` function were to be modified without an equivalent change to `tsup.config.ts` -or vice-versa- then the migration test would report:
-
-```plaintext
-Oh no! Running the migrate script unexpectedly modified:
- - tsup.config.ts
-```
diff --git a/README.md b/README.md
index efc7f1e78..a5894ebbe 100644
--- a/README.md
+++ b/README.md
@@ -31,10 +31,11 @@ First make sure you have the following installed:
Then in an existing repository or in your directory where you'd like to make a new repository:
```shell
-npx create-typescript-app
+npx create typescript-app
```
-That setup script will walk you through using the template.
+That will launch `create-typescript-app` using the [`create` runner](https://create.bingo).
+
You can read more about the supported setup modes in their docs pages:
- [**Creating from the terminal**](./docs/Creation.md): creating a new repository locally on the command-line _(recommended)_
diff --git a/bin/index.js b/bin/index.js
index 1b7a28434..a2fcfd5e6 100755
--- a/bin/index.js
+++ b/bin/index.js
@@ -1,4 +1,16 @@
#!/usr/bin/env node
-import { bin } from "../lib/bin/index.js";
+import chalk from "chalk";
-process.exitCode = await bin(process.argv.slice(2));
+console.log(
+ [
+ "create-typescript-app is now run using ",
+ chalk.bold("create"),
+ ".\n\nRun:\n ",
+ chalk.bold("npx create", process.argv.slice(2).join(" ")),
+ "\n\nYou can read more on:\n https://",
+ chalk.bold("create.bingo"),
+ "\n\nThanks for using create-typescript-app! 🎁",
+ ].join(""),
+);
+
+process.exitCode = 1;
diff --git a/docs/Tooling.md b/docs/Blocks.md
similarity index 66%
rename from docs/Tooling.md
rename to docs/Blocks.md
index 983f8063b..6ff111b38 100644
--- a/docs/Tooling.md
+++ b/docs/Blocks.md
@@ -1,48 +1,54 @@
-# Tooling
+# Blocks
`create-typescript-app` provides over two dozen pieces of tooling, ranging from code building and formatting to various forms of GitHub repository management.
-Most can be individually turned off or on.
-
-The `create-typescript-app` setup scripts -[creation](./Creation.md), [initialization](./Initialization.md), and [migration](./Migration.md)- will prompt you to choose a "base" template level to initialize from.
-Those template levels provide common presets of which tooling pieces to enable.
-
-```plaintext
-◆ How much tooling would you like the template to set up for you?
-│ ○ minimal Just the bare starter tooling most repositories should ideally include.
-│ ○ common Important additions to the minimal starters such as releases and tests.
-│ ○ everything The most thorough tooling imaginable: sorting, spellchecking, and more!
-│ ○ prompt (allow me to customize)
-└
-```
-
-This table summarizes each tooling piece and which base levels they're included in:
-
-| Tooling Piece | Exclusion Flag | Minimal | Common | Everything |
-| --------------------------------------------- | ------------------------------ | ------- | ------ | ---------- |
-| [Building](#building) | | ✔️ | ✅ | 💯 |
-| [Compliance](#compliance) | `--exclude-compliance` | | | 💯 |
-| [Contributors](#contributors) | `--exclude-contributors` | | ✅ | 💯 |
-| [Formatting](#formatting) | | ✔️ | ✅ | 💯 |
-| [Lint ESLint](#lint-eslint) | `--exclude-lint-eslint` | | | 💯 |
-| [Lint JSDoc](#lint-jsdoc) | `--exclude-lint-jsdoc` | | | 💯 |
-| [Lint JSON](#lint-json) | `--exclude-lint-json` | | | 💯 |
-| [Lint Knip](#lint-knip) | `--exclude-lint-knip` | | ✅ | 💯 |
-| [Lint MD](#lint-md) | `--exclude-lint-md` | | | 💯 |
-| [Lint Package JSON](#lint-package-json) | `--exclude-lint-package-json` | | | 💯 |
-| [Lint Packages](#lint-packages) | `--exclude-lint-packages` | | | 💯 |
-| [Lint Perfectionist](#lint-perfectionist) | `--exclude-lint-perfectionist` | | | 💯 |
-| [Lint Regexp](#lint-regexp) | `--exclude-lint-regexp` | | | 💯 |
-| [Lint Spelling](#lint-spelling) | `--exclude-lint-spelling` | | | 💯 |
-| [Lint Strict](#lint-strict) | `--exclude-lint-strict` | | | 💯 |
-| [Lint Stylistic](#lint-stylistic) | `--exclude-lint-stylistic` | | | 💯 |
-| [Lint YML](#lint-yml) | `--exclude-lint-yml` | | | 💯 |
-| [Linting](#linting) | | ✔️ | ✅ | 💯 |
-| [Package Management](#package-management) | | ✔️ | ✅ | 💯 |
-| [Releases](#releases) | `--exclude-releases` | | ✅ | 💯 |
-| [Renovate](#renovate) | `--exclude-renovate` | | ✅ | 💯 |
-| [Repository Templates](#repository-templates) | | ✔️ | ✅ | 💯 |
-| [Testing](#testing) | `--exclude-tests` | | ✅ | 💯 |
-| [Type Checking](#type-checking) | | ✔️ | ✅ | 💯 |
+Each can be individually turned off or on.
+
+This table summarizes each block and which base levels they're included in:
+
+| Block | Exclusion Flag | Minimal | Common | Everything |
+| ---------------------------- | ---------------------------------------- | ------- | ------ | ---------- |
+| AllContributors | `--exclude-allcontributors` | | ✅ | 💯 |
+| Are The Types Wrong | `--exclude-are-the-types-wrong` | | | |
+| Contributing Docs | `--exclude-contributing-docs` | ✔️ | ✅ | 💯 |
+| Contributor Covenant | `--exclude-contributor-covenant` | ✔️ | ✅ | 💯 |
+| CSpell | `--exclude-cspell` | | | 💯 |
+| Development Docs | `--exclude-development-docs` | ✔️ | ✅ | 💯 |
+| ESLint | `--exclude-eslint` | ✔️ | ✅ | 💯 |
+| ESLint Comments Plugin | `--exclude-eslint-comments-plugin` | | | 💯 |
+| ESLint JSDoc Plugin | `--exclude-eslint-jsdoc-plugin` | | | 💯 |
+| ESLint JSONC Plugin | `--exclude-eslint-jsonc-plugin` | | | 💯 |
+| ESLint Markdown Plugin | `--exclude-eslint-markdown-plugin` | | | 💯 |
+| ESLint More Styling | `--exclude-eslint-more-styling` | | | 💯 |
+| ESLint Node Plugin | `--exclude-eslint-node-plugin` | | | 💯 |
+| ESLint package.json Plugin | `--exclude-eslint-package-json-plugin` | | | 💯 |
+| ESLint Perfectionist Plugin | `--exclude-eslint-perfectionist-plugin` | | | 💯 |
+| ESLint Regexp Plugin | `--exclude-eslint-regexp-plugin` | | | 💯 |
+| ESLint YML Plugin | `--exclude-eslint-yml-plugin` | | | 💯 |
+| Funding | `--exclude-funding` | ✔️ | ✅ | 💯 |
+| GitHub Actions CI | `--exclude-github-actions-ci` | ✔️ | ✅ | 💯 |
+| GitHub Issue Templates | `--exclude-github-issue-templates` | ✔️ | ✅ | 💯 |
+| GitHub Issue Templates | `--exclude-github-issue-templates` | ✔️ | ✅ | 💯 |
+| Gitignore | `--exclude-gitignore` | ✔️ | ✅ | 💯 |
+| Knip | `--exclude-knip` | | | 💯 |
+| Markdownlint | `--exclude-markdownlint` | | | 💯 |
+| MIT License | `--exclude-mit-license` | ✔️ | ✅ | 💯 |
+| nvmrc | `--exclude-nvmrc` | | | 💯 |
+| Package JSON | `--exclude-package-json` | ✔️ | ✅ | 💯 |
+| pnpm Dedupe | `--exclude-pnpm-dedupe` | | | 💯 |
+| PR Compliance | `--exclude-pr-compliance` | | | 💯 |
+| Prettier | `--exclude-prettier` | ✔️ | ✅ | 💯 |
+| Prettier Plugin Curly | `--exclude-prettier-plugin-curly` | | | 💯 |
+| Prettier Plugin Package JSON | `--exclude-prettier-plugin-package-json` | | | 💯 |
+| Prettier Plugin Sh | `--exclude-prettier-plugin-sh` | | | 💯 |
+| README.md | `--exclude-readme-md` | ✔️ | ✅ | 💯 |
+| release-it | `--exclude-release-it` | | ✅ | 💯 |
+| Renovate | `--exclude-renovate` | | | 💯 |
+| Security Docs | `--exclude-security-docs` | | | 💯 |
+| Templated By Notice | `--exclude-templated-by-notice` | ✔️ | ✅ | 💯 |
+| TSup | `--exclude-tsup` | ✔️ | ✅ | 💯 |
+| TypeScript | `--exclude-typescript` | ✔️ | ✅ | 💯 |
+| Vitest | `--exclude-vitest` | | ✅ | 💯 |
+| VS Code | `--exclude-vs-code` | | | 💯 |
See also [Options](./Options.md) for how to customize the way template is run.
diff --git a/docs/Creation.md b/docs/Creation.md
index 5572d9d15..ba1f3ac04 100644
--- a/docs/Creation.md
+++ b/docs/Creation.md
@@ -1,9 +1,9 @@
# Creating from the Terminal
-You can run `npx create-typescript-app` in your terminal to interactively create a new repository in a child directory:
+You can run `npx create typescript-app` in your terminal to interactively create a new repository in a child directory:
```shell
-npx create-typescript-app
+npx create typescript-app
```
The creation script will by default:
@@ -31,10 +31,10 @@ Hooray! 🥳
You can explicitly provide some or all of the options the script would prompt for as command-line flags.
See [Options.md](./Options.md).
-For example, running the creation script and skipping all GitHub-related APIs:
+For example, running the creation script and skipping the _"This package was templated with..."_ block:
```shell
-npx create-typescript-app --mode create --skip-all-contributors-api --skip-github-api
+npx create typescript-app --mode create --exclude-templated-with
```
See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
diff --git a/docs/FAQs.md b/docs/FAQs.md
index 71eb53746..b65837f51 100644
--- a/docs/FAQs.md
+++ b/docs/FAQs.md
@@ -147,43 +147,24 @@ That's reasonable.
Unless you know a package needs to support a CJS consumer, please strongly consider keeping it ESM-only (the `create-typescript-app` default).
ESM-only packages have a smaller footprint by virtue of including fewer files.
-## Is there a way to pull in template updates to previously created repositories?
-
-Not directly.
-You can always copy & paste them in manually, and/or re-run `npx create-typescript-app --mode migrate`.
-
-See [🚀 Feature: Add a script to sync the tooling updates from forked template repo #498](https://github.com/JoshuaKGoldberg/create-typescript-app/issues/498): it will likely eventually be possible.
-
## What about `eslint-config-prettier`?
[`eslint-config-prettier`](https://github.com/prettier/eslint-config-prettier) is an ESLint plugin that serves only to turn off all rules that are unnecessary or might conflict with formatters such as Prettier.
None of the ESLint configs enabled by this repository's tooling leave any rules enabled that would need to be disabled.
Using `eslint-config-prettier` would be redundant.
-## What determines which "base" a tool goes into?
+## What determines which preset a tool goes into?
-The four bases correspond to what have seemed to be the most common user needs of template consumers:
+The four presets correspond to what have seemed to be the most common user needs of template consumers:
1. **Minimal**: Developers who just want the barest of starting templates.
- They may be very new to TypeScript tooling, or they may have made an informed decision that the additional tooling isn't worth the complexity and/or time investment.
- - Tooling in this base is only what would be essential for a small TypeScript package that can be built, formatted, linted, and released.
+ - Tooling in this preset is only what would be essential for a small TypeScript package that can be built, formatted, linted, and released.
2. **Common**: The common case of users who want the minimal tooling along with common repository management.
- - Tooling added in this base should be essential for a TypeScript repository that additionally automates useful GitHub tasks: contributor recognition, release management, and testing.
+ - Tooling added in this preset should be essential for a TypeScript repository that additionally automates useful GitHub tasks: contributor recognition, release management, and testing.
3. **Everything**: Power users (including this repository) who want as much of the latest and greatest safety checks and standardization as possible.
-Note that users can always customize exactly with portions are kept in with `--base` **`prompt`**.
-
-## Which tools can't I remove?
-
-The following pieces of this template's tooling don't have options to be removed:
-
-- Linting with ESLint and `pnpm run lint`
-- GitHub repository metadata such as the code of conduct and issue templates
-- Prettier and `pnpm run format`
-- tsup and `pnpm run build`
-- TypeScript and `pnpm run tsc`
-
-If you have a strong desire to add an `--exclude-*` flag for any of them, please do [file a feature request](https://github.com/JoshuaKGoldberg/create-typescript-app/issues/new?assignees=&labels=type%3A+feature&projects=&template=03-feature.yml&title=%F0%9F%9A%80+Feature%3A+%3Cshort+description+of+the+feature%3E).
+Note that you can always customize exactly which preset you use per [Options](./Options.md).
## Why does this package include so many tools?
diff --git a/docs/Initialization.md b/docs/Initialization.md
index f2f17d3f7..4ec14ead3 100644
--- a/docs/Initialization.md
+++ b/docs/Initialization.md
@@ -1,21 +1,12 @@
# Initializing from the Template
-As an alternative to [creating with `npx create-typescript-app`](./Creation.md), the [_Use this template_](https://github.com/JoshuaKGoldberg/create-typescript-app/generate) button on GitHub can be used to quickly create a new repository from the template.
+As an alternative to [creating with `npx create typescript-app`](./Creation.md), the [_Use this template_](https://github.com/JoshuaKGoldberg/create-typescript-app/generate) button on GitHub can be used to quickly create a new repository from the template.
You can set up the new repository locally by cloning it and installing packages:
```shell
git clone https://github.com/YourUsername/YourRepositoryName
cd YourRepositoryName
-pnpm i
-```
-
-> 💡 If you don't want to clone it locally, you can always [develop in a codespace](https://docs.github.com/en/codespaces/developing-in-codespaces/developing-in-a-codespace) instead.
-
-Once the repository's packages are installed, you can run `pnpm run initialize` to fill out your repository's details and install necessary packages.
-It will then remove itself and uninstall dependencies only used for initialization.
-
-```shell
-pnpm run initialize
+npx create typescript-app
```
You'll then need to manually go through the following two steps to set up tooling on GitHub:
@@ -37,10 +28,10 @@ See [Options.md](./Options.md).
`pnpm run initialize` will set `--mode` to `initialize`.
-For example, running the initialization script and skipping all GitHub-related APIs:
+For example, running the creation script and skipping the _"This package was templated with..."_ block:
```shell
-pnpm run initialize --skip-all-contributors-api --skip-github-api
+npx create typescript-app --mode create --exclude-templated-with
```
See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
diff --git a/docs/Migration.md b/docs/Migration.md
index ef03d0276..3526c53ae 100644
--- a/docs/Migration.md
+++ b/docs/Migration.md
@@ -1,9 +1,9 @@
# Migrating an Existing Repository
-If you have an existing repository that you'd like to give the files from this repository, you can run `npx create-typescript-app` in it to "migrate" its tooling to this template's.
+If you have an existing repository that you'd like to give the files from this repository, you can run `npx create typescript-app` in it to "migrate" its tooling to this template's.
```shell
-npx create-typescript-app
+npx create typescript-app
```
The migration script will:
@@ -17,6 +17,7 @@ The migration script will:
For example, if the repository previously using Jest for testing:
- `eslint-plugin-jest`, `jest`, and other Jest-related packages will be uninstalled
+ - 🛑 TODO: Missing in `blockVitest`
- Any Jest config file like `jest.config.js` will be deleted
- `@vitest/eslint-plugin`, `vitest`, and other Vitest-related packages will be installed
- A `vitest.config.ts` file will be created
@@ -40,20 +41,13 @@ Hooray! 🥳
## Options
-`create-typescript-app` will detect whether it's being run in an existing repository.
-It also allows specifying `--mode migrate` if that detection misinterprets the current directory:
-
-```shell
-npx create-typescript-app --mode migrate
-```
-
You can explicitly provide some or all of the options the script would prompt for as command-line flags.
See [Options.md](./Options.md).
-For example, running the migration script and skipping all GitHub-related APIs:
+For example, running the migration script and skipping the _"This package was templated with..."_ block:
```shell
-npx create-typescript-app --skip-all-contributors-api --skip-github-api
+npx create typescript-app --mode migrate --exclude-templated-with
```
See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
diff --git a/docs/Options.md b/docs/Options.md
index 79fd90faa..8e67bdfe4 100644
--- a/docs/Options.md
+++ b/docs/Options.md
@@ -1,55 +1,50 @@
# Options
-All three of `create-typescript-app`'s setup scripts -[creation](./Creation.md), [initialization](./Initialization.md), and [migration](./Migration.md)- support a shared set of input options.
+`create-typescript-app` is built on top of [`create`](https://create.bingo).
+`npx create typescript-app` supports all the flags defined by the [`create` CLI](https://www.create.bingo/cli).
+It provides three Presets:
-> This page uses `npx create-typescript-app` in its code examples, but initialization's `pnpm run initialize` works the same.
+1. **Minimal**: Just bare starter tooling: building, formatting, linting, and type checking.
+2. **Common**: Bare starters plus testing and automation for all-contributors and releases.
+3. **Everything**: The most comprehensive tooling imaginable: sorting, spellchecking, and more!
-## Required Options
+For example, to create a new repository on the _everything_ preset:
-The following required options will be prompted for interactively if not provided as CLI flags.
-
-### Base and Mode
+```shell
+npx create typescript-app --preset everything
+```
-These required options determine how the creation script will set up and scaffold the repository:
+`create-typescript-app` itself adds in two sections of flags:
-- `--base`: Whether to scaffold the repository with:
- - `minimal`: Just the bare starter tooling most repositories should ideally include
- - `common`: Important additions to the minimal starters such as releases and tests
- - `everything`: The most thorough tooling imaginable: sorting, spellchecking, and more!
- - `prompt`: Fine-grained control over which tooling pieces to use
-- `--mode`: Whether to:
- - `create` a new repository in a child directory
- - `initialize` a newly created repository in the current directory
- - `migrate` an existing repository in the current directory
+- [Base Options](#base-options)
+- [Block Exclusions](#block-exclusions)
-For example, scaffolding a full new repository under the current directory and also linking it to a new repository on github.com:
+## Base Options
-```shell
-npx create-typescript-app --base everything --mode create
-```
+Per [`create` > CLI > Template Options](https://www.create.bingo/cli#template-options), options defined by `create-typescript-app` may be provided on the CLI.
-See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
+### Required Base Options
-### Core Options
+🛑 TODO: Mention `--repository` and `--owner` in `create`, and add _"you can't override these"_ issue
-These required options determine the options that will be substituted into the template's files:
+These options can only be inferred when running on an existing repository.
+Each will be prompted for when creating a new repository if not explicitly provided:
-- `--description` _(`string`)_: Sentence case description of the repository (e.g. `Quickstart-friendly TypeScript package with lots of great repository tooling. ✨`)
-- `--owner` _(`string`)_: GitHub organization or user the repository is underneath (e.g. `JoshuaKGoldberg`)
-- `--repository` _(`string`)_: The kebab-case name of the repository (e.g. `create-typescript-app`)
-- `--title` _(`string`)_: Title Case title for the repository (e.g. `Create TypeScript App`)
+- `--description` _(`string`)_: Sentence case description of the repository
+- `--title` _(`string`)_: Title Case title for the repository
-For example, pre-populating all core required options and also creating a new repository:
+For example, pre-populating both required base options:
```shell
-npx create-typescript-app --base everything --mode create --repository testing-repository --title "Testing Title" --owner TestingOwner --description "Test Description"
+npx create typescript-app --description "My awesome TypeScript app! 💖" --title "My TypeScript App"
```
That script will run completely autonomously, no prompted inputs required. ✨
-## Optional Options
+### Optional Base Options
-The setup scripts also allow for optional overrides of the following inputs whose defaults are based on other options:
+These optional options do not need to be provided explicitly.
+They will be inferred from the running user, and if migrating an existing repository, its files on disk.
- `--access` _(`"public" | "restricted"`)_: Which [`npm publish --access`](https://docs.npmjs.com/cli/commands/npm-publish#access) to release npm packages with (by default, `"public"`)
- `--author` _(`string`)_: Username on npm to publish packages under (by default, an existing npm author, or the currently logged in npm user, or `owner.toLowerCase()`)
@@ -68,109 +63,19 @@ The setup scripts also allow for optional overrides of the following inputs whos
- `--logo-width` _(`number`)_: If `--logo` is provided or detected from an existing README.md, an explicit width style (by default, read from the image, capped to `128`)
- `--preserve-generated-from` _(`boolean`)_: Whether to keep the GitHub repository _generated from_ notice (by default, `false`)
-For example, customizing the ownership and users associated with a new repository:
-
-```shell
-npx create-typescript-app --author my-npm-username --email example@joshuakgoldberg.com --funding MyGitHubOrganization
-```
-
-> 💡 You can always manually edit files such as `package.json` after running a setup script.
-
-## Opt-Outs
-
-The setup scripts can be directed with CLI flags to opt out tooling portions and/or using API calls.
-
-### Excluding Tooling Portions
-
-The setup scripts normally will prompt you to select how much of the tooling you'd like to enable in a new repository.
-Alternately, you can bypass that prompt by providing any number of the following CLI flags:
-
-- `--exclude-all-contributors`: Don't add all-contributors to track contributions and display them in a README.md table.
-- `--exclude-build`: Don't add a build task to generate built `.js`, `.d.ts`, and related output.
-- `--exclude-compliance`: Don't add a GitHub Actions workflow to verify that PRs match an expected format.
-- `--exclude-lint-json`: Don't apply linting and sorting to `*.json` and `*.jsonc` files.
-- `--exclude-lint-knip`: Don't add Knip to detect unused files, dependencies, and code exports.
-- `--exclude-lint-md`: Don't apply linting to `*.md` files.
-- `--exclude-lint-package-json`: Don't add eslint-plugin-package-json to lint for package.json correctness.
-- `--exclude-lint-eslint`: Don't use eslint-plugin-eslint-comment to enforce good practices around ESLint comment directives.
-- `--exclude-lint-jsdoc`: Don't use eslint-plugin-jsdoc to enforce good practices around JSDoc comments.
-- `--exclude-lint-packages`: Don't add a pnpm dedupe workflow to ensure packages aren't duplicated unnecessarily.
-- `--exclude-lint-perfectionist`: Don't apply eslint-plugin-perfectionist to ensure imports, keys, and so on are in sorted order.
-- `--exclude-lint-regexp`: Don't add eslint-plugin-regexp to enforce good practices around regular expressions.
-- `--exclude-lint-strict`: Don't augment the recommended logical lint rules with typescript-eslint's strict config.
-- `--exclude-lint-stylistic`: Don't add stylistic rules such as typescript-eslint's stylistic config.
-- `--exclude-lint-spelling`: Don't add cspell to spell check against dictionaries of known words.
-- `--exclude-lint-yml`: Don't apply linting and sorting to `*.yaml` and `*.yml` files.
-- `--exclude-releases`: Don't add release-it to generate changelogs, package bumps, and publishes based on conventional commits.
-- `--exclude-renovate`: Don't add a Renovate config to dependencies up-to-date with PRs.
-- `--exclude-templated-by`: Don't add a _"This package was templated with create-typescript-app"_ notice at the end of the README.md.
-- `--exclude-tests`: Don't add Vitest tooling for fast unit tests, configured with coverage tracking.
-
-For example, initializing with all tooling except for `package.json` checks and Renovate:
+For example, customizing the npm author and funding source:
```shell
-npx create-typescript-app --exclude-lint-package-json --exclude-lint-packages --exclude-renovate
+npx create typescript-app --author my-npm-username --funding MyGitHubOrganization
```
-> **Warning**
-> Specifying any `--exclude-*` flag on the command-line will cause the setup script to skip prompting for more excludes.
-
-See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
-
-### Skipping API Calls
-
-> Alternately, see [Offline Mode](#offline-mode) to skip API calls without disabling features
-
-You can prevent the migration script from making some network-based changes using any or all of the following CLI flags:
-
-- `--skip-all-contributors-api` _(`boolean`)_: Skips network calls that fetch all-contributors data from GitHub
- - This flag does nothing if `--exclude-all-contributors` was specified.
-- `--skip-github-api` _(`boolean`)_: Skips calling to GitHub APIs.
-- `--skip-install` _(`boolean`)_: Skips installing all the new template packages with `pnpm`.
+## Block Exclusions
-For example, providing all three flags will completely skip all network requests:
+Per [`create` > CLI > Template Options > Block Exclusions](https://www.create.bingo/cli#block-exclusions), individual Blocks may be excluded from running.
+For example, initializing with all tooling except for Renovate:
```shell
-npx create-typescript-app --skip-all-contributors-api --skip-github-api --skip-install
+npx create typescript-app --exclude-renovate
```
-> 💡 Tip: To temporarily preview what the script would apply without making changes on GitHub, you can run with all `--skip-*-api` flags, then `git add -A; git reset --hard HEAD` to completely reset all changes.
-
-### Skipping Local Changes
-
-You can prevent the migration script from making some changes on disk using any or all of the following CLI flags:
-
-- `--skip-removal` _(`boolean`)_: Skips removing setup docs and scripts, including this `docs/` directory
-- `--skip-restore` _(`boolean`)_: Skips the prompt offering to restore the repository if an error occurs during setup
-- `--skip-uninstall` _(`boolean`)_: Skips uninstalling packages only used for setup scripts
-
-For example, providing all local change skip flags:
-
-```shell
-npx create-typescript-app --skip-removal --skip-restore --skip-uninstall
-```
-
-## Automatic Mode
-
-You can run `create-typescript-app` in an "automatic" manner with `--auto` and `--mode migrate`.
-Doing so will:
-
-- Use the default inference for all options
-- Bail out if any required [core options](#core-options) are missing
-
-```shell
-npx create-typescript-app --auto --mode migrate
-```
-
-## Offline Mode
-
-You can run `create-typescript-app` in an "offline" manner with `--offline`.
-Doing so will:
-
-- Enable `--exclude-all-contributors-api` and `--skip-github-api`
-- Skip network calls when setting up contributors
-- Run pnpm commands with pnpm's `--offline` mode
-
-```shell
-npx create-typescript-app --offline
-```
+See [Blocks.md](./Blocks.md) for the list of blocks and their corresponding presets.
diff --git a/package.json b/package.json
index 25174f5f4..caaa9fd58 100644
--- a/package.json
+++ b/package.json
@@ -24,7 +24,6 @@
"scripts": {
"build": "tsup",
"format": "prettier .",
- "initialize": "pnpm build --no-dts && tsx ./bin/index.js --mode initialize",
"lint": "eslint . --max-warnings 0",
"lint:knip": "knip",
"lint:md": "markdownlint \"**/*.md\" \".github/**/*.md\" --rules sentences-per-line",
@@ -32,17 +31,12 @@
"lint:spelling": "cspell \"**\" \".github/**/*\"",
"prepare": "husky",
"test": "vitest",
- "test:create": "npx tsx script/create-test-e2e.ts",
- "test:initialize": "npx tsx script/initialize-test-e2e.ts",
- "test:migrate": "vitest run -r script/",
"tsc": "tsc"
},
"lint-staged": {
"*": "prettier --ignore-unknown --write"
},
"dependencies": {
- "@clack/prompts": "^0.9.0",
- "@prettier/sync": "^0.5.2",
"chalk": "^5.4.1",
"create": "0.1.0-alpha.11",
"cspell-populate-words": "^0.3.0",
@@ -58,25 +52,18 @@
"lazy-value": "^3.0.0",
"npm-user": "^6.1.1",
"object-strings-deep": "^0.1.1",
- "octokit": "^4.0.2",
- "octokit-from-auth": "^0.3.0",
"parse-author": "^2.0.0",
"parse-package-name": "^1.0.0",
- "populate-all-contributors-for-repository": "^0.1.2",
"prettier": "^3.4.2",
"remove-undefined-objects": "^5.0.0",
- "replace-in-file": "^8.3.0",
- "rimraf": "^6.0.1",
"set-github-repository-labels": "^0.1.0",
"sort-package-json": "^2.12.0",
"title-case": "^4.3.2",
- "zod": "^3.24.1",
- "zod-validation-error": "^3.4.0"
+ "zod": "^3.24.1"
},
"devDependencies": {
"@eslint-community/eslint-plugin-eslint-comments": "4.4.1",
"@eslint/js": "9.17.0",
- "@octokit/request-error": "6.1.5",
"@release-it/conventional-changelog": "9.0.3",
"@types/eslint-plugin-markdown": "2.0.2",
"@types/git-url-parse": "9.0.3",
@@ -87,7 +74,6 @@
"@vitest/coverage-v8": "2.1.8",
"@vitest/eslint-plugin": "1.1.20",
"all-contributors-cli": "6.26.1",
- "c8": "10.1.3",
"console-fail-test": "0.5.0",
"create-testers": "0.1.0-alpha.11",
"cspell": "8.17.1",
@@ -100,7 +86,6 @@
"eslint-plugin-perfectionist": "4.4.0",
"eslint-plugin-regexp": "2.7.0",
"eslint-plugin-yml": "1.16.0",
- "globby": "14.0.2",
"husky": "9.1.7",
"knip": "5.41.1",
"lint-staged": "15.3.0",
@@ -112,7 +97,6 @@
"release-it": "17.10.0",
"sentences-per-line": "0.3.0",
"tsup": "8.3.5",
- "tsx": "4.19.2",
"typescript": "5.7.2",
"typescript-eslint": "8.19.0",
"vitest": "2.1.8"
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index 668e7cc19..325da0b11 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -8,12 +8,6 @@ importers:
.:
dependencies:
- '@clack/prompts':
- specifier: ^0.9.0
- version: 0.9.0
- '@prettier/sync':
- specifier: ^0.5.2
- version: 0.5.2(prettier@3.4.2)
chalk:
specifier: ^5.4.1
version: 5.4.1
@@ -59,33 +53,18 @@ importers:
object-strings-deep:
specifier: ^0.1.1
version: 0.1.1
- octokit:
- specifier: ^4.0.2
- version: 4.0.2
- octokit-from-auth:
- specifier: ^0.3.0
- version: 0.3.0
parse-author:
specifier: ^2.0.0
version: 2.0.0
parse-package-name:
specifier: ^1.0.0
version: 1.0.0
- populate-all-contributors-for-repository:
- specifier: ^0.1.2
- version: 0.1.2
prettier:
specifier: ^3.4.2
version: 3.4.2
remove-undefined-objects:
specifier: ^5.0.0
version: 5.0.0
- replace-in-file:
- specifier: ^8.3.0
- version: 8.3.0
- rimraf:
- specifier: ^6.0.1
- version: 6.0.1
set-github-repository-labels:
specifier: ^0.1.0
version: 0.1.0
@@ -98,9 +77,6 @@ importers:
zod:
specifier: ^3.24.1
version: 3.24.1
- zod-validation-error:
- specifier: ^3.4.0
- version: 3.4.0(zod@3.24.1)
devDependencies:
'@eslint-community/eslint-plugin-eslint-comments':
specifier: 4.4.1
@@ -108,9 +84,6 @@ importers:
'@eslint/js':
specifier: 9.17.0
version: 9.17.0
- '@octokit/request-error':
- specifier: 6.1.5
- version: 6.1.5
'@release-it/conventional-changelog':
specifier: 9.0.3
version: 9.0.3(conventional-commits-filter@5.0.0)(conventional-commits-parser@6.0.0)(release-it@17.10.0(typescript@5.7.2))
@@ -141,9 +114,6 @@ importers:
all-contributors-cli:
specifier: 6.26.1
version: 6.26.1
- c8:
- specifier: 10.1.3
- version: 10.1.3
console-fail-test:
specifier: 0.5.0
version: 0.5.0
@@ -180,9 +150,6 @@ importers:
eslint-plugin-yml:
specifier: 1.16.0
version: 1.16.0(eslint@9.17.0(jiti@2.4.0))
- globby:
- specifier: 14.0.2
- version: 14.0.2
husky:
specifier: 9.1.7
version: 9.1.7
@@ -216,9 +183,6 @@ importers:
tsup:
specifier: 8.3.5
version: 8.3.5(jiti@2.4.0)(postcss@8.4.36)(tsx@4.19.2)(typescript@5.7.2)(yaml@2.6.1)
- tsx:
- specifier: 4.19.2
- version: 4.19.2
typescript:
specifier: 5.7.2
version: 5.7.2
@@ -286,10 +250,6 @@ packages:
'@bcoe/v8-coverage@0.2.3':
resolution: {integrity: sha512-0hYQ8SB4Db5zvZB4axdMHGwEaQjkZzFjQiN9LVYvIFB2nSUHW9tYpxWriPrWDASIxiaXax83REcLxuSdnGPZtw==}
- '@bcoe/v8-coverage@1.0.1':
- resolution: {integrity: sha512-W+a0/JpU28AqH4IKtwUPcEUnUyXMDLALcn5/JLczGGT9fHE2sIby/xP/oQnx3nxkForzgzPy201RAKcB4xPAFQ==}
- engines: {node: '>=18'}
-
'@clack/core@0.4.0':
resolution: {integrity: sha512-YJCYBsyJfNDaTbvDUVSJ3SgSuPrcujarRgkJ5NLjexDZKvaOiVVJvAQYx8lIgG0qRT8ff0fPgqyBCVivanIZ+A==}
@@ -1221,11 +1181,6 @@ packages:
resolution: {integrity: sha512-UA91GwWPhFExt3IizW6bOeY/pQ0BkuNwKjk9iQW9KqxluGCrg4VenZ0/L+2Y0+ZOtme72EVvg6v0zo3AMQRCeA==}
engines: {node: '>=12'}
- '@prettier/sync@0.5.2':
- resolution: {integrity: sha512-Yb569su456XNx5BsH/Vyem7xD6g/y9iLmLUzRKM1a/dhU/D7HqqvkAG72znulXlMXztbV0iiu9O5AL8K98TzZQ==}
- peerDependencies:
- prettier: '*'
-
'@release-it/conventional-changelog@9.0.3':
resolution: {integrity: sha512-+3TL+B89Kc+VTbfGxpTvJkbegWt5XIzkovsYVJyoZpOZDG07v25FU8c5R5Q8yNUs76Ikfq0sp+ZTTxmefG4Hiw==}
engines: {node: ^18.18.0 || ^20.9.0 || ^22.0.0}
@@ -1365,9 +1320,6 @@ packages:
'@types/html-to-text@9.0.4':
resolution: {integrity: sha512-pUY3cKH/Nm2yYrEmDlPR1mR7yszjGx4DrwPjQ702C4/D5CwHuZTgZdIdwPkRbcuhs7BAh2L5rg3CL5cbRiGTCQ==}
- '@types/istanbul-lib-coverage@2.0.4':
- resolution: {integrity: sha512-z/QT1XN4K4KYuslS23k62yDIDLwLFkzxOuMplDtObz0+y7VqJCaO2o+SPwHCvLFZh7xazvvoor2tA/hPz9ee7g==}
-
'@types/js-yaml@4.0.9':
resolution: {integrity: sha512-k4MGaQl5TGo/iipqb2UDG2UwjXziSWkh0uysQelTlJpX1qGlpUZYm8PnO4DxG1qBomtJUdYJ6qR6xdIah10JLg==}
@@ -1532,10 +1484,6 @@ packages:
engines: {node: '>=4'}
hasBin: true
- all-contributors-for-repository@0.4.0:
- resolution: {integrity: sha512-qGJHdVaRnoKMEZ6IcDvrFQJIDA9KPSsCVAMk+mjqVkHMYrgvp05TU94I8x4SjwWIfYBLp9heC9Onh55BIrOA4A==}
- engines: {node: '>=18'}
-
ansi-align@3.0.1:
resolution: {integrity: sha512-IOfwwBF5iczOjp/WeY4YxyjqAFMQoZufdQWDd19SEExbVLNXqvpzSJ/M7Za4/sCPmQ0+GRquoA7bGcINcxew6w==}
@@ -1662,24 +1610,10 @@ packages:
peerDependencies:
esbuild: '>=0.18'
- c8@10.1.3:
- resolution: {integrity: sha512-LvcyrOAaOnrrlMpW22n690PUvxiq4Uf9WMhQwNJ9vgagkL/ph1+D4uvjvDA5XCbykrc0sx+ay6pVi9YZ1GnhyA==}
- engines: {node: '>=18'}
- hasBin: true
- peerDependencies:
- monocart-coverage-reports: ^2
- peerDependenciesMeta:
- monocart-coverage-reports:
- optional: true
-
cac@6.7.14:
resolution: {integrity: sha512-b6Ilus+c3RrdDk+JhLKUAQfzzgLEPy6wcXqS7f/xe1EETvsDP6GORG7SFuOs6cID5YkqchW/LXZbX5bc8j7ZcQ==}
engines: {node: '>=8'}
- cached-factory@0.0.2:
- resolution: {integrity: sha512-4mhebGQ8YyvlDAX+zOHso5MezPi2pP11ZFE7vYhDIOJifsxEi7R5geB/nrLBYzQH5nwZ9kNsLAjM+WBj6osLDg==}
- engines: {node: '>=18'}
-
cached-factory@0.1.0:
resolution: {integrity: sha512-IGOSWu+NuED5UzCRmBeqQPZ8z7SkgrD/nN67W2iY1Qv83CVhevyMexkGclJ86saXisIqxoOnbeiTWvsCHRqJBw==}
engines: {node: '>=18'}
@@ -1811,10 +1745,6 @@ packages:
resolution: {integrity: sha512-JQHZ2QMW6l3aH/j6xCqQThY/9OH4D/9ls34cgkUBiEeocRTU04tHfKPBsUK1PqZCUQM7GiA0IIXJSuXHI64Kbg==}
engines: {node: '>=0.8'}
- co-author-to-username@0.1.1:
- resolution: {integrity: sha512-KtQvKq2m52MncBa9hmYSiUae/hRRjLIOpXY29Ll43mSK/KreIRcoNy+K5/U5UHyVMNNjux1Xi4Ya744BO4G+RA==}
- engines: {node: '>=18'}
-
color-convert@1.9.3:
resolution: {integrity: sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==}
@@ -1943,9 +1873,6 @@ packages:
engines: {node: '>=18'}
hasBin: true
- convert-source-map@2.0.0:
- resolution: {integrity: sha512-Kvp459HrV2FEJ1CAsi1Ku+MY3kasH19TFykTz2xWmMeq6bk2NU3XXvfJ+Q61m0xktWwt+1HSYf3JZsTms3aRJg==}
-
core-util-is@1.0.3:
resolution: {integrity: sha512-ZQBvi1DcpJ4GDqanjucZ2Hj3wEO5pZDS89BWbkcrvdxksJorwUDDZamX9ldFkp9aw2lmBDLgkObEA4DWNJ9FYQ==}
@@ -2089,10 +2016,6 @@ packages:
resolution: {integrity: sha512-0je+qPKHEMohvfRTCEo3CrPG6cAzAYgmzKyxRiYSSDkS6eGJdyVJm7WaYA5ECaAD9wLB2T4EEeymA5aFVcYXCA==}
engines: {node: '>=6'}
- description-to-co-authors@0.3.0:
- resolution: {integrity: sha512-GaMDuvOCmtkQ3tjjawAEl3YhxNEIsR9tmlXV4xeypGyiO75YLXRm3iHKltqrmxNgtNlppVzT2pdsPM2Yzm67CA==}
- engines: {node: '>=18.3.0'}
-
detect-indent@6.1.0:
resolution: {integrity: sha512-reYkTUJAZb9gUuZ2RvVCNhVHdg62RHnJ7WJl8ftMi4diZ6NWlciOzQN88pUhSELEwflJht4oQDv0F0BMlwaYtA==}
engines: {node: '>=8'}
@@ -3106,9 +3029,6 @@ packages:
resolution: {integrity: sha512-hXdUTZYIVOt1Ex//jAQi+wTZZpUpwBj/0QsOzqegb3rGMMeJiSEu5xLHnYfBrRV4RH2+OCSOO95Is/7x1WJ4bw==}
engines: {node: '>=10'}
- make-synchronized@0.2.9:
- resolution: {integrity: sha512-4wczOs8SLuEdpEvp3vGo83wh8rjJ78UsIk7DIX5fxdfmfMJGog4bQzxfvOwq7Q3yCHLC4jp1urPHIxRS/A93gA==}
-
markdown-it@14.1.0:
resolution: {integrity: sha512-a54IwgWPaeBCAAsv13YgmALOF1elABB08FxO9i+r4VFk5Vl4pKokRPeX8u5TCgSsPi6ec1otfLjdOpVcgbpshg==}
hasBin: true
@@ -3364,10 +3284,6 @@ packages:
resolution: {integrity: sha512-UMo+WQkWi1PS/HZ15SMM1sgPhhXykqPJKCCAi6R0+JAhTbLBcfeKCJ3AwUVdJBJQKSQ37BUqPpME4iGAjZQtkw==}
engines: {node: '>=18.3.0'}
- octokit-from-auth@0.3.0:
- resolution: {integrity: sha512-HeS+YVPExV5nkdRZyQQRepZVV+pL4ahVmh9KfKXLAbNc3D82G2xeuSWfKzHSIlqB1taQnZYPJWdpfcIe45d1lw==}
- engines: {node: '>=18.3.0'}
-
octokit@4.0.2:
resolution: {integrity: sha512-wbqF4uc1YbcldtiBFfkSnquHtECEIpYD78YUXI6ri1Im5OO2NLo6ZVpRdbJpdnpZ05zMrVPssNiEo6JQtea+Qg==}
engines: {node: '>= 18'}
@@ -3580,11 +3496,6 @@ packages:
resolution: {integrity: sha512-saLsH7WeYYPiD25LDuLRRY/i+6HaPYr6G1OUlN39otzkSTxKnubR9RTxS3/Kk50s1g2JTgFwWQDQyplC5/SHZg==}
engines: {node: '>= 6'}
- populate-all-contributors-for-repository@0.1.2:
- resolution: {integrity: sha512-PQ93fI7OpetHtW0c+iu6Dx2aaiRurj07LTu51pAw/qkky0eb1CdcZ+4Y+DYbzZXId4UhHGH/ednm46Pwf1P8FA==}
- engines: {node: '>=18.3.0'}
- hasBin: true
-
postcss-load-config@6.0.1:
resolution: {integrity: sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==}
engines: {node: '>= 18'}
@@ -3732,11 +3643,6 @@ packages:
resolution: {integrity: sha512-PV0dzCYDNfRi1jCDbJzpW7jNNDRuCOG/jI5ctQcGKt/clZD+YcPS3yIlWuTJMmESC8aevCFmWJy5wjAFgNqN6w==}
engines: {node: '>=0.10'}
- replace-in-file@8.3.0:
- resolution: {integrity: sha512-4VhddQiMCPIuypiwHDTM+XHjZoVu9h7ngBbSCnwGRcwdHwxltjt/m//Ep3GDwqaOx1fDSrKFQ+n7uo4uVcEz9Q==}
- engines: {node: '>=18'}
- hasBin: true
-
require-directory@2.1.1:
resolution: {integrity: sha512-fGxEI7+wsG9xrvdjsrlmL22OMTTiHRwAMroiEeMgq8gzoLC/PQr7RsRDSTLUg/bZAZtF+TVIkHc6/4RIKrui+Q==}
engines: {node: '>=0.10.0'}
@@ -3778,11 +3684,6 @@ packages:
rfdc@1.4.1:
resolution: {integrity: sha512-q1b3N5QkRUWUl7iyylaaj3kOpIT0N2i9MqIEQXP73GVsN9cw3fdx8X63cEmWhJGi2PPCF23Ijp7ktmd39rawIA==}
- rimraf@6.0.1:
- resolution: {integrity: sha512-9dkvaxAsk/xNXSJzMgFqqMCuFgt2+KsOFek3TMLfo8NCPfWpBmqwyNn5Y+NX56QUYfCtsyhF3ayiboEoUmJk/A==}
- engines: {node: 20 || >=22}
- hasBin: true
-
rollup@4.28.0:
resolution: {integrity: sha512-G9GOrmgWHBma4YfCcX8PjH0qhXSdH8B4HDE2o4/jaxj93S4DPCIDoLcXz99eWMji4hB29UFCEd7B2gwGJDR9cQ==}
engines: {node: '>=18.0.0', npm: '>=8.0.0'}
@@ -4225,10 +4126,6 @@ packages:
util-deprecate@1.0.2:
resolution: {integrity: sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==}
- v8-to-istanbul@9.2.0:
- resolution: {integrity: sha512-/EH/sDgxU2eGxajKdwLCDmQ4FWq+kpi3uCmBGpw1xJtnAxEjlD8j8PEiGWpCIMIs3ciNAgH0d3TTJiUkYzyZjA==}
- engines: {node: '>=10.12.0'}
-
validate-npm-package-license@3.0.4:
resolution: {integrity: sha512-DpKm2Ui/xN7/HQKCtpZxoRWBhZ9Z0kqtygG8XCgNQ8ZlDnxuQmWhj566j8fN4Cu3/JmbhsDo7fcAJq4s9h27Ew==}
@@ -4497,8 +4394,6 @@ snapshots:
'@bcoe/v8-coverage@0.2.3': {}
- '@bcoe/v8-coverage@1.0.1': {}
-
'@clack/core@0.4.0':
dependencies:
picocolors: 1.1.1
@@ -5264,11 +5159,6 @@ snapshots:
'@pnpm/network.ca-file': 1.0.2
config-chain: 1.1.13
- '@prettier/sync@0.5.2(prettier@3.4.2)':
- dependencies:
- make-synchronized: 0.2.9
- prettier: 3.4.2
-
'@release-it/conventional-changelog@9.0.3(conventional-commits-filter@5.0.0)(conventional-commits-parser@6.0.0)(release-it@17.10.0(typescript@5.7.2))':
dependencies:
concat-stream: 2.0.0
@@ -5376,8 +5266,6 @@ snapshots:
'@types/html-to-text@9.0.4': {}
- '@types/istanbul-lib-coverage@2.0.4': {}
-
'@types/js-yaml@4.0.9': {}
'@types/json-schema@7.0.15': {}
@@ -5595,14 +5483,6 @@ snapshots:
transitivePeerDependencies:
- encoding
- all-contributors-for-repository@0.4.0:
- dependencies:
- co-author-to-username: 0.1.1
- conventional-commits-parser: 6.0.0
- description-to-co-authors: 0.3.0
- octokit: 4.0.2
- octokit-from-auth: 0.3.0
-
ansi-align@3.0.1:
dependencies:
string-width: 4.2.3
@@ -5722,24 +5602,8 @@ snapshots:
esbuild: 0.24.0
load-tsconfig: 0.2.5
- c8@10.1.3:
- dependencies:
- '@bcoe/v8-coverage': 1.0.1
- '@istanbuljs/schema': 0.1.3
- find-up: 5.0.0
- foreground-child: 3.3.0
- istanbul-lib-coverage: 3.2.2
- istanbul-lib-report: 3.0.1
- istanbul-reports: 3.1.7
- test-exclude: 7.0.1
- v8-to-istanbul: 9.2.0
- yargs: 17.7.2
- yargs-parser: 21.1.1
-
cac@6.7.14: {}
- cached-factory@0.0.2: {}
-
cached-factory@0.1.0: {}
callsites@3.1.0: {}
@@ -5862,11 +5726,6 @@ snapshots:
clone@1.0.4: {}
- co-author-to-username@0.1.1:
- dependencies:
- cached-factory: 0.0.2
- octokit: 4.0.2
-
color-convert@1.9.3:
dependencies:
color-name: 1.1.3
@@ -6006,8 +5865,6 @@ snapshots:
conventional-commits-parser: 6.0.0
meow: 13.2.0
- convert-source-map@2.0.0: {}
-
core-util-is@1.0.3: {}
cosmiconfig@9.0.0(typescript@5.7.2):
@@ -6199,8 +6056,6 @@ snapshots:
dequal@2.0.3: {}
- description-to-co-authors@0.3.0: {}
-
detect-indent@6.1.0: {}
detect-indent@7.0.1: {}
@@ -6326,6 +6181,7 @@ snapshots:
'@esbuild/win32-arm64': 0.23.0
'@esbuild/win32-ia32': 0.23.0
'@esbuild/win32-x64': 0.23.0
+ optional: true
esbuild@0.24.0:
optionalDependencies:
@@ -7371,8 +7227,6 @@ snapshots:
dependencies:
semver: 7.6.3
- make-synchronized@0.2.9: {}
-
markdown-it@14.1.0:
dependencies:
argparse: 2.0.1
@@ -7731,11 +7585,6 @@ snapshots:
get-github-auth-token: 0.1.0
octokit: 4.0.2
- octokit-from-auth@0.3.0:
- dependencies:
- get-github-auth-token: 0.1.0
- octokit: 4.0.2
-
octokit@4.0.2:
dependencies:
'@octokit/app': 15.0.1
@@ -7987,14 +7836,6 @@ snapshots:
pirates@4.0.6: {}
- populate-all-contributors-for-repository@0.1.2:
- dependencies:
- all-contributors-cli: 6.26.1
- all-contributors-for-repository: 0.4.0
- execa: 9.5.2
- transitivePeerDependencies:
- - encoding
-
postcss-load-config@6.0.1(jiti@2.4.0)(postcss@8.4.36)(tsx@4.19.2)(yaml@2.6.1):
dependencies:
lilconfig: 3.1.3
@@ -8162,12 +8003,6 @@ snapshots:
repeat-string@1.6.1: {}
- replace-in-file@8.3.0:
- dependencies:
- chalk: 5.4.1
- glob: 10.4.5
- yargs: 17.7.2
-
require-directory@2.1.1: {}
require-main-filename@2.0.0: {}
@@ -8200,11 +8035,6 @@ snapshots:
rfdc@1.4.1: {}
- rimraf@6.0.1:
- dependencies:
- glob: 11.0.0
- package-json-from-dist: 1.0.1
-
rollup@4.28.0:
dependencies:
'@types/estree': 1.0.6
@@ -8564,6 +8394,7 @@ snapshots:
get-tsconfig: 4.8.1
optionalDependencies:
fsevents: 2.3.3
+ optional: true
type-check@0.4.0:
dependencies:
@@ -8633,12 +8464,6 @@ snapshots:
util-deprecate@1.0.2: {}
- v8-to-istanbul@9.2.0:
- dependencies:
- '@jridgewell/trace-mapping': 0.3.25
- '@types/istanbul-lib-coverage': 2.0.4
- convert-source-map: 2.0.0
-
validate-npm-package-license@3.0.4:
dependencies:
spdx-correct: 3.1.1
diff --git a/script/__snapshots__/migrate-test-e2e.ts.snap b/script/__snapshots__/migrate-test-e2e.ts.snap
deleted file mode 100644
index 9806ef36d..000000000
--- a/script/__snapshots__/migrate-test-e2e.ts.snap
+++ /dev/null
@@ -1,273 +0,0 @@
-// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
-
-exports[`expected file changes > .github/renovate.json 1`] = `
-"--- a/.github/renovate.json
-+++ b/.github/renovate.json
-@@ ... @@
- "$schema": "https://docs.renovatebot.com/renovate-schema.json",
- "automerge": true,
- "extends": ["config:best-practices", "replacements:all"],
-- "github-actions": { "enabled": false },
- "ignoreDeps": ["codecov/codecov-action"],
- "labels": ["dependencies"],
- "minimumReleaseAge": "7 days","
-`;
-
-exports[`expected file changes > .github/workflows/ci.yml 1`] = `
-"--- a/.github/workflows/ci.yml
-+++ b/.github/workflows/ci.yml
-@@ ... @@
- jobs:
-- are_the_types_wrong:
-- name: Are The Types Wrong?
-- runs-on: ubuntu-latest
-- steps:
-- - uses: actions/checkout@v4
-- - uses: ./.github/actions/prepare
-- - run: pnpm build
-- - run: npx --yes @arethetypeswrong/cli --pack . --ignore-rules cjs-resolves-to-esm
- build:
- name: Build
- runs-on: ubuntu-latest
-@@ ... @@ jobs:
- - uses: actions/checkout@v4
- - uses: ./.github/actions/prepare
- - run: pnpm build
-- - run: node ./lib/index.js --version
-+ - run: node ./lib/index.js
- lint:
- name: Lint
- runs-on: ubuntu-latest
-@@ ... @@ jobs:
- - uses: actions/checkout@v4
- - uses: ./.github/actions/prepare
- - run: pnpm run test --coverage
-- - env:
-- CODECOV_TOKEN: \${{ secrets.CODECOV_TOKEN }}
-- if: always()
-- uses: codecov/codecov-action@v3
-- with:
-- flags: unit
-- test_creation_script:
-- name: Test Creation Script
-- runs-on: ubuntu-latest
-- steps:
-- - uses: actions/checkout@v4
-- - uses: ./.github/actions/prepare
-- - run: pnpm run build
-- - run: pnpm run test:create
-- - env:
-- CODECOV_TOKEN: \${{ secrets.CODECOV_TOKEN }}
-- if: always()
-- uses: codecov/codecov-action@v3
-- with:
-- files: coverage-create/lcov.info
-- flags: create
-- test_initialization_script:
-- name: Test Initialization Script
-- runs-on: ubuntu-latest
-- steps:
-- - uses: actions/checkout@v4
-- - uses: ./.github/actions/prepare
-- - run: pnpm run build
-- - run: pnpm run test:initialize
-- - env:
-- CODECOV_TOKEN: \${{ secrets.CODECOV_TOKEN }}
-- if: always()
-- uses: codecov/codecov-action@v3
-- with:
-- files: coverage-initialize/lcov.info
-- flags: initialize
-- test_migration_script:
-- name: Test Migration Script
-- runs-on: ubuntu-latest
-- steps:
-- - uses: actions/checkout@v4
-- - uses: ./.github/actions/prepare
-- - run: pnpm run build
-- - run: pnpm run test:migrate
-- - env:
-- CODECOV_TOKEN: \${{ secrets.CODECOV_TOKEN }}
-- if: always()
-+ - if: always()
- uses: codecov/codecov-action@v3
-- with:
-- files: coverage-migrate/lcov.info
-- flags: migrate
- type_check:
- name: Type Check
- runs-on: ubuntu-latest"
-`;
-
-exports[`expected file changes > .gitignore 1`] = `
-"--- a/.gitignore
-+++ b/.gitignore
-@@ ... @@
--/coverage*
-+/coverage
- /lib
- /node_modules"
-`;
-
-exports[`expected file changes > .prettierignore 1`] = `
-"--- a/.prettierignore
-+++ b/.prettierignore
-@@ ... @@
- /.husky
--/coverage*
-+/coverage
- /lib
- /pnpm-lock.yaml"
-`;
-
-exports[`expected file changes > README.md 1`] = `
-"--- a/README.md
-+++ b/README.md
-@@ ... @@
- Create TypeScript App
-
--Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling. 🎁
-+A very lovely package. Hooray!
-
-
-
-@@ ... @@ Thanks! 💖
-
-
-
-+
-+
-+
-+> 💙 This package was templated with [\`create-typescript-app\`](https://github.com/JoshuaKGoldberg/create-typescript-app)."
-`;
-
-exports[`expected file changes > eslint.config.js 1`] = `
-"--- a/eslint.config.js
-+++ b/eslint.config.js
-@@ ... @@
--/*
--👋 Hi! This ESLint configuration contains a lot more stuff than many repos'!
--You can read from it to see all sorts of linting goodness, but don't worry -
--it's not something you need to exhaustively understand immediately. 💙
--
--If you're interested in learning more, see the 'getting started' docs on:
--- ESLint: https://eslint.org
--- typescript-eslint: https://typescript-eslint.io
--*/
--
- import comments from "@eslint-community/eslint-plugin-eslint-comments/configs";
- import eslint from "@eslint/js";
- import vitest from "@vitest/eslint-plugin";
-@@ ... @@ import tseslint from "typescript-eslint";
-
- export default tseslint.config(
- {
-- ignores: [
-- "**/*.snap",
-- "coverage*",
-- "lib",
-- "node_modules",
-- "pnpm-lock.yaml",
-- ],
-+ ignores: ["**/*.snap", "coverage", "lib", "node_modules", "pnpm-lock.yaml"],
- },
- { linterOptions: { reportUnusedDisableDirectives: "error" } },
- eslint.configs.recommended,
-@@ ... @@ export default tseslint.config(
- files: ["**/*.js", "**/*.ts"],
- languageOptions: {
- parserOptions: {
-- projectService: {
-- allowDefaultProject: ["*.config.*s", "bin/*.js"],
-- },
-+ projectService: { allowDefaultProject: ["*.config.*s"] },
- tsconfigRootDir: import.meta.dirname,
- },
- },
- rules: {
-- // These on-by-default rules work well for this repo if configured
-- "@typescript-eslint/no-unnecessary-condition": [
-- "error",
-- {
-- allowConstantLoopConditions: true,
-- },
-- ],
-- "@typescript-eslint/prefer-nullish-coalescing": [
-- "error",
-- { ignorePrimitives: true },
-- ],
-- "@typescript-eslint/restrict-template-expressions": [
-- "error",
-- { allowBoolean: true, allowNullish: true, allowNumber: true },
-- ],
-- "n/no-unsupported-features/node-builtins": [
-- "error",
-- { allowExperimental: true },
-- ],
--
- // Stylistic concerns that don't interfere with Prettier
- "logical-assignment-operators": [
- "error",
-@@ ... @@ export default tseslint.config(
- {
- extends: [tseslint.configs.disableTypeChecked],
- files: ["**/*.md/*.ts"],
-+ rules: {
-+ "n/no-missing-import": [
-+ "error",
-+ { allowModules: ["create-typescript-app"] },
-+ ],
-+ },
- },
- {
- extends: [vitest.configs.recommended],"
-`;
-
-exports[`expected file changes > knip.json 1`] = `
-"--- a/knip.json
-+++ b/knip.json
-@@ ... @@
- {
- "$schema": "https://unpkg.com/knip@5.41.1/schema.json",
-- "entry": ["script/*e2e.js", "src/index.ts!", "src/**/*.test.*"],
-- "ignoreDependencies": ["all-contributors-cli", "cspell-populate-words"],
-+ "entry": ["src/index.ts!"],
- "ignoreExportsUsedInFile": { "interface": true, "type": true },
-- "project": ["src/**/*.ts!", "script/**/*.js"]
-+ "project": ["src/**/*.ts!"]
- }"
-`;
-
-exports[`expected file changes > package.json 1`] = `
-"--- a/package.json
-+++ b/package.json
-@@ ... @@
- {
- "name": "create-typescript-app",
- "version": "1.79.0",
-- "description": "Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling. 🎁",
-+ "description": "A very lovely package. Hooray! 💖",
- "repository": {
- "type": "git",
- "url": "https://github.com/JoshuaKGoldberg/create-typescript-app"
-@@ ... @@
- "lint-staged": "15.2.11",
- "markdownlint": "0.37.2",
- "markdownlint-cli": "0.43.0",
-+ "prettier": "^3.4.2",
- "prettier-plugin-curly": "0.3.1",
- "prettier-plugin-packagejson": "2.5.6",
- "prettier-plugin-sh": "0.14.0","
-`;
-
-exports[`expected file changes > tsconfig.json 1`] = `
-"--- a/tsconfig.json
-+++ b/tsconfig.json
-@@ ... @@
- "strict": true,
- "target": "ES2022"
- },
-- "include": ["src", "script"]
-+ "include": ["src"]
- }"
-`;
diff --git a/script/create-test-e2e.ts b/script/create-test-e2e.ts
deleted file mode 100644
index 2419e8b9a..000000000
--- a/script/create-test-e2e.ts
+++ /dev/null
@@ -1,44 +0,0 @@
-import { $, execaCommand } from "execa";
-import { strict as assert } from "node:assert";
-import { rimraf } from "rimraf";
-
-const author = "Test Author";
-const description = "Test description.";
-const email = "test@email.com";
-const repository = "testing-repository";
-const owner = "TestOwner";
-const title = "Test Title";
-
-await rimraf(["coverage*", repository]);
-
-// Fist we run with --mode create to create a new new local repository,
-// asserting that pnpm i passes in that repository's directory.
-await $({
- stdio: "inherit",
-})`c8 -o ./coverage-create -r html -r lcov --src src node ./bin/index.js --base everything --mode create --author ${author} --email ${email} --description ${description} --owner ${owner} --title ${title} --repository ${repository} --skip-all-contributors-api --skip-github-api`;
-
-process.chdir(repository);
-
-const failures = [];
-
-// Then we run each of the CI commands to assert that they pass too.
-for (const command of [
- `pnpm i`,
- `pnpm run build`,
- `pnpm run format --list-different`,
- `pnpm run lint`,
- `pnpm run lint:md`,
- `pnpm run lint:packages`,
- `pnpm run lint:spelling`,
- `pnpm run lint:knip`,
- `pnpm run test run`,
- `pnpm run tsc`,
-]) {
- const result = await execaCommand(command, { stdio: "inherit" });
-
- if (result.exitCode) {
- failures.push({ command, result });
- }
-}
-
-assert.deepEqual(failures, []);
diff --git a/script/initialize-test-e2e.ts b/script/initialize-test-e2e.ts
deleted file mode 100644
index a64be02ec..000000000
--- a/script/initialize-test-e2e.ts
+++ /dev/null
@@ -1,61 +0,0 @@
-import { $ } from "execa";
-import { globby } from "globby";
-import { strict as assert } from "node:assert";
-import * as fs from "node:fs/promises";
-import { rimraf } from "rimraf";
-
-const description = "New Description Test";
-const owner = "RNR1";
-const title = "New Title Test";
-const repository = "new-repository-test";
-
-await rimraf("coverage*");
-
-// Fist we run with --mode initialize to modify the local repository files,
-// asserting that the created package.json keeps the general description.
-await $({
- stdio: "inherit",
-})`pnpm run initialize --base everything --mode initialize --description ${description} --owner ${owner} --title ${title} --repository ${repository} --skip-all-contributors-api --skip-github-api --skip-restore`;
-
-const newPackageJson = JSON.parse(
- (await fs.readFile("./package.json")).toString(),
-) as Record;
-console.log("New package JSON:", newPackageJson);
-
-assert.equal(newPackageJson.description, description);
-assert.equal(newPackageJson.name, repository);
-
-// Assert that the initialize script used the provided values in files,
-// except for the 'This package was templated with ...' attribution notice.
-const files = await globby(["*.*", "**/*.*"], {
- gitignore: true,
- ignoreFiles: ["script/initialize-test-e2e.js"],
-});
-
-for (const search of [`/JoshuaKGoldberg/`, "create-typescript-app"]) {
- const { stdout } = await $`grep -i ${search} ${files}`;
- assert.equal(
- stdout,
- `README.md:> 💙 This package was templated with [\`create-typescript-app\`](https://github.com/JoshuaKGoldberg/create-typescript-app).`,
- );
-}
-
-// Use Knip to assert that none of the template-only dependencies remain.
-// They should have been removed as part of initialization.
-try {
- await $`pnpm run lint:knip`;
-} catch (error) {
- throw new Error(
- `Error running lint:knip: ${(error as Error).stack ?? (error as string)}`,
- );
-}
-
-// Now that initialize has passed normal steps, we reset everything,
-// then run again without removing files - so we can capture test coverage
-await $`git add -A`;
-await $`git reset --hard HEAD`;
-await $`pnpm i`;
-await $`pnpm run build`;
-await $({
- stdio: "inherit",
-})`c8 -o ./coverage -r html -r lcov --src src node ./bin/index.js --base everything --mode initialize --description ${description} --owner ${owner} --title ${title} --repository ${repository} --skip-all-contributors-api --skip-github-api --skip-removal --skip-restore`;
diff --git a/script/migrate-test-e2e.ts b/script/migrate-test-e2e.ts
deleted file mode 100644
index e713d67be..000000000
--- a/script/migrate-test-e2e.ts
+++ /dev/null
@@ -1,151 +0,0 @@
-import chalk from "chalk";
-import { $, execaCommand } from "execa";
-import * as fs from "node:fs/promises";
-import { rimraf } from "rimraf";
-import { assert, describe, expect, test } from "vitest";
-
-const filesExpectedToBeChanged = [
- "README.md",
- "knip.json",
- ".github/renovate.json",
- ".github/workflows/ci.yml",
- ".gitignore",
- ".prettierignore",
- "eslint.config.js",
- "package.json",
- "tsconfig.json",
-];
-
-const filesThatMightBeChanged = new Set([
- // For now, ignore typos cspell is picking up from migration snapshots.
- "cspell.json",
-
- "script/__snapshots__/migrate-test-e2e.ts.snap",
- ...filesExpectedToBeChanged,
-]);
-
-await rimraf("coverage*");
-
-const originalReadme = (await fs.readFile("README.md")).toString();
-
-const originalSnapshots = (
- await fs.readFile("script/__snapshots__/migrate-test-e2e.ts.snap")
-).toString();
-
-await $({
- stdio: "inherit",
-})`c8 -o ./coverage -r html -r lcov --src src node ./bin/index.js --auto --mode migrate --skip-all-contributors-api --skip-github-api --skip-install`;
-
-// All Contributors seems to not be using Prettier to format files...
-await fs.writeFile(
- ".all-contributorsrc",
- JSON.stringify(
- JSON.parse((await fs.readFile(".all-contributorsrc")).toString()),
- null,
- 2,
- ) + "\n",
-);
-
-// Ignore changes to the README.md all-contributor count and contributors table...
-const updatedReadme = (await fs.readFile("README.md")).toString();
-await fs.writeFile(
- "README.md",
- [
- updatedReadme.slice(0, updatedReadme.indexOf("## Contributors")) +
- originalReadme.slice(
- originalReadme.indexOf("## Contributors"),
- originalReadme.indexOf(""),
- ),
- updatedReadme.slice(updatedReadme.indexOf("")),
- ]
- .join("")
- .replace(
- /All Contributors: \d+/g,
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
- /All Contributors: \d+/.exec(originalReadme)![0],
- )
- .replace(
- /all_contributors-\d+/g,
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
- /all_contributors-\d+/.exec(originalReadme)![0],
- ),
-);
-
-// ...and even to the snapshot file, so diffs don't mind it.
-await fs.writeFile(
- "script/__snapshots__/migrate-test-e2e.ts.snap",
- originalSnapshots
- .replace(
- /All Contributors: \d+/g,
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
- /All Contributors: \d+/.exec(originalReadme)![0],
- )
- .replace(
- /all_contributors-\d+/g,
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
- /all_contributors-\d+/.exec(originalReadme)![0],
- ),
-);
-
-describe("expected file changes", () => {
- test.each(filesExpectedToBeChanged)("%s", async (file) => {
- const { stdout } = await execaCommand(`git diff HEAD -- ${file}`);
- const contentsAfterGitMarkers = stdout
- .split("\n")
- .slice(2)
- .join("\n")
- .replace(/@@ -\d+,\d+ \+\d+,\d+ @@/g, "@@ ... @@");
-
- assert(
- stdout,
- `Looks like there were no changes to ${file} from migration?`,
- );
-
- // If this fails, see .github/DEVELOPMENT.md > Setup Scripts for context.
- // Then see .github/DEVELOPMENT.md > Migration Snapshot Failures.
- expect(contentsAfterGitMarkers).toMatchSnapshot();
- });
-});
-
-test("unexpected file changes", async () => {
- const { stdout: gitStatus } = await $`git status`;
- console.log(`Stdout from running \`git status\`:\n${gitStatus}`);
-
- const indexOfUnstagedFilesMessage = gitStatus.indexOf(
- "Changes not staged for commit:",
- );
- assert(
- indexOfUnstagedFilesMessage !== -1,
- `Looks like migrate didn't cause any file changes? That's ...probably incorrect? 😬`,
- );
-
- const unstagedModifiedFiles = gitStatus
- .slice(indexOfUnstagedFilesMessage)
- .match(/modified: {3}\S+\n/g)
- ?.map((match) => match.split(/\s+/)[1])
- .filter((filePath) => !filesThatMightBeChanged.has(filePath));
-
- console.log("Unexpected modified files are:", unstagedModifiedFiles);
-
- if (unstagedModifiedFiles?.length) {
- const gitDiffCommand = `git diff HEAD -- ${unstagedModifiedFiles.join(
- " ",
- )}`;
- const { stdout } = await execaCommand(gitDiffCommand);
-
- console.log(`Stdout from running \`${gitDiffCommand}\`:\n${stdout}`);
-
- throw new Error(
- [
- "",
- "Oh no! Running the migrate script unexpectedly modified:",
- ...unstagedModifiedFiles.map((filePath) => ` - ${filePath}`),
- "",
- "See .github/DEVELOPMENT.md > Setup Scripts for context.",
- "Then see .github/DEVELOPMENT.md > Unexpected File Modifications.",
- ]
- .map((line) => chalk.red(line))
- .join("\n"),
- );
- }
-});
diff --git a/script/vitest.config.ts b/script/vitest.config.ts
deleted file mode 100644
index 663590231..000000000
--- a/script/vitest.config.ts
+++ /dev/null
@@ -1,3 +0,0 @@
-import { defineConfig } from "vitest/config";
-
-export default defineConfig({ test: { include: ["./migrate-test-e2e.ts"] } });
diff --git a/src/base.test.ts b/src/base.test.ts
new file mode 100644
index 000000000..ca22a4735
--- /dev/null
+++ b/src/base.test.ts
@@ -0,0 +1,53 @@
+import { produceBase } from "create";
+import { readFile } from "fs/promises";
+import { describe, expect, test } from "vitest";
+
+import { base } from "./base.js";
+import { AllContributorsData } from "./types.js";
+
+describe("base", () => {
+ test("production from create-typescript-app", async () => {
+ const options = await produceBase(base);
+
+ expect(options).toEqual({
+ access: "public",
+ author: "Josh Goldberg ✨",
+ bin: "./bin/index.js",
+ contributors: (
+ JSON.parse(
+ (await readFile(".all-contributorsrc")).toString(),
+ ) as AllContributorsData
+ ).contributors,
+ description:
+ "Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling. 🎁",
+ documentation: "",
+ email: {
+ github: "github@joshuakgoldberg.com",
+ npm: "npm@joshuakgoldberg.com",
+ },
+ funding: "JoshuaKGoldberg",
+ guide: {
+ href: "https://www.joshuakgoldberg.com/blog/contributing-to-a-create-typescript-app-repository",
+ title: "Contributing to a create-typescript-app Repository",
+ },
+ login: "Josh Goldberg ✨",
+ logo: {
+ alt: "Project logo: the TypeScript blue square with rounded corners, but a plus sign instead of 'TS'",
+ height: 128,
+ src: "./docs/create-typescript-app.png",
+ type: "png",
+ width: 128,
+ },
+ node: {
+ minimum: expect.any(String),
+ pinned: expect.any(String),
+ },
+ owner: "JoshuaKGoldberg",
+ packageData: expect.any(Object),
+ repository: "create-typescript-app",
+ title: "Create TypeScript App",
+ usage: expect.any(String),
+ version: expect.any(String),
+ });
+ });
+});
diff --git a/src/next/base.ts b/src/base.ts
similarity index 88%
rename from src/next/base.ts
rename to src/base.ts
index 80797ab13..f94d5ae43 100644
--- a/src/next/base.ts
+++ b/src/base.ts
@@ -9,18 +9,18 @@ import lazyValue from "lazy-value";
import npmUser from "npm-user";
import { z } from "zod";
-import { parsePackageAuthor } from "../shared/options/createOptionDefaults/parsePackageAuthor.js";
-import { readDefaultsFromReadme } from "../shared/options/createOptionDefaults/readDefaultsFromReadme.js";
-import { readEmails } from "../shared/options/createOptionDefaults/readEmails.js";
-import { readFunding } from "../shared/options/createOptionDefaults/readFunding.js";
-import { readGuide } from "../shared/options/createOptionDefaults/readGuide.js";
-import { readPackageData } from "../shared/packages.js";
-import { readFileSafe } from "../shared/readFileSafe.js";
-import { tryCatchLazyValueAsync } from "../shared/tryCatchLazyValueAsync.js";
-import { AllContributorsData } from "../shared/types.js";
-import { readDescription } from "./readDescription.js";
-import { readDocumentation } from "./readDocumentation.js";
+import { parsePackageAuthor } from "./options/parsePackageAuthor.js";
+import { readDefaultsFromReadme } from "./options/readDefaultsFromReadme.js";
+import { readDescription } from "./options/readDescription.js";
+import { readDocumentation } from "./options/readDocumentation.js";
+import { readEmails } from "./options/readEmails.js";
+import { readFileSafe } from "./options/readFileSafe.js";
+import { readFunding } from "./options/readFunding.js";
+import { readGuide } from "./options/readGuide.js";
+import { readPackageData } from "./options/readPackageData.js";
+import { AllContributorsData } from "./types.js";
import { swallowError } from "./utils/swallowError.js";
+import { tryCatchLazyValueAsync } from "./utils/tryCatchLazyValueAsync.js";
export const base = createBase({
options: {
diff --git a/src/bin/help.test.ts b/src/bin/help.test.ts
deleted file mode 100644
index 42f0965f2..000000000
--- a/src/bin/help.test.ts
+++ /dev/null
@@ -1,354 +0,0 @@
-import chalk from "chalk";
-import { beforeEach, describe, expect, it, MockInstance, vi } from "vitest";
-
-import { logHelpText } from "./help.js";
-
-function makeProxy(receiver: T): T {
- return new Proxy(receiver, {
- get: () => makeProxy((input: string) => input),
- });
-}
-
-vi.mock("chalk", () => ({
- default: makeProxy({}),
-}));
-
-let mockConsoleLog: MockInstance;
-
-describe("logHelpText", () => {
- beforeEach(() => {
- mockConsoleLog = vi
- .spyOn(console, "log")
- .mockImplementation(() => undefined);
- });
-
- it("logs help text when called", () => {
- logHelpText([
- chalk.yellow(
- "⚠️ This template is early stage, opinionated, and not endorsed by the TypeScript team. ⚠️",
- ),
- chalk.yellow(
- "⚠️ If any tooling it sets displeases you, you can always remove that portion manually. ⚠️",
- ),
- ]);
-
- expect(mockConsoleLog.mock.calls).toMatchInlineSnapshot(`
- [
- [
- "⚠️ This template is early stage, opinionated, and not endorsed by the TypeScript team. ⚠️",
- ],
- [
- " ",
- ],
- [
- "⚠️ If any tooling it sets displeases you, you can always remove that portion manually. ⚠️",
- ],
- [
- " ",
- ],
- [
- "
- A quickstart-friendly TypeScript template with comprehensive formatting,
- linting, releases, testing, and other great tooling built-in.
- ",
- ],
- [
- " ",
- ],
- [
- "Core options:",
- ],
- [
- "
- --base (string): Whether to scaffold the repository with:
- • everything: that comes with the template (recommended)
- • common: additions to the minimal starters such as releases and tests
- • minimal: amounts of tooling, essentially opting out of everything
- • prompt: for which portions to exclude",
- ],
- [
- "
- --create-repository: Whether to create a corresponding repository on github.com
- (if it doesn't yet exist)",
- ],
- [
- "
- --description (string): Sentence case description of the repository
- (e.g. A quickstart-friendly TypeScript package with lots of great
- repository tooling. ✨)",
- ],
- [
- "
- --mode (string): Whether to:
- • create: a new repository in a child directory
- • initialize: a freshly repository in the current directory
- • migrate: an existing repository in the current directory",
- ],
- [
- "
- --owner (string): GitHub organization or user the repository is underneath
- (e.g. JoshuaKGoldberg)",
- ],
- [
- "
- --repository (string): The kebab-case name of the repository
- (e.g. create-typescript-app)",
- ],
- [
- "
- --title (string): Title Case title for the repository to be used in
- documentation (e.g. Create TypeScript App)",
- ],
- [],
- [
- " ",
- ],
- [
- "Optional options:",
- ],
- [
- "
- --access (string): ("public" | "restricted"): Which npm publish --access to
- release npm packages with (by default, "public")",
- ],
- [
- "
- --author (string): Username on npm to publish packages under (by
- default, an existing npm author, or the currently logged in npm user, or
- owner.toLowerCase())",
- ],
- [
- "
- --auto: Whether to infer all options from files on disk.",
- ],
- [
- "
- --bin (string): package.json bin value to include for npx-style running.",
- ],
- [
- "
- --directory (string): Directory to create the repository in (by default, the same
- name as the repository)",
- ],
- [
- "
- --email (string): Email address to be listed as the point of contact in docs
- and packages (e.g. example@joshuakgoldberg.com)",
- ],
- [
- "
- --email-github (string): Optionally, may be provided to use different emails in .md
- files",
- ],
- [
- "
- --email-npm (string): Optionally, may be provided to use different emails in
- package.json",
- ],
- [
- "
- --funding (string): GitHub organization or username to mention in funding.yml
- (by default, owner)",
- ],
- [
- "
- --guide (string): Link to a contribution guide to place at the top of the
- development docs",
- ],
- [
- "
- --guide-title (string): If --guide is provided or detected from an existing
- DEVELOPMENT.md, the text title to place in the guide link",
- ],
- [
- "
- --keywords (string): Any number of keywords to include in package.json (by default,
- none). This can be specified any number of times, like
- --keywords apple --keywords "banana cherry"",
- ],
- [
- "
- --logo (string): Local image file in the repository to display near the top of
- the README.md as a logo",
- ],
- [
- "
- --logo-alt (string): If --logo is provided or detected from an existing README.md,
- alt text that describes the image will be prompted for if not provided",
- ],
- [
- "
- --logo-height (string): If --logo is provided or detected from an existing README.md,
- an explicit height style",
- ],
- [
- "
- --logo-width (string): If --logo is provided or detected from an existing README.md,
- an explicit width style",
- ],
- [
- "
- --preserve-generated-from: Whether to keep the GitHub repository generated from
- notice (by default, false)",
- ],
- [],
- [
- " ",
- ],
- [
- "Opt-outs:",
- ],
- [
- "
- ⚠️ Warning: Specifying any --exclude-* flag on the command-line will
- cause the setup script to skip prompting for more excludes. ⚠️",
- ],
- [
- "
- --exclude-all-contributors: Don't add all-contributors to track contributions
- and display them in a README.md table.",
- ],
- [
- "
- --exclude-compliance: Don't add a GitHub Actions workflow to verify that PRs match
- an expected format.",
- ],
- [
- "
- --exclude-lint-jsdoc: Don't use eslint-plugin-jsdoc to enforce good practices around
- JSDoc comments.",
- ],
- [
- "
- --exclude-lint-json: Don't apply linting and sorting to *.json, and
- *.jsonc files.",
- ],
- [
- "
- --exclude-lint-knip: Don't add Knip to detect unused files, dependencies, and code
- exports.",
- ],
- [
- "
- --exclude-lint-md: Don't apply linting to *.md files.",
- ],
- [
- "
- --exclude-lint-package-json: Don't add eslint-plugin-package-json to lint for
- package.json correctness.",
- ],
- [
- "
- --exclude-lint-packages: Don't add a pnpm dedupe workflow to ensure packages
- aren't duplicated unnecessarily.",
- ],
- [
- "
- --exclude-lint-perfectionist: Don't apply eslint-plugin-perfectionist to ensure
- imports, keys, and so on are in sorted order.",
- ],
- [
- "
- --exclude-lint-regexp: Don't add eslint-plugin-regexp to enforce good practices around
- regular expressions.",
- ],
- [
- "
- --exclude-lint-spelling: Don't add cspell to spell check against dictionaries
- of known words.",
- ],
- [
- "
- --exclude-lint-strict: Don't augment the recommended logical lint rules with
- typescript-eslint's strict config.",
- ],
- [
- "
- --exclude-lint-stylistic: Don't add stylistic rules such as typescript-eslint's
- stylistic config.",
- ],
- [
- "
- --exclude-lint-yml: Don't apply linting and sorting to *.yaml and *.yml files.",
- ],
- [
- "
- --exclude-releases: Don't add release-it to generate changelogs, package bumps,
- and publishes based on conventional commits.",
- ],
- [
- "
- --exclude-renovate: Don't add a Renovate config to dependencies up-to-date with
- PRs.",
- ],
- [
- "
- --exclude-templated-by: Don't add a _"This package was templated with create-typescript-app"_ notice at the end of the README.md.",
- ],
- [
- "
- --exclude-tests: Don't add Vitest tooling for fast unit tests, configured
- with coverage tracking.",
- ],
- [
- "
- You can prevent the migration script from making some network-based
- changes using any or all of the following CLI flags:",
- ],
- [
- "
- --exclude-contributors: Skips network calls that fetch all-contributors
- data from GitHub",
- ],
- [
- "
- --skip-all-contributors-api: Skips network calls that fetch all-contributors data from
- GitHub. This flag does nothing if --exclude-all-contributors was specified.",
- ],
- [
- "
- --skip-github-api: Skips calling to GitHub APIs.",
- ],
- [
- "
- --skip-install: Skips installing all the new template packages with pnpm.",
- ],
- [
- "
- You can prevent the migration script from making some changes on disk
- using any or all of the following CLI flags:",
- ],
- [
- "
- --skip-removal: Skips removing setup docs and scripts, including this docs/
- directory",
- ],
- [
- "
- --skip-restore: Skips the prompt offering to restore the repository if an
- error occurs during setup",
- ],
- [
- "
- --skip-uninstall: Skips uninstalling packages only used for setup scripts",
- ],
- [],
- [
- " ",
- ],
- [
- "Offline Mode:",
- ],
- [
- "
- --offline: You can run create-typescript-app in an "offline" mode.
- Doing so will:
- • Enable --exclude-all-contributors-api and --skip-github-api
- • Skip network calls when setting up contributors
- • Run pnpm commands with pnpm's --offline mode",
- ],
- [],
- ]
- `);
- });
-});
diff --git a/src/bin/help.ts b/src/bin/help.ts
deleted file mode 100644
index a36b8737a..000000000
--- a/src/bin/help.ts
+++ /dev/null
@@ -1,148 +0,0 @@
-import chalk from "chalk";
-
-import { allArgOptions } from "../shared/options/args.js";
-
-interface HelpTextSection {
- sectionHeading: string;
- subsections: {
- flags: SubsectionFlag[];
- subheading?: string;
- warning?: string;
- }[];
-}
-
-interface SubsectionFlag {
- description: string;
- flag: string;
- type: string;
-}
-
-export function logHelpText(introLogs: string[]): void {
- const helpTextSections = createHelpTextSections();
-
- for (const log of introLogs) {
- console.log(log);
- console.log(" ");
- }
-
- console.log(
- chalk.cyan(
- `
-A quickstart-friendly TypeScript template with comprehensive formatting,
-linting, releases, testing, and other great tooling built-in.
- `,
- ),
- );
-
- for (const section of helpTextSections) {
- logHelpTextSection(section);
-
- console.log();
- }
-}
-
-function createHelpTextSections(): HelpTextSection[] {
- const core: HelpTextSection = {
- sectionHeading: "Core options:",
- subsections: [
- {
- flags: [],
- },
- ],
- };
-
- const optional: HelpTextSection = {
- sectionHeading: "Optional options:",
- subsections: [
- {
- flags: [],
- },
- ],
- };
-
- const optOut: HelpTextSection = {
- sectionHeading: "Opt-outs:",
- subsections: [
- {
- flags: [],
- warning: `
- ⚠️ Warning: Specifying any --exclude-* flag on the command-line will
- cause the setup script to skip prompting for more excludes. ⚠️`,
- },
- {
- flags: [
- {
- description: `Skips network calls that fetch all-contributors
- data from GitHub`,
- flag: "exclude-contributors",
- type: "boolean",
- },
- ],
- subheading: `
-You can prevent the migration script from making some network-based
-changes using any or all of the following CLI flags:`,
- },
- {
- flags: [],
- subheading: `
-You can prevent the migration script from making some changes on disk
-using any or all of the following CLI flags:`,
- },
- ],
- };
-
- const offline: HelpTextSection = {
- sectionHeading: "Offline Mode:",
- subsections: [
- {
- flags: [],
- },
- ],
- };
-
- const subsections = {
- core: core.subsections[0],
- offline: offline.subsections[0],
- "opt-out": optOut.subsections[0],
- optional: optional.subsections[0],
- "skip-disk": optOut.subsections[2],
- "skip-net": optOut.subsections[1],
- };
-
- for (const [option, data] of Object.entries(allArgOptions)) {
- subsections[data.docsSection].flags.push({
- description: data.description,
- flag: option,
- type: data.type,
- });
- }
-
- return [core, optional, optOut, offline];
-}
-
-function logHelpTextSection(section: HelpTextSection): void {
- console.log(" ");
-
- console.log(chalk.black.bgGreenBright(section.sectionHeading));
-
- for (const subsection of section.subsections) {
- if (subsection.warning) {
- console.log(chalk.yellow(subsection.warning));
- }
-
- if (subsection.subheading) {
- console.log(chalk.green(subsection.subheading));
- }
-
- for (const { description, flag, type } of subsection.flags) {
- console.log(
- chalk.cyan(
- `
- --${flag}${
- type !== "boolean" ? ` (${chalk.cyanBright(type)})` : ""
- }: ${description}`,
- ),
- );
- }
- }
-}
diff --git a/src/bin/index.test.ts b/src/bin/index.test.ts
deleted file mode 100644
index 8ed7fa0a9..000000000
--- a/src/bin/index.test.ts
+++ /dev/null
@@ -1,205 +0,0 @@
-import chalk from "chalk";
-import { beforeEach, describe, expect, it, vi } from "vitest";
-import z from "zod";
-
-import { bin } from "./index.js";
-import { getVersionFromPackageJson } from "./packageJson.js";
-
-const mockCancel = vi.fn();
-const mockOutro = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- get cancel() {
- return mockCancel;
- },
- intro: vi.fn(),
- log: {
- info: vi.fn(),
- },
- get outro() {
- return mockOutro;
- },
-}));
-
-const mockLogLine = vi.fn();
-
-vi.mock("../shared/cli/lines.js", () => ({
- get logLine() {
- return mockLogLine;
- },
-}));
-
-const mockCreate = vi.fn();
-
-vi.mock("../create/index.js", () => ({
- get create() {
- return mockCreate;
- },
-}));
-
-const mockInitialize = vi.fn();
-
-vi.mock("../initialize/index.js", () => ({
- get initialize() {
- return mockInitialize;
- },
-}));
-
-const mockMigrate = vi.fn();
-
-vi.mock("../migrate/index.js", () => ({
- get migrate() {
- return mockMigrate;
- },
-}));
-
-const mockPromptForMode = vi.fn();
-
-vi.mock("./promptForMode.js", () => ({
- get promptForMode() {
- return mockPromptForMode;
- },
-}));
-
-describe("bin", () => {
- beforeEach(() => {
- vi.spyOn(console, "clear").mockImplementation(() => undefined);
- vi.spyOn(console, "log").mockImplementation(() => undefined);
- });
-
- it("returns 1 when promptForMode returns an undefined mode", async () => {
- mockPromptForMode.mockResolvedValue({});
-
- const result = await bin([]);
-
- expect(mockOutro).toHaveBeenCalledWith(
- chalk.red("Operation cancelled. Exiting - maybe another time? 👋"),
- );
- expect(result).toBe(1);
- });
-
- it("returns 1 when promptForMode returns an error mode", async () => {
- const error = new Error("Oh no!");
- mockPromptForMode.mockResolvedValue({ mode: error });
-
- const result = await bin([]);
-
- expect(mockOutro).toHaveBeenCalledWith(chalk.red(error.message));
- expect(result).toBe(1);
- });
-
- it("returns the success result of the corresponding runner without cancel logging when promptForMode returns a mode that succeeds", async () => {
- const mode = "create";
- const args = ["--owner", "abc123"];
- const code = 0;
- const promptedOptions = { directory: "." };
-
- mockPromptForMode.mockResolvedValue({ mode, options: promptedOptions });
- mockCreate.mockResolvedValue({ code, options: {} });
-
- const result = await bin(args);
-
- expect(mockCreate).toHaveBeenCalledWith(args, promptedOptions);
- expect(mockCancel).not.toHaveBeenCalled();
- expect(mockInitialize).not.toHaveBeenCalled();
- expect(mockMigrate).not.toHaveBeenCalled();
- expect(result).toEqual(code);
- });
-
- it("returns the cancel result of the corresponding runner and cancel logs when promptForMode returns a mode that cancels", async () => {
- const mode = "create";
- const args = ["--owner", "abc123"];
- const code = 2;
- const promptedOptions = { directory: "." };
-
- mockPromptForMode.mockResolvedValue({ mode, options: promptedOptions });
- mockCreate.mockResolvedValue({ code, options: {} });
-
- const result = await bin(args);
-
- expect(mockCreate).toHaveBeenCalledWith(args, promptedOptions);
- expect(mockCancel).toHaveBeenCalledWith(
- `Operation cancelled. Exiting - maybe another time? 👋`,
- );
- expect(result).toEqual(code);
- });
-
- it("returns the cancel result containing a zod error of the corresponding runner and output plus cancel logs when promptForMode returns a mode that cancels with a string error", async () => {
- const mode = "initialize";
- const args = ["--email", "abc123"];
- const code = 2;
- const error = "Oh no!";
-
- mockPromptForMode.mockResolvedValue({ mode });
- mockInitialize.mockResolvedValue({
- code: 2,
- error,
- options: {},
- });
-
- const result = await bin(args);
-
- expect(mockInitialize).toHaveBeenCalledWith(args, undefined);
- expect(mockLogLine).toHaveBeenCalledWith(chalk.red(error));
- expect(mockCancel).toHaveBeenCalledWith(
- `Operation cancelled. Exiting - maybe another time? 👋`,
- );
- expect(result).toEqual(code);
- });
-
- it("returns the cancel result containing a zod error of the corresponding runner and output plus cancel logs when promptForMode returns a mode that cancels with a zod error", async () => {
- const mode = "initialize";
- const args = ["--email", "abc123"];
- const code = 2;
-
- const validationResult = z
- .object({ email: z.string().email() })
- .safeParse({ email: "abc123" });
-
- mockPromptForMode.mockResolvedValue({ mode });
- mockInitialize.mockResolvedValue({
- code: 2,
- error: (validationResult as z.SafeParseError<{ email: string }>).error,
- options: {},
- });
-
- const result = await bin(args);
-
- expect(mockInitialize).toHaveBeenCalledWith(args, undefined);
- expect(mockLogLine).toHaveBeenCalledWith(
- chalk.red('Validation error: Invalid email at "email"'),
- );
- expect(mockCancel).toHaveBeenCalledWith(
- `Operation cancelled. Exiting - maybe another time? 👋`,
- );
- expect(result).toEqual(code);
- });
-
- it("returns the cancel result of the corresponding runner and cancel logs when promptForMode returns a mode that fails", async () => {
- const mode = "create";
- const args = ["--owner", "abc123"];
- const code = 1;
- const promptedOptions = { directory: "." };
-
- mockPromptForMode.mockResolvedValue({ mode, options: promptedOptions });
- mockCreate.mockResolvedValue({ code, options: {} });
-
- const result = await bin(args);
-
- expect(mockCreate).toHaveBeenCalledWith(args, promptedOptions);
- expect(mockCancel).toHaveBeenCalledWith(
- `Operation failed. Exiting - maybe another time? 👋`,
- );
- expect(result).toEqual(code);
- });
-
- it("prints the version when the --version flag is passed", async () => {
- const args = ["--version"];
- const version = await getVersionFromPackageJson();
-
- const result = await bin(args);
-
- expect(console.log).toHaveBeenCalledWith(version);
- expect(result).toBe(0);
- });
-});
diff --git a/src/bin/index.ts b/src/bin/index.ts
deleted file mode 100644
index 923727759..000000000
--- a/src/bin/index.ts
+++ /dev/null
@@ -1,108 +0,0 @@
-import * as prompts from "@clack/prompts";
-import chalk from "chalk";
-import { parseArgs } from "node:util";
-import { fromZodError } from "zod-validation-error";
-
-import { createRerunSuggestion } from "../create/createRerunSuggestion.js";
-import { create } from "../create/index.js";
-import { initialize } from "../initialize/index.js";
-import { migrate } from "../migrate/index.js";
-import { logLine } from "../shared/cli/lines.js";
-import { StatusCodes } from "../shared/codes.js";
-import { logHelpText } from "./help.js";
-import { getVersionFromPackageJson } from "./packageJson.js";
-import { promptForMode } from "./promptForMode.js";
-
-const operationMessage = (verb: string) =>
- `Operation ${verb}. Exiting - maybe another time? 👋`;
-
-export async function bin(args: string[]) {
- console.clear();
-
- const version = await getVersionFromPackageJson();
-
- const introPrompts = [
- chalk.greenBright(`✨ Welcome to`),
- chalk.bgGreenBright.black(`create-typescript-app`),
- chalk.greenBright(`${version}! ✨`),
- ].join(" ");
-
- const introWarnings = [
- chalk.yellow(
- "⚠️ This template is early stage, opinionated, and not endorsed by the TypeScript team. ⚠️",
- ),
- chalk.yellow(
- "⚠️ If any tooling it sets displeases you, you can always remove that portion manually. ⚠️",
- ),
- ];
-
- const { values } = parseArgs({
- args,
- options: {
- help: {
- short: "h",
- type: "boolean",
- },
- mode: { type: "string" },
- version: {
- short: "v",
- type: "boolean",
- },
- },
- strict: false,
- });
-
- if (values.help) {
- logHelpText([introPrompts, ...introWarnings]);
- return 0;
- }
-
- if (values.version) {
- console.log(version);
- return 0;
- }
-
- prompts.intro(introPrompts);
-
- logLine();
- logLine(introWarnings[0]);
- logLine(introWarnings[1]);
-
- const { mode, options: promptedOptions } = await promptForMode(
- !!values.auto,
- values.mode,
- );
- if (typeof mode !== "string") {
- prompts.outro(chalk.red(mode?.message ?? operationMessage("cancelled")));
- return 1;
- }
-
- const runners = { create, initialize, migrate };
- const { code, error, options } = await runners[mode](args, promptedOptions);
-
- prompts.log.info(
- [
- chalk.italic(`Tip: to run again with the same input values, use:`),
- chalk.blue(createRerunSuggestion(options)),
- ].join(" "),
- );
-
- if (code) {
- logLine();
-
- if (error) {
- logLine(
- chalk.red(typeof error === "string" ? error : fromZodError(error)),
- );
- logLine();
- }
-
- prompts.cancel(
- code === StatusCodes.Cancelled
- ? operationMessage("cancelled")
- : operationMessage("failed"),
- );
- }
-
- return code;
-}
diff --git a/src/bin/packageJson.test.ts b/src/bin/packageJson.test.ts
deleted file mode 100644
index 7e1be8dee..000000000
--- a/src/bin/packageJson.test.ts
+++ /dev/null
@@ -1,31 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { getVersionFromPackageJson } from "./packageJson.js";
-
-const mockReadFileSafeAsJson = vi.fn();
-
-vi.mock("../shared/readFileSafeAsJson.js", () => ({
- get readFileSafeAsJson() {
- return mockReadFileSafeAsJson;
- },
-}));
-
-describe("getVersionFromPackageJson", () => {
- it("returns the current version number from the package.json", async () => {
- mockReadFileSafeAsJson.mockResolvedValue({
- version: "1.40.0",
- });
-
- const version = await getVersionFromPackageJson();
-
- expect(version).toBe("1.40.0");
- });
-
- it("throws an error when there is no version number", async () => {
- mockReadFileSafeAsJson.mockResolvedValue({});
-
- await expect(() => getVersionFromPackageJson()).rejects.toEqual(
- new Error("Cannot find version number"),
- );
- });
-});
diff --git a/src/bin/packageJson.ts b/src/bin/packageJson.ts
deleted file mode 100644
index 18d7004e2..000000000
--- a/src/bin/packageJson.ts
+++ /dev/null
@@ -1,16 +0,0 @@
-import { readFileSafeAsJson } from "../shared/readFileSafeAsJson.js";
-
-interface PackageWithVersion {
- version?: string;
-}
-
-export async function getVersionFromPackageJson(): Promise {
- const path = new URL("../../package.json", import.meta.url);
- const data = (await readFileSafeAsJson(path)) as PackageWithVersion;
-
- if (typeof data === "object" && typeof data.version === "string") {
- return data.version;
- }
-
- throw new Error("Cannot find version number");
-}
diff --git a/src/bin/promptForMode.test.ts b/src/bin/promptForMode.test.ts
deleted file mode 100644
index f508c6d14..000000000
--- a/src/bin/promptForMode.test.ts
+++ /dev/null
@@ -1,129 +0,0 @@
-import chalk from "chalk";
-import { describe, expect, it, vi } from "vitest";
-
-import { promptForMode } from "./promptForMode.js";
-
-const mockSelect = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- isCancel: () => false,
- get select() {
- return mockSelect;
- },
-}));
-
-const mockReaddir = vi.fn();
-
-vi.mock("node:fs/promises", () => ({
- get readdir() {
- return mockReaddir;
- },
-}));
-
-const mockCwd = vi.fn();
-
-vi.mock("node:process", () => ({
- get cwd() {
- return mockCwd;
- },
-}));
-
-const mockLogLine = vi.fn();
-
-vi.mock("../shared/cli/lines.js", () => ({
- get logLine() {
- return mockLogLine;
- },
-}));
-describe("promptForMode", () => {
- it("returns an error when auto exists and input is not migrate", async () => {
- const mode = await promptForMode(true, "create");
-
- expect(mode).toMatchInlineSnapshot(
- `
- {
- "mode": [Error: --auto can only be used with --mode migrate.],
- }
- `,
- );
- });
-
- it("returns an error when the input exists and is not a mode", async () => {
- const mode = await promptForMode(false, "other");
-
- expect(mode).toMatchInlineSnapshot(
- `
- {
- "mode": [Error: Unknown --mode: other. Allowed modes are: create, initialize, migrate.],
- }
- `,
- );
- });
-
- it("returns the input when it is a mode", async () => {
- const input = "create";
-
- const mode = await promptForMode(false, input);
-
- expect(mode).toEqual({ mode: input });
- });
-
- it("returns creating in the current directory when the current directory is empty and the user selects create-current", async () => {
- mockSelect.mockResolvedValueOnce("create-current");
- const directory = "test-directory";
-
- mockReaddir.mockResolvedValueOnce([]);
- mockCwd.mockReturnValueOnce(`/path/to/${directory}`);
-
- const actual = await promptForMode(false, undefined);
-
- expect(actual).toEqual({
- mode: "create",
- options: { directory: ".", repository: directory },
- });
- expect(mockLogLine).not.toHaveBeenCalled();
- });
-
- it("returns creating in a child directory when the current directory is empty and the user selects create-child", async () => {
- mockSelect.mockResolvedValueOnce("create-child");
- const directory = "test-directory";
-
- mockReaddir.mockResolvedValueOnce([]);
- mockCwd.mockReturnValueOnce(`/path/to/${directory}`);
-
- const actual = await promptForMode(false, undefined);
-
- expect(actual).toEqual({
- mode: "create",
- });
- expect(mockLogLine).not.toHaveBeenCalled();
- });
-
- it("returns the user selection when the current directory is a Git directory", async () => {
- const mode = "initialize";
- mockSelect.mockResolvedValueOnce(mode);
-
- mockReaddir.mockResolvedValueOnce([".git"]);
-
- const actual = await promptForMode(false, undefined);
-
- expect(actual).toEqual({ mode });
- expect(mockLogLine).not.toHaveBeenCalled();
- });
-
- it("returns create without prompting when the current directory contains children but is not a Git directory", async () => {
- const mode = "create";
-
- mockReaddir.mockResolvedValueOnce(["file"]);
-
- const actual = await promptForMode(false, undefined);
-
- expect(actual).toEqual({ mode });
- expect(mockSelect).not.toHaveBeenCalled();
- expect(mockLogLine).toHaveBeenCalledWith(
- chalk.gray(
- "Defaulting to --mode create because the directory contains children and isn't a Git repository.",
- ),
- );
- });
-});
diff --git a/src/bin/promptForMode.ts b/src/bin/promptForMode.ts
deleted file mode 100644
index 8bb6ab43c..000000000
--- a/src/bin/promptForMode.ts
+++ /dev/null
@@ -1,127 +0,0 @@
-import * as prompts from "@clack/prompts";
-import chalk from "chalk";
-import * as fs from "fs/promises";
-import path from "node:path";
-import * as process from "node:process";
-
-import { logLine } from "../shared/cli/lines.js";
-import { isUsingCreateEngine } from "../shared/isUsingCreateEngine.js";
-import { filterPromptCancel } from "../shared/prompts.js";
-import { Mode, PromptedOptions } from "../shared/types.js";
-
-const allowedModes = ["create", "initialize", "migrate"] satisfies Mode[];
-
-export interface PromptedMode {
- mode: Error | Mode | undefined;
- options?: PromptedOptions;
-}
-
-export async function promptForMode(
- auto: boolean,
- input: boolean | string | undefined,
-): Promise {
- if (auto && input !== "migrate") {
- return {
- mode: new Error("--auto can only be used with --mode migrate."),
- };
- }
-
- if (input) {
- if (!isMode(input)) {
- return {
- mode: new Error(
- `Unknown --mode: ${input}. Allowed modes are: ${allowedModes.join(
- ", ",
- )}.`,
- ),
- };
- }
-
- return { mode: input };
- }
-
- const dir = await fs.readdir(".");
-
- if (dir.length === 0 && !isUsingCreateEngine()) {
- const mode = filterPromptCancel(
- (await prompts.select({
- message: chalk.blue("How would you like to use the template?"),
- options: [
- {
- label: label(
- "create",
- "a new repository in the current empty directory",
- ),
- value: "create-current",
- },
- {
- label: label("create", "a new repository in a new child directory"),
- value: "create-child",
- },
- ],
- })) as string,
- );
-
- const directory = path.basename(process.cwd());
-
- return {
- mode: "create",
- ...(mode === "create-current" && {
- options: {
- directory: ".",
- repository: directory,
- },
- }),
- };
- }
-
- if (dir.includes(".git")) {
- return {
- mode: filterPromptCancel(
- (await prompts.select({
- initialValue: "migrate" as Mode,
- message: chalk.blue("How would you like to use the template?"),
- options: [
- {
- label: label("create", "a new repository in a child directory"),
- value: "create",
- },
- {
- label: label(
- "initialize",
- "a freshly cloned repository in the current directory",
- ),
- value: "initialize",
- },
- {
- label: label(
- "migrate",
- "the existing repository in the current directory",
- ),
- value: "migrate",
- },
- ],
- })) as Mode | symbol,
- ),
- };
- }
-
- logLine();
- logLine(
- chalk.gray(
- "Defaulting to --mode create because the directory contains children and isn't a Git repository.",
- ),
- );
-
- return {
- mode: "create",
- };
-}
-
-function isMode(input: boolean | string): input is Mode {
- return allowedModes.includes(input as Mode);
-}
-
-function label(base: string, text: string) {
- return `${chalk.bold(base)} ${text}`;
-}
diff --git a/src/next/blocks/blockAllContributors.test.ts b/src/blocks/blockAllContributors.test.ts
similarity index 100%
rename from src/next/blocks/blockAllContributors.test.ts
rename to src/blocks/blockAllContributors.test.ts
diff --git a/src/next/blocks/blockAllContributors.ts b/src/blocks/blockAllContributors.ts
similarity index 94%
rename from src/next/blocks/blockAllContributors.ts
rename to src/blocks/blockAllContributors.ts
index 1ecd3f85d..0c91d3b0b 100644
--- a/src/next/blocks/blockAllContributors.ts
+++ b/src/blocks/blockAllContributors.ts
@@ -1,6 +1,6 @@
-import { createSoloWorkflowFile } from "../../steps/writing/creation/dotGitHub/createSoloWorkflowFile.js";
import { base } from "../base.js";
import { blockRepositorySecrets } from "./blockRepositorySecrets.js";
+import { createSoloWorkflowFile } from "./files/createSoloWorkflowFile.js";
import { CommandPhase } from "./phases.js";
export const blockAllContributors = base.createBlock({
diff --git a/src/next/blocks/blockAreTheTypesWrong.test.ts b/src/blocks/blockAreTheTypesWrong.test.ts
similarity index 100%
rename from src/next/blocks/blockAreTheTypesWrong.test.ts
rename to src/blocks/blockAreTheTypesWrong.test.ts
diff --git a/src/next/blocks/blockAreTheTypesWrong.ts b/src/blocks/blockAreTheTypesWrong.ts
similarity index 93%
rename from src/next/blocks/blockAreTheTypesWrong.ts
rename to src/blocks/blockAreTheTypesWrong.ts
index 79b32105d..dfe35c49e 100644
--- a/src/next/blocks/blockAreTheTypesWrong.ts
+++ b/src/blocks/blockAreTheTypesWrong.ts
@@ -3,7 +3,7 @@ import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
export const blockAreTheTypesWrong = base.createBlock({
about: {
- name: "README.md",
+ name: "Are The Types Wrong",
},
produce() {
return {
diff --git a/src/next/blocks/blockCSpell.test.ts b/src/blocks/blockCSpell.test.ts
similarity index 100%
rename from src/next/blocks/blockCSpell.test.ts
rename to src/blocks/blockCSpell.test.ts
diff --git a/src/next/blocks/blockCSpell.ts b/src/blocks/blockCSpell.ts
similarity index 97%
rename from src/next/blocks/blockCSpell.ts
rename to src/blocks/blockCSpell.ts
index 130991124..5a3cebb14 100644
--- a/src/next/blocks/blockCSpell.ts
+++ b/src/blocks/blockCSpell.ts
@@ -2,12 +2,12 @@ import { getObjectStringsDeep } from "object-strings-deep";
import { z } from "zod";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { resolveBin } from "../utils/resolveBin.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
import { blockPackageJson } from "./blockPackageJson.js";
import { blockVSCode } from "./blockVSCode.js";
-import { getPackageDependencies } from "./packageData.js";
import { CommandPhase } from "./phases.js";
const filesGlob = `"**" ".github/**/*"`;
diff --git a/src/next/blocks/blockContributingDocs.test.ts b/src/blocks/blockContributingDocs.test.ts
similarity index 100%
rename from src/next/blocks/blockContributingDocs.test.ts
rename to src/blocks/blockContributingDocs.test.ts
diff --git a/src/next/blocks/blockContributingDocs.ts b/src/blocks/blockContributingDocs.ts
similarity index 100%
rename from src/next/blocks/blockContributingDocs.ts
rename to src/blocks/blockContributingDocs.ts
diff --git a/src/next/blocks/blockContributorCovenant.test.ts b/src/blocks/blockContributorCovenant.test.ts
similarity index 100%
rename from src/next/blocks/blockContributorCovenant.test.ts
rename to src/blocks/blockContributorCovenant.test.ts
diff --git a/src/next/blocks/blockContributorCovenant.ts b/src/blocks/blockContributorCovenant.ts
similarity index 100%
rename from src/next/blocks/blockContributorCovenant.ts
rename to src/blocks/blockContributorCovenant.ts
diff --git a/src/next/blocks/blockDevelopmentDocs.test.ts b/src/blocks/blockDevelopmentDocs.test.ts
similarity index 100%
rename from src/next/blocks/blockDevelopmentDocs.test.ts
rename to src/blocks/blockDevelopmentDocs.test.ts
diff --git a/src/next/blocks/blockDevelopmentDocs.ts b/src/blocks/blockDevelopmentDocs.ts
similarity index 100%
rename from src/next/blocks/blockDevelopmentDocs.ts
rename to src/blocks/blockDevelopmentDocs.ts
diff --git a/src/next/blocks/blockESLint.test.ts b/src/blocks/blockESLint.test.ts
similarity index 98%
rename from src/next/blocks/blockESLint.test.ts
rename to src/blocks/blockESLint.test.ts
index 9fce1268e..4c24fa1b5 100644
--- a/src/next/blocks/blockESLint.test.ts
+++ b/src/blocks/blockESLint.test.ts
@@ -71,7 +71,7 @@ describe("blockESLint", () => {
"@eslint/js": "9.17.0",
"@types/node": "22.10.2",
"eslint": "9.17.0",
- "typescript-eslint": "8.18.1",
+ "typescript-eslint": "8.19.0",
},
"scripts": {
"lint": "eslint . --max-warnings 0",
@@ -200,7 +200,7 @@ describe("blockESLint", () => {
"@eslint/js": "9.17.0",
"@types/node": "22.10.2",
"eslint": "9.17.0",
- "typescript-eslint": "8.18.1",
+ "typescript-eslint": "8.19.0",
},
"scripts": {
"lint": "eslint . --max-warnings 0",
@@ -366,7 +366,7 @@ describe("blockESLint", () => {
"eslint": "9.17.0",
"eslint-plugin-markdown": "5.1.0",
"eslint-plugin-regexp": "2.7.0",
- "typescript-eslint": "8.18.1",
+ "typescript-eslint": "8.19.0",
},
"scripts": {
"lint": "eslint . --max-warnings 0",
diff --git a/src/next/blocks/blockESLint.ts b/src/blocks/blockESLint.ts
similarity index 99%
rename from src/next/blocks/blockESLint.ts
rename to src/blocks/blockESLint.ts
index cbbf38ffd..c4280bea6 100644
--- a/src/next/blocks/blockESLint.ts
+++ b/src/blocks/blockESLint.ts
@@ -3,12 +3,12 @@ import { parse as parsePackageName } from "parse-package-name";
import { z } from "zod";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
import { blockPackageJson } from "./blockPackageJson.js";
import { blockVSCode } from "./blockVSCode.js";
-import { getPackageDependencies } from "./packageData.js";
import { CommandPhase } from "./phases.js";
const zRuleOptions = z.union([
diff --git a/src/next/blocks/blockESLintComments.ts b/src/blocks/blockESLintComments.ts
similarity index 100%
rename from src/next/blocks/blockESLintComments.ts
rename to src/blocks/blockESLintComments.ts
diff --git a/src/next/blocks/blockESLintJSDoc.ts b/src/blocks/blockESLintJSDoc.ts
similarity index 100%
rename from src/next/blocks/blockESLintJSDoc.ts
rename to src/blocks/blockESLintJSDoc.ts
diff --git a/src/next/blocks/blockESLintJSONC.ts b/src/blocks/blockESLintJSONC.ts
similarity index 100%
rename from src/next/blocks/blockESLintJSONC.ts
rename to src/blocks/blockESLintJSONC.ts
diff --git a/src/next/blocks/blockESLintMarkdown.ts b/src/blocks/blockESLintMarkdown.ts
similarity index 100%
rename from src/next/blocks/blockESLintMarkdown.ts
rename to src/blocks/blockESLintMarkdown.ts
diff --git a/src/next/blocks/blockESLintMoreStyling.ts b/src/blocks/blockESLintMoreStyling.ts
similarity index 100%
rename from src/next/blocks/blockESLintMoreStyling.ts
rename to src/blocks/blockESLintMoreStyling.ts
diff --git a/src/next/blocks/blockESLintNode.ts b/src/blocks/blockESLintNode.ts
similarity index 100%
rename from src/next/blocks/blockESLintNode.ts
rename to src/blocks/blockESLintNode.ts
diff --git a/src/next/blocks/blockESLintPackageJson.ts b/src/blocks/blockESLintPackageJson.ts
similarity index 100%
rename from src/next/blocks/blockESLintPackageJson.ts
rename to src/blocks/blockESLintPackageJson.ts
diff --git a/src/next/blocks/blockESLintPerfectionist.ts b/src/blocks/blockESLintPerfectionist.ts
similarity index 100%
rename from src/next/blocks/blockESLintPerfectionist.ts
rename to src/blocks/blockESLintPerfectionist.ts
diff --git a/src/next/blocks/blockESLintRegexp.ts b/src/blocks/blockESLintRegexp.ts
similarity index 100%
rename from src/next/blocks/blockESLintRegexp.ts
rename to src/blocks/blockESLintRegexp.ts
diff --git a/src/next/blocks/blockESLintYML.ts b/src/blocks/blockESLintYML.ts
similarity index 100%
rename from src/next/blocks/blockESLintYML.ts
rename to src/blocks/blockESLintYML.ts
diff --git a/src/next/blocks/blockExampleFiles.test.ts b/src/blocks/blockExampleFiles.test.ts
similarity index 100%
rename from src/next/blocks/blockExampleFiles.test.ts
rename to src/blocks/blockExampleFiles.test.ts
diff --git a/src/next/blocks/blockExampleFiles.ts b/src/blocks/blockExampleFiles.ts
similarity index 100%
rename from src/next/blocks/blockExampleFiles.ts
rename to src/blocks/blockExampleFiles.ts
diff --git a/src/next/blocks/blockFunding.ts b/src/blocks/blockFunding.ts
similarity index 77%
rename from src/next/blocks/blockFunding.ts
rename to src/blocks/blockFunding.ts
index 934d3ed19..64cff2929 100644
--- a/src/next/blocks/blockFunding.ts
+++ b/src/blocks/blockFunding.ts
@@ -1,5 +1,5 @@
-import { formatYaml } from "../../steps/writing/creation/formatters/formatYaml.js";
import { base } from "../base.js";
+import { formatYaml } from "./files/formatYaml.js";
export const blockFunding = base.createBlock({
about: {
diff --git a/src/next/blocks/blockGitHubActionsCI.test.ts b/src/blocks/blockGitHubActionsCI.test.ts
similarity index 96%
rename from src/next/blocks/blockGitHubActionsCI.test.ts
rename to src/blocks/blockGitHubActionsCI.test.ts
index af9efb099..1cc55e004 100644
--- a/src/next/blocks/blockGitHubActionsCI.test.ts
+++ b/src/blocks/blockGitHubActionsCI.test.ts
@@ -70,7 +70,7 @@ describe("blockGitHubActionsCI", () => {
steps:
- uses: actions-ecosystem/action-remove-labels@v1
with:
- labels: "status: waiting for author"
+ labels: 'status: waiting for author'
- if: failure()
run: |
echo "Don't worry if the previous step failed."
@@ -159,7 +159,7 @@ describe("blockGitHubActionsCI", () => {
steps:
- uses: actions-ecosystem/action-remove-labels@v1
with:
- labels: "status: waiting for author"
+ labels: 'status: waiting for author'
- if: failure()
run: |
echo "Don't worry if the previous step failed."
@@ -271,11 +271,11 @@ describe("blockGitHubActionsCI", () => {
- uses: actions/checkout@v4
- uses: ./.github/actions/prepare
- env:
- VAR_ENV: "true"
+ VAR_ENV: 'true'
if: always()
run: pnpm validate
with:
- VAR_WITH: "true"
+ VAR_WITH: 'true'
name: CI
@@ -291,7 +291,7 @@ describe("blockGitHubActionsCI", () => {
steps:
- uses: actions-ecosystem/action-remove-labels@v1
with:
- labels: "status: waiting for author"
+ labels: 'status: waiting for author'
- if: failure()
run: |
echo "Don't worry if the previous step failed."
diff --git a/src/next/blocks/blockGitHubActionsCI.ts b/src/blocks/blockGitHubActionsCI.ts
similarity index 93%
rename from src/next/blocks/blockGitHubActionsCI.ts
rename to src/blocks/blockGitHubActionsCI.ts
index ad5f9d6e7..b7c5ca522 100644
--- a/src/next/blocks/blockGitHubActionsCI.ts
+++ b/src/blocks/blockGitHubActionsCI.ts
@@ -1,9 +1,9 @@
import jsYaml from "js-yaml";
import { z } from "zod";
-import { createMultiWorkflowFile } from "../../steps/writing/creation/dotGitHub/createMultiWorkflowFile.js";
-import { createSoloWorkflowFile } from "../../steps/writing/creation/dotGitHub/createSoloWorkflowFile.js";
import { base } from "../base.js";
+import { createMultiWorkflowFile } from "./files/createMultiWorkflowFile.js";
+import { createSoloWorkflowFile } from "./files/createSoloWorkflowFile.js";
import { CommandPhase } from "./phases.js";
export const blockGitHubActionsCI = base.createBlock({
diff --git a/src/next/blocks/blockGitHubApps.ts b/src/blocks/blockGitHubApps.ts
similarity index 100%
rename from src/next/blocks/blockGitHubApps.ts
rename to src/blocks/blockGitHubApps.ts
diff --git a/src/next/blocks/blockGitHubIssueTemplates.ts b/src/blocks/blockGitHubIssueTemplates.ts
similarity index 98%
rename from src/next/blocks/blockGitHubIssueTemplates.ts
rename to src/blocks/blockGitHubIssueTemplates.ts
index 23bd3e977..6d3ef2bb3 100644
--- a/src/next/blocks/blockGitHubIssueTemplates.ts
+++ b/src/blocks/blockGitHubIssueTemplates.ts
@@ -1,5 +1,5 @@
-import { formatYaml } from "../../steps/writing/creation/formatters/formatYaml.js";
import { base } from "../base.js";
+import { formatYaml } from "./files/formatYaml.js";
export const blockGitHubIssueTemplates = base.createBlock({
about: {
diff --git a/src/next/blocks/blockGitHubPRTemplate.ts b/src/blocks/blockGitHubPRTemplate.ts
similarity index 100%
rename from src/next/blocks/blockGitHubPRTemplate.ts
rename to src/blocks/blockGitHubPRTemplate.ts
diff --git a/src/next/blocks/blockGitignore.test.ts b/src/blocks/blockGitignore.test.ts
similarity index 100%
rename from src/next/blocks/blockGitignore.test.ts
rename to src/blocks/blockGitignore.test.ts
diff --git a/src/next/blocks/blockGitignore.ts b/src/blocks/blockGitignore.ts
similarity index 79%
rename from src/next/blocks/blockGitignore.ts
rename to src/blocks/blockGitignore.ts
index b10e4bf6c..fe607e3ef 100644
--- a/src/next/blocks/blockGitignore.ts
+++ b/src/blocks/blockGitignore.ts
@@ -1,7 +1,7 @@
import { z } from "zod";
-import { formatIgnoreFile } from "../../steps/writing/creation/formatters/formatIgnoreFile.js";
import { base } from "../base.js";
+import { formatIgnoreFile } from "./files/formatIgnoreFile.js";
export const blockGitignore = base.createBlock({
about: {
diff --git a/src/next/blocks/blockKnip.ts b/src/blocks/blockKnip.ts
similarity index 93%
rename from src/next/blocks/blockKnip.ts
rename to src/blocks/blockKnip.ts
index fadbb2021..1883a6431 100644
--- a/src/next/blocks/blockKnip.ts
+++ b/src/blocks/blockKnip.ts
@@ -1,8 +1,11 @@
import { base } from "../base.js";
+import {
+ getPackageDependencies,
+ getPackageDependency,
+} from "../data/packageData.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
import { blockPackageJson } from "./blockPackageJson.js";
-import { getPackageDependencies, getPackageDependency } from "./packageData.js";
import { CommandPhase } from "./phases.js";
export const blockKnip = base.createBlock({
diff --git a/src/next/blocks/blockMITLicense.ts b/src/blocks/blockMITLicense.ts
similarity index 100%
rename from src/next/blocks/blockMITLicense.ts
rename to src/blocks/blockMITLicense.ts
diff --git a/src/next/blocks/blockMarkdownlint.test.ts b/src/blocks/blockMarkdownlint.test.ts
similarity index 100%
rename from src/next/blocks/blockMarkdownlint.test.ts
rename to src/blocks/blockMarkdownlint.test.ts
diff --git a/src/next/blocks/blockMarkdownlint.ts b/src/blocks/blockMarkdownlint.ts
similarity index 96%
rename from src/next/blocks/blockMarkdownlint.ts
rename to src/blocks/blockMarkdownlint.ts
index 0f1cd9002..4410f601c 100644
--- a/src/next/blocks/blockMarkdownlint.ts
+++ b/src/blocks/blockMarkdownlint.ts
@@ -1,12 +1,12 @@
import { z } from "zod";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
import { blockPackageJson } from "./blockPackageJson.js";
import { blockVSCode } from "./blockVSCode.js";
-import { getPackageDependencies } from "./packageData.js";
import { CommandPhase } from "./phases.js";
export const blockMarkdownlint = base.createBlock({
diff --git a/src/next/blocks/blockNvmrc.test.ts b/src/blocks/blockNvmrc.test.ts
similarity index 100%
rename from src/next/blocks/blockNvmrc.test.ts
rename to src/blocks/blockNvmrc.test.ts
diff --git a/src/next/blocks/blockNvmrc.ts b/src/blocks/blockNvmrc.ts
similarity index 100%
rename from src/next/blocks/blockNvmrc.ts
rename to src/blocks/blockNvmrc.ts
diff --git a/src/next/blocks/blockPRCompliance.ts b/src/blocks/blockPRCompliance.ts
similarity index 89%
rename from src/next/blocks/blockPRCompliance.ts
rename to src/blocks/blockPRCompliance.ts
index eac9a64cd..a671555ca 100644
--- a/src/next/blocks/blockPRCompliance.ts
+++ b/src/blocks/blockPRCompliance.ts
@@ -1,5 +1,5 @@
-import { createSoloWorkflowFile } from "../../steps/writing/creation/dotGitHub/createSoloWorkflowFile.js";
import { base } from "../base.js";
+import { createSoloWorkflowFile } from "./files/createSoloWorkflowFile.js";
export const blockPRCompliance = base.createBlock({
about: {
diff --git a/src/next/blocks/blockPackageJson.test.ts b/src/blocks/blockPackageJson.test.ts
similarity index 100%
rename from src/next/blocks/blockPackageJson.test.ts
rename to src/blocks/blockPackageJson.test.ts
diff --git a/src/next/blocks/blockPackageJson.ts b/src/blocks/blockPackageJson.ts
similarity index 100%
rename from src/next/blocks/blockPackageJson.ts
rename to src/blocks/blockPackageJson.ts
diff --git a/src/next/blocks/blockPnpmDedupe.ts b/src/blocks/blockPnpmDedupe.ts
similarity index 100%
rename from src/next/blocks/blockPnpmDedupe.ts
rename to src/blocks/blockPnpmDedupe.ts
diff --git a/src/next/blocks/blockPrettier.test.ts b/src/blocks/blockPrettier.test.ts
similarity index 98%
rename from src/next/blocks/blockPrettier.test.ts
rename to src/blocks/blockPrettier.test.ts
index e54d2c6c7..ef447c7d8 100644
--- a/src/next/blocks/blockPrettier.test.ts
+++ b/src/blocks/blockPrettier.test.ts
@@ -60,7 +60,7 @@ describe("blockPrettier", () => {
"properties": {
"devDependencies": {
"husky": "9.1.7",
- "lint-staged": "15.2.11",
+ "lint-staged": "15.3.0",
"prettier": "^3.4.2",
},
"lint-staged": {
@@ -171,7 +171,7 @@ describe("blockPrettier", () => {
"properties": {
"devDependencies": {
"husky": "9.1.7",
- "lint-staged": "15.2.11",
+ "lint-staged": "15.3.0",
"prettier": "^3.4.2",
},
"lint-staged": {
@@ -296,7 +296,7 @@ describe("blockPrettier", () => {
"properties": {
"devDependencies": {
"husky": "9.1.7",
- "lint-staged": "15.2.11",
+ "lint-staged": "15.3.0",
"prettier": "^3.4.2",
"prettier-plugin-curly": "0.3.1",
"prettier-plugin-packagejson": "2.5.6",
diff --git a/src/next/blocks/blockPrettier.ts b/src/blocks/blockPrettier.ts
similarity index 97%
rename from src/next/blocks/blockPrettier.ts
rename to src/blocks/blockPrettier.ts
index bce284e4c..f08f1c4fb 100644
--- a/src/next/blocks/blockPrettier.ts
+++ b/src/blocks/blockPrettier.ts
@@ -1,12 +1,12 @@
import { z } from "zod";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
import { blockPackageJson } from "./blockPackageJson.js";
import { blockVSCode } from "./blockVSCode.js";
-import { getPackageDependencies } from "./packageData.js";
import { CommandPhase } from "./phases.js";
export const blockPrettier = base.createBlock({
diff --git a/src/next/blocks/blockPrettierPluginCurly.ts b/src/blocks/blockPrettierPluginCurly.ts
similarity index 100%
rename from src/next/blocks/blockPrettierPluginCurly.ts
rename to src/blocks/blockPrettierPluginCurly.ts
diff --git a/src/next/blocks/blockPrettierPluginPackageJson.ts b/src/blocks/blockPrettierPluginPackageJson.ts
similarity index 100%
rename from src/next/blocks/blockPrettierPluginPackageJson.ts
rename to src/blocks/blockPrettierPluginPackageJson.ts
diff --git a/src/next/blocks/blockPrettierPluginSh.ts b/src/blocks/blockPrettierPluginSh.ts
similarity index 100%
rename from src/next/blocks/blockPrettierPluginSh.ts
rename to src/blocks/blockPrettierPluginSh.ts
diff --git a/src/next/blocks/blockREADME.test.ts b/src/blocks/blockREADME.test.ts
similarity index 100%
rename from src/next/blocks/blockREADME.test.ts
rename to src/blocks/blockREADME.test.ts
diff --git a/src/next/blocks/blockREADME.ts b/src/blocks/blockREADME.ts
similarity index 100%
rename from src/next/blocks/blockREADME.ts
rename to src/blocks/blockREADME.ts
diff --git a/src/next/blocks/blockReleaseIt.ts b/src/blocks/blockReleaseIt.ts
similarity index 95%
rename from src/next/blocks/blockReleaseIt.ts
rename to src/blocks/blockReleaseIt.ts
index 29d4e2d01..fb909f95c 100644
--- a/src/next/blocks/blockReleaseIt.ts
+++ b/src/blocks/blockReleaseIt.ts
@@ -1,9 +1,9 @@
-import { createSoloWorkflowFile } from "../../steps/writing/creation/dotGitHub/createSoloWorkflowFile.js";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockPackageJson } from "./blockPackageJson.js";
import { blockRepositorySecrets } from "./blockRepositorySecrets.js";
-import { getPackageDependencies } from "./packageData.js";
+import { createSoloWorkflowFile } from "./files/createSoloWorkflowFile.js";
export const blockReleaseIt = base.createBlock({
about: {
diff --git a/src/next/blocks/blockRenovate.ts b/src/blocks/blockRenovate.ts
similarity index 100%
rename from src/next/blocks/blockRenovate.ts
rename to src/blocks/blockRenovate.ts
diff --git a/src/next/blocks/blockRepositoryBranchRuleset.test.ts b/src/blocks/blockRepositoryBranchRuleset.test.ts
similarity index 100%
rename from src/next/blocks/blockRepositoryBranchRuleset.test.ts
rename to src/blocks/blockRepositoryBranchRuleset.test.ts
diff --git a/src/next/blocks/blockRepositoryBranchRuleset.ts b/src/blocks/blockRepositoryBranchRuleset.ts
similarity index 100%
rename from src/next/blocks/blockRepositoryBranchRuleset.ts
rename to src/blocks/blockRepositoryBranchRuleset.ts
diff --git a/src/next/blocks/blockRepositoryLabels.ts b/src/blocks/blockRepositoryLabels.ts
similarity index 84%
rename from src/next/blocks/blockRepositoryLabels.ts
rename to src/blocks/blockRepositoryLabels.ts
index b8980788b..4b00f32dc 100644
--- a/src/next/blocks/blockRepositoryLabels.ts
+++ b/src/blocks/blockRepositoryLabels.ts
@@ -1,7 +1,7 @@
import { setGitHubRepositoryLabels } from "set-github-repository-labels";
-import { outcomeLabels } from "../../steps/initializeGitHubRepository/outcomeLabels.js";
import { base } from "../base.js";
+import { outcomeLabels } from "./outcomeLabels.js";
export const blockRepositoryLabels = base.createBlock({
about: {
diff --git a/src/next/blocks/blockRepositorySecrets.ts b/src/blocks/blockRepositorySecrets.ts
similarity index 100%
rename from src/next/blocks/blockRepositorySecrets.ts
rename to src/blocks/blockRepositorySecrets.ts
diff --git a/src/next/blocks/blockRepositorySettings.ts b/src/blocks/blockRepositorySettings.ts
similarity index 100%
rename from src/next/blocks/blockRepositorySettings.ts
rename to src/blocks/blockRepositorySettings.ts
diff --git a/src/next/blocks/blockSecurityDocs.ts b/src/blocks/blockSecurityDocs.ts
similarity index 100%
rename from src/next/blocks/blockSecurityDocs.ts
rename to src/blocks/blockSecurityDocs.ts
diff --git a/src/next/blocks/blockTSup.test.ts b/src/blocks/blockTSup.test.ts
similarity index 100%
rename from src/next/blocks/blockTSup.test.ts
rename to src/blocks/blockTSup.test.ts
diff --git a/src/next/blocks/blockTSup.ts b/src/blocks/blockTSup.ts
similarity index 96%
rename from src/next/blocks/blockTSup.ts
rename to src/blocks/blockTSup.ts
index dd7f01fcb..56efc77a2 100644
--- a/src/next/blocks/blockTSup.ts
+++ b/src/blocks/blockTSup.ts
@@ -1,11 +1,11 @@
import { z } from "zod";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockESLint } from "./blockESLint.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
import { blockPackageJson } from "./blockPackageJson.js";
-import { getPackageDependencies } from "./packageData.js";
import { CommandPhase } from "./phases.js";
export const blockTSup = base.createBlock({
diff --git a/src/next/blocks/blockTemplatedBy.ts b/src/blocks/blockTemplatedWith.ts
similarity index 91%
rename from src/next/blocks/blockTemplatedBy.ts
rename to src/blocks/blockTemplatedWith.ts
index ab85c72cb..9d9d2c38f 100644
--- a/src/next/blocks/blockTemplatedBy.ts
+++ b/src/blocks/blockTemplatedWith.ts
@@ -2,7 +2,7 @@ import { base } from "../base.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockREADME } from "./blockREADME.js";
-export const blockTemplatedBy = base.createBlock({
+export const blockTemplatedWith = base.createBlock({
about: {
name: "Templated By Notice",
},
diff --git a/src/next/blocks/blockTypeScript.test.ts b/src/blocks/blockTypeScript.test.ts
similarity index 100%
rename from src/next/blocks/blockTypeScript.test.ts
rename to src/blocks/blockTypeScript.test.ts
diff --git a/src/next/blocks/blockTypeScript.ts b/src/blocks/blockTypeScript.ts
similarity index 98%
rename from src/next/blocks/blockTypeScript.ts
rename to src/blocks/blockTypeScript.ts
index 9da798617..ccd575f1a 100644
--- a/src/next/blocks/blockTypeScript.ts
+++ b/src/blocks/blockTypeScript.ts
@@ -1,4 +1,5 @@
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockExampleFiles } from "./blockExampleFiles.js";
import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
@@ -7,7 +8,6 @@ import { blockMarkdownlint } from "./blockMarkdownlint.js";
import { blockPackageJson } from "./blockPackageJson.js";
import { blockVitest } from "./blockVitest.js";
import { blockVSCode } from "./blockVSCode.js";
-import { getPackageDependencies } from "./packageData.js";
export const blockTypeScript = base.createBlock({
about: {
diff --git a/src/next/blocks/blockVSCode.test.ts b/src/blocks/blockVSCode.test.ts
similarity index 100%
rename from src/next/blocks/blockVSCode.test.ts
rename to src/blocks/blockVSCode.test.ts
diff --git a/src/next/blocks/blockVSCode.ts b/src/blocks/blockVSCode.ts
similarity index 100%
rename from src/next/blocks/blockVSCode.ts
rename to src/blocks/blockVSCode.ts
diff --git a/src/next/blocks/blockVitest.test.ts b/src/blocks/blockVitest.test.ts
similarity index 100%
rename from src/next/blocks/blockVitest.test.ts
rename to src/blocks/blockVitest.test.ts
diff --git a/src/next/blocks/blockVitest.ts b/src/blocks/blockVitest.ts
similarity index 98%
rename from src/next/blocks/blockVitest.ts
rename to src/blocks/blockVitest.ts
index d1281b79d..ee03c306c 100644
--- a/src/next/blocks/blockVitest.ts
+++ b/src/blocks/blockVitest.ts
@@ -1,6 +1,7 @@
import { z } from "zod";
import { base } from "../base.js";
+import { getPackageDependencies } from "../data/packageData.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockESLint } from "./blockESLint.js";
@@ -12,7 +13,6 @@ import { blockPackageJson } from "./blockPackageJson.js";
import { blockPrettier } from "./blockPrettier.js";
import { blockTSup } from "./blockTSup.js";
import { blockVSCode } from "./blockVSCode.js";
-import { getPackageDependencies } from "./packageData.js";
import { CommandPhase } from "./phases.js";
export const blockVitest = base.createBlock({
diff --git a/src/steps/writing/creation/dotGitHub/createMultiWorkflowFile.ts b/src/blocks/files/createMultiWorkflowFile.ts
similarity index 100%
rename from src/steps/writing/creation/dotGitHub/createMultiWorkflowFile.ts
rename to src/blocks/files/createMultiWorkflowFile.ts
diff --git a/src/steps/writing/creation/dotGitHub/createSoloWorkflowFile.ts b/src/blocks/files/createSoloWorkflowFile.ts
similarity index 100%
rename from src/steps/writing/creation/dotGitHub/createSoloWorkflowFile.ts
rename to src/blocks/files/createSoloWorkflowFile.ts
diff --git a/src/steps/writing/creation/formatters/formatIgnoreFile.ts b/src/blocks/files/formatIgnoreFile.ts
similarity index 100%
rename from src/steps/writing/creation/formatters/formatIgnoreFile.ts
rename to src/blocks/files/formatIgnoreFile.ts
diff --git a/src/steps/writing/creation/dotGitHub/formatWorkflowYaml.ts b/src/blocks/files/formatWorkflowYaml.ts
similarity index 82%
rename from src/steps/writing/creation/dotGitHub/formatWorkflowYaml.ts
rename to src/blocks/files/formatWorkflowYaml.ts
index b2dbfefe6..c7e088372 100644
--- a/src/steps/writing/creation/dotGitHub/formatWorkflowYaml.ts
+++ b/src/blocks/files/formatWorkflowYaml.ts
@@ -1,4 +1,4 @@
-import { formatYaml } from "../formatters/formatYaml.js";
+import { formatYaml } from "./formatYaml.js";
export function formatWorkflowYaml(value: unknown) {
return (
diff --git a/src/steps/writing/creation/formatters/formatYaml.ts b/src/blocks/files/formatYaml.ts
similarity index 78%
rename from src/steps/writing/creation/formatters/formatYaml.ts
rename to src/blocks/files/formatYaml.ts
index ef7245d44..8ad1ff528 100644
--- a/src/steps/writing/creation/formatters/formatYaml.ts
+++ b/src/blocks/files/formatYaml.ts
@@ -1,4 +1,3 @@
-import prettier from "@prettier/sync";
import jsYaml from "js-yaml";
const options: jsYaml.DumpOptions = {
@@ -22,6 +21,5 @@ const options: jsYaml.DumpOptions = {
};
export function formatYaml(value: unknown) {
- const dumped = jsYaml.dump(value, options);
- return prettier.format(dumped, { parser: "yaml" });
+ return jsYaml.dump(value, options);
}
diff --git a/src/blocks/index.ts b/src/blocks/index.ts
new file mode 100644
index 000000000..efc0fb385
--- /dev/null
+++ b/src/blocks/index.ts
@@ -0,0 +1,130 @@
+import { blockAllContributors } from "./blockAllContributors.js";
+import { blockAreTheTypesWrong } from "./blockAreTheTypesWrong.js";
+import { blockContributingDocs } from "./blockContributingDocs.js";
+import { blockContributorCovenant } from "./blockContributorCovenant.js";
+import { blockCSpell } from "./blockCSpell.js";
+import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
+import { blockESLint } from "./blockESLint.js";
+import { blockESLintComments } from "./blockESLintComments.js";
+import { blockESLintJSDoc } from "./blockESLintJSDoc.js";
+import { blockESLintJSONC } from "./blockESLintJSONC.js";
+import { blockESLintMarkdown } from "./blockESLintMarkdown.js";
+import { blockESLintMoreStyling } from "./blockESLintMoreStyling.js";
+import { blockESLintNode } from "./blockESLintNode.js";
+import { blockESLintPackageJson } from "./blockESLintPackageJson.js";
+import { blockESLintPerfectionist } from "./blockESLintPerfectionist.js";
+import { blockESLintRegexp } from "./blockESLintRegexp.js";
+import { blockESLintYML } from "./blockESLintYML.js";
+import { blockFunding } from "./blockFunding.js";
+import { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
+import { blockGitHubIssueTemplates } from "./blockGitHubIssueTemplates.js";
+import { blockGitHubPRTemplate } from "./blockGitHubPRTemplate.js";
+import { blockGitignore } from "./blockGitignore.js";
+import { blockKnip } from "./blockKnip.js";
+import { blockMarkdownlint } from "./blockMarkdownlint.js";
+import { blockMITLicense } from "./blockMITLicense.js";
+import { blockNvmrc } from "./blockNvmrc.js";
+import { blockPackageJson } from "./blockPackageJson.js";
+import { blockPnpmDedupe } from "./blockPnpmDedupe.js";
+import { blockPRCompliance } from "./blockPRCompliance.js";
+import { blockPrettier } from "./blockPrettier.js";
+import { blockPrettierPluginCurly } from "./blockPrettierPluginCurly.js";
+import { blockPrettierPluginPackageJson } from "./blockPrettierPluginPackageJson.js";
+import { blockPrettierPluginSh } from "./blockPrettierPluginSh.js";
+import { blockREADME } from "./blockREADME.js";
+import { blockReleaseIt } from "./blockReleaseIt.js";
+import { blockRenovate } from "./blockRenovate.js";
+import { blockSecurityDocs } from "./blockSecurityDocs.js";
+import { blockTemplatedWith } from "./blockTemplatedWith.js";
+import { blockTSup } from "./blockTSup.js";
+import { blockTypeScript } from "./blockTypeScript.js";
+import { blockVitest } from "./blockVitest.js";
+import { blockVSCode } from "./blockVSCode.js";
+
+export const blocks = {
+ blockAllContributors,
+ blockAreTheTypesWrong,
+ blockContributingDocs,
+ blockContributorCovenant,
+ blockCSpell,
+ blockDevelopmentDocs,
+ blockESLint,
+ blockESLintComments,
+ blockESLintJSDoc,
+ blockESLintJSONC,
+ blockESLintMarkdown,
+ blockESLintMoreStyling,
+ blockESLintNode,
+ blockESLintPackageJson,
+ blockESLintPerfectionist,
+ blockESLintRegexp,
+ blockESLintYML,
+ blockFunding,
+ blockGitHubActionsCI,
+ blockGitHubIssueTemplates,
+ blockGitHubPRTemplate,
+ blockGitignore,
+ blockKnip,
+ blockMarkdownlint,
+ blockMITLicense,
+ blockNvmrc,
+ blockPackageJson,
+ blockPnpmDedupe,
+ blockPRCompliance,
+ blockPrettier,
+ blockPrettierPluginCurly,
+ blockPrettierPluginPackageJson,
+ blockPrettierPluginSh,
+ blockREADME,
+ blockReleaseIt,
+ blockRenovate,
+ blockSecurityDocs,
+ blockTemplatedWith,
+ blockTSup,
+ blockTypeScript,
+ blockVitest,
+ blockVSCode,
+};
+
+export { blockAllContributors } from "./blockAllContributors.js";
+export { blockAreTheTypesWrong } from "./blockAreTheTypesWrong.js";
+export { blockContributingDocs } from "./blockContributingDocs.js";
+export { blockContributorCovenant } from "./blockContributorCovenant.js";
+export { blockCSpell } from "./blockCSpell.js";
+export { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
+export { blockESLint } from "./blockESLint.js";
+export { blockESLintComments } from "./blockESLintComments.js";
+export { blockESLintJSDoc } from "./blockESLintJSDoc.js";
+export { blockESLintJSONC } from "./blockESLintJSONC.js";
+export { blockESLintMarkdown } from "./blockESLintMarkdown.js";
+export { blockESLintMoreStyling } from "./blockESLintMoreStyling.js";
+export { blockESLintNode } from "./blockESLintNode.js";
+export { blockESLintPackageJson } from "./blockESLintPackageJson.js";
+export { blockESLintPerfectionist } from "./blockESLintPerfectionist.js";
+export { blockESLintRegexp } from "./blockESLintRegexp.js";
+export { blockESLintYML } from "./blockESLintYML.js";
+export { blockFunding } from "./blockFunding.js";
+export { blockGitHubActionsCI } from "./blockGitHubActionsCI.js";
+export { blockGitHubIssueTemplates } from "./blockGitHubIssueTemplates.js";
+export { blockGitHubPRTemplate } from "./blockGitHubPRTemplate.js";
+export { blockGitignore } from "./blockGitignore.js";
+export { blockKnip } from "./blockKnip.js";
+export { blockMarkdownlint } from "./blockMarkdownlint.js";
+export { blockMITLicense } from "./blockMITLicense.js";
+export { blockNvmrc } from "./blockNvmrc.js";
+export { blockPackageJson } from "./blockPackageJson.js";
+export { blockPnpmDedupe } from "./blockPnpmDedupe.js";
+export { blockPRCompliance } from "./blockPRCompliance.js";
+export { blockPrettier } from "./blockPrettier.js";
+export { blockPrettierPluginCurly } from "./blockPrettierPluginCurly.js";
+export { blockPrettierPluginPackageJson } from "./blockPrettierPluginPackageJson.js";
+export { blockPrettierPluginSh } from "./blockPrettierPluginSh.js";
+export { blockREADME } from "./blockREADME.js";
+export { blockReleaseIt } from "./blockReleaseIt.js";
+export { blockRenovate } from "./blockRenovate.js";
+export { blockSecurityDocs } from "./blockSecurityDocs.js";
+export { blockTemplatedWith } from "./blockTemplatedWith.js";
+export { blockTSup } from "./blockTSup.js";
+export { blockTypeScript } from "./blockTypeScript.js";
+export { blockVitest } from "./blockVitest.js";
+export { blockVSCode } from "./blockVSCode.js";
diff --git a/src/next/blocks/options.fakes.ts b/src/blocks/options.fakes.ts
similarity index 100%
rename from src/next/blocks/options.fakes.ts
rename to src/blocks/options.fakes.ts
diff --git a/src/steps/initializeGitHubRepository/outcomeLabels.ts b/src/blocks/outcomeLabels.ts
similarity index 98%
rename from src/steps/initializeGitHubRepository/outcomeLabels.ts
rename to src/blocks/outcomeLabels.ts
index 092bfd5bc..e589e97eb 100644
--- a/src/steps/initializeGitHubRepository/outcomeLabels.ts
+++ b/src/blocks/outcomeLabels.ts
@@ -1,4 +1,3 @@
-/* spellchecker: disable */
export const outcomeLabels = [
{
aliases: ["docs"],
@@ -75,6 +74,7 @@ export const outcomeLabels = [
},
{
aliases: ["enhancement"],
+ /* spellchecker: disable-next-line */
color: "a2eeef",
description: "New enhancement or request 🚀",
name: "type: feature",
diff --git a/src/next/blocks/phases.ts b/src/blocks/phases.ts
similarity index 100%
rename from src/next/blocks/phases.ts
rename to src/blocks/phases.ts
diff --git a/src/create/createAndEnterGitDirectory.test.ts b/src/create/createAndEnterGitDirectory.test.ts
deleted file mode 100644
index 85ceb3a02..000000000
--- a/src/create/createAndEnterGitDirectory.test.ts
+++ /dev/null
@@ -1,82 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { createAndEnterGitDirectory } from "./createAndEnterGitDirectory.js";
-
-const mockMkdir = vi.fn();
-const mockReaddir = vi.fn();
-
-vi.mock("node:fs/promises", () => ({
- get mkdir() {
- return mockMkdir;
- },
- get readdir() {
- return mockReaddir;
- },
-}));
-
-const mockChdir = vi.fn();
-const mockCwd = "/path/to/cwd";
-
-vi.mock("node:process", () => ({
- get chdir() {
- return mockChdir;
- },
- cwd() {
- return mockCwd;
- },
-}));
-
-const mock$$ = vi.fn();
-
-const mock$ = vi.fn().mockReturnValue(mock$$);
-
-vi.mock("execa", () => ({
- get $() {
- return mock$;
- },
-}));
-
-describe("createAndEnterGitDirectory", () => {
- it("returns false and doesn't run fs.mkdir when the directory is '.' and has children", async () => {
- mockReaddir.mockResolvedValueOnce(["file"]);
-
- const actual = await createAndEnterGitDirectory(".");
-
- expect(actual).toBeUndefined();
- expect(mockMkdir).not.toHaveBeenCalled();
- });
-
- it("returns true and doesn't run fs.mkdir when the directory is '.' and is empty", async () => {
- mockReaddir.mockResolvedValueOnce([]);
-
- const actual = await createAndEnterGitDirectory(".");
-
- expect(actual).toBe(mockCwd);
- expect(mockMkdir).not.toHaveBeenCalled();
- });
-
- it("returns false and doesn't run process.chdir when the directory is a child directory with children", async () => {
- const directory = "dir";
- mockReaddir
- .mockResolvedValueOnce([directory])
- .mockResolvedValueOnce(["file"]);
-
- const actual = await createAndEnterGitDirectory(directory);
-
- expect(actual).toBeUndefined();
- expect(mockChdir).not.toHaveBeenCalled();
- expect(mock$).not.toHaveBeenCalled();
- });
-
- it("returns true and runs process.chdir when the directory is a child directory that doesn't exist", async () => {
- const directory = "dir";
- mockReaddir.mockResolvedValueOnce([directory]).mockResolvedValueOnce([]);
-
- const actual = await createAndEnterGitDirectory(directory);
-
- expect(actual).toBe(mockCwd);
- expect(mockChdir).toHaveBeenCalledWith(directory);
- expect(mock$).toHaveBeenCalledWith({ cwd: mockCwd });
- expect(mock$$).toHaveBeenCalledWith(["git init -b main"]);
- });
-});
diff --git a/src/create/createAndEnterGitDirectory.ts b/src/create/createAndEnterGitDirectory.ts
deleted file mode 100644
index c249d823d..000000000
--- a/src/create/createAndEnterGitDirectory.ts
+++ /dev/null
@@ -1,26 +0,0 @@
-import { $ } from "execa";
-import * as fs from "node:fs/promises";
-import * as process from "node:process";
-
-export async function createAndEnterGitDirectory(directory: string) {
- if (directory !== "." && !(await fs.readdir(".")).includes(directory)) {
- await fs.mkdir(directory);
- } else if ((await fs.readdir(directory)).length) {
- return undefined;
- }
-
- if (directory !== ".") {
- process.chdir(directory);
- }
-
- // https://github.com/JoshuaKGoldberg/create-typescript-app/pull/1781
- // For some reason, after this function returns, the cwd was resetting back?
- // Also $ commands seem to preserve the cwd from the start of the script?!
- // Maybe some parallelized running of scripts was messing with it?
- // All this code will be gone soon once the create engine takes over anyway.
- const cwd = process.cwd();
-
- await $({ cwd })`git init -b main`;
-
- return cwd;
-}
diff --git a/src/create/createRerunDirectorySuggestion.test.ts b/src/create/createRerunDirectorySuggestion.test.ts
deleted file mode 100644
index 9b4798348..000000000
--- a/src/create/createRerunDirectorySuggestion.test.ts
+++ /dev/null
@@ -1,48 +0,0 @@
-import { describe, expect, it } from "vitest";
-
-import { createRerunDirectorySuggestion } from "./createRerunDirectorySuggestion.js";
-
-const directory = "test-directory";
-const repository = "test-repository";
-
-describe("createRerunDirectorySuggestion", () => {
- it("returns undefined when mode is create and directory matches repository", () => {
- const suggestion = createRerunDirectorySuggestion({
- directory: repository,
- mode: "create",
- repository,
- });
-
- expect(suggestion).toBe(undefined);
- });
-
- it("returns directory when mode is create and directory doesn't match repository", () => {
- const suggestion = createRerunDirectorySuggestion({
- directory,
- mode: "create",
- repository,
- });
-
- expect(suggestion).toBe(directory);
- });
-
- it("returns undefined when mode is initialize and directory is .", () => {
- const suggestion = createRerunDirectorySuggestion({
- directory: ".",
- mode: "initialize",
- repository,
- });
-
- expect(suggestion).toBe(undefined);
- });
-
- it("returns directory when mode is initialize and directory is not .", () => {
- const suggestion = createRerunDirectorySuggestion({
- directory,
- mode: "initialize",
- repository,
- });
-
- expect(suggestion).toBe(directory);
- });
-});
diff --git a/src/create/createRerunDirectorySuggestion.ts b/src/create/createRerunDirectorySuggestion.ts
deleted file mode 100644
index 307299423..000000000
--- a/src/create/createRerunDirectorySuggestion.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-import { Options } from "../shared/types.js";
-
-export function createRerunDirectorySuggestion(options: Partial) {
- const defaultValue = options.mode === "create" ? options.repository : ".";
-
- return options.directory === defaultValue ? undefined : options.directory;
-}
diff --git a/src/create/createRerunSuggestion.test.ts b/src/create/createRerunSuggestion.test.ts
deleted file mode 100644
index 856d55c9c..000000000
--- a/src/create/createRerunSuggestion.test.ts
+++ /dev/null
@@ -1,185 +0,0 @@
-import { describe, expect, it } from "vitest";
-
-import { getExclusions } from "../shared/options/exclusionKeys.js";
-import { Options } from "../shared/types.js";
-import { createRerunSuggestion } from "./createRerunSuggestion.js";
-
-const options = {
- access: "public",
- author: "TestAuthor",
- base: "everything",
- description: "Test description.",
- directory: ".",
- email: {
- github: "github@email.com",
- npm: "npm@email.com",
- },
- excludeAllContributors: true,
- excludeCompliance: true,
- excludeLintJSDoc: true,
- excludeLintJson: true,
- excludeLintKnip: true,
- excludeLintMd: false,
- excludeLintPackageJson: true,
- excludeLintPackages: false,
- excludeLintPerfectionist: true,
- excludeLintSpelling: false,
- excludeLintYml: false,
- excludeReleases: false,
- excludeRenovate: undefined,
- excludeTemplatedBy: undefined,
- excludeTests: undefined,
- funding: undefined,
- keywords: ["abc", "def ghi", "jkl mno pqr"],
- mode: "create",
- owner: "TestOwner",
- repository: "test-repository",
- skipGitHubApi: true,
- skipInstall: true,
- skipRemoval: true,
- skipRestore: undefined,
- skipUninstall: undefined,
- title: "Test Title",
-} satisfies Options;
-
-describe("createRerunSuggestion", () => {
- it("prints no options when no options are provided", () => {
- const actual = createRerunSuggestion({});
-
- expect(actual).toMatchInlineSnapshot(`"npx create-typescript-app"`);
- });
-
- it("prints only mode when no other options are provided", () => {
- const actual = createRerunSuggestion({
- mode: "create",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --mode create"`,
- );
- });
-
- it("includes key-value pairs with mixed truthy and falsy values", () => {
- const actual = createRerunSuggestion(options);
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --directory . --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --keywords "abc def ghi jkl mno pqr" --mode create --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("includes a non-default value when specified", () => {
- const actual = createRerunSuggestion({
- ...options,
- access: "restricted",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --access restricted --author TestAuthor --description "Test description." --directory . --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --keywords "abc def ghi jkl mno pqr" --mode create --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("includes stringified guide when it exists", () => {
- const actual = createRerunSuggestion({
- ...options,
- guide: {
- href: "https://example.com",
- title: "Test Title",
- },
- mode: "initialize",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --guide https://example.com --guide-title "Test Title" --keywords "abc def ghi jkl mno pqr" --mode initialize --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("includes stringified logo when it exists", () => {
- const actual = createRerunSuggestion({
- ...options,
- logo: {
- alt: "Test alt.",
- src: "test/src.png",
- },
- mode: "initialize",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --keywords "abc def ghi jkl mno pqr" --logo test/src.png --logo-alt "Test alt." --mode initialize --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("does not include directory when it is repository and the mode is create", () => {
- const actual = createRerunSuggestion({
- ...options,
- directory: options.repository,
- mode: "create",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --keywords "abc def ghi jkl mno pqr" --mode create --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("includes directory when it is repository and the mode is migrate", () => {
- const actual = createRerunSuggestion({
- ...options,
- directory: options.repository,
- mode: "migrate",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --directory test-repository --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --keywords "abc def ghi jkl mno pqr" --mode migrate --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("includes exclusions when they exist", () => {
- const actual = createRerunSuggestion({
- ...options,
- excludeCompliance: true,
- excludeLintMd: true,
- excludeLintSpelling: true,
- mode: "initialize",
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-md --exclude-lint-package-json --exclude-lint-perfectionist --exclude-lint-spelling --keywords "abc def ghi jkl mno pqr" --mode initialize --owner TestOwner --repository test-repository --skip-github-api --skip-install --skip-removal --title "Test Title""`,
- );
- });
-
- it("does not list all excludes when using common base", () => {
- const common = createRerunSuggestion({
- base: "common",
- ...getExclusions(options, "common"),
- excludeLintKnip: undefined,
- });
-
- expect(common).toMatchInlineSnapshot(
- `"npx create-typescript-app --base common"`,
- );
- });
-
- it("does not list all excludes when using minimal base", () => {
- const minimal = createRerunSuggestion({
- base: "minimal",
- ...getExclusions(options, "minimal"),
- excludeLintKnip: undefined,
- });
-
- expect(minimal).toMatchInlineSnapshot(
- `"npx create-typescript-app --base minimal"`,
- );
- });
-
- it("does not list API skip flags when --offline is true", () => {
- const actual = createRerunSuggestion({
- ...options,
- offline: true,
- skipAllContributorsApi: true,
- skipGitHubApi: true,
- });
-
- expect(actual).toMatchInlineSnapshot(
- `"npx create-typescript-app --base everything --author TestAuthor --description "Test description." --directory . --email-github github@email.com --email-npm npm@email.com --exclude-all-contributors --exclude-compliance --exclude-lint-jsdoc --exclude-lint-json --exclude-lint-knip --exclude-lint-package-json --exclude-lint-perfectionist --keywords "abc def ghi jkl mno pqr" --mode create --offline --owner TestOwner --repository test-repository --skip-install --skip-removal --title "Test Title""`,
- );
- });
-});
diff --git a/src/create/createRerunSuggestion.ts b/src/create/createRerunSuggestion.ts
deleted file mode 100644
index 0ffb97865..000000000
--- a/src/create/createRerunSuggestion.ts
+++ /dev/null
@@ -1,81 +0,0 @@
-import { allArgOptions } from "../shared/options/args.js";
-import {
- ExclusionKey,
- getExclusions,
-} from "../shared/options/exclusionKeys.js";
-import { Options } from "../shared/types.js";
-import { createRerunDirectorySuggestion } from "./createRerunDirectorySuggestion.js";
-
-function getFirstMatchingArg(key: string) {
- return Object.keys(allArgOptions).find(
- (arg) => arg.replaceAll("-", "") === key.toLowerCase(),
- );
-}
-
-const defaultValues = new Map([["access", "public"]]);
-
-export function createRerunSuggestion(options: Partial): string {
- const optionsNormalized = {
- ...options,
- email: undefined,
- ...(options.email && {
- emailGitHub: options.email.github,
- emailNpm: options.email.npm,
- }),
- ...(options.guide
- ? {
- guide: options.guide.href,
- guideTitle: options.guide.title,
- }
- : { guide: undefined }),
- ...(options.logo
- ? {
- logo: options.logo.src,
- logoAlt: options.logo.alt,
- }
- : { logo: undefined }),
- ...(options.offline && {
- skipAllContributorsApi: undefined,
- skipGitHubApi: undefined,
- }),
- directory: createRerunDirectorySuggestion(options),
- };
-
- const args = Object.entries(optionsNormalized)
- // Sort so the base is first, then the rest are sorted alphabetically
- .sort(([a], [b]) =>
- a === "base" ? -1 : b === "base" ? 1 : a.localeCompare(b),
- )
- // Filter out entries with an excluded key or a default or falsy value
- .filter(
- ([key, value]) =>
- getExclusions(options, optionsNormalized.base)[key as ExclusionKey] ==
- undefined &&
- !!value &&
- value !== defaultValues.get(key),
- )
- .map(([key, value]) => {
- return `--${getFirstMatchingArg(key)}${stringifyValue(value)}`;
- })
- .join(" ");
-
- return ["npx create-typescript-app", args].filter(Boolean).join(" ");
-}
-
-function stringifyValue(
- value: boolean | string | string[] | undefined,
-): string {
- if (Array.isArray(value)) {
- return stringifyValue(value.join(" "));
- }
-
- if (typeof value === "boolean" && value) {
- return "";
- }
-
- const valueStringified = `${value}`;
-
- return valueStringified.includes(" ")
- ? ` "${valueStringified}"`
- : ` ${valueStringified}`;
-}
diff --git a/src/create/createWithOptions.test.ts b/src/create/createWithOptions.test.ts
deleted file mode 100644
index cbb5ce483..000000000
--- a/src/create/createWithOptions.test.ts
+++ /dev/null
@@ -1,195 +0,0 @@
-import { Octokit } from "octokit";
-import { describe, expect, it, vi } from "vitest";
-
-import { SpinnerTask } from "../shared/cli/spinners.js";
-import { doesRepositoryExist } from "../shared/doesRepositoryExist.js";
-import { Options } from "../shared/types.js";
-import { addToolAllContributors } from "../steps/addToolAllContributors.js";
-import { finalizeDependencies } from "../steps/finalizeDependencies.js";
-import { initializeGitHubRepository } from "../steps/initializeGitHubRepository/index.js";
-import { runCleanup } from "../steps/runCleanup.js";
-import { createWithOptions } from "./createWithOptions.js";
-
-const optionsBase: Options = {
- access: "public",
- author: "Test Author",
- base: "common",
- description: "Test Description",
- directory: "test-directory",
- email: { github: "github@example.com", npm: "npm@example.com" },
- funding: "Test Funding",
- keywords: ["test", "keywords"],
- logo: { alt: "Test Alt", src: "test.png" },
- mode: "create",
- owner: "Test Owner",
- repository: "test-repo",
- title: "Test Title",
-};
-
-const mock$$ = vi.fn();
-
-const mock$ = vi.fn().mockReturnValue(mock$$);
-
-vi.mock("execa", () => ({
- get $() {
- return mock$;
- },
-}));
-
-const mockOctokit = new Octokit();
-
-vi.mock("../shared/cli/spinners.js", () => ({
- withSpinner: async (label: string, task: SpinnerTask) => {
- return await task();
- },
- withSpinners: async (
- label: string,
- tasks: [string, SpinnerTask][],
- ) => {
- for (const [, task] of tasks) {
- await task();
- }
- },
-}));
-
-vi.mock("../steps/writing/writeStructure.js");
-
-vi.mock("../steps/writeReadme/index.js");
-
-vi.mock("../steps/finalizeDependencies.js");
-
-vi.mock("../steps/clearLocalGitTags.js");
-
-vi.mock("../steps/runCleanup.js");
-
-vi.mock("../shared/doesRepositoryExist.js", () => ({
- doesRepositoryExist: vi.fn().mockResolvedValue(true),
-}));
-
-vi.mock("../steps/initializeGitHubRepository/index.js", () => ({
- initializeGitHubRepository: vi.fn().mockResolvedValue(true),
-}));
-
-vi.mock("../shared/getGitHubUserAsAllContributor.js", () => ({
- getGitHubUserAsAllContributor: vi.fn().mockResolvedValue({
- contributions: ["code", "doc"],
- name: "Test User",
- }),
-}));
-
-vi.mock("../steps/addToolAllContributors.js");
-
-describe("createWithOptions", () => {
- it("calls addToolAllContributors with options when excludeAllContributors is false", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: false,
- skipAllContributorsApi: false,
- };
-
- await createWithOptions({ octokit: mockOctokit, options }, ".");
- expect(addToolAllContributors).toHaveBeenCalledWith(mockOctokit, options);
- });
-
- it("does not call addToolAllContributors when excludeAllContributors is true", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: true,
- };
-
- await createWithOptions({ octokit: mockOctokit, options }, ".");
- expect(addToolAllContributors).not.toHaveBeenCalled();
- });
-
- it("does not call addToolAllContributors when skipAllContributorsApi is true", async () => {
- const options = {
- ...optionsBase,
- skipAllContributorsApi: true,
- };
-
- await createWithOptions({ octokit: mockOctokit, options }, ".");
- expect(addToolAllContributors).not.toHaveBeenCalled();
- });
-
- it("does not call finalizeDependencies or runCleanup when skipInstall is true", async () => {
- const options = {
- ...optionsBase,
- skipInstall: true,
- };
-
- await createWithOptions({ octokit: mockOctokit, options }, ".");
- expect(finalizeDependencies).not.toHaveBeenCalled();
- expect(runCleanup).not.toHaveBeenCalled();
- });
-
- it("calls finalizeDependencies and runCleanup when skipInstall is false", async () => {
- const options = {
- ...optionsBase,
- skipInstall: false,
- };
-
- await createWithOptions({ octokit: mockOctokit, options }, ".");
-
- expect(finalizeDependencies).toHaveBeenCalledWith(options);
- expect(runCleanup).toHaveBeenCalled();
- });
-
- it("does not initialize GitHub repository if repository does not exist", async () => {
- const options = optionsBase;
- vi.mocked(doesRepositoryExist).mockResolvedValueOnce(false);
- await createWithOptions({ octokit: mockOctokit, options }, ".");
- expect(initializeGitHubRepository).not.toHaveBeenCalled();
- expect(mock$$).not.toHaveBeenCalled();
- });
-
- it("executes git commands when initializing GitHub repository when doesRepositoryExist is true", async () => {
- const options = optionsBase;
-
- vi.mocked(doesRepositoryExist).mockResolvedValueOnce(true);
- await createWithOptions({ octokit: mockOctokit, options }, ".");
-
- expect(mock$$.mock.calls).toMatchInlineSnapshot(`
- [
- [
- [
- "git remote add origin https://github.com/",
- "/",
- "",
- ],
- "Test Owner",
- "test-repo",
- ],
- [
- [
- "git add -A",
- ],
- ],
- [
- [
- "git commit --message ",
- " --no-gpg-sign",
- ],
- "feat: initialized repo ✨",
- ],
- [
- [
- "git push -u origin main --force",
- ],
- ],
- ]
- `);
- });
-
- it("calls doesRepositoryExist with github and options when doesRepositoryExist is true", async () => {
- const options = optionsBase;
- vi.mocked(doesRepositoryExist).mockResolvedValueOnce(true);
-
- await createWithOptions({ octokit: mockOctokit, options }, ".");
-
- expect(doesRepositoryExist).toHaveBeenCalledWith(mockOctokit, options);
- expect(initializeGitHubRepository).toHaveBeenCalledWith(
- mockOctokit,
- options,
- );
- });
-});
diff --git a/src/create/createWithOptions.ts b/src/create/createWithOptions.ts
deleted file mode 100644
index a87275160..000000000
--- a/src/create/createWithOptions.ts
+++ /dev/null
@@ -1,81 +0,0 @@
-import { $ } from "execa";
-
-import { runCreateEnginePreset } from "../next/runCreateEnginePreset.js";
-import {
- LabeledSpinnerTask,
- withSpinner,
- withSpinners,
-} from "../shared/cli/spinners.js";
-import { createCleanupCommands } from "../shared/createCleanupCommands.js";
-import { doesRepositoryExist } from "../shared/doesRepositoryExist.js";
-import { isUsingCreateEngine } from "../shared/isUsingCreateEngine.js";
-import { OctokitAndOptions } from "../shared/options/readOptions.js";
-import { addToolAllContributors } from "../steps/addToolAllContributors.js";
-import { clearLocalGitTags } from "../steps/clearLocalGitTags.js";
-import { finalizeDependencies } from "../steps/finalizeDependencies.js";
-import { initializeGitHubRepository } from "../steps/initializeGitHubRepository/index.js";
-import { runCleanup } from "../steps/runCleanup.js";
-import { writeReadme } from "../steps/writeReadme/index.js";
-import { writeStructure } from "../steps/writing/writeStructure.js";
-
-export async function createWithOptions(
- { octokit, options }: OctokitAndOptions,
- repositoryDirectory: string,
-) {
- // https://github.com/JoshuaKGoldberg/create-typescript-app/pull/1781
- // I don't know why the chdir wasn't sticking from createAndEnterGitDirectory
- process.chdir(repositoryDirectory);
- const $$ = $({ cwd: repositoryDirectory });
-
- if (isUsingCreateEngine()) {
- await withSpinner("Creating repository", async () => {
- await runCreateEnginePreset(options);
- });
- return { sentToGitHub: false };
- }
-
- await withSpinners("Creating repository structure", [
- [
- "Writing structure",
- async () => {
- await writeStructure(options);
- },
- ],
- [
- "Writing README.md",
- async () => {
- await writeReadme(options);
- },
- ] satisfies LabeledSpinnerTask,
- ]);
-
- if (!options.excludeAllContributors && !options.skipAllContributorsApi) {
- await withSpinner("Adding contributors to table", async () => {
- await addToolAllContributors(octokit, options);
- });
- }
-
- if (!options.skipInstall) {
- await withSpinner("Installing packages", async () =>
- finalizeDependencies(options),
- );
-
- await runCleanup(createCleanupCommands(options.bin), options.mode);
- }
-
- await withSpinner("Clearing any local Git tags", clearLocalGitTags);
-
- const sendToGitHub = octokit && (await doesRepositoryExist(octokit, options));
-
- if (sendToGitHub) {
- await withSpinner("Initializing GitHub repository", async () => {
- await $$`git remote add origin https://github.com/${options.owner}/${options.repository}`;
- await $$`git add -A`;
- await $$`git commit --message ${"feat: initialized repo ✨"} --no-gpg-sign`;
- await $$`git push -u origin main --force`;
- await initializeGitHubRepository(octokit, options);
- });
- }
-
- return { sentToGitHub: sendToGitHub };
-}
diff --git a/src/create/index.test.ts b/src/create/index.test.ts
deleted file mode 100644
index d86ec6b3f..000000000
--- a/src/create/index.test.ts
+++ /dev/null
@@ -1,72 +0,0 @@
-import chalk from "chalk";
-import { describe, expect, it, vi } from "vitest";
-
-import { StatusCodes } from "../shared/codes.js";
-import { create } from "./index.js";
-
-const mockOutro = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- get outro() {
- return mockOutro;
- },
- spinner: vi.fn(),
-}));
-
-const mockReadOptions = vi.fn();
-
-vi.mock("../shared/options/readOptions.js", () => ({
- get readOptions() {
- return mockReadOptions;
- },
-}));
-
-const mockCreateAndEnterGitDirectory = vi.fn();
-
-vi.mock("./createAndEnterGitDirectory.js", () => ({
- get createAndEnterGitDirectory() {
- return mockCreateAndEnterGitDirectory;
- },
-}));
-
-const optionsBase = {
- directory: "TestDirectory",
- repository: "TestRepository",
-};
-
-describe("create", () => {
- it("returns a cancellation code when readOptions cancels", async () => {
- mockReadOptions.mockResolvedValue({
- cancelled: true,
- options: optionsBase,
- });
-
- const result = await create([]);
-
- expect(result).toEqual({
- code: StatusCodes.Cancelled,
- options: optionsBase,
- });
- });
-
- it("returns a failure code when createAndEnterGitDirectory returns false", async () => {
- mockReadOptions.mockResolvedValue({
- cancelled: false,
- options: optionsBase,
- });
-
- mockCreateAndEnterGitDirectory.mockResolvedValue(false);
-
- const result = await create([]);
-
- expect(result).toEqual({
- code: StatusCodes.Failure,
- options: optionsBase,
- });
- expect(mockOutro).toHaveBeenCalledWith(
- chalk.red(
- "The TestDirectory directory already exists and is not empty. Please clear the directory, run with --mode initialize, or try a different directory.",
- ),
- );
- });
-});
diff --git a/src/create/index.ts b/src/create/index.ts
deleted file mode 100644
index 5d1ce6610..000000000
--- a/src/create/index.ts
+++ /dev/null
@@ -1,74 +0,0 @@
-import * as prompts from "@clack/prompts";
-import chalk from "chalk";
-
-import { outro } from "../shared/cli/outro.js";
-import { StatusCodes } from "../shared/codes.js";
-import { generateNextSteps } from "../shared/generateNextSteps.js";
-import { readOptions } from "../shared/options/readOptions.js";
-import { runOrRestore } from "../shared/runOrRestore.js";
-import { ModeRunner } from "../shared/types.js";
-import { createAndEnterGitDirectory } from "./createAndEnterGitDirectory.js";
-import { createRerunSuggestion } from "./createRerunSuggestion.js";
-import { createWithOptions } from "./createWithOptions.js";
-
-export const create: ModeRunner = async (args, promptedOptions) => {
- const inputs = await readOptions(args, "create", promptedOptions);
- if (inputs.cancelled) {
- return {
- code: StatusCodes.Cancelled,
- error: inputs.error,
- options: inputs.options,
- };
- }
-
- const repositoryDirectory = await createAndEnterGitDirectory(
- inputs.options.directory,
- );
- if (!repositoryDirectory) {
- prompts.outro(
- chalk.red(
- `The ${inputs.options.directory} directory already exists and is not empty. Please clear the directory, run with --mode initialize, or try a different directory.`,
- ),
- );
- return { code: StatusCodes.Failure, options: inputs.options };
- }
-
- return {
- code: await runOrRestore({
- run: async () => {
- const { sentToGitHub } = await createWithOptions(
- inputs,
- repositoryDirectory,
- );
- const nextSteps = generateNextSteps(inputs.options);
-
- outro(
- sentToGitHub
- ? nextSteps
- : [
- {
- label:
- "Consider creating a GitHub repository from the new directory:",
- lines: [
- `cd ${inputs.options.repository}`,
- createRerunSuggestion({
- ...inputs.options,
- mode: "initialize",
- skipGitHubApi: false,
- skipInstall: false,
- }),
- `git add -A`,
- `git commit -m "feat: initial commit ✨"`,
- `git push -u origin main`,
- ],
- variant: "code",
- },
- ...nextSteps,
- ],
- );
- },
- skipRestore: inputs.options.skipRestore,
- }),
- options: inputs.options,
- };
-};
diff --git a/src/next/blocks/packageData.test.ts b/src/data/packageData.test.ts
similarity index 95%
rename from src/next/blocks/packageData.test.ts
rename to src/data/packageData.test.ts
index 14eafebdd..17a0dc58f 100644
--- a/src/next/blocks/packageData.test.ts
+++ b/src/data/packageData.test.ts
@@ -13,7 +13,7 @@ vi.mock("node:module", () => ({
}),
}));
-describe("packageData", () => {
+describe("getPackageDependencies", () => {
it("returns a devDependency when it exists", () => {
const actual = getPackageDependencies("package-dev-dep");
diff --git a/src/next/blocks/packageData.ts b/src/data/packageData.ts
similarity index 68%
rename from src/next/blocks/packageData.ts
rename to src/data/packageData.ts
index a72bf3adc..14d7bc033 100644
--- a/src/next/blocks/packageData.ts
+++ b/src/data/packageData.ts
@@ -1,4 +1,10 @@
-import { sourcePackageJson } from "./sourcePackageJson.js";
+import { createRequire } from "node:module";
+
+const require = createRequire(import.meta.url);
+
+export const packageData =
+ // Importing from above src/ would expand the TS build rootDir
+ require("../../package.json") as typeof import("../../package.json");
export function getPackageDependencies(...names: string[]) {
return Object.fromEntries(
@@ -26,7 +32,7 @@ function getPackageInner(
key: "dependencies" | "devDependencies",
name: string,
) {
- const inner = sourcePackageJson[key];
+ const inner = packageData[key];
return inner[name as keyof typeof inner] as string | undefined;
}
diff --git a/src/docs.test.ts b/src/docs.test.ts
new file mode 100644
index 000000000..19fabd8fd
--- /dev/null
+++ b/src/docs.test.ts
@@ -0,0 +1,74 @@
+import { Block } from "create";
+import * as fs from "node:fs/promises";
+import * as prettier from "prettier";
+import { describe, expect, test } from "vitest";
+
+import { blocks, presets } from "./index.js";
+
+const actualLines = await createActualLines();
+const expectedLines = await createExpectedLines();
+
+describe("Blocks.md", () => {
+ for (const [i, line] of expectedLines.entries()) {
+ const name = line.split(" | ")[0].replace("| ", "").trim();
+ if (!name) {
+ continue;
+ }
+
+ // eslint-disable-next-line vitest/valid-title
+ test(name, () => {
+ const actualLine = actualLines.find((line) => line.includes(`| ${name}`));
+ const expectedLine = expectedLines[i];
+
+ expect(actualLine).toBe(expectedLine);
+ });
+ }
+});
+
+async function createActualLines() {
+ const actualFile = (await fs.readFile("docs/Blocks.md")).toString();
+
+ actualFile
+ .split("\n")
+ .filter((line) => !line.includes("----"))
+ .map((line) => line.toLowerCase());
+
+ return splitTable(actualFile);
+}
+
+async function createExpectedLines() {
+ const lines = [
+ "| Block | Exclusion Flag | Minimal | Common | Everything |",
+ "| ---------------------------- | ---------------------------------------- | ------- | ------ | ---------- |",
+ ];
+
+ for (const block of Object.values(blocks) as Block[]) {
+ // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
+ const name = block.about!.name!;
+
+ lines.push(
+ [
+ name,
+ `\`--exclude-${name.replaceAll(/\W+/g, "-").toLowerCase()}\``,
+ presets.minimal.blocks.includes(block) ? "✔️" : " ",
+ presets.common.blocks.includes(block) ? "✅" : " ",
+ presets.everything.blocks.includes(block) ? "💯" : " ",
+ "",
+ ].join(" | "),
+ );
+ }
+
+ const expectedTable = await prettier.format(lines.join("\n"), {
+ parser: "markdown",
+ useTabs: true,
+ });
+
+ return splitTable(expectedTable);
+}
+
+function splitTable(table: string) {
+ return table
+ .split("\n")
+ .filter((line) => !line.includes("----"))
+ .map((line) => line.toLowerCase());
+}
diff --git a/src/greet.test.ts b/src/greet.test.ts
deleted file mode 100644
index f729115fc..000000000
--- a/src/greet.test.ts
+++ /dev/null
@@ -1,44 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { greet } from "./greet.js";
-
-const message = "Yay, testing!";
-
-describe("greet", () => {
- it("logs to the console once when message is provided as a string", () => {
- const logger = vi.spyOn(console, "log").mockImplementation(() => undefined);
-
- greet(message);
-
- expect(logger).toHaveBeenCalledWith(message);
- expect(logger).toHaveBeenCalledTimes(1);
- });
-
- it("logs to the console once when message is provided as an object", () => {
- const logger = vi.spyOn(console, "log").mockImplementation(() => undefined);
-
- greet({ message });
-
- expect(logger).toHaveBeenCalledWith(message);
- expect(logger).toHaveBeenCalledTimes(1);
- });
-
- it("logs once when times is not provided in an object", () => {
- const logger = vi.fn();
-
- greet({ logger, message });
-
- expect(logger).toHaveBeenCalledWith(message);
- expect(logger).toHaveBeenCalledTimes(1);
- });
-
- it("logs a specified number of times when times is provided", () => {
- const logger = vi.fn();
- const times = 7;
-
- greet({ logger, message, times });
-
- expect(logger).toHaveBeenCalledWith(message);
- expect(logger).toHaveBeenCalledTimes(7);
- });
-});
diff --git a/src/greet.ts b/src/greet.ts
deleted file mode 100644
index a0d3b4c67..000000000
--- a/src/greet.ts
+++ /dev/null
@@ -1,13 +0,0 @@
-import { GreetOptions } from "./types.js";
-
-export function greet(options: GreetOptions | string) {
- const {
- logger = console.log.bind(console),
- message,
- times = 1,
- } = typeof options === "string" ? { message: options } : options;
-
- for (let i = 0; i < times; i += 1) {
- logger(message);
- }
-}
diff --git a/src/index.test.ts b/src/index.test.ts
new file mode 100644
index 000000000..01fc1a2f8
--- /dev/null
+++ b/src/index.test.ts
@@ -0,0 +1,16 @@
+import { produceBase, producePreset } from "create";
+import { describe, test } from "vitest";
+
+import { base, BaseOptions, presets } from "./index.js";
+
+describe("index", () => {
+ test("end-to-end", async () => {
+ const created = await producePreset(presets.everything, {
+ options: (await produceBase(base)) as BaseOptions,
+ });
+
+ const actual = await readFilesForCreate(".");
+
+ console.log(created);
+ });
+});
diff --git a/src/index.ts b/src/index.ts
index 39b62f22e..d2fbcef84 100644
--- a/src/index.ts
+++ b/src/index.ts
@@ -1,9 +1,4 @@
-export * from "./greet.js";
-
-// If you're using create-typescript-app as a template, ignore these.
-// They're plumbing for the create engine. :)
-export * from "./next/blocks/index.js";
-export * from "./next/presets/index.js";
-export { default } from "./next/template.js";
-
-export * from "./types.js";
+export * from "./base.js";
+export * from "./blocks/index.js";
+export * from "./presets/index.js";
+export { default } from "./template.js";
diff --git a/src/initialize/index.test.ts b/src/initialize/index.test.ts
deleted file mode 100644
index 703dcdd83..000000000
--- a/src/initialize/index.test.ts
+++ /dev/null
@@ -1,108 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { StatusCodes } from "../shared/codes.js";
-import { RunOrRestoreOptions } from "../shared/runOrRestore.js";
-import { initialize } from "./index.js";
-
-const mockOutro = vi.fn();
-
-vi.mock("../shared/cli/outro.js", () => ({
- get outro() {
- return mockOutro;
- },
-}));
-
-const mockEnsureGitRepository = vi.fn();
-
-vi.mock("../shared/ensureGitRepository.js", () => ({
- get ensureGitRepository() {
- return mockEnsureGitRepository;
- },
-}));
-
-const mockReadOptions = vi.fn();
-
-vi.mock("../shared/options/readOptions.js", () => ({
- get readOptions() {
- return mockReadOptions;
- },
-}));
-
-vi.mock("../shared/runOrRestore.js", () => ({
- async runOrRestore({ run }: RunOrRestoreOptions) {
- await run();
- return StatusCodes.Success;
- },
-}));
-
-const mockInitializeWithOptions = vi.fn();
-
-vi.mock("./initializeWithOptions.js", () => ({
- get initializeWithOptions() {
- return mockInitializeWithOptions;
- },
-}));
-
-const optionsBase = {
- repository: "TestRepository",
-};
-
-describe("initialize", () => {
- it("returns a cancellation code when readOptions cancels", async () => {
- mockReadOptions.mockResolvedValue({
- cancelled: true,
- options: optionsBase,
- });
-
- const result = await initialize([]);
-
- expect(result).toEqual({
- code: StatusCodes.Cancelled,
- options: optionsBase,
- });
- expect(mockEnsureGitRepository).not.toHaveBeenCalled();
- });
-
- it("runs initializeWithOptions when readOptions returns inputs", async () => {
- mockReadOptions.mockResolvedValue({
- cancelled: false,
- options: optionsBase,
- });
-
- const result = await initialize([]);
-
- expect(result).toEqual({
- code: StatusCodes.Success,
- options: optionsBase,
- });
- expect(mockEnsureGitRepository).toHaveBeenCalled();
- expect(mockOutro.mock.calls).toMatchInlineSnapshot(`
- [
- [
- [
- {
- "label": "You may consider committing these changes:",
- "lines": [
- "git add -A",
- "git commit -m "feat: initialized repo ✨",
- "git push",
- ],
- "variant": "code",
- },
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the GitHub apps:
- - Codecov (https://github.com/apps/codecov)
- - Renovate (https://github.com/apps/renovate)",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
- ],
- ],
- ]
- `);
- });
-});
diff --git a/src/initialize/index.ts b/src/initialize/index.ts
deleted file mode 100644
index a2b4caa3e..000000000
--- a/src/initialize/index.ts
+++ /dev/null
@@ -1,44 +0,0 @@
-import { outro } from "../shared/cli/outro.js";
-import { StatusCodes } from "../shared/codes.js";
-import { ensureGitRepository } from "../shared/ensureGitRepository.js";
-import { generateNextSteps } from "../shared/generateNextSteps.js";
-import { readOptions } from "../shared/options/readOptions.js";
-import { runOrRestore } from "../shared/runOrRestore.js";
-import { ModeRunner } from "../shared/types.js";
-import { initializeWithOptions } from "./initializeWithOptions.js";
-
-export const initialize: ModeRunner = async (args) => {
- const inputs = await readOptions(args, "initialize");
- if (inputs.cancelled) {
- return {
- code: StatusCodes.Cancelled,
- error: inputs.error,
- options: inputs.options,
- };
- }
-
- await ensureGitRepository();
-
- return {
- code: await runOrRestore({
- run: async () => {
- await initializeWithOptions(inputs);
-
- outro([
- {
- label: "You may consider committing these changes:",
- lines: [
- `git add -A`,
- `git commit -m "feat: initialized repo ✨`,
- `git push`,
- ],
- variant: "code",
- },
- ...generateNextSteps(inputs.options),
- ]);
- },
- skipRestore: inputs.options.skipRestore,
- }),
- options: inputs.options,
- };
-};
diff --git a/src/initialize/initializeWithOptions.test.ts b/src/initialize/initializeWithOptions.test.ts
deleted file mode 100644
index 3910a7c00..000000000
--- a/src/initialize/initializeWithOptions.test.ts
+++ /dev/null
@@ -1,188 +0,0 @@
-import { Octokit } from "octokit";
-import { describe, expect, it, vi } from "vitest";
-
-import { Options } from "../shared/types.js";
-import { initializeWithOptions } from "./initializeWithOptions.js";
-
-vi.mock("../shared/cli/spinners.js", () => ({
- async withSpinner(_label: string, task: () => Promise) {
- await task();
- },
- withSpinners: vi.fn(),
-}));
-
-const mockAddOwnerAsAllContributor = vi.fn();
-
-vi.mock("../steps/addOwnerAsAllContributor.js", () => ({
- get addOwnerAsAllContributor() {
- return mockAddOwnerAsAllContributor;
- },
-}));
-
-const mockClearChangelog = vi.fn();
-
-vi.mock("../steps/clearChangelog.js", () => ({
- get clearChangelog() {
- return mockClearChangelog;
- },
-}));
-
-const mockInitializeGitHubRepository = vi.fn();
-
-vi.mock("../steps/initializeGitHubRepository/index.js", () => ({
- get initializeGitHubRepository() {
- return mockInitializeGitHubRepository;
- },
-}));
-
-const mockRemoveSetupScripts = vi.fn();
-
-vi.mock("../steps/removeSetupScripts.js", () => ({
- get removeSetupScripts() {
- return mockRemoveSetupScripts;
- },
-}));
-
-const mockResetGitTags = vi.fn();
-
-vi.mock("../steps/resetGitTags.js", () => ({
- get resetGitTags() {
- return mockResetGitTags;
- },
-}));
-
-const mockRunCleanup = vi.fn();
-
-vi.mock("../steps/runCleanup.js", () => ({
- get runCleanup() {
- return mockRunCleanup;
- },
-}));
-
-const mockUninstallPackages = vi.fn();
-
-vi.mock("../steps/uninstallPackages.js", () => ({
- get uninstallPackages() {
- return mockUninstallPackages;
- },
-}));
-
-const optionsBase = {} as Options;
-
-describe("initializeWithOptions", () => {
- it("runs addOwnerAsAllContributor when excludeAllContributors is false", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: false,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockAddOwnerAsAllContributor).toHaveBeenCalledWith(
- undefined,
- options,
- );
- });
-
- it("does not run addOwnerAsAllContributor when excludeAllContributors is true", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: true,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockAddOwnerAsAllContributor).not.toHaveBeenCalled();
- });
-
- it("runs initializeGitHubRepository when github is truthy", async () => {
- const octokit = {} as Octokit;
-
- await initializeWithOptions({
- octokit,
- options: optionsBase,
- });
-
- expect(mockInitializeGitHubRepository).toHaveBeenCalledWith(
- octokit,
- optionsBase,
- );
- });
-
- it("does not run initializeGitHubRepository when github is falsy", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: true,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockInitializeGitHubRepository).not.toHaveBeenCalled();
- });
-
- it("runs removeSetupScripts when skipRemoval is false", async () => {
- const options = {
- ...optionsBase,
- skipRemoval: false,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockRemoveSetupScripts).toHaveBeenCalled();
- });
-
- it("does not run removeSetupScripts when skipRemoval is true", async () => {
- const options = {
- ...optionsBase,
- skipRemoval: true,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockRemoveSetupScripts).not.toHaveBeenCalled();
- });
-
- it("runs uninstallPackages when skipUninstall is false", async () => {
- const options = {
- ...optionsBase,
- offline: true,
- skipUninstall: false,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockUninstallPackages).toHaveBeenCalledWith(options.offline);
- });
-
- it("does not run uninstallPackages when skipUninstall is true", async () => {
- const options = {
- ...optionsBase,
- skipUninstall: true,
- };
-
- await initializeWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockUninstallPackages).not.toHaveBeenCalled();
- });
-});
diff --git a/src/initialize/initializeWithOptions.ts b/src/initialize/initializeWithOptions.ts
deleted file mode 100644
index f6e37cbf6..000000000
--- a/src/initialize/initializeWithOptions.ts
+++ /dev/null
@@ -1,68 +0,0 @@
-import { withSpinner, withSpinners } from "../shared/cli/spinners.js";
-import { createCleanupCommands } from "../shared/createCleanupCommands.js";
-import { OctokitAndOptions } from "../shared/options/readOptions.js";
-import { addOwnerAsAllContributor } from "../steps/addOwnerAsAllContributor.js";
-import { clearChangelog } from "../steps/clearChangelog.js";
-import { initializeGitHubRepository } from "../steps/initializeGitHubRepository/index.js";
-import { removeSetupScripts } from "../steps/removeSetupScripts.js";
-import { resetGitTags } from "../steps/resetGitTags.js";
-import { runCleanup } from "../steps/runCleanup.js";
-import { uninstallPackages } from "../steps/uninstallPackages.js";
-import { updateAllContributorsTable } from "../steps/updateAllContributorsTable.js";
-import { updateLocalFiles } from "../steps/updateLocalFiles.js";
-import { updateReadme } from "../steps/updateReadme.js";
-
-export async function initializeWithOptions({
- octokit,
- options,
-}: OctokitAndOptions) {
- await withSpinners("Initializing local files", [
- [
- "Updating local files",
- async () => {
- await updateLocalFiles(options);
- },
- ],
- [
- "Updating README.md",
- async () => {
- await updateReadme(options);
- },
- ],
- ["Clearing changelog", clearChangelog],
- [
- "Updating all-contributors table",
- async () => {
- await updateAllContributorsTable(options);
- },
- ],
- ["Resetting Git tags", resetGitTags],
- ]);
-
- if (!options.excludeAllContributors) {
- await withSpinner("Updating existing contributor details", async () => {
- await addOwnerAsAllContributor(octokit, options);
- });
- }
-
- if (octokit) {
- await withSpinner("Initializing GitHub repository", async () => {
- await initializeGitHubRepository(octokit, options);
- });
- }
-
- if (!options.skipRemoval) {
- await withSpinner("Removing setup scripts", removeSetupScripts);
- }
-
- if (!options.skipUninstall) {
- await withSpinner("Uninstalling initialization-only packages", async () =>
- uninstallPackages(options.offline),
- );
- }
-
- await runCleanup(
- createCleanupCommands(options.bin, "pnpm dedupe --offline"),
- options.mode,
- );
-}
diff --git a/src/migrate/index.test.ts b/src/migrate/index.test.ts
deleted file mode 100644
index c13782e55..000000000
--- a/src/migrate/index.test.ts
+++ /dev/null
@@ -1,107 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { StatusCodes } from "../shared/codes.js";
-import { RunOrRestoreOptions } from "../shared/runOrRestore.js";
-import { migrate } from "./index.js";
-
-const mockOutro = vi.fn();
-
-vi.mock("../shared/cli/outro.js", () => ({
- get outro() {
- return mockOutro;
- },
-}));
-
-const mockEnsureGitRepository = vi.fn();
-
-vi.mock("../shared/ensureGitRepository.js", () => ({
- get ensureGitRepository() {
- return mockEnsureGitRepository;
- },
-}));
-
-const mockReadOptions = vi.fn();
-
-vi.mock("../shared/options/readOptions.js", () => ({
- get readOptions() {
- return mockReadOptions;
- },
-}));
-
-vi.mock("../shared/runOrRestore.js", () => ({
- async runOrRestore({ run }: RunOrRestoreOptions) {
- await run();
- return StatusCodes.Success;
- },
-}));
-
-const mockMigrateWithOptions = vi.fn();
-
-vi.mock("./migrateWithOptions.js", () => ({
- get migrateWithOptions() {
- return mockMigrateWithOptions;
- },
-}));
-
-const optionsBase = {
- repository: "TestRepository",
-};
-
-describe("migrate", () => {
- it("returns a cancellation code when readOptions cancels", async () => {
- mockReadOptions.mockResolvedValue({
- cancelled: true,
- options: optionsBase,
- });
-
- const result = await migrate([]);
-
- expect(result).toEqual({
- code: StatusCodes.Cancelled,
- options: optionsBase,
- });
- });
-
- it("runs migrateWithOptions when readOptions returns inputs", async () => {
- mockReadOptions.mockResolvedValue({
- cancelled: false,
- options: optionsBase,
- });
-
- const result = await migrate([]);
-
- expect(result).toEqual({
- code: StatusCodes.Success,
- options: optionsBase,
- });
- expect(mockEnsureGitRepository).toHaveBeenCalled();
- expect(mockOutro.mock.calls).toMatchInlineSnapshot(`
- [
- [
- [
- {
- "label": "You may consider committing these changes:",
- "lines": [
- "git add -A",
- "git commit -m "migrated repo to create-typescript-app ✨",
- "git push",
- ],
- "variant": "code",
- },
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the GitHub apps:
- - Codecov (https://github.com/apps/codecov)
- - Renovate (https://github.com/apps/renovate)",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
- ],
- ],
- ]
- `);
- });
-});
diff --git a/src/migrate/index.ts b/src/migrate/index.ts
deleted file mode 100644
index 7f4fd02ed..000000000
--- a/src/migrate/index.ts
+++ /dev/null
@@ -1,44 +0,0 @@
-import { outro } from "../shared/cli/outro.js";
-import { StatusCodes } from "../shared/codes.js";
-import { ensureGitRepository } from "../shared/ensureGitRepository.js";
-import { generateNextSteps } from "../shared/generateNextSteps.js";
-import { readOptions } from "../shared/options/readOptions.js";
-import { runOrRestore } from "../shared/runOrRestore.js";
-import { ModeRunner } from "../shared/types.js";
-import { migrateWithOptions } from "./migrateWithOptions.js";
-
-export const migrate: ModeRunner = async (args) => {
- const inputs = await readOptions(args, "migrate");
- if (inputs.cancelled) {
- return {
- code: StatusCodes.Cancelled,
- error: inputs.error,
- options: inputs.options,
- };
- }
-
- await ensureGitRepository();
-
- return {
- code: await runOrRestore({
- run: async () => {
- await migrateWithOptions(inputs);
-
- outro([
- {
- label: "You may consider committing these changes:",
- lines: [
- `git add -A`,
- `git commit -m "migrated repo to create-typescript-app ✨`,
- `git push`,
- ],
- variant: "code",
- },
- ...generateNextSteps(inputs.options),
- ]);
- },
- skipRestore: inputs.options.skipRestore,
- }),
- options: inputs.options,
- };
-};
diff --git a/src/migrate/migrateWithOptions.test.ts b/src/migrate/migrateWithOptions.test.ts
deleted file mode 100644
index 5215b27a9..000000000
--- a/src/migrate/migrateWithOptions.test.ts
+++ /dev/null
@@ -1,199 +0,0 @@
-import { Octokit } from "octokit";
-import { describe, expect, it, vi } from "vitest";
-
-import { Options } from "../shared/types.js";
-import { migrateWithOptions } from "./migrateWithOptions.js";
-
-vi.mock("../shared/cli/spinners.js", () => ({
- async withSpinner(_label: string, task: () => Promise) {
- await task();
- },
- withSpinners: vi.fn(),
-}));
-
-const mockPopulateAllContributorsForRepository = vi.fn();
-
-vi.mock("populate-all-contributors-for-repository", () => ({
- get populateAllContributorsForRepository() {
- return mockPopulateAllContributorsForRepository;
- },
-}));
-
-const mockFinalizeDependencies = vi.fn();
-
-vi.mock("../steps/finalizeDependencies.js", () => ({
- get finalizeDependencies() {
- return mockFinalizeDependencies;
- },
-}));
-
-const mockInitializeGitHubRepository = vi.fn();
-
-vi.mock("../steps/initializeGitHubRepository/index.js", () => ({
- get initializeGitHubRepository() {
- return mockInitializeGitHubRepository;
- },
-}));
-
-const mockPopulateCSpellDictionary = vi.fn();
-
-vi.mock("../steps/populateCSpellDictionary.js", () => ({
- get populateCSpellDictionary() {
- return mockPopulateCSpellDictionary;
- },
-}));
-
-vi.mock("../steps/runCleanup.js", () => ({
- runCleanup: vi.fn(),
-}));
-
-vi.mock("../shared/cli/spinners.js", () => ({
- async withSpinner(_label: string, task: () => Promise) {
- await task();
- },
- withSpinners: vi.fn(),
-}));
-
-const mockGitHubAndOptions = vi.fn();
-
-vi.mock("../shared/options/readOptions.js", () => ({
- get GitHubAndOptions() {
- return mockGitHubAndOptions;
- },
-}));
-
-const optionsBase = {} as Options;
-
-describe("migrateWithOptions", () => {
- it("runs initializeGitHubRepository when octokit is truthy", async () => {
- const octokit = {} as Octokit;
-
- await migrateWithOptions({
- octokit,
- options: optionsBase,
- });
-
- expect(mockInitializeGitHubRepository).toHaveBeenCalledWith(
- octokit,
- optionsBase,
- );
- });
-
- it("does not run initializeGitHubRepository when github is falsy", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: true,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockInitializeGitHubRepository).not.toHaveBeenCalled();
- });
-
- it("does not run populateAllContributorsForRepository excludeAllContributors is true", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: true,
- skipAllContributorsApi: false,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockPopulateAllContributorsForRepository).not.toHaveBeenCalled();
- });
-
- it("does not run populateAllContributorsForRepository skipAllContributorsApi is true", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: false,
- skipAllContributorsApi: true,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockPopulateAllContributorsForRepository).not.toHaveBeenCalled();
- });
-
- it("runs populateAllContributorsForRepository excludeAllContributors and skipAllContributorsApi are false", async () => {
- const options = {
- ...optionsBase,
- excludeAllContributors: false,
- skipAllContributorsApi: false,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockPopulateAllContributorsForRepository).toHaveBeenCalledWith({
- owner: options.owner,
- repo: options.repository,
- });
- });
-
- it("runs finalizeDependencies when skipInstall is false", async () => {
- const options = {
- ...optionsBase,
- skipInstall: false,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockFinalizeDependencies).toHaveBeenCalledWith(options);
- });
-
- it("does not run finalizeDependencies when skipInstall is true", async () => {
- const options = {
- ...optionsBase,
- skipInstall: true,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockFinalizeDependencies).not.toHaveBeenCalled();
- });
-
- it("runs populateCSpellDictionary when excludeLintSpelling is false", async () => {
- const options = {
- ...optionsBase,
- excludeLintSpelling: false,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockPopulateCSpellDictionary).toHaveBeenCalled();
- });
-
- it("does not run populateCSpellDictionary when excludeLintSpelling is true", async () => {
- const options = {
- ...optionsBase,
- excludeLintSpelling: true,
- };
-
- await migrateWithOptions({
- octokit: undefined,
- options,
- });
-
- expect(mockPopulateCSpellDictionary).not.toHaveBeenCalled();
- });
-});
diff --git a/src/migrate/migrateWithOptions.ts b/src/migrate/migrateWithOptions.ts
deleted file mode 100644
index 00ea83c6f..000000000
--- a/src/migrate/migrateWithOptions.ts
+++ /dev/null
@@ -1,74 +0,0 @@
-import { populateAllContributorsForRepository } from "populate-all-contributors-for-repository";
-
-import { withSpinner, withSpinners } from "../shared/cli/spinners.js";
-import { createCleanupCommands } from "../shared/createCleanupCommands.js";
-import { OctokitAndOptions } from "../shared/options/readOptions.js";
-import { clearUnnecessaryFiles } from "../steps/clearUnnecessaryFiles.js";
-import { finalizeDependencies } from "../steps/finalizeDependencies.js";
-import { initializeGitHubRepository } from "../steps/initializeGitHubRepository/index.js";
-import { populateCSpellDictionary } from "../steps/populateCSpellDictionary.js";
-import { runCleanup } from "../steps/runCleanup.js";
-import { updateAllContributorsTable } from "../steps/updateAllContributorsTable.js";
-import { updateLocalFiles } from "../steps/updateLocalFiles.js";
-import { writeReadme } from "../steps/writeReadme/index.js";
-import { writeStructure } from "../steps/writing/writeStructure.js";
-
-export async function migrateWithOptions({
- octokit,
- options,
-}: OctokitAndOptions) {
- await withSpinners("Migrating repository structure", [
- ["Clearing unnecessary files", clearUnnecessaryFiles],
- [
- "Writing structure",
- async () => {
- await writeStructure(options);
- },
- ],
- [
- "Writing README.md",
- async () => {
- await writeReadme(options);
- },
- ],
- [
- "Updating local files",
- async () => {
- await updateLocalFiles(options);
- },
- ],
- [
- "Updating all-contributors table",
- async () => {
- await updateAllContributorsTable(options);
- },
- ],
- ]);
-
- if (octokit) {
- await withSpinner("Initializing GitHub repository", async () => {
- await initializeGitHubRepository(octokit, options);
- });
- }
-
- if (!options.excludeAllContributors && !options.skipAllContributorsApi) {
- await withSpinner("Detecting existing contributors", async () =>
- populateAllContributorsForRepository({
- owner: options.owner,
- repo: options.repository,
- }),
- );
- }
-
- if (!options.skipInstall) {
- await withSpinner("Installing packages", async () =>
- finalizeDependencies(options),
- );
- }
-
- if (!options.excludeLintSpelling) {
- await withSpinner("Populating CSpell dictionary", populateCSpellDictionary);
- }
-
- await runCleanup(createCleanupCommands(options.bin), options.mode);
-}
diff --git a/src/next/blocks/index.ts b/src/next/blocks/index.ts
deleted file mode 100644
index 26d79f044..000000000
--- a/src/next/blocks/index.ts
+++ /dev/null
@@ -1,42 +0,0 @@
-export * from "./blockAllContributors.js";
-export * from "./blockAreTheTypesWrong.js";
-export * from "./blockContributingDocs.js";
-export * from "./blockContributorCovenant.js";
-export * from "./blockCSpell.js";
-export * from "./blockDevelopmentDocs.js";
-export * from "./blockESLint.js";
-export * from "./blockESLintComments.js";
-export * from "./blockESLintJSDoc.js";
-export * from "./blockESLintJSONC.js";
-export * from "./blockESLintMarkdown.js";
-export * from "./blockESLintMoreStyling.js";
-export * from "./blockESLintNode.js";
-export * from "./blockESLintPackageJson.js";
-export * from "./blockESLintPerfectionist.js";
-export * from "./blockESLintRegexp.js";
-export * from "./blockESLintYML.js";
-export * from "./blockFunding.js";
-export * from "./blockGitHubActionsCI.js";
-export * from "./blockGitHubIssueTemplates.js";
-export * from "./blockGitHubPRTemplate.js";
-export * from "./blockGitignore.js";
-export * from "./blockKnip.js";
-export * from "./blockMarkdownlint.js";
-export * from "./blockMITLicense.js";
-export * from "./blockNvmrc.js";
-export * from "./blockPackageJson.js";
-export * from "./blockPnpmDedupe.js";
-export * from "./blockPRCompliance.js";
-export * from "./blockPrettier.js";
-export * from "./blockPrettierPluginCurly.js";
-export * from "./blockPrettierPluginPackageJson.js";
-export * from "./blockPrettierPluginSh.js";
-export * from "./blockREADME.js";
-export * from "./blockReleaseIt.js";
-export * from "./blockRenovate.js";
-export * from "./blockSecurityDocs.js";
-export * from "./blockTemplatedBy.js";
-export * from "./blockTSup.js";
-export * from "./blockTypeScript.js";
-export * from "./blockVitest.js";
-export * from "./blockVSCode.js";
diff --git a/src/next/blocks/sourcePackageJson.ts b/src/next/blocks/sourcePackageJson.ts
deleted file mode 100644
index 75867f05d..000000000
--- a/src/next/blocks/sourcePackageJson.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-import { createRequire } from "node:module";
-
-const require = createRequire(import.meta.url);
-
-export const sourcePackageJson =
- // Importing from above src/ would expand the TS build rootDir
- require("../../../package.json") as typeof import("../../../package.json");
diff --git a/src/next/presets/index.ts b/src/next/presets/index.ts
deleted file mode 100644
index d995968ea..000000000
--- a/src/next/presets/index.ts
+++ /dev/null
@@ -1,3 +0,0 @@
-export * from "./presetCommon.js";
-export * from "./presetEverything.js";
-export * from "./presetMinimal.js";
diff --git a/src/next/runCreateEnginePreset.ts b/src/next/runCreateEnginePreset.ts
deleted file mode 100644
index 30196a235..000000000
--- a/src/next/runCreateEnginePreset.ts
+++ /dev/null
@@ -1,23 +0,0 @@
-import { runPreset } from "create";
-
-import { Options } from "../shared/types.js";
-import { presetCommon } from "./presets/presetCommon.js";
-import { presetEverything } from "./presets/presetEverything.js";
-import { presetMinimal } from "./presets/presetMinimal.js";
-
-const presets = {
- common: presetCommon,
- everything: presetEverything,
- minimal: presetMinimal,
-};
-
-export async function runCreateEnginePreset(options: Options) {
- const preset =
- options.base && options.base !== "prompt" && presets[options.base];
-
- if (!preset) {
- throw new Error(`Cannot yet use create engine with base ${options.base}.`);
- }
-
- await runPreset(preset, { mode: "initialize", options });
-}
diff --git a/src/shared/options/createOptionDefaults/getUsageFromReadme.test.ts b/src/options/getUsageFromReadme.test.ts
similarity index 100%
rename from src/shared/options/createOptionDefaults/getUsageFromReadme.test.ts
rename to src/options/getUsageFromReadme.test.ts
diff --git a/src/shared/options/createOptionDefaults/getUsageFromReadme.ts b/src/options/getUsageFromReadme.ts
similarity index 100%
rename from src/shared/options/createOptionDefaults/getUsageFromReadme.ts
rename to src/options/getUsageFromReadme.ts
diff --git a/src/shared/options/createOptionDefaults/parsePackageAuthor.test.ts b/src/options/parsePackageAuthor.test.ts
similarity index 75%
rename from src/shared/options/createOptionDefaults/parsePackageAuthor.test.ts
rename to src/options/parsePackageAuthor.test.ts
index 078768d18..931a2df94 100644
--- a/src/shared/options/createOptionDefaults/parsePackageAuthor.test.ts
+++ b/src/options/parsePackageAuthor.test.ts
@@ -1,15 +1,7 @@
-import { describe, expect, it, vi } from "vitest";
+import { describe, expect, it } from "vitest";
import { parsePackageAuthor } from "./parsePackageAuthor.js";
-const mockReadFileSafe = vi.fn();
-
-vi.mock("../../shared/readFileSafe.js", () => ({
- get readFileSafe() {
- return mockReadFileSafe;
- },
-}));
-
describe("parsePackageAuthor", () => {
it.each([
[{}, {}],
diff --git a/src/shared/options/createOptionDefaults/parsePackageAuthor.ts b/src/options/parsePackageAuthor.ts
similarity index 91%
rename from src/shared/options/createOptionDefaults/parsePackageAuthor.ts
rename to src/options/parsePackageAuthor.ts
index 5fa833d19..b7ffd000a 100644
--- a/src/shared/options/createOptionDefaults/parsePackageAuthor.ts
+++ b/src/options/parsePackageAuthor.ts
@@ -1,6 +1,6 @@
import parse from "parse-author";
-import { PartialPackageData } from "../../types.js";
+import { PartialPackageData } from "../types.js";
export function parsePackageAuthor(packageData: PartialPackageData) {
let packageAuthor: string | undefined;
diff --git a/src/shared/options/createOptionDefaults/readDefaultsFromReadme.test.ts b/src/options/readDefaultsFromReadme.test.ts
similarity index 98%
rename from src/shared/options/createOptionDefaults/readDefaultsFromReadme.test.ts
rename to src/options/readDefaultsFromReadme.test.ts
index c9559ccc0..a994a776e 100644
--- a/src/shared/options/createOptionDefaults/readDefaultsFromReadme.test.ts
+++ b/src/options/readDefaultsFromReadme.test.ts
@@ -4,7 +4,7 @@ import { readDefaultsFromReadme } from "./readDefaultsFromReadme.js";
const mockReadLogoSizing = vi.fn().mockResolvedValue({});
-vi.mock("../readLogoSizing.js", () => ({
+vi.mock("./readLogoSizing.js", () => ({
get readLogoSizing() {
return mockReadLogoSizing;
},
@@ -107,7 +107,7 @@ describe("readDefaultsFromReadme", () => {
const logo = await readDefaultsFromReadme(
() =>
Promise.resolve(`
-`),
+`),
() => Promise.resolve(undefined),
).logo();
diff --git a/src/shared/options/createOptionDefaults/readDefaultsFromReadme.ts b/src/options/readDefaultsFromReadme.ts
similarity index 85%
rename from src/shared/options/createOptionDefaults/readDefaultsFromReadme.ts
rename to src/options/readDefaultsFromReadme.ts
index 41fb6ed82..f27492b21 100644
--- a/src/shared/options/createOptionDefaults/readDefaultsFromReadme.ts
+++ b/src/options/readDefaultsFromReadme.ts
@@ -1,8 +1,8 @@
import lazyValue from "lazy-value";
import { titleCase } from "title-case";
-import { readLogoSizing } from "../readLogoSizing.js";
import { getUsageFromReadme } from "./getUsageFromReadme.js";
+import { readLogoSizing } from "./readLogoSizing.js";
export function readDefaultsFromReadme(
readme: () => Promise,
@@ -22,14 +22,17 @@ export function readDefaultsFromReadme(
const src = /src\s*=(.+)['"/]>/
.exec(tag)?.[1]
- ?.replaceAll(/^['"]|['"]$/g, "");
+ ?.split(/\s*\w+=/)[0]
+ .replaceAll(/^['"]|['"]$/g, "");
if (!src) {
return undefined;
}
return {
- alt: /alt=['"](.+)['"]\s*src=/.exec(tag)?.[1] ?? "Project logo",
+ alt:
+ /alt=['"](.+)['"]\s*src=/.exec(tag)?.[1].split(/['"]?\s*\w+=/)[0] ??
+ "Project logo",
src,
...readLogoSizing(src),
};
diff --git a/src/next/readDescription.test.ts b/src/options/readDescription.test.ts
similarity index 79%
rename from src/next/readDescription.test.ts
rename to src/options/readDescription.test.ts
index 8f44f250a..0fb5da8e5 100644
--- a/src/next/readDescription.test.ts
+++ b/src/options/readDescription.test.ts
@@ -2,12 +2,12 @@ import { describe, expect, it, vi } from "vitest";
import { readDescription } from "./readDescription.js";
-const mockSourcePackageJson = vi.fn<() => string>();
+const mockPackageDataDescription = vi.fn<() => string>();
-vi.mock("./blocks/sourcePackageJson", () => ({
- sourcePackageJson: {
+vi.mock("../data/packageData.js", () => ({
+ packageData: {
get description() {
- return mockSourcePackageJson();
+ return mockPackageDataDescription();
},
},
}));
@@ -16,7 +16,7 @@ describe("readDescription", () => {
it("returns undefined when the description matches the current package.json description", async () => {
const existing = "Same description.";
- mockSourcePackageJson.mockReturnValueOnce(existing);
+ mockPackageDataDescription.mockReturnValueOnce(existing);
const description = await readDescription(
() => Promise.resolve({ description: existing }),
@@ -29,7 +29,7 @@ describe("readDescription", () => {
it("returns the updated description when neither description nor name match the current package.json", async () => {
const updated = "Updated description.";
- mockSourcePackageJson.mockReturnValueOnce("Existing description");
+ mockPackageDataDescription.mockReturnValueOnce("Existing description");
const description = await readDescription(
() => Promise.resolve({ description: updated }),
@@ -43,7 +43,7 @@ describe("readDescription", () => {
const plaintext = "Updated description.";
const encoded = "Updated description
.";
- mockSourcePackageJson.mockReturnValueOnce("Existing description");
+ mockPackageDataDescription.mockReturnValueOnce("Existing description");
const description = await readDescription(
() => Promise.resolve({ description: plaintext }),
@@ -57,7 +57,7 @@ describe("readDescription", () => {
const plaintext = "Updated description.";
const encoded = "Incorrect description
.";
- mockSourcePackageJson.mockReturnValueOnce("Existing description");
+ mockPackageDataDescription.mockReturnValueOnce("Existing description");
const description = await readDescription(
() => Promise.resolve({ description: plaintext }),
diff --git a/src/next/readDescription.ts b/src/options/readDescription.ts
similarity index 74%
rename from src/next/readDescription.ts
rename to src/options/readDescription.ts
index 5caa5c53e..ced7370e6 100644
--- a/src/next/readDescription.ts
+++ b/src/options/readDescription.ts
@@ -1,5 +1,5 @@
-import { PartialPackageData } from "../shared/types.js";
-import { sourcePackageJson } from "./blocks/sourcePackageJson.js";
+import { packageData } from "../data/packageData.js";
+import { PartialPackageData } from "../types.js";
import { readDescriptionFromReadme } from "./readDescriptionFromReadme.js";
export async function readDescription(
@@ -11,7 +11,7 @@ export async function readDescription(
return undefined;
}
- const { description: existing } = sourcePackageJson;
+ const { description: existing } = packageData;
const fromReadme = await readDescriptionFromReadme(getReadme);
if (fromReadme?.replaceAll(/<\s*(?:\/\s*)?\w+\s*>/gu, "") === inferred) {
diff --git a/src/next/readDescriptionFromReadme.test.ts b/src/options/readDescriptionFromReadme.test.ts
similarity index 100%
rename from src/next/readDescriptionFromReadme.test.ts
rename to src/options/readDescriptionFromReadme.test.ts
diff --git a/src/next/readDescriptionFromReadme.ts b/src/options/readDescriptionFromReadme.ts
similarity index 100%
rename from src/next/readDescriptionFromReadme.ts
rename to src/options/readDescriptionFromReadme.ts
diff --git a/src/next/readDocumentation.test.ts b/src/options/readDocumentation.test.ts
similarity index 100%
rename from src/next/readDocumentation.test.ts
rename to src/options/readDocumentation.test.ts
diff --git a/src/next/readDocumentation.ts b/src/options/readDocumentation.ts
similarity index 92%
rename from src/next/readDocumentation.ts
rename to src/options/readDocumentation.ts
index 5fd0d0be9..e2cac2c4d 100644
--- a/src/next/readDocumentation.ts
+++ b/src/options/readDocumentation.ts
@@ -1,7 +1,7 @@
import { TakeInput } from "create";
import { inputFromFile } from "input-from-file";
-import { swallowError } from "./utils/swallowError.js";
+import { swallowError } from "../utils/swallowError.js";
const knownHeadings = new Set([
"building",
diff --git a/src/shared/options/createOptionDefaults/readEmails.ts b/src/options/readEmails.ts
similarity index 91%
rename from src/shared/options/createOptionDefaults/readEmails.ts
rename to src/options/readEmails.ts
index d26579e98..0fa4af380 100644
--- a/src/shared/options/createOptionDefaults/readEmails.ts
+++ b/src/options/readEmails.ts
@@ -1,7 +1,7 @@
import { $ } from "execa";
import { UserInfo } from "npm-user";
-import { tryCatchAsync } from "../../tryCatchAsync.js";
+import { tryCatchAsync } from "../utils/tryCatchAsync.js";
import { readGitHubEmail } from "./readGitHubEmail.js";
export async function readEmails(
diff --git a/src/shared/readFileAsJson.test.ts b/src/options/readFileAsJson.test.ts
similarity index 100%
rename from src/shared/readFileAsJson.test.ts
rename to src/options/readFileAsJson.test.ts
diff --git a/src/shared/readFileAsJson.ts b/src/options/readFileAsJson.ts
similarity index 100%
rename from src/shared/readFileAsJson.ts
rename to src/options/readFileAsJson.ts
diff --git a/src/shared/readFileSafe.test.ts b/src/options/readFileSafe.test.ts
similarity index 100%
rename from src/shared/readFileSafe.test.ts
rename to src/options/readFileSafe.test.ts
diff --git a/src/shared/readFileSafe.ts b/src/options/readFileSafe.ts
similarity index 100%
rename from src/shared/readFileSafe.ts
rename to src/options/readFileSafe.ts
diff --git a/src/shared/options/createOptionDefaults/readFunding.ts b/src/options/readFunding.ts
similarity index 77%
rename from src/shared/options/createOptionDefaults/readFunding.ts
rename to src/options/readFunding.ts
index 7de124697..9d78ab08c 100644
--- a/src/shared/options/createOptionDefaults/readFunding.ts
+++ b/src/options/readFunding.ts
@@ -1,6 +1,6 @@
import fs from "node:fs/promises";
-import { tryCatchAsync } from "../../tryCatchAsync.js";
+import { tryCatchAsync } from "../utils/tryCatchAsync.js";
export async function readFunding() {
return await tryCatchAsync(async () =>
diff --git a/src/shared/options/createOptionDefaults/readGitHubEmail.test.ts b/src/options/readGitHubEmail.test.ts
similarity index 96%
rename from src/shared/options/createOptionDefaults/readGitHubEmail.test.ts
rename to src/options/readGitHubEmail.test.ts
index f472abeb1..a001e7667 100644
--- a/src/shared/options/createOptionDefaults/readGitHubEmail.test.ts
+++ b/src/options/readGitHubEmail.test.ts
@@ -4,7 +4,7 @@ import { readGitHubEmail } from "./readGitHubEmail.js";
const mockReadFileSafe = vi.fn();
-vi.mock("../../readFileSafe.js", () => ({
+vi.mock("./readFileSafe.js", () => ({
get readFileSafe() {
return mockReadFileSafe;
},
diff --git a/src/shared/options/createOptionDefaults/readGitHubEmail.ts b/src/options/readGitHubEmail.ts
similarity index 88%
rename from src/shared/options/createOptionDefaults/readGitHubEmail.ts
rename to src/options/readGitHubEmail.ts
index ea52beacb..884274e4a 100644
--- a/src/shared/options/createOptionDefaults/readGitHubEmail.ts
+++ b/src/options/readGitHubEmail.ts
@@ -1,4 +1,4 @@
-import { readFileSafe } from "../../readFileSafe.js";
+import { readFileSafe } from "./readFileSafe.js";
export async function readGitHubEmail() {
// The create-typescript-app template puts the GitHub email in the CoC.
diff --git a/src/shared/options/createOptionDefaults/readGuide.test.ts b/src/options/readGuide.test.ts
similarity index 96%
rename from src/shared/options/createOptionDefaults/readGuide.test.ts
rename to src/options/readGuide.test.ts
index 9a438babc..d0fcdb761 100644
--- a/src/shared/options/createOptionDefaults/readGuide.test.ts
+++ b/src/options/readGuide.test.ts
@@ -4,7 +4,7 @@ import { readGuide } from "./readGuide.js";
const mockReadFileSafe = vi.fn();
-vi.mock("../../readFileSafe.js", () => ({
+vi.mock("./readFileSafe.js", () => ({
get readFileSafe() {
return mockReadFileSafe;
},
diff --git a/src/shared/options/createOptionDefaults/readGuide.ts b/src/options/readGuide.ts
similarity index 84%
rename from src/shared/options/createOptionDefaults/readGuide.ts
rename to src/options/readGuide.ts
index dbbf6ad03..05c44904f 100644
--- a/src/shared/options/createOptionDefaults/readGuide.ts
+++ b/src/options/readGuide.ts
@@ -1,4 +1,4 @@
-import { readFileSafe } from "../../readFileSafe.js";
+import { readFileSafe } from "./readFileSafe.js";
export async function readGuide() {
const development = await readFileSafe(".github/DEVELOPMENT.md", "");
diff --git a/src/shared/options/readLogoSizing.test.ts b/src/options/readLogoSizing.test.ts
similarity index 100%
rename from src/shared/options/readLogoSizing.test.ts
rename to src/options/readLogoSizing.test.ts
diff --git a/src/shared/options/readLogoSizing.ts b/src/options/readLogoSizing.ts
similarity index 100%
rename from src/shared/options/readLogoSizing.ts
rename to src/options/readLogoSizing.ts
diff --git a/src/options/readPackageData.ts b/src/options/readPackageData.ts
new file mode 100644
index 000000000..359c70a31
--- /dev/null
+++ b/src/options/readPackageData.ts
@@ -0,0 +1,10 @@
+import { PartialPackageData } from "../types.js";
+import { readFileSafe } from "./readFileSafe.js";
+
+export async function readPackageData() {
+ return (
+ (JSON.parse(await readFileSafe("./package.json", "{}")) as
+ | PartialPackageData
+ | undefined) ?? {}
+ );
+}
diff --git a/src/next/presets/presetCommon.ts b/src/presets/common.ts
similarity index 90%
rename from src/next/presets/presetCommon.ts
rename to src/presets/common.ts
index 2f402f447..b33f738ac 100644
--- a/src/next/presets/presetCommon.ts
+++ b/src/presets/common.ts
@@ -2,7 +2,7 @@ import { base } from "../base.js";
import { blockAllContributors } from "../blocks/blockAllContributors.js";
import { blockReleaseIt } from "../blocks/blockReleaseIt.js";
import { blockVitest } from "../blocks/blockVitest.js";
-import { presetMinimal } from "./presetMinimal.js";
+import { presetMinimal } from "./minimal.js";
export const presetCommon = base.createPreset({
about: {
diff --git a/src/next/presets/presetEverything.ts b/src/presets/everything.ts
similarity index 97%
rename from src/next/presets/presetEverything.ts
rename to src/presets/everything.ts
index 557dce519..5cff5fc53 100644
--- a/src/next/presets/presetEverything.ts
+++ b/src/presets/everything.ts
@@ -21,7 +21,7 @@ import { blockPrettierPluginSh } from "../blocks/blockPrettierPluginSh.js";
import { blockRenovate } from "../blocks/blockRenovate.js";
import { blockSecurityDocs } from "../blocks/blockSecurityDocs.js";
import { blockVSCode } from "../blocks/blockVSCode.js";
-import { presetCommon } from "../presets/presetCommon.js";
+import { presetCommon } from "./common.js";
export const presetEverything = base.createPreset({
about: {
diff --git a/src/presets/index.ts b/src/presets/index.ts
new file mode 100644
index 000000000..5b140abdb
--- /dev/null
+++ b/src/presets/index.ts
@@ -0,0 +1,13 @@
+import { presetCommon } from "./common.js";
+import { presetEverything } from "./everything.js";
+import { presetMinimal } from "./minimal.js";
+
+export const presets = {
+ common: presetCommon,
+ everything: presetEverything,
+ minimal: presetMinimal,
+};
+
+export { presetCommon } from "./common.js";
+export { presetEverything } from "./everything.js";
+export { presetMinimal } from "./minimal.js";
diff --git a/src/next/presets/presetMinimal.ts b/src/presets/minimal.ts
similarity index 95%
rename from src/next/presets/presetMinimal.ts
rename to src/presets/minimal.ts
index 587f9ea0b..0ff44447f 100644
--- a/src/next/presets/presetMinimal.ts
+++ b/src/presets/minimal.ts
@@ -18,7 +18,7 @@ import { blockRepositoryBranchRuleset } from "../blocks/blockRepositoryBranchRul
import { blockRepositoryLabels } from "../blocks/blockRepositoryLabels.js";
import { blockRepositorySecrets } from "../blocks/blockRepositorySecrets.js";
import { blockRepositorySettings } from "../blocks/blockRepositorySettings.js";
-import { blockTemplatedBy } from "../blocks/blockTemplatedBy.js";
+import { blockTemplatedWith } from "../blocks/blockTemplatedWith.js";
import { blockTSup } from "../blocks/blockTSup.js";
import { blockTypeScript } from "../blocks/blockTypeScript.js";
@@ -48,7 +48,7 @@ export const presetMinimal = base.createPreset({
blockRepositoryLabels,
blockRepositorySecrets,
blockRepositorySettings,
- blockTemplatedBy,
+ blockTemplatedWith,
blockTSup,
blockTypeScript,
],
diff --git a/src/shared/__snapshots__/generateNextSteps.test.ts.snap b/src/shared/__snapshots__/generateNextSteps.test.ts.snap
deleted file mode 100644
index df9a0ec6e..000000000
--- a/src/shared/__snapshots__/generateNextSteps.test.ts.snap
+++ /dev/null
@@ -1,183 +0,0 @@
-// Vitest Snapshot v1, https://vitest.dev/guide/snapshot.html
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":false,"excludeRenovate":false,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the GitHub apps:
- - Codecov (https://github.com/apps/codecov)
- - Renovate (https://github.com/apps/renovate)",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":false,"excludeRenovate":false,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the Renovate GitHub app (https://github.com/apps/renovate).",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":false,"excludeRenovate":true,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the Codecov GitHub app (https://github.com/apps/codecov).",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":false,"excludeRenovate":true,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":true,"excludeRenovate":false,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the GitHub apps:
- - Codecov (https://github.com/apps/codecov)
- - Renovate (https://github.com/apps/renovate)",
- "- populate the ACCESS_TOKEN secret (a GitHub PAT with repo and workflow permissions).",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":true,"excludeRenovate":false,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the Renovate GitHub app (https://github.com/apps/renovate).",
- "- populate the ACCESS_TOKEN secret (a GitHub PAT with repo and workflow permissions).",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":true,"excludeRenovate":true,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the Codecov GitHub app (https://github.com/apps/codecov).",
- "- populate the ACCESS_TOKEN secret (a GitHub PAT with repo and workflow permissions).",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":false,"excludeReleases":true,"excludeRenovate":true,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to populate the ACCESS_TOKEN secret (a GitHub PAT with repo and workflow permissions).",
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":false,"excludeRenovate":false,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the GitHub apps:
- - Codecov (https://github.com/apps/codecov)
- - Renovate (https://github.com/apps/renovate)",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":false,"excludeRenovate":false,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the Renovate GitHub app (https://github.com/apps/renovate).",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":false,"excludeRenovate":true,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to:",
- "lines": [
- "- enable the Codecov GitHub app (https://github.com/apps/codecov).",
- "- populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- ],
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":false,"excludeRenovate":true,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to populate the secrets:
- - ACCESS_TOKEN (a GitHub PAT with repo and workflow permissions)
- - NPM_TOKEN (an npm access token with automation permissions)",
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":true,"excludeRenovate":false,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to enable the GitHub apps:
- - Codecov (https://github.com/apps/codecov)
- - Renovate (https://github.com/apps/renovate)",
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":true,"excludeRenovate":false,"excludeTests":true} 1`] = `
-[
- {
- "label": "Be sure to enable the Renovate GitHub app (https://github.com/apps/renovate).",
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":true,"excludeRenovate":true,"excludeTests":false} 1`] = `
-[
- {
- "label": "Be sure to enable the Codecov GitHub app (https://github.com/apps/codecov).",
- },
-]
-`;
-
-exports[`generateNextSteps > {"excludeAllContributors":true,"excludeReleases":true,"excludeRenovate":true,"excludeTests":true} 1`] = `[]`;
diff --git a/src/shared/cli/lines.ts b/src/shared/cli/lines.ts
deleted file mode 100644
index d1075ad4c..000000000
--- a/src/shared/cli/lines.ts
+++ /dev/null
@@ -1,14 +0,0 @@
-import chalk from "chalk";
-
-export function logLine(line?: string) {
- console.log(makeLine(line));
-}
-
-export function logNewSection(line: string) {
- logLine();
- console.log(`◇ ${line}`);
-}
-
-export function makeLine(line: string | undefined) {
- return [chalk.gray("│"), line].filter(Boolean).join(" ");
-}
diff --git a/src/shared/cli/lowerFirst.ts b/src/shared/cli/lowerFirst.ts
deleted file mode 100644
index 3b96e98f2..000000000
--- a/src/shared/cli/lowerFirst.ts
+++ /dev/null
@@ -1,3 +0,0 @@
-export function lowerFirst(text: string) {
- return text[0].toLowerCase() + text.slice(1);
-}
diff --git a/src/shared/cli/outro.test.ts b/src/shared/cli/outro.test.ts
deleted file mode 100644
index 9f228a5ab..000000000
--- a/src/shared/cli/outro.test.ts
+++ /dev/null
@@ -1,61 +0,0 @@
-import chalk from "chalk";
-import { beforeEach, describe, expect, it, MockInstance, vi } from "vitest";
-
-import { outro } from "./outro.js";
-
-const mockOutro = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- get outro() {
- return mockOutro;
- },
-}));
-
-let mockConsoleLog: MockInstance;
-
-describe("outro", () => {
- beforeEach(() => {
- mockConsoleLog = vi
- .spyOn(console, "log")
- .mockImplementation(() => undefined);
- });
-
- it("logs only basic statements when no lines are provided", () => {
- outro([{ label: "Abc 123" }]);
-
- expect(mockConsoleLog.mock.calls).toEqual([
- [chalk.blue("Abc 123")],
- [],
- [chalk.greenBright(`See ya! 👋`)],
- [],
- ]);
- });
-
- it("also logs lines when provided", () => {
- outro([{ label: "Abc 123", lines: ["one", "two"] }]);
-
- expect(mockConsoleLog.mock.calls).toEqual([
- [chalk.blue("Abc 123")],
- [],
- ["one"],
- ["two"],
- [],
- [chalk.greenBright(`See ya! 👋`)],
- [],
- ]);
- });
-
- it("logs lines as code when variant is specified", () => {
- outro([{ label: "Abc 123", lines: ["one", "two"], variant: "code" }]);
-
- expect(mockConsoleLog.mock.calls).toEqual([
- [chalk.blue("Abc 123")],
- [],
- [chalk.gray("one")],
- [chalk.gray("two")],
- [],
- [chalk.greenBright(`See ya! 👋`)],
- [],
- ]);
- });
-});
diff --git a/src/shared/cli/outro.ts b/src/shared/cli/outro.ts
deleted file mode 100644
index 444891de1..000000000
--- a/src/shared/cli/outro.ts
+++ /dev/null
@@ -1,28 +0,0 @@
-import * as prompts from "@clack/prompts";
-import chalk from "chalk";
-
-export interface OutroGroup {
- label: string;
- lines?: string[];
- variant?: "code";
-}
-
-export function outro(groups: OutroGroup[]) {
- prompts.outro(chalk.blue(`Great, looks like the script finished! 🎉`));
-
- for (const { label, lines, variant } of groups) {
- console.log(chalk.blue(label));
- console.log();
-
- if (lines) {
- for (const line of lines) {
- console.log(variant === "code" ? chalk.gray(line) : line);
- }
-
- console.log();
- }
- }
-
- console.log(chalk.greenBright(`See ya! 👋`));
- console.log();
-}
diff --git a/src/shared/cli/spinners.ts b/src/shared/cli/spinners.ts
deleted file mode 100644
index 00a271f86..000000000
--- a/src/shared/cli/spinners.ts
+++ /dev/null
@@ -1,72 +0,0 @@
-import * as prompts from "@clack/prompts";
-import chalk from "chalk";
-import readline from "readline";
-
-import { logLine, logNewSection, makeLine } from "./lines.js";
-import { lowerFirst } from "./lowerFirst.js";
-import { startLineWithDots } from "./startLineWithDots.js";
-
-const s = prompts.spinner();
-
-export type LabeledSpinnerTask = [string, SpinnerTask];
-
-export type SpinnerTask = () => Promise;
-
-export async function withSpinner(
- label: string,
- task: SpinnerTask,
-) {
- s.start(`${label}...`);
-
- try {
- const result = await task();
-
- s.stop(chalk.green(`✅ Passed ${lowerFirst(label)}.`));
-
- return result;
- } catch (error) {
- s.stop(chalk.red(`❌ Error ${lowerFirst(label)}.`));
-
- throw new Error(`Failed ${lowerFirst(label)}`, { cause: error });
- }
-}
-
-export async function withSpinners(
- label: string,
- tasks: LabeledSpinnerTask[],
-) {
- logNewSection(`${label}...`);
-
- let currentLabel!: string;
- let lastLogged!: string;
-
- for (const [label, run] of tasks) {
- currentLabel = label;
-
- const line = makeLine(chalk.gray(` - ${label}`));
- const stopWriting = startLineWithDots(line);
-
- try {
- await run();
- } catch (error) {
- const descriptor = `${lowerFirst(label)} > ${lowerFirst(currentLabel)}`;
-
- logLine(chalk.red(`❌ Error ${descriptor}.`));
-
- throw new Error(`Failed ${descriptor}`, { cause: error });
- } finally {
- const lineLength = stopWriting();
- readline.clearLine(process.stdout, -1);
- readline.moveCursor(process.stdout, -lineLength, 0);
- }
-
- lastLogged = chalk.gray(`${line} ✔️\n`);
-
- process.stdout.write(lastLogged);
- }
-
- readline.moveCursor(process.stdout, -lastLogged.length, -tasks.length - 2);
- readline.clearScreenDown(process.stdout);
-
- logNewSection(chalk.green(`✅ Passed ${lowerFirst(label)}.`));
-}
diff --git a/src/shared/cli/startLineWithDots.ts b/src/shared/cli/startLineWithDots.ts
deleted file mode 100644
index e80378160..000000000
--- a/src/shared/cli/startLineWithDots.ts
+++ /dev/null
@@ -1,38 +0,0 @@
-import readline from "readline";
-
-export function startLineWithDots(line: string) {
- const timer = [setTimeout(tick, 500)];
- let dots = 0;
- let lastLogged!: string;
-
- function clearLine() {
- readline.clearLine(process.stdout, -1);
- readline.moveCursor(process.stdout, -lastLogged.length, 0);
- }
-
- function writeLine() {
- dots = (dots + 1) % 4;
-
- const toLog = `${line}${".".repeat(dots)}`;
-
- process.stdout.write(toLog);
-
- lastLogged = toLog;
-
- return toLog;
- }
-
- function tick() {
- clearLine();
- writeLine();
- timer[0] = setTimeout(tick, 500);
- }
-
- writeLine();
-
- return () => {
- clearLine();
- clearInterval(timer[0]);
- return lastLogged.length;
- };
-}
diff --git a/src/shared/codes.ts b/src/shared/codes.ts
deleted file mode 100644
index 89b39571e..000000000
--- a/src/shared/codes.ts
+++ /dev/null
@@ -1,7 +0,0 @@
-export const StatusCodes = {
- Cancelled: 2,
- Failure: 1,
- Success: 0,
-} as const;
-
-export type StatusCode = (typeof StatusCodes)[keyof typeof StatusCodes];
diff --git a/src/shared/createCleanupCommands.test.ts b/src/shared/createCleanupCommands.test.ts
deleted file mode 100644
index 8c2117f7e..000000000
--- a/src/shared/createCleanupCommands.test.ts
+++ /dev/null
@@ -1,32 +0,0 @@
-import { describe, expect, it } from "vitest";
-
-import { createCleanupCommands } from "./createCleanupCommands.js";
-
-describe("createCleanupCommands", () => {
- it("only lints and formats when bin is not enabled and there are no prepended commands", () => {
- const actual = createCleanupCommands(undefined);
-
- expect(actual).toEqual(["pnpm lint --fix", "pnpm format --write"]);
- });
-
- it("runs prepended commands before it lints and formats when bin is not enabled", () => {
- const actual = createCleanupCommands(undefined, "prepended");
-
- expect(actual).toEqual([
- "prepended",
- "pnpm lint --fix",
- "pnpm format --write",
- ]);
- });
-
- it("runs prepended commands before it builds, lints, and formats when bin is not enabled", () => {
- const actual = createCleanupCommands("bin/index.js", "prepended");
-
- expect(actual).toEqual([
- "prepended",
- "pnpm build",
- "pnpm lint --fix",
- "pnpm format --write",
- ]);
- });
-});
diff --git a/src/shared/createCleanupCommands.ts b/src/shared/createCleanupCommands.ts
deleted file mode 100644
index d48c6477f..000000000
--- a/src/shared/createCleanupCommands.ts
+++ /dev/null
@@ -1,12 +0,0 @@
-export function createCleanupCommands(
- bin: string | undefined,
- ...prependedCommands: string[]
-) {
- return [
- ...prependedCommands,
- // n/no-missing-import rightfully reports on a missing the bin .js file
- ...(bin ? ["pnpm build"] : []),
- "pnpm lint --fix",
- "pnpm format --write",
- ];
-}
diff --git a/src/shared/doesRepositoryExist.test.ts b/src/shared/doesRepositoryExist.test.ts
deleted file mode 100644
index 5932d647e..000000000
--- a/src/shared/doesRepositoryExist.test.ts
+++ /dev/null
@@ -1,47 +0,0 @@
-import { Octokit } from "octokit";
-import { describe, expect, it, MockInstance, vi } from "vitest";
-
-import { doesRepositoryExist } from "./doesRepositoryExist.js";
-
-const createMockOctokit = (
- repos: Partial>,
-) =>
- ({
- rest: {
- repos,
- },
- }) as unknown as Octokit;
-
-const owner = "StubOwner";
-const repository = "stub-repository";
-
-describe("doesRepositoryExist", () => {
- it("returns true when the octokit GET resolves", async () => {
- const octokit = createMockOctokit({ get: vi.fn().mockResolvedValue({}) });
-
- const actual = await doesRepositoryExist(octokit, { owner, repository });
-
- expect(actual).toBe(true);
- });
-
- it("returns false when the octokit GET rejects with a 404", async () => {
- const octokit = createMockOctokit({
- get: vi.fn().mockRejectedValue({ status: 404 }),
- });
-
- const actual = await doesRepositoryExist(octokit, { owner, repository });
-
- expect(actual).toBe(false);
- });
-
- it("throws the error when awaiting the octokit GET throws a non-404 error", async () => {
- const error = new Error("Oh no!");
- const octokit = createMockOctokit({
- get: vi.fn().mockRejectedValue(error),
- });
-
- await expect(
- async () => await doesRepositoryExist(octokit, { owner, repository }),
- ).rejects.toEqual(error);
- });
-});
diff --git a/src/shared/doesRepositoryExist.ts b/src/shared/doesRepositoryExist.ts
deleted file mode 100644
index 8868a2f94..000000000
--- a/src/shared/doesRepositoryExist.ts
+++ /dev/null
@@ -1,27 +0,0 @@
-import { Octokit, RequestError } from "octokit";
-
-export interface DoesRepositoryExistOptions {
- owner: string;
- repository: string;
-}
-
-export async function doesRepositoryExist(
- octokit: Octokit,
- options: DoesRepositoryExistOptions,
-) {
- // Because the Octokit SDK throws on 404s (😡),
- // we try/catch to check whether the repo exists.
- try {
- await octokit.rest.repos.get({
- owner: options.owner,
- repo: options.repository,
- });
- return true;
- } catch (error) {
- if ((error as RequestError).status !== 404) {
- throw error;
- }
-
- return false;
- }
-}
diff --git a/src/shared/ensureGitRepository.test.ts b/src/shared/ensureGitRepository.test.ts
deleted file mode 100644
index a9b4b7ec6..000000000
--- a/src/shared/ensureGitRepository.test.ts
+++ /dev/null
@@ -1,29 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { ensureGitRepository } from "./ensureGitRepository.js";
-
-const mock$ = vi.fn();
-
-vi.mock("execa", () => ({
- get $() {
- return mock$;
- },
-}));
-
-describe("ensureGitRepository", () => {
- it("does not run git init when git status succeeds", async () => {
- mock$.mockResolvedValue(0);
-
- await ensureGitRepository();
-
- expect(mock$).toHaveBeenCalledTimes(1);
- });
-
- it("runs git init when git status fails", async () => {
- mock$.mockRejectedValueOnce(1);
-
- await ensureGitRepository();
-
- expect(mock$).toHaveBeenCalledWith(["git init"]);
- });
-});
diff --git a/src/shared/ensureGitRepository.ts b/src/shared/ensureGitRepository.ts
deleted file mode 100644
index 97ad7074d..000000000
--- a/src/shared/ensureGitRepository.ts
+++ /dev/null
@@ -1,9 +0,0 @@
-import { $ } from "execa";
-
-export async function ensureGitRepository() {
- try {
- await $`git status`;
- } catch {
- await $`git init`;
- }
-}
diff --git a/src/shared/generateNextSteps.test.ts b/src/shared/generateNextSteps.test.ts
deleted file mode 100644
index 88c83812a..000000000
--- a/src/shared/generateNextSteps.test.ts
+++ /dev/null
@@ -1,50 +0,0 @@
-import { describe, expect, test } from "vitest";
-
-import { generateNextSteps } from "./generateNextSteps.js";
-import { Options } from "./types.js";
-
-const options = {
- access: "public",
- base: "everything",
- description: "Test description.",
- directory: ".",
- email: {
- github: "github@email.com",
- npm: "npm@email.com",
- },
- mode: "create",
- owner: "TestOwner",
- repository: "test-repository",
- title: "Test Title",
-} satisfies Options;
-
-describe("generateNextSteps", () => {
- for (const excludeAllContributors of [false, true]) {
- for (const excludeReleases of [false, true]) {
- for (const excludeRenovate of [false, true]) {
- for (const excludeTests of [false, true]) {
- test(
- // eslint-disable-next-line vitest/valid-title
- JSON.stringify({
- excludeAllContributors,
- excludeReleases,
- excludeRenovate,
- excludeTests,
- }),
- () => {
- expect(
- generateNextSteps({
- ...options,
- excludeAllContributors,
- excludeReleases,
- excludeRenovate,
- excludeTests,
- }),
- ).toMatchSnapshot();
- },
- );
- }
- }
- }
- }
-});
diff --git a/src/shared/generateNextSteps.ts b/src/shared/generateNextSteps.ts
deleted file mode 100644
index c8eb8b998..000000000
--- a/src/shared/generateNextSteps.ts
+++ /dev/null
@@ -1,69 +0,0 @@
-import { OutroGroup } from "./cli/outro.js";
-import { Options } from "./types.js";
-
-export function generateNextSteps(options: Options): OutroGroup[] {
- const lines = [
- joinedMessage(
- "enable the",
- [
- options.excludeTests || ["Codecov", "https://github.com/apps/codecov"],
- options.excludeRenovate || [
- "Renovate",
- "https://github.com/apps/renovate",
- ],
- ],
- "GitHub app",
- ),
- joinedMessage(
- "populate the",
- [
- (options.excludeAllContributors && options.excludeReleases) || [
- "ACCESS_TOKEN",
- "a GitHub PAT with repo and workflow permissions",
- ],
- options.excludeReleases || [
- "NPM_TOKEN",
- "an npm access token with automation permissions",
- ],
- ],
- "secret",
- ),
- ].filter((line): line is string => !!line);
-
- switch (lines.length) {
- case 0:
- return [];
- case 1:
- return [{ label: `Be sure to ${lines[0]}` }];
- default:
- return [
- {
- label: "Be sure to:",
- lines: lines.map((line) => `- ${line}`),
- },
- ];
- }
-}
-
-const itemBreak = "\n - ";
-
-function joinedMessage(
- prefix: string,
- entries: ([string, string] | true)[],
- suffix: string,
-) {
- const realEntries = entries.filter(
- (entry): entry is [string, string] => entry !== true,
- );
-
- switch (realEntries.length) {
- case 0:
- return undefined;
- case 1:
- return `${prefix} ${realEntries[0][0]} ${suffix} (${realEntries[0][1]}).`;
- default:
- return `${prefix} ${suffix}s: ${itemBreak}${realEntries
- .map((entry) => `${entry[0]} (${entry[1]})`)
- .join(itemBreak)}`;
- }
-}
diff --git a/src/shared/getGitHubUserAsAllContributor.test.ts b/src/shared/getGitHubUserAsAllContributor.test.ts
deleted file mode 100644
index 25fe80e68..000000000
--- a/src/shared/getGitHubUserAsAllContributor.test.ts
+++ /dev/null
@@ -1,103 +0,0 @@
-import chalk from "chalk";
-import { Octokit } from "octokit";
-import { beforeEach, describe, expect, it, MockInstance, vi } from "vitest";
-
-import { getGitHubUserAsAllContributor } from "./getGitHubUserAsAllContributor.js";
-
-const mock$ = vi.fn();
-
-vi.mock("execa", () => ({
- get $() {
- return mock$;
- },
-}));
-
-let mockConsoleWarn: MockInstance;
-
-const createMockOctokit = (getAuthenticated: MockInstance = vi.fn()) =>
- ({
- rest: { users: { getAuthenticated } },
- }) as unknown as Octokit;
-
-const owner = "TestOwner";
-
-describe("getGitHubUserAsAllContributor", () => {
- beforeEach(() => {
- mockConsoleWarn = vi
- .spyOn(console, "warn")
- .mockImplementation(() => undefined);
- });
-
- it("defaults to owner with a log when options.offline is true", async () => {
- const octokit = createMockOctokit();
- const actual = await getGitHubUserAsAllContributor(octokit, {
- offline: true,
- owner,
- });
-
- expect(actual).toEqual(owner);
- expect(mockConsoleWarn).toHaveBeenCalledWith(
- chalk.gray(
- `Skipping populating all-contributors contributions for TestOwner because in --offline mode.`,
- ),
- );
- });
-
- it("defaults to owner without a log when octokit is undefined", async () => {
- const actual = await getGitHubUserAsAllContributor(undefined, { owner });
-
- expect(actual).toEqual(owner);
- expect(mockConsoleWarn).not.toHaveBeenCalled();
- });
-
- it("uses the user from gh api user when it succeeds", async () => {
- const login = "gh-api-user";
- const octokit = createMockOctokit(
- vi.fn().mockResolvedValue({ data: { login } }),
- );
-
- await getGitHubUserAsAllContributor(octokit, { owner });
-
- expect(mockConsoleWarn).not.toHaveBeenCalled();
- expect(mock$.mock.calls).toMatchInlineSnapshot(`
- [
- [
- [
- "pnpx all-contributors-cli@6.25 add ",
- " ",
- "",
- ],
- "gh-api-user",
- "code,content,doc,ideas,infra,maintenance,projectManagement,tool",
- ],
- ]
- `);
- });
-
- it("defaults the user to the owner when gh api user fails", async () => {
- const octokit = createMockOctokit(
- vi.fn().mockRejectedValue(new Error("Oh no!")),
- );
-
- await getGitHubUserAsAllContributor(octokit, { owner });
-
- expect(mockConsoleWarn).toHaveBeenCalledWith(
- chalk.gray(
- `Couldn't authenticate GitHub user, falling back to the provided owner name '${owner}'.`,
- ),
- );
- expect(mock$.mock.calls).toMatchInlineSnapshot(`
- [
- [
- [
- "pnpx all-contributors-cli@6.25 add ",
- " ",
- "",
- ],
- "TestOwner",
- "code,content,doc,ideas,infra,maintenance,projectManagement,tool",
- ],
- ]
- `);
- });
-});
diff --git a/src/shared/getGitHubUserAsAllContributor.ts b/src/shared/getGitHubUserAsAllContributor.ts
deleted file mode 100644
index 95948e26f..000000000
--- a/src/shared/getGitHubUserAsAllContributor.ts
+++ /dev/null
@@ -1,50 +0,0 @@
-import chalk from "chalk";
-import { $ } from "execa";
-import { Octokit } from "octokit";
-
-import { Options } from "./types.js";
-
-export async function getGitHubUserAsAllContributor(
- octokit: Octokit | undefined,
- options: Pick,
-) {
- if (options.offline) {
- console.warn(
- chalk.gray(
- `Skipping populating all-contributors contributions for ${options.owner} because in --offline mode.`,
- ),
- );
- return options.owner;
- }
-
- let user: string;
-
- if (octokit) {
- try {
- user = (await octokit.rest.users.getAuthenticated()).data.login;
- } catch {
- console.warn(
- chalk.gray(
- `Couldn't authenticate GitHub user, falling back to the provided owner name '${options.owner}'.`,
- ),
- );
- user = options.owner;
- }
- } else {
- user = options.owner;
- }
-
- const contributions = [
- "code",
- "content",
- "doc",
- "ideas",
- "infra",
- "maintenance",
- "projectManagement",
- "tool",
- ].join(",");
- await $`pnpx all-contributors-cli@6.25 add ${user} ${contributions}`;
-
- return user;
-}
diff --git a/src/shared/isUsingCreateEngine.ts b/src/shared/isUsingCreateEngine.ts
deleted file mode 100644
index da251858c..000000000
--- a/src/shared/isUsingCreateEngine.ts
+++ /dev/null
@@ -1,5 +0,0 @@
-export function isUsingCreateEngine() {
- return (
- !!process.env.CTA_CREATE_ENGINE && Boolean(process.env.CTA_CREATE_ENGINE)
- );
-}
diff --git a/src/shared/options/args.ts b/src/shared/options/args.ts
deleted file mode 100644
index 670197edb..000000000
--- a/src/shared/options/args.ts
+++ /dev/null
@@ -1,332 +0,0 @@
-import chalk from "chalk";
-
-export const allArgOptions = {
- access: {
- description: `(${chalk.cyanBright(
- '"public" | "restricted"',
- )}): Which ${chalk.cyanBright("npm publish --access")} to
- release npm packages with (by default, "public")`,
- docsSection: "optional",
- type: "string",
- },
- author: {
- description: `Username on npm to publish packages under (by
- default, an existing npm author, or the currently logged in npm user, or
- ${chalk.cyanBright("owner.toLowerCase()")})`,
- docsSection: "optional",
- type: "string",
- },
- auto: {
- description: `Whether to infer all options from files on disk.`,
- docsSection: "optional",
- type: "boolean",
- },
- base: {
- description: `Whether to scaffold the repository with:
- • everything: that comes with the template (${chalk.cyanBright.bold(
- "recommended",
- )})
- • common: additions to the minimal starters such as releases and tests
- • minimal: amounts of tooling, essentially opting out of everything
- • prompt: for which portions to exclude`,
- docsSection: "core",
- type: "string",
- },
- bin: {
- description: `package.json bin value to include for npx-style running.`,
- docsSection: "optional",
- type: "string",
- },
- "create-repository": {
- description: `Whether to create a corresponding repository on github.com
- (if it doesn't yet exist)`,
- docsSection: "core",
- type: "boolean",
- },
- description: {
- description: `Sentence case description of the repository
- (e.g. A quickstart-friendly TypeScript package with lots of great
- repository tooling. ✨)`,
- docsSection: "core",
- type: "string",
- },
- directory: {
- description: `Directory to create the repository in (by default, the same
- name as the repository)`,
- docsSection: "optional",
- type: "string",
- },
- email: {
- description: `Email address to be listed as the point of contact in docs
- and packages (e.g. example@joshuakgoldberg.com)`,
- docsSection: "optional",
- type: "string",
- },
- "email-github": {
- description: `Optionally, may be provided to use different emails in ${chalk.cyanBright(
- ".md",
- )}
- files`,
- docsSection: "optional",
- type: "string",
- },
- "email-npm": {
- description: `Optionally, may be provided to use different emails in
- ${chalk.cyanBright("package.json")}`,
- docsSection: "optional",
- type: "string",
- },
- "exclude-all-contributors": {
- description: `Don't add all-contributors to track contributions
- and display them in a README.md table.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-compliance": {
- description: `Don't add a GitHub Actions workflow to verify that PRs match
- an expected format.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-jsdoc": {
- description: `Don't use eslint-plugin-jsdoc to enforce good practices around
- JSDoc comments.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-json": {
- description: `Don't apply linting and sorting to ${chalk.cyanBright(
- "*.json",
- )}, and
- ${chalk.cyanBright("*.jsonc")} files.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-knip": {
- description: `Don't add Knip to detect unused files, dependencies, and code
- exports.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-md": {
- description: `Don't apply linting to ${chalk.cyanBright("*.md")} files.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-package-json": {
- description: `Don't add eslint-plugin-package-json to lint for
- package.json correctness.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-packages": {
- description: `Don't add a pnpm dedupe workflow to ensure packages
- aren't duplicated unnecessarily.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-perfectionist": {
- description: `Don't apply eslint-plugin-perfectionist to ensure
- imports, keys, and so on are in sorted order.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-regexp": {
- description: `Don't add eslint-plugin-regexp to enforce good practices around
- regular expressions.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-spelling": {
- description: `Don't add cspell to spell check against dictionaries
- of known words.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-strict": {
- description: `Don't augment the recommended logical lint rules with
- typescript-eslint's strict config.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-stylistic": {
- description: `Don't add stylistic rules such as typescript-eslint's
- stylistic config.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-lint-yml": {
- description: `Don't apply linting and sorting to ${chalk.cyanBright(
- "*.yaml",
- )} and ${chalk.cyanBright("*.yml")} files.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-releases": {
- description: `Don't add release-it to generate changelogs, package bumps,
- and publishes based on conventional commits.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-renovate": {
- description: `Don't add a Renovate config to dependencies up-to-date with
- PRs.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-templated-by": {
- description: `Don't add a _"This package was templated with create-typescript-app"_ notice at the end of the README.md.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- "exclude-tests": {
- description: `Don't add Vitest tooling for fast unit tests, configured
- with coverage tracking.`,
- docsSection: "opt-out",
- type: "boolean",
- },
- funding: {
- description: `GitHub organization or username to mention in ${chalk.cyanBright(
- "funding.yml",
- )}
- (by default, owner)`,
- docsSection: "optional",
- type: "string",
- },
- guide: {
- description: `Link to a contribution guide to place at the top of the
- development docs`,
- docsSection: "optional",
- type: "string",
- },
- "guide-title": {
- description: `If ${chalk.cyanBright(
- "--guide",
- )} is provided or detected from an existing
- DEVELOPMENT.md, the text title to place in the guide link`,
- docsSection: "optional",
- type: "string",
- },
- keywords: {
- description: `Any number of keywords to include in ${chalk.cyanBright(
- "package.json",
- )} (by default,
- none). This can be specified any number of times, like
- ${chalk.cyanBright('--keywords apple --keywords "banana cherry"')}`,
- docsSection: "optional",
- multiple: true,
- type: "string",
- },
- logo: {
- description: `Local image file in the repository to display near the top of
- the README.md as a logo`,
- docsSection: "optional",
- type: "string",
- },
- "logo-alt": {
- description: `If ${chalk.cyanBright(
- "--logo",
- )} is provided or detected from an existing README.md,
- alt text that describes the image will be prompted for if not provided`,
- docsSection: "optional",
- type: "string",
- },
- "logo-height": {
- description: `If ${chalk.cyanBright(
- "--logo",
- )} is provided or detected from an existing README.md,
- an explicit height style`,
- docsSection: "optional",
- type: "string",
- },
- "logo-width": {
- description: `If ${chalk.cyanBright(
- "--logo",
- )} is provided or detected from an existing README.md,
- an explicit width style`,
- docsSection: "optional",
- type: "string",
- },
- mode: {
- description: `Whether to:
- • create: a new repository in a child directory
- • initialize: a freshly repository in the current directory
- • migrate: an existing repository in the current directory`,
- docsSection: "core",
- type: "string",
- },
- offline: {
- description: `You can run ${chalk.cyanBright(
- "create-typescript-app",
- )} in an "offline" mode.
- Doing so will:
- • Enable ${chalk.cyanBright(
- "--exclude-all-contributors-api",
- )} and ${chalk.cyanBright("--skip-github-api")}
- • Skip network calls when setting up contributors
- • Run pnpm commands with pnpm's ${chalk.cyanBright("--offline")} mode`,
- docsSection: "offline",
- type: "boolean",
- },
- owner: {
- description: `GitHub organization or user the repository is underneath
- (e.g. JoshuaKGoldberg)`,
- docsSection: "core",
- type: "string",
- },
- "preserve-generated-from": {
- description: `Whether to keep the GitHub repository generated from
- notice (by default, false)`,
- docsSection: "optional",
- type: "boolean",
- },
- repository: {
- description: `The kebab-case name of the repository
- (e.g. create-typescript-app)`,
- docsSection: "core",
- type: "string",
- },
- "skip-all-contributors-api": {
- description: `Skips network calls that fetch all-contributors data from
- GitHub. This flag does nothing if ${chalk.cyanBright(
- "--exclude-all-contributors",
- )} was specified.`,
- docsSection: "skip-net",
- type: "boolean",
- },
- "skip-github-api": {
- description: `Skips calling to GitHub APIs.`,
- docsSection: "skip-net",
- type: "boolean",
- },
- "skip-install": {
- description: `Skips installing all the new template packages with pnpm.`,
- docsSection: "skip-net",
- type: "boolean",
- },
- "skip-removal": {
- description: `Skips removing setup docs and scripts, including this ${chalk.cyanBright(
- "docs/",
- )}
- directory`,
- docsSection: "skip-disk",
- type: "boolean",
- },
- "skip-restore": {
- description: `Skips the prompt offering to restore the repository if an
- error occurs during setup`,
- docsSection: "skip-disk",
- type: "boolean",
- },
- "skip-uninstall": {
- description: `Skips uninstalling packages only used for setup scripts`,
- docsSection: "skip-disk",
- type: "boolean",
- },
- title: {
- description: `Title Case title for the repository to be used in
- documentation (e.g. Create TypeScript App)`,
- docsSection: "core",
- type: "string",
- },
-} as const;
diff --git a/src/shared/options/augmentOptionsWithExcludes.test.ts b/src/shared/options/augmentOptionsWithExcludes.test.ts
deleted file mode 100644
index ed95d1354..000000000
--- a/src/shared/options/augmentOptionsWithExcludes.test.ts
+++ /dev/null
@@ -1,189 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { Options } from "../types.js";
-import { augmentOptionsWithExcludes } from "./augmentOptionsWithExcludes.js";
-
-const mockMultiselect = vi.fn();
-const mockSelect = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- isCancel: () => false,
- get multiselect() {
- return mockMultiselect;
- },
- get select() {
- return mockSelect;
- },
-}));
-
-const optionsBase = {
- access: "public",
- author: undefined,
- base: undefined,
- description: "",
- directory: ".",
- email: {
- github: "github@email.com",
- npm: "npm@email.com",
- },
- excludeAllContributors: undefined,
- excludeBuild: undefined,
- excludeCompliance: undefined,
- excludeLintESLint: undefined,
- excludeLintJSDoc: undefined,
- excludeLintJson: undefined,
- excludeLintKnip: undefined,
- excludeLintMd: undefined,
- excludeLintPackageJson: undefined,
- excludeLintPackages: undefined,
- excludeLintPerfectionist: undefined,
- excludeLintRegexp: undefined,
- excludeLintSpelling: undefined,
- excludeLintStrict: undefined,
- excludeLintStylistic: undefined,
- excludeLintYml: undefined,
- excludeReleases: undefined,
- excludeRenovate: undefined,
- excludeTemplatedBy: undefined,
- excludeTests: undefined,
- funding: undefined,
- logo: undefined,
- mode: "create",
- offline: true,
- owner: "",
- repository: "",
- skipGitHubApi: false,
- skipInstall: undefined,
- skipRemoval: undefined,
- skipRestore: undefined,
- skipUninstall: undefined,
- title: "",
-} satisfies Options;
-
-describe("augmentOptionsWithExcludes", () => {
- it("returns options directly when no exclusions are provided and 'base' is provided for the prompt", async () => {
- const base = "everything";
-
- mockSelect.mockResolvedValueOnce(base);
-
- const actual = await augmentOptionsWithExcludes(optionsBase);
-
- expect(actual).toEqual({
- ...optionsBase,
- base,
- });
- });
-
- it("returns options based on the select when no exclusions are provided and 'prompt' is provided for the prompt", async () => {
- const base = "prompt";
-
- mockSelect.mockResolvedValueOnce(base);
- mockMultiselect.mockResolvedValue([]);
-
- const actual = await augmentOptionsWithExcludes(optionsBase);
-
- expect(actual).toEqual({
- ...optionsBase,
- base,
- excludeAllContributors: true,
- excludeBuild: true,
- excludeCompliance: true,
- excludeLintESLint: true,
- excludeLintJSDoc: true,
- excludeLintJson: true,
- excludeLintKnip: true,
- excludeLintMd: true,
- excludeLintPackageJson: true,
- excludeLintPackages: true,
- excludeLintPerfectionist: true,
- excludeLintRegexp: true,
- excludeLintSpelling: true,
- excludeLintStrict: true,
- excludeLintStylistic: true,
- excludeLintYml: true,
- excludeReleases: true,
- excludeRenovate: true,
- excludeTemplatedBy: true,
- excludeTests: true,
- });
- });
-
- it("skips prompting returns options directly when exclusions are provided manually", async () => {
- const options = {
- ...optionsBase,
- excludeCompliance: true,
- } satisfies Options;
-
- const actual = await augmentOptionsWithExcludes(options);
-
- expect(actual).toBe(options);
- });
-
- it("uses the 'common' base without prompting when provided manually", async () => {
- const options = {
- ...optionsBase,
- base: "common",
- } satisfies Options;
-
- const actual = await augmentOptionsWithExcludes(options);
-
- expect(actual).toEqual({
- ...options,
- excludeCompliance: true,
- excludeLintESLint: true,
- excludeLintJSDoc: true,
- excludeLintJson: true,
- excludeLintMd: true,
- excludeLintPackageJson: true,
- excludeLintPackages: true,
- excludeLintPerfectionist: true,
- excludeLintRegexp: true,
- excludeLintSpelling: true,
- excludeLintStrict: true,
- excludeLintStylistic: true,
- excludeLintYml: true,
- });
- });
-
- it("uses the 'minimal' base without prompting when provided manually", async () => {
- const options = {
- ...optionsBase,
- base: "minimal",
- } satisfies Options;
-
- const actual = await augmentOptionsWithExcludes(options);
-
- expect(actual).toEqual({
- ...options,
- excludeAllContributors: true,
- excludeCompliance: true,
- excludeLintESLint: true,
- excludeLintJSDoc: true,
- excludeLintJson: true,
- excludeLintKnip: true,
- excludeLintMd: true,
- excludeLintPackageJson: true,
- excludeLintPackages: true,
- excludeLintPerfectionist: true,
- excludeLintRegexp: true,
- excludeLintSpelling: true,
- excludeLintStrict: true,
- excludeLintStylistic: true,
- excludeLintYml: true,
- excludeReleases: true,
- excludeRenovate: true,
- excludeTests: true,
- });
- });
-
- it("uses the 'everything' base without prompting when provided manually", async () => {
- const options = {
- ...optionsBase,
- base: "everything",
- } satisfies Options;
-
- const actual = await augmentOptionsWithExcludes(options);
-
- expect(actual).toStrictEqual(options);
- });
-});
diff --git a/src/shared/options/augmentOptionsWithExcludes.ts b/src/shared/options/augmentOptionsWithExcludes.ts
deleted file mode 100644
index 47070d669..000000000
--- a/src/shared/options/augmentOptionsWithExcludes.ts
+++ /dev/null
@@ -1,108 +0,0 @@
-import * as prompts from "@clack/prompts";
-import chalk from "chalk";
-
-import { filterPromptCancel } from "../prompts.js";
-import { Options, OptionsBase } from "../types.js";
-import {
- exclusionDescriptions,
- ExclusionKey,
- exclusionKeys,
- getExclusions,
-} from "./exclusionKeys.js";
-
-export async function augmentOptionsWithExcludes(
- options: Options,
-): Promise {
- if (
- Object.keys(options).some(
- (key) =>
- key in exclusionDescriptions &&
- options[key as keyof typeof options] !== undefined,
- )
- ) {
- return options;
- }
-
- const base =
- options.base ??
- filterPromptCancel(
- await prompts.select({
- initialValue: "common" as OptionsBase,
- message: `How much tooling would you like the template to set up for you?`,
- options: [
- {
- label: makeLabel(
- "everything",
- "The most comprehensive tooling imaginable: sorting, spellchecking, and more!",
- ),
- value: "everything",
- },
- {
- hint: "recommended",
- label: makeLabel(
- "common",
- "Bare starters plus testing and automation for all-contributors and releases.",
- ),
- value: "common",
- },
- {
- label: makeLabel(
- "minimal",
- "Just bare starter tooling: building, formatting, linting, and type checking.",
- ),
- value: "minimal",
- },
- {
- label: makeLabel("prompt", "(allow me to customize)"),
- value: "prompt",
- },
- ],
- }),
- );
-
- switch (base) {
- case "common":
- case "everything":
- case "minimal":
- return {
- ...options,
- base,
- ...getExclusions(options, base),
- };
- case "prompt": {
- const exclusionsNotEnabled = new Set(
- filterPromptCancel(
- await prompts.multiselect({
- initialValues: exclusionKeys,
- message:
- "Select the tooling portions you'd like to remove. All are enabled by default. Press ↑ or ↓ to change the selected item, then space to select.",
- options: Object.entries(exclusionDescriptions).map(
- ([value, { hint, label }]) => ({
- hint,
- label,
- value: value as ExclusionKey,
- }),
- ),
- }),
- ),
- );
-
- return {
- ...options,
- base,
- ...Object.fromEntries(
- exclusionKeys.map(
- (exclusionKey) =>
- [exclusionKey, !exclusionsNotEnabled.has(exclusionKey)] as const,
- ),
- ),
- };
- }
- case undefined:
- return undefined;
- }
-}
-
-function makeLabel(label: string, message: string) {
- return [chalk.bold(label), message].join("\t ");
-}
diff --git a/src/shared/options/createOptionDefaults/index.test.ts b/src/shared/options/createOptionDefaults/index.test.ts
deleted file mode 100644
index 54189becd..000000000
--- a/src/shared/options/createOptionDefaults/index.test.ts
+++ /dev/null
@@ -1,218 +0,0 @@
-import { beforeEach, describe, expect, it, vi } from "vitest";
-
-import { createOptionDefaults } from "./index.js";
-
-const mock$ = vi.fn();
-
-vi.mock("execa", () => ({
- get $() {
- return mock$;
- },
-}));
-
-const mockGitUrlParse = vi.fn();
-
-vi.mock("git-url-parse", () => ({
- get default() {
- return mockGitUrlParse;
- },
-}));
-
-const mockNpmUser = vi.fn();
-
-vi.mock("npm-user", () => ({
- get default() {
- return mockNpmUser;
- },
-}));
-
-const mockReadPackageData = vi.fn();
-
-vi.mock("../../packages.js", () => ({
- get readPackageData() {
- return mockReadPackageData;
- },
-}));
-
-const mockReadGitHubEmail = vi.fn();
-
-vi.mock("./readGitHubEmail.js", () => ({
- get readGitHubEmail() {
- return mockReadGitHubEmail;
- },
-}));
-
-describe("createOptionDefaults", () => {
- describe("bin", () => {
- it("returns undefined when package data does not have a bin", async () => {
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().bin();
-
- expect(actual).toBeUndefined();
- });
-
- it("returns the bin when package data has a bin", async () => {
- const bin = "./lib/index.js";
-
- mockReadPackageData.mockResolvedValue({ bin });
-
- const actual = await createOptionDefaults().bin();
-
- expect(actual).toBe(bin);
- });
- });
-
- describe("email", () => {
- beforeEach(() => {
- mockNpmUser.mockImplementation((username: string) => ({
- email: `npm-${username}@test.com`,
- }));
- });
-
- it("returns the npm whoami email from npm when only an npm exists", async () => {
- mock$.mockImplementation(([command]: string[]) =>
- command === "npm whoami" ? { stdout: "username" } : undefined,
- );
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "npm-username@test.com",
- npm: "npm-username@test.com",
- });
- });
-
- it("returns the npm whoami email from npm when only a package author email exists", async () => {
- mock$.mockResolvedValue({ stdout: "" });
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({
- author: {
- email: "test@package.com",
- },
- });
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "test@package.com",
- npm: "test@package.com",
- });
- });
-
- it("returns the github email when only a github email exists", async () => {
- mock$.mockResolvedValue({ stdout: "" });
- mockReadPackageData.mockResolvedValueOnce({});
- mockReadGitHubEmail.mockResolvedValueOnce("github@test.com");
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "github@test.com",
- npm: "github@test.com",
- });
- });
-
- it("returns the git user email when only a git user email exists", async () => {
- mock$.mockImplementation(([command]: string[]) =>
- command === "git config --get user.email"
- ? { stdout: "git@test.com" }
- : undefined,
- );
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "git@test.com",
- npm: "git@test.com",
- });
- });
-
- it("returns both the git user email and the npm user email when only those two exist", async () => {
- mock$.mockImplementation(([command]: string[]) => ({
- stdout:
- command === "git config --get user.email"
- ? "git@test.com"
- : "username",
- }));
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "git@test.com",
- npm: "npm-username@test.com",
- });
- });
-
- it("returns all three emails when they all exist", async () => {
- mock$.mockImplementation(([command]: string[]) => ({
- stdout:
- command === "git config --get user.email"
- ? "git@test.com"
- : "username",
- }));
- mockReadGitHubEmail.mockResolvedValueOnce("github@test.com");
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "github@test.com",
- npm: "npm-username@test.com",
- });
- });
-
- it("returns undefined when none of the emails exist", async () => {
- mock$.mockResolvedValue({ stdout: "" });
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toBeUndefined();
- });
- });
-
- describe("repository", () => {
- it("returns promptedOptions.repository when it exists", async () => {
- const repository = "test-prompted-repository";
- const promptedOptions = { repository };
- const actual = await createOptionDefaults(promptedOptions).repository();
-
- expect(actual).toBe(repository);
- });
-
- it("returns the Git name when it exists and promptedOptions.repository doesn't", async () => {
- const name = "test-git-repository";
- mockGitUrlParse.mockResolvedValueOnce({ name });
-
- const actual = await createOptionDefaults().repository();
-
- expect(actual).toBe(name);
- });
-
- it("returns the package name when it exists and promptedOptions.repository and Git name don't", async () => {
- const name = "test-package-name";
- mockReadPackageData.mockResolvedValueOnce({ name });
-
- const actual = await createOptionDefaults().repository();
-
- expect(actual).toBe(name);
- });
-
- it("returns the directory when it exists and promptedOptions.repository, Git name, and package name don't", async () => {
- mockReadPackageData.mockResolvedValueOnce({});
- const directory = "test-prompted-directory";
- const promptedOptions = { directory };
-
- const actual = await createOptionDefaults(promptedOptions).repository();
-
- expect(actual).toBe(directory);
- });
- });
-});
diff --git a/src/shared/options/createOptionDefaults/index.ts b/src/shared/options/createOptionDefaults/index.ts
deleted file mode 100644
index 3049463c8..000000000
--- a/src/shared/options/createOptionDefaults/index.ts
+++ /dev/null
@@ -1,61 +0,0 @@
-import { $ } from "execa";
-import gitRemoteOriginUrl from "git-remote-origin-url";
-import gitUrlParse from "git-url-parse";
-import lazyValue from "lazy-value";
-import * as fs from "node:fs/promises";
-import npmUser from "npm-user";
-
-import { readDescription } from "../../../next/readDescription.js";
-import { readPackageData } from "../../packages.js";
-import { readFileSafe } from "../../readFileSafe.js";
-import { tryCatchAsync } from "../../tryCatchAsync.js";
-import { tryCatchLazyValueAsync } from "../../tryCatchLazyValueAsync.js";
-import { PromptedOptions } from "../../types.js";
-import { parsePackageAuthor } from "./parsePackageAuthor.js";
-import { readDefaultsFromReadme } from "./readDefaultsFromReadme.js";
-import { readEmails } from "./readEmails.js";
-import { readGuide } from "./readGuide.js";
-
-export function createOptionDefaults(promptedOptions?: PromptedOptions) {
- const gitDefaults = tryCatchLazyValueAsync(async () =>
- gitUrlParse(await gitRemoteOriginUrl()),
- );
-
- const npmDefaults = tryCatchLazyValueAsync(async () => {
- const whoami = (await $`npm whoami`).stdout;
- return whoami ? await npmUser(whoami) : undefined;
- });
-
- const packageData = lazyValue(readPackageData);
- const packageAuthor = lazyValue(async () =>
- parsePackageAuthor(await packageData()),
- );
-
- const readme = lazyValue(async () => await readFileSafe("README.md", ""));
-
- return {
- author: async () =>
- (await packageAuthor()).author ?? (await npmDefaults())?.name,
- bin: async () => (await packageData()).bin,
- description: async () => await readDescription(packageData, readme),
- email: async () => readEmails(npmDefaults, packageAuthor),
- funding: async () =>
- await tryCatchAsync(async () =>
- (await fs.readFile(".github/FUNDING.yml"))
- .toString()
- .split(":")[1]
- ?.trim(),
- ),
- guide: readGuide,
- owner: async () =>
- (await gitDefaults())?.organization ?? (await packageAuthor()).author,
- repository: async () =>
- promptedOptions?.repository ??
- (await gitDefaults())?.name ??
- (await packageData()).name ??
- promptedOptions?.directory,
- ...readDefaultsFromReadme(readme, () =>
- Promise.resolve(promptedOptions?.repository),
- ),
- };
-}
diff --git a/src/shared/options/createRepositoryWithApi.test.ts b/src/shared/options/createRepositoryWithApi.test.ts
deleted file mode 100644
index 5f4ca68e9..000000000
--- a/src/shared/options/createRepositoryWithApi.test.ts
+++ /dev/null
@@ -1,71 +0,0 @@
-import { Octokit } from "octokit";
-import { describe, expect, it, vi } from "vitest";
-
-import { createRepositoryWithApi } from "./createRepositoryWithApi.js";
-
-const options = { owner: "StubOwner", repository: "stub-repository" };
-
-const mockCreateUsingTemplate = vi.fn();
-const mockCreateInOrg = vi.fn();
-const mockCreateForAuthenticatedUser = vi.fn();
-const mockGetAuthenticated = vi.fn();
-
-const createMockOctokit = () =>
- ({
- rest: {
- repos: {
- createForAuthenticatedUser: mockCreateForAuthenticatedUser,
- createInOrg: mockCreateInOrg,
- createUsingTemplate: mockCreateUsingTemplate,
- },
- users: {
- getAuthenticated: mockGetAuthenticated,
- },
- },
- }) as unknown as Octokit;
-
-describe("createRepositoryWithApi", () => {
- it("creates using a template when preserveGeneratedFrom is true", async () => {
- await createRepositoryWithApi(createMockOctokit(), {
- ...options,
- preserveGeneratedFrom: true,
- });
-
- expect(mockCreateForAuthenticatedUser).not.toHaveBeenCalled();
- expect(mockCreateInOrg).not.toHaveBeenCalled();
- expect(mockCreateUsingTemplate).toHaveBeenCalledWith({
- name: options.repository,
- owner: options.owner,
- template_owner: "JoshuaKGoldberg",
- template_repo: "create-typescript-app",
- });
- });
-
- it("creates under the user when the user is the owner", async () => {
- mockGetAuthenticated.mockResolvedValueOnce({
- data: {
- login: options.owner,
- },
- });
- await createRepositoryWithApi(createMockOctokit(), options);
-
- expect(mockCreateForAuthenticatedUser).toHaveBeenCalledWith({
- name: options.repository,
- });
- expect(mockCreateInOrg).not.toHaveBeenCalled();
- expect(mockCreateUsingTemplate).not.toHaveBeenCalled();
- });
-
- it("creates under an org when the user is not the owner", async () => {
- const login = "other-user";
- mockGetAuthenticated.mockResolvedValueOnce({ data: { login } });
- await createRepositoryWithApi(createMockOctokit(), options);
-
- expect(mockCreateForAuthenticatedUser).not.toHaveBeenCalled();
- expect(mockCreateInOrg).toHaveBeenCalledWith({
- name: options.repository,
- org: options.owner,
- });
- expect(mockCreateUsingTemplate).not.toHaveBeenCalled();
- });
-});
diff --git a/src/shared/options/createRepositoryWithApi.ts b/src/shared/options/createRepositoryWithApi.ts
deleted file mode 100644
index f8be9b3fd..000000000
--- a/src/shared/options/createRepositoryWithApi.ts
+++ /dev/null
@@ -1,41 +0,0 @@
-import { Octokit } from "octokit";
-
-export interface CreateRepositoryWithApiOptions {
- owner: string;
- preserveGeneratedFrom?: boolean;
- repository: string;
-}
-
-export async function createRepositoryWithApi(
- octokit: Octokit,
- options: CreateRepositoryWithApiOptions,
-) {
- if (options.preserveGeneratedFrom) {
- await octokit.rest.repos.createUsingTemplate({
- name: options.repository,
- owner: options.owner,
- template_owner: "JoshuaKGoldberg",
- template_repo: "create-typescript-app",
- });
- return;
- }
-
- const currentUser = await octokit.rest.users.getAuthenticated();
-
- try {
- if (currentUser.data.login === options.owner) {
- await octokit.rest.repos.createForAuthenticatedUser({
- name: options.repository,
- });
- } else {
- await octokit.rest.repos.createInOrg({
- name: options.repository,
- org: options.owner,
- });
- }
- } catch (error) {
- throw new Error("Failed to create new repository on GitHub.", {
- cause: error,
- });
- }
-}
diff --git a/src/shared/options/detectEmailRedundancy.test.ts b/src/shared/options/detectEmailRedundancy.test.ts
deleted file mode 100644
index 3091a6096..000000000
--- a/src/shared/options/detectEmailRedundancy.test.ts
+++ /dev/null
@@ -1,38 +0,0 @@
-import { describe, expect, it } from "vitest";
-
-import { detectEmailRedundancy } from "./detectEmailRedundancy.js";
-
-describe("detectEmailRedundancy", () => {
- it("returns undefined when only email is specified", () => {
- expect(detectEmailRedundancy({ email: "test@email.com" })).toBeUndefined();
- });
-
- it("returns undefined when email-github and email-npm are specified while email is not", () => {
- expect(
- detectEmailRedundancy({
- "email-github": "test@email.com",
- "email-npm": "test@email.com",
- }),
- ).toBeUndefined();
- });
-
- it("returns a complaint when email-github is specified while email and email-npm are not", () => {
- expect(
- detectEmailRedundancy({
- "email-github": "test@email.com",
- }),
- ).toBe(
- "If --email-github is specified, either --email or --email-npm should be.",
- );
- });
-
- it("returns a complaint when email-npm is specified while email and email-github are not", () => {
- expect(
- detectEmailRedundancy({
- "email-npm": "test@email.com",
- }),
- ).toBe(
- "If --email-npm is specified, either --email or --email-github should be.",
- );
- });
-});
diff --git a/src/shared/options/detectEmailRedundancy.ts b/src/shared/options/detectEmailRedundancy.ts
deleted file mode 100644
index 15044b475..000000000
--- a/src/shared/options/detectEmailRedundancy.ts
+++ /dev/null
@@ -1,23 +0,0 @@
-export interface EmailValues {
- email?: boolean | string;
- "email-github"?: boolean | string;
- "email-npm"?: boolean | string;
-}
-
-export function detectEmailRedundancy(values: EmailValues) {
- if (values.email) {
- return values["email-github"] && values["email-npm"]
- ? "--email should not be specified if both --email-github and --email-npm are specified."
- : undefined;
- }
-
- if (values["email-github"] && !values["email-npm"]) {
- return "If --email-github is specified, either --email or --email-npm should be.";
- }
-
- if (values["email-npm"] && !values["email-github"]) {
- return "If --email-npm is specified, either --email or --email-github should be.";
- }
-
- return undefined;
-}
diff --git a/src/shared/options/ensureRepositoryExists.test.ts b/src/shared/options/ensureRepositoryExists.test.ts
deleted file mode 100644
index 3345f3adf..000000000
--- a/src/shared/options/ensureRepositoryExists.test.ts
+++ /dev/null
@@ -1,170 +0,0 @@
-import { Octokit } from "octokit";
-import { describe, expect, it, vi } from "vitest";
-
-import { ensureRepositoryExists } from "./ensureRepositoryExists.js";
-
-const mockSelect = vi.fn();
-const mockText = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- intro: vi.fn(),
- isCancel: vi.fn(),
- outro: vi.fn(),
- get select() {
- return mockSelect;
- },
- get text() {
- return mockText;
- },
-}));
-
-const mockDoesRepositoryExist = vi.fn();
-
-vi.mock("../doesRepositoryExist.js", () => ({
- get doesRepositoryExist() {
- return mockDoesRepositoryExist;
- },
-}));
-
-const owner = "StubOwner";
-const repository = "stub-repository";
-
-const mockCreateRepositoryWithApi = vi.fn();
-
-vi.mock("./createRepositoryWithApi.js", () => ({
- get createRepositoryWithApi() {
- return mockCreateRepositoryWithApi;
- },
-}));
-
-const createMockOctokit = () => ({}) as unknown as Octokit;
-
-describe("ensureRepositoryExists", () => {
- it("returns the repository when octokit is undefined", async () => {
- const actual = await ensureRepositoryExists(undefined, {
- mode: "initialize",
- owner,
- repository,
- });
-
- expect(actual).toEqual({ octokit: undefined, repository });
- });
-
- it("returns the repository when octokit is defined and the repository exists", async () => {
- mockDoesRepositoryExist.mockResolvedValue(true);
- const octokit = createMockOctokit();
- const actual = await ensureRepositoryExists(octokit, {
- mode: "initialize",
- owner,
- repository,
- });
-
- expect(actual).toEqual({ octokit, repository });
- });
-
- it("creates a new repository when the prompt is 'create' and the repository does not exist", async () => {
- const octokit = createMockOctokit();
-
- mockDoesRepositoryExist.mockResolvedValue(false);
- mockSelect.mockResolvedValue("create");
-
- const actual = await ensureRepositoryExists(octokit, {
- mode: "initialize",
- owner,
- repository,
- });
-
- expect(actual).toEqual({ octokit, repository });
- expect(mockCreateRepositoryWithApi).toHaveBeenCalledWith(octokit, {
- owner,
- preserveGeneratedFrom: undefined,
- repository,
- });
- });
-
- it("defaults to creating a repository when mode is 'create'", async () => {
- const octokit = createMockOctokit();
-
- mockDoesRepositoryExist.mockResolvedValue(false);
-
- const actual = await ensureRepositoryExists(octokit, {
- mode: "create",
- owner,
- repository,
- });
-
- expect(actual).toEqual({ octokit, repository });
- expect(mockCreateRepositoryWithApi).toHaveBeenCalledWith(octokit, {
- owner,
- preserveGeneratedFrom: undefined,
- repository,
- });
- expect(mockSelect).not.toHaveBeenCalled();
- });
-
- it("returns the second repository when the prompt is 'different', the first repository does not exist, and the second repository exists", async () => {
- const octokit = createMockOctokit();
- const newRepository = "new-repository";
-
- mockDoesRepositoryExist
- .mockResolvedValueOnce(false)
- .mockResolvedValueOnce(true);
- mockSelect.mockResolvedValueOnce("different");
- mockText.mockResolvedValue(newRepository);
-
- const actual = await ensureRepositoryExists(octokit, {
- mode: "initialize",
- owner,
- repository,
- });
-
- expect(actual).toEqual({
- octokit,
- repository: newRepository,
- });
- expect(mockCreateRepositoryWithApi).not.toHaveBeenCalled();
- });
-
- it("creates the second repository when the prompt is 'different', the first repository does not exist, and the second repository does not exist", async () => {
- const octokit = createMockOctokit();
- const newRepository = "new-repository";
-
- mockDoesRepositoryExist.mockResolvedValue(false);
- mockSelect
- .mockResolvedValueOnce("different")
- .mockResolvedValueOnce("create");
- mockText.mockResolvedValue(newRepository);
-
- const actual = await ensureRepositoryExists(octokit, {
- mode: "initialize",
- owner,
- repository,
- });
-
- expect(actual).toEqual({
- octokit,
- repository: newRepository,
- });
- expect(mockCreateRepositoryWithApi).toHaveBeenCalledWith(octokit, {
- owner,
- preserveGeneratedFrom: undefined,
- repository: newRepository,
- });
- });
-
- it("switches octokit to undefined when the prompt is 'local' and the repository does not exist", async () => {
- const octokit = createMockOctokit();
-
- mockDoesRepositoryExist.mockResolvedValue(false);
- mockSelect.mockResolvedValue("local");
-
- const actual = await ensureRepositoryExists(octokit, {
- mode: "initialize",
- owner,
- repository,
- });
-
- expect(actual).toEqual({ octokit: undefined, repository });
- expect(mockCreateRepositoryWithApi).not.toHaveBeenCalled();
- });
-});
diff --git a/src/shared/options/ensureRepositoryExists.ts b/src/shared/options/ensureRepositoryExists.ts
deleted file mode 100644
index a04c9b707..000000000
--- a/src/shared/options/ensureRepositoryExists.ts
+++ /dev/null
@@ -1,95 +0,0 @@
-import * as prompts from "@clack/prompts";
-import { Octokit } from "octokit";
-
-import { doesRepositoryExist } from "../doesRepositoryExist.js";
-import { isUsingCreateEngine } from "../isUsingCreateEngine.js";
-import { filterPromptCancel } from "../prompts.js";
-import { Options } from "../types.js";
-import { createRepositoryWithApi } from "./createRepositoryWithApi.js";
-
-export type EnsureRepositoryExistsOptions = Pick<
- Options,
- "mode" | "owner" | "preserveGeneratedFrom" | "repository"
->;
-
-export interface RepositoryExistsResult {
- octokit: Octokit | undefined;
- repository: string;
-}
-
-export async function ensureRepositoryExists(
- octokit: Octokit | undefined,
- options: EnsureRepositoryExistsOptions,
-): Promise> {
- // We'll only respect input options once before prompting for them
- let { repository } = options;
- let createRepository = options.mode === "create";
-
- // We'll continuously pester the user for a repository
- // until they bail, create a new one, or it exists.
- while (octokit) {
- if (
- isUsingCreateEngine() ||
- (await doesRepositoryExist(octokit, options))
- ) {
- return { octokit, repository };
- }
-
- const selection = createRepository
- ? "create"
- : filterPromptCancel(
- (await prompts.select({
- message: `Repository ${options.repository} doesn't seem to exist under ${options.owner}. What would you like to do?`,
- options: [
- { label: "Create a new repository", value: "create" },
- {
- label: "Switch to a different repository name",
- value: "different",
- },
- {
- label: "Keep changes local",
- value: "local",
- },
- { label: "Bail out and maybe try again later", value: "bail" },
- ],
- })) as "bail" | "create" | "different" | "local",
- );
-
- createRepository = false;
-
- switch (selection) {
- case "bail":
- case undefined:
- return {};
-
- case "create":
- await createRepositoryWithApi(octokit, {
- owner: options.owner,
- preserveGeneratedFrom: options.preserveGeneratedFrom,
- repository,
- });
- return { octokit, repository };
-
- case "different": {
- const newRepository = filterPromptCancel(
- await prompts.text({
- message: `What would you like to call the repository?`,
- }),
- );
-
- if (!newRepository) {
- return {};
- }
-
- repository = newRepository;
- break;
- }
-
- case "local":
- octokit = undefined;
- break;
- }
- }
-
- return { octokit, repository };
-}
diff --git a/src/shared/options/exclusionKeys.ts b/src/shared/options/exclusionKeys.ts
deleted file mode 100644
index 8d3b03425..000000000
--- a/src/shared/options/exclusionKeys.ts
+++ /dev/null
@@ -1,158 +0,0 @@
-import { Options, OptionsBase } from "../types.js";
-
-export interface ExclusionDescription {
- hint: string;
- label: string;
- level?: "common" | "minimal";
-}
-
-export type ExclusionKey = `exclude${string}` & keyof Options;
-
-export const exclusionDescriptions: Record =
- {
- excludeAllContributors: {
- hint: "--exclude-all-contributors",
- label:
- "Add all-contributors to track contributions and display them in a README.md table.",
- },
- excludeBuild: {
- hint: "--exclude-build",
- label: "Add a tsup build step to generate built output files.",
- level: "minimal",
- },
- excludeCompliance: {
- hint: "--exclude-compliance",
- label:
- "Add a GitHub Actions workflow to verify that PRs match an expected format.",
- level: "common",
- },
- excludeLintESLint: {
- hint: "--exclude-lint-eslint",
- label:
- "Include eslint-plugin-eslint-comment to enforce good practices around ESLint comment directives.",
- level: "common",
- },
- excludeLintJSDoc: {
- hint: "--exclude-lint-jsdoc",
- label:
- "Include eslint-plugin-jsdoc to enforce good practices around JSDoc comments.",
- level: "common",
- },
- excludeLintJson: {
- hint: "--exclude-lint-json",
- label: "Apply linting and sorting to *.json and *.jsonc files.",
- level: "common",
- },
- excludeLintKnip: {
- hint: "--exclude-lint-knip",
- label: "Add Knip to detect unused files, dependencies, and code exports.",
- },
- excludeLintMd: {
- hint: "--exclude-lint-md",
- label: "Apply linting to *.md files.",
- level: "common",
- },
- excludeLintPackageJson: {
- hint: "--exclude-lint-package-json",
- label:
- "Add eslint-plugin-package-json to lint for package.json correctness.",
- level: "common",
- },
- excludeLintPackages: {
- hint: "--exclude-lint-packages",
- label:
- "Add a pnpm dedupe workflow to ensure packages aren't duplicated unnecessarily.",
- level: "common",
- },
- excludeLintPerfectionist: {
- hint: "--exclude-lint-perfectionist",
- label:
- "Apply eslint-plugin-perfectionist to ensure imports, keys, and so on are in sorted order.",
- level: "common",
- },
- excludeLintRegexp: {
- hint: "--exclude-lint-regexp",
- label:
- "Include eslint-plugin-regexp to enforce good practices around regular expressions.",
- level: "common",
- },
- excludeLintSpelling: {
- hint: "--exclude-lint-spelling",
- label: "Add cspell to spell check against dictionaries of known words.",
- level: "common",
- },
- excludeLintStrict: {
- hint: "--exclude-lint-strict",
- label:
- "Include strict logical lint rules such as typescript-eslint's strict config. ",
- level: "common",
- },
- excludeLintStylistic: {
- hint: "--exclude-lint-stylistic",
- label:
- "Include stylistic lint rules such as typescript-eslint's stylistic config.",
- level: "common",
- },
- excludeLintYml: {
- hint: "--exclude-lint-yml",
- label: "Apply linting and sorting to *.yaml and *.yml files.",
- level: "common",
- },
- excludeReleases: {
- hint: "--exclude-releases",
- label:
- "Add release-it to generate changelogs, package bumps, and publishes based on conventional commits.",
- },
- excludeRenovate: {
- hint: "--exclude-renovate",
- label: "Add a Renovate config to keep dependencies up-to-date with PRs.",
- },
- excludeTemplatedBy: {
- hint: "--exclude-templated-by",
- label:
- 'Add a _"This package was templated with create-typescript-app"_ notice at the end of the README.md.',
- level: "minimal",
- },
- excludeTests: {
- hint: "--exclude-tests",
- label:
- "Add Vitest tooling for fast unit tests, configured with coverage tracking.",
- },
- };
-
-export const exclusionKeys = Object.keys(
- exclusionDescriptions,
-) as ExclusionKey[];
-
-export function getExclusions(
- options: Partial,
- base?: OptionsBase,
-): Partial {
- switch (base) {
- case "common":
- return {
- ...Object.fromEntries(
- exclusionKeys
- .filter(
- (exclusion) =>
- exclusionDescriptions[exclusion].level === "common",
- )
- .map((exclusion) => [exclusion, options[exclusion] ?? true]),
- ),
- };
- case "minimal":
- return {
- ...Object.fromEntries(
- exclusionKeys
- .filter(
- (exclusion) =>
- exclusionDescriptions[exclusion].level !== "minimal",
- )
- .map((exclusion) => [exclusion, options[exclusion] ?? true]),
- ),
- };
- // We only really care about exclusions on the common and minimal bases
- default:
- return {};
- }
-}
diff --git a/src/shared/options/getBase.test.ts b/src/shared/options/getBase.test.ts
deleted file mode 100644
index 9f50ce440..000000000
--- a/src/shared/options/getBase.test.ts
+++ /dev/null
@@ -1,59 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { getBase } from "./getBase.js";
-
-const mockReadPackageData = vi.fn();
-vi.mock("../packages.js", () => ({
- get readPackageData() {
- return mockReadPackageData;
- },
-}));
-
-describe("getBase", () => {
- it("should return minimal with minimal scripts", async () => {
- mockReadPackageData.mockImplementationOnce(() =>
- Promise.resolve({
- scripts: {
- build: "build",
- lint: "lint",
- test: "test",
- },
- }),
- );
-
- expect(await getBase()).toBe("minimal");
- });
-
- it("should return common with common scripts", async () => {
- mockReadPackageData.mockImplementationOnce(() =>
- Promise.resolve({
- scripts: {
- build: "build",
- lint: "lint",
- "lint:knip": "knip",
- test: "test",
- },
- }),
- );
-
- expect(await getBase()).toBe("common");
- });
-
- it("should return everything with everything scripts", async () => {
- mockReadPackageData.mockImplementationOnce(() =>
- Promise.resolve({
- scripts: {
- build: "build",
- lint: "lint",
- "lint:knip": "knip",
- "lint:md": "md",
- "lint:packages": "packages",
- "lint:spelling": "spelling",
- test: "test",
- },
- }),
- );
-
- expect(await getBase()).toBe("everything");
- });
-});
diff --git a/src/shared/options/getBase.ts b/src/shared/options/getBase.ts
deleted file mode 100644
index af5e8bf6e..000000000
--- a/src/shared/options/getBase.ts
+++ /dev/null
@@ -1,34 +0,0 @@
-import { readPackageData } from "../packages.js";
-import { OptionsBase } from "../types.js";
-
-const commonScripts = new Set(["lint:knip", "test"]);
-
-const everythingScripts = new Set([
- "lint:md",
- "lint:packages",
- "lint:spelling",
-]);
-
-export async function getBase(): Promise {
- const scripts = Object.keys((await readPackageData()).scripts ?? {});
-
- if (
- scripts.reduce(
- (acc, curr) => (everythingScripts.has(curr) ? acc + 1 : acc),
- 0,
- ) >= 3
- ) {
- return "everything";
- }
-
- if (
- scripts.reduce(
- (acc, curr) => (commonScripts.has(curr) ? acc + 1 : acc),
- 0,
- ) >= 2
- ) {
- return "common";
- }
-
- return "minimal";
-}
diff --git a/src/shared/options/getPrefillOrPromptedOption.test.ts b/src/shared/options/getPrefillOrPromptedOption.test.ts
deleted file mode 100644
index ef190dd49..000000000
--- a/src/shared/options/getPrefillOrPromptedOption.test.ts
+++ /dev/null
@@ -1,106 +0,0 @@
-import { TextOptions } from "@clack/prompts";
-import { describe, expect, it, vi } from "vitest";
-
-import { getPrefillOrPromptedOption } from "./getPrefillOrPromptedOption.js";
-
-const mockText = vi.fn();
-
-vi.mock("@clack/prompts", () => ({
- isCancel: () => false,
- get text() {
- return mockText;
- },
-}));
-
-describe("getPrefillOrPromptedValue", () => {
- it("returns the provided when it exists", async () => {
- const value = "Test Value";
-
- const actual = await getPrefillOrPromptedOption({
- auto: true,
- getDefaultValue: vi.fn().mockResolvedValue("default value"),
- message: "Input message.",
- name: "field",
- provided: value,
- });
-
- expect(actual).toEqual({ error: undefined, value });
- });
-
- it("returns the default value when auto is true and it exists", async () => {
- const value = "Test Value";
-
- const actual = await getPrefillOrPromptedOption({
- auto: true,
- getDefaultValue: vi.fn().mockResolvedValue(value),
- message: "Input message.",
- name: "field",
- });
-
- expect(actual).toEqual({ error: undefined, value });
- });
-
- it("returns an error when auto is true and no default value exists", async () => {
- const actual = await getPrefillOrPromptedOption({
- auto: true,
- getDefaultValue: vi.fn().mockResolvedValue(undefined),
- message: "Input message.",
- name: "field",
- });
-
- expect(actual).toEqual({
- error: "Could not infer a default value for field.",
- value: undefined,
- });
- });
-
- it("provides no placeholder when one is not provided and auto is false", async () => {
- const message = "Test message";
-
- await getPrefillOrPromptedOption({ auto: false, message, name: "field" });
-
- expect(mockText).toHaveBeenCalledWith({
- message,
- placeholder: undefined,
- validate: expect.any(Function),
- });
- });
-
- it("prompts with the default value as a placeholder when a placeholder function is provided and auto is false", async () => {
- const message = "Test message";
- const placeholder = "Test placeholder";
-
- await getPrefillOrPromptedOption({
- auto: false,
- getDefaultValue: vi.fn().mockResolvedValue(placeholder),
- message,
- name: "field",
- });
-
- expect(mockText).toHaveBeenCalledWith({
- message,
- placeholder,
- validate: expect.any(Function),
- });
- });
-
- it("validates entered text when it's not blank and auto is false", async () => {
- const message = "Test message";
-
- await getPrefillOrPromptedOption({ auto: false, message, name: "field" });
-
- const { validate } = (mockText.mock.calls[0] as [Required])[0];
-
- expect(validate(message)).toBeUndefined();
- });
-
- it("invalidates entered text when it's blank and auto is false", async () => {
- const message = "";
-
- await getPrefillOrPromptedOption({ auto: false, message, name: "field" });
-
- const { validate } = (mockText.mock.calls[0] as [Required])[0];
-
- expect(validate(message)).toBe("Please enter a value.");
- });
-});
diff --git a/src/shared/options/getPrefillOrPromptedOption.ts b/src/shared/options/getPrefillOrPromptedOption.ts
deleted file mode 100644
index 17384ba34..000000000
--- a/src/shared/options/getPrefillOrPromptedOption.ts
+++ /dev/null
@@ -1,48 +0,0 @@
-import * as prompts from "@clack/prompts";
-
-import { filterPromptCancel } from "../prompts.js";
-
-export interface GetPrefillOrPromptedOptionOptions {
- auto: boolean;
- getDefaultValue?: () => Promise;
- message: string;
- name: string;
- provided?: string | undefined;
-}
-
-export async function getPrefillOrPromptedOption({
- auto,
- getDefaultValue,
- message,
- name,
- provided,
-}: GetPrefillOrPromptedOptionOptions) {
- if (provided) {
- return { value: provided };
- }
-
- const defaultValue = await getDefaultValue?.();
-
- if (auto) {
- return {
- error: defaultValue
- ? undefined
- : `Could not infer a default value for ${name}.`,
- value: defaultValue,
- };
- }
-
- return {
- value: filterPromptCancel(
- await prompts.text({
- message,
- placeholder: defaultValue,
- validate: (val) => {
- if (val.length === 0) {
- return "Please enter a value.";
- }
- },
- }),
- ),
- };
-}
diff --git a/src/shared/options/logInferredOptions.test.ts b/src/shared/options/logInferredOptions.test.ts
deleted file mode 100644
index f3a8ec9e2..000000000
--- a/src/shared/options/logInferredOptions.test.ts
+++ /dev/null
@@ -1,118 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-
-import { InferredOptions, logInferredOptions } from "./logInferredOptions.js";
-
-function makeProxy(receiver: T): T {
- return new Proxy(receiver, {
- get: () => makeProxy((input: string) => input),
- });
-}
-
-vi.mock("chalk", () => ({
- default: makeProxy({}),
-}));
-
-const mockLogLine = vi.fn();
-
-vi.mock("../cli/lines.js", () => ({
- get logLine() {
- return mockLogLine;
- },
-}));
-
-const options = {
- description: "Test description.",
- email: {
- github: "github@email.com",
- npm: "npm@email.com",
- },
- owner: "TestOwner",
- repository: "test-repository",
- title: "Test Title",
-} satisfies InferredOptions;
-
-describe("logInferredOptions", () => {
- it("logs the required inferred values when only they exist", () => {
- logInferredOptions(options);
-
- expect(mockLogLine.mock.calls).toMatchInlineSnapshot(`
- [
- [],
- [
- "--auto inferred the following values:",
- ],
- [
- "- description: Test description.",
- ],
- [
- "- email-github: github@email.com",
- ],
- [
- "- email-npm: github@email.com",
- ],
- [
- "- owner: TestOwner",
- ],
- [
- "- repository: test-repository",
- ],
- [
- "- title: Test Title",
- ],
- ]
- `);
- });
-
- it("logs additional and required inferred values when all they exist", () => {
- logInferredOptions({
- ...options,
- guide: {
- href: "https://example.com/guide",
- title: "Example Guide",
- },
- logo: {
- alt: "Logo text.",
- src: "https://example.com/logo",
- },
- });
-
- expect(mockLogLine.mock.calls).toMatchInlineSnapshot(`
- [
- [],
- [
- "--auto inferred the following values:",
- ],
- [
- "- description: Test description.",
- ],
- [
- "- email-github: github@email.com",
- ],
- [
- "- email-npm: github@email.com",
- ],
- [
- "- guide: https://example.com/guide",
- ],
- [
- "- guide-title: Example Guide",
- ],
- [
- "- logo: https://example.com/logo",
- ],
- [
- "- logo-alt: Logo text.",
- ],
- [
- "- owner: TestOwner",
- ],
- [
- "- repository: test-repository",
- ],
- [
- "- title: Test Title",
- ],
- ]
- `);
- });
-});
diff --git a/src/shared/options/logInferredOptions.ts b/src/shared/options/logInferredOptions.ts
deleted file mode 100644
index 833d25c78..000000000
--- a/src/shared/options/logInferredOptions.ts
+++ /dev/null
@@ -1,39 +0,0 @@
-import chalk from "chalk";
-
-import { logLine } from "../cli/lines.js";
-import { Options } from "../types.js";
-
-export type InferredOptions = Pick<
- Options,
- "description" | "email" | "guide" | "logo" | "owner" | "repository" | "title"
->;
-
-export function logInferredOptions(augmentedOptions: InferredOptions) {
- logLine();
- logLine(chalk.gray("--auto inferred the following values:"));
- logLine(chalk.gray(`- description: ${augmentedOptions.description}`));
- logLine(chalk.gray(`- email-github: ${augmentedOptions.email.github}`));
- logLine(chalk.gray(`- email-npm: ${augmentedOptions.email.github}`));
-
- if (augmentedOptions.guide) {
- logLine(chalk.gray(`- guide: ${augmentedOptions.guide.href}`));
- logLine(chalk.gray(`- guide-title: ${augmentedOptions.guide.title}`));
- }
-
- if (augmentedOptions.logo) {
- logLine(chalk.gray(`- logo: ${augmentedOptions.logo.src}`));
- logLine(chalk.gray(`- logo-alt: ${augmentedOptions.logo.alt}`));
-
- if (augmentedOptions.logo.height) {
- logLine(chalk.gray(`- logo-height: ${augmentedOptions.logo.height}`));
- }
-
- if (augmentedOptions.logo.width) {
- logLine(chalk.gray(`- logo-width: ${augmentedOptions.logo.width}`));
- }
- }
-
- logLine(chalk.gray(`- owner: ${augmentedOptions.owner}`));
- logLine(chalk.gray(`- repository: ${augmentedOptions.repository}`));
- logLine(chalk.gray(`- title: ${augmentedOptions.title}`));
-}
diff --git a/src/shared/options/optionsSchema.ts b/src/shared/options/optionsSchema.ts
deleted file mode 100644
index 3ac1aa70c..000000000
--- a/src/shared/options/optionsSchema.ts
+++ /dev/null
@@ -1,66 +0,0 @@
-import { z } from "zod";
-
-export const optionsSchemaShape = {
- access: z.union([z.literal("public"), z.literal("restricted")]).optional(),
- author: z.string().optional(),
- auto: z.boolean().optional(),
- base: z
- .union([
- z.literal("common"),
- z.literal("everything"),
- z.literal("minimal"),
- z.literal("prompt"),
- ])
- .optional(),
- bin: z.string().optional(),
- description: z.string().optional(),
- directory: z.string().optional(),
- email: z
- .object({
- github: z.string().email(),
- npm: z.string().email(),
- })
- .optional(),
- excludeAllContributors: z.boolean().optional(),
- excludeBuild: z.boolean().optional(),
- excludeCompliance: z.boolean().optional(),
- excludeLintESLint: z.boolean().optional(),
- excludeLintJSDoc: z.boolean().optional(),
- excludeLintJson: z.boolean().optional(),
- excludeLintKnip: z.boolean().optional(),
- excludeLintMd: z.boolean().optional(),
- excludeLintPackageJson: z.boolean().optional(),
- excludeLintPackages: z.boolean().optional(),
- excludeLintPerfectionist: z.boolean().optional(),
- excludeLintRegexp: z.boolean().optional(),
- excludeLintSpelling: z.boolean().optional(),
- excludeLintStrict: z.boolean().optional(),
- excludeLintStylistic: z.boolean().optional(),
- excludeLintYml: z.boolean().optional(),
- excludeReleases: z.boolean().optional(),
- excludeRenovate: z.boolean().optional(),
- excludeTemplatedBy: z.boolean().optional(),
- excludeTests: z.boolean().optional(),
- funding: z.string().optional(),
- guide: z.string().url().optional(),
- guideTitle: z.string().optional(),
- keywords: z.array(z.string()).optional(),
- logo: z.string().optional(),
- logoAlt: z.string().optional(),
- mode: z
- .union([z.literal("create"), z.literal("initialize"), z.literal("migrate")])
- .optional(),
- offline: z.boolean().optional(),
- owner: z.string().optional(),
- preserveGeneratedFrom: z.boolean().optional(),
- repository: z.string().optional(),
- skipAllContributorsApi: z.boolean().optional(),
- skipGitHubApi: z.boolean().optional(),
- skipInstall: z.boolean().optional(),
- skipRemoval: z.boolean().optional(),
- skipRestore: z.boolean().optional(),
- skipUninstall: z.boolean().optional(),
- title: z.string().optional(),
-};
-
-export const optionsSchema = z.object(optionsSchemaShape);
diff --git a/src/shared/options/readOptions.test.ts b/src/shared/options/readOptions.test.ts
deleted file mode 100644
index e9c741bfc..000000000
--- a/src/shared/options/readOptions.test.ts
+++ /dev/null
@@ -1,843 +0,0 @@
-import { describe, expect, it, vi } from "vitest";
-import z from "zod";
-
-import { Options } from "../types.js";
-import { GetPrefillOrPromptedOptionOptions } from "./getPrefillOrPromptedOption.js";
-import { optionsSchemaShape } from "./optionsSchema.js";
-import { readOptions } from "./readOptions.js";
-
-const emptyOptions = {
- access: undefined,
- author: undefined,
- auto: false,
- base: undefined,
- bin: undefined,
- description: undefined,
- directory: undefined,
- email: undefined,
- excludeAllContributors: undefined,
- excludeCompliance: undefined,
- excludeLintESLint: undefined,
- excludeLintJSDoc: undefined,
- excludeLintJson: undefined,
- excludeLintKnip: undefined,
- excludeLintMd: undefined,
- excludeLintPackageJson: undefined,
- excludeLintPackages: undefined,
- excludeLintPerfectionist: undefined,
- excludeLintRegexp: undefined,
- excludeLintSpelling: undefined,
- excludeLintStrict: undefined,
- excludeLintYml: undefined,
- excludeReleases: undefined,
- excludeRenovate: undefined,
- excludeTemplatedBy: undefined,
- excludeTests: undefined,
- funding: undefined,
- guide: undefined,
- guideTitle: undefined,
- logo: undefined,
- logoAlt: undefined,
- offline: undefined,
- owner: undefined,
- preserveGeneratedFrom: false,
- repository: undefined,
- skipAllContributorsApi: undefined,
- skipGitHubApi: undefined,
- skipInstall: undefined,
- skipRemoval: undefined,
- skipRestore: false,
- skipUninstall: undefined,
- title: undefined,
-};
-
-const mockOptions = {
- base: "prompt",
- github: "mock.git",
- repository: "mock.repository",
-};
-
-vi.mock("../cli/spinners.ts", () => ({
- withSpinner() {
- return () => ({});
- },
-}));
-
-const mockReadPackageData = vi.fn().mockResolvedValue({});
-
-vi.mock("../packages.js", () => ({
- get readPackageData() {
- return mockReadPackageData;
- },
-}));
-
-const mockAugmentOptionsWithExcludes = vi.fn();
-
-vi.mock("./augmentOptionsWithExcludes.js", () => ({
- get augmentOptionsWithExcludes() {
- return mockAugmentOptionsWithExcludes;
- },
-}));
-
-const mockDetectEmailRedundancy = vi.fn();
-
-vi.mock("./detectEmailRedundancy.js", () => ({
- get detectEmailRedundancy() {
- return mockDetectEmailRedundancy;
- },
-}));
-
-const mockGetPrefillOrPromptedOption = vi.fn();
-
-vi.mock("./getPrefillOrPromptedOption.js", () => ({
- get getPrefillOrPromptedOption() {
- return mockGetPrefillOrPromptedOption;
- },
-}));
-
-const mockEnsureRepositoryExists = vi.fn();
-
-vi.mock("./ensureRepositoryExists.js", () => ({
- get ensureRepositoryExists() {
- return mockEnsureRepositoryExists;
- },
-}));
-
-vi.mock("./getGitHub.js", () => ({
- getGitHub() {
- return undefined;
- },
-}));
-
-vi.mock("./createOptionDefaults/index.js", () => ({
- createOptionDefaults() {
- return {
- author: vi.fn(),
- bin: vi.fn(),
- description: vi.fn(),
- email: vi.fn(),
- funding: vi.fn(),
- guide: vi.fn(),
- logo: vi.fn(),
- owner: vi.fn(),
- repository: vi.fn(),
- title: vi.fn(),
- };
- },
-}));
-
-const mockLogInferredOptions = vi.fn();
-
-vi.mock("./logInferredOptions.js", () => ({
- get logInferredOptions() {
- return mockLogInferredOptions;
- },
-}));
-
-describe("readOptions", () => {
- it("returns a cancellation when an arg is invalid", async () => {
- const validationResult = z
- .object({ base: optionsSchemaShape.base })
- .safeParse({ base: "b" });
-
- const actual = await readOptions(["--base", "b"], "migrate");
-
- expect(actual).toStrictEqual({
- cancelled: true,
- error: (validationResult as z.SafeParseError<{ base: string }>).error,
- options: { ...emptyOptions, base: "b" },
- });
- });
-
- it("returns a cancellation when an email redundancy is detected", async () => {
- const error = "Too many emails!";
- mockDetectEmailRedundancy.mockReturnValue(error);
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: undefined,
- }));
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error,
- options: {
- ...emptyOptions,
- base: "minimal",
- },
- });
- });
-
- it("returns a cancellation when the owner prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: undefined,
- }));
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- },
- });
- });
-
- it("returns a cancellation when the repository prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementation(() => ({ value: undefined }));
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- owner: "MockOwner",
- },
- });
- });
-
- it("returns a cancellation when ensureRepositoryPrompt does not return a repository", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({});
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- owner: "MockOwner",
- repository: "MockRepository",
- },
- });
- });
-
- it("returns a cancellation when the description prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- owner: "MockOwner",
- repository: "MockRepository",
- },
- });
- });
-
- it("returns a cancellation when the title prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementationOnce(() => ({ value: "Mock description." }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- description: "Mock description.",
- owner: "MockOwner",
- repository: "MockRepository",
- },
- });
- });
-
- it("returns a cancellation when the guide title prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementationOnce(() => ({ value: "Mock description." }))
- .mockImplementationOnce(() => ({ value: "Mock Title" }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(
- await readOptions(["--guide", "https://example.com"], "migrate"),
- ).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- description: "Mock description.",
- guide: "https://example.com",
- owner: "MockOwner",
- repository: "MockRepository",
- title: "Mock Title",
- },
- });
- });
-
- it("returns a cancellation when the guide alt prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementationOnce(() => ({ value: "Mock description." }))
- .mockImplementationOnce(() => ({ value: "Mock Title" }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(
- await readOptions(["--guide", "https://example.com"], "migrate"),
- ).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- description: "Mock description.",
- guide: "https://example.com",
- owner: "MockOwner",
- repository: "MockRepository",
- title: "Mock Title",
- },
- });
- });
-
- it("returns a cancellation when the logo alt prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementationOnce(() => ({ value: "Mock description." }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(await readOptions(["--logo", "logo.svg"], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- description: "Mock description.",
- logo: "logo.svg",
- owner: "MockOwner",
- repository: "MockRepository",
- },
- });
- });
-
- it("returns a cancellation when the email prompt is cancelled", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementationOnce(() => ({ value: "Mock description." }))
- .mockImplementationOnce(() => ({ value: "Mock title." }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- description: "Mock description.",
- owner: "MockOwner",
- repository: "MockRepository",
- title: "Mock title.",
- },
- });
- });
-
- it("returns a cancellation when augmentOptionsWithExcludes returns undefined", async () => {
- mockDetectEmailRedundancy.mockReturnValue(false);
- mockGetPrefillOrPromptedOption
- .mockImplementationOnce(() => ({ value: "MockOwner" }))
- .mockImplementationOnce(() => ({ value: "MockRepository" }))
- .mockImplementationOnce(() => ({ value: "Mock description." }))
- .mockImplementationOnce(() => ({ value: "Mock title." }))
- .mockImplementation(() => ({ value: undefined }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
- mockAugmentOptionsWithExcludes.mockResolvedValue(undefined);
-
- expect(await readOptions([], "migrate")).toStrictEqual({
- cancelled: true,
- error: undefined,
- options: {
- ...emptyOptions,
- base: "minimal",
- description: "Mock description.",
- owner: "MockOwner",
- repository: "MockRepository",
- title: "Mock title.",
- },
- });
- });
-
- it("returns success options when --base is valid", async () => {
- mockAugmentOptionsWithExcludes.mockResolvedValue({
- ...emptyOptions,
- ...mockOptions,
- });
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(
- await readOptions(["--base", mockOptions.base], "migrate"),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: {
- ...emptyOptions,
- ...mockOptions,
- },
- });
- });
-
- it("returns success options when --base is valid with all optional options", async () => {
- mockAugmentOptionsWithExcludes.mockResolvedValue({
- ...emptyOptions,
- ...mockOptions,
- });
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(
- await readOptions(
- [
- "--base",
- mockOptions.base,
- "--guide",
- "https://example.com",
- "--logo",
- "logo.svg",
- ],
- "migrate",
- ),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: {
- ...emptyOptions,
- ...mockOptions,
- },
- });
- });
-
- it("returns cancelled options when augmentOptionsWithExcludes returns undefined", async () => {
- mockAugmentOptionsWithExcludes.mockResolvedValue(undefined);
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
-
- expect(
- await readOptions(["--base", mockOptions.base], "migrate"),
- ).toStrictEqual({
- cancelled: true,
- options: {
- ...emptyOptions,
- base: mockOptions.base,
- description: "mock",
- owner: "mock",
- repository: "mock",
- title: "mock",
- },
- });
- });
-
- it("defaults preserveGeneratedFrom to false when the owner is not JoshuaKGoldberg", async () => {
- mockAugmentOptionsWithExcludes.mockImplementationOnce(
- (options: Partial) => ({
- ...options,
- ...mockOptions,
- }),
- );
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
-
- expect(
- await readOptions(["--base", mockOptions.base], "migrate"),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: expect.objectContaining({
- preserveGeneratedFrom: false,
- }),
- });
- });
-
- it("defaults preserveGeneratedFrom to true when the owner is JoshuaKGoldberg", async () => {
- mockAugmentOptionsWithExcludes.mockImplementationOnce(
- (options: Partial) => ({
- ...options,
- ...mockOptions,
- }),
- );
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
-
- expect(
- await readOptions(
- ["--base", mockOptions.base, "--owner", "JoshuaKGoldberg"],
- "migrate",
- ),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: expect.objectContaining({
- preserveGeneratedFrom: true,
- }),
- });
- });
-
- it("defaults skipRestore to false when the mode is not create", async () => {
- mockAugmentOptionsWithExcludes.mockImplementationOnce(
- (options: Partial) => ({
- ...options,
- ...mockOptions,
- }),
- );
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
-
- expect(
- await readOptions(["--base", mockOptions.base], "migrate"),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: expect.objectContaining({
- skipRestore: false,
- }),
- });
- });
-
- it("defaults skipRestore to true when the mode is create", async () => {
- mockAugmentOptionsWithExcludes.mockImplementationOnce(
- (options: Partial) => ({
- ...options,
- ...mockOptions,
- }),
- );
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
-
- expect(
- await readOptions(["--base", mockOptions.base], "create"),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: expect.objectContaining({
- skipRestore: true,
- }),
- });
- });
-
- it("skips API calls when --offline is true", async () => {
- mockAugmentOptionsWithExcludes.mockImplementation((options: Options) => ({
- ...emptyOptions,
- ...mockOptions,
- ...options,
- }));
- mockGetPrefillOrPromptedOption.mockImplementation(() => ({
- value: "mock",
- }));
- mockEnsureRepositoryExists.mockResolvedValue({
- octokit: undefined,
- repository: mockOptions.repository,
- });
-
- expect(
- await readOptions(["--base", mockOptions.base, "--offline"], "migrate"),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: {
- ...emptyOptions,
- ...mockOptions,
- access: "public",
- description: "mock",
- directory: "mock",
- email: {
- github: "mock",
- npm: "mock",
- },
- guide: undefined,
- logo: undefined,
- mode: "migrate",
- offline: true,
- owner: "mock",
- skipAllContributorsApi: true,
- skipGitHubApi: true,
- title: "mock",
- },
- });
- });
-
- it("infers base from package scripts during migration", async () => {
- mockReadPackageData.mockImplementationOnce(() =>
- Promise.resolve({
- scripts: {
- build: "build",
- lint: "lint",
- test: "test",
- },
- }),
- );
- expect(await readOptions(["--offline"], "migrate")).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: {
- ...emptyOptions,
- ...mockOptions,
- access: "public",
- base: "minimal",
- description: "mock",
- directory: "mock",
- email: {
- github: "mock",
- npm: "mock",
- },
- guide: undefined,
- logo: undefined,
- mode: "migrate",
- offline: true,
- owner: "mock",
- skipAllContributorsApi: true,
- skipGitHubApi: true,
- title: "mock",
- },
- });
- });
-
- it("uses values when provided on the CLI", async () => {
- mockReadPackageData.mockImplementationOnce(() => ({}));
- mockGetPrefillOrPromptedOption.mockImplementation(
- async ({ getDefaultValue }: GetPrefillOrPromptedOptionOptions) => ({
- value: (await getDefaultValue?.()) ?? "mock",
- }),
- );
-
- const description = "Test description.";
- const owner = "TestOwner";
- const repository = "test-repository";
- const title = "Test Title";
-
- expect(
- await readOptions(
- [
- "--offline",
- "--description",
- description,
- "--owner",
- owner,
- "--repository",
- repository,
- "--title",
- title,
- ],
- "migrate",
- ),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: {
- ...emptyOptions,
- ...mockOptions,
- access: "public",
- base: "minimal",
- description,
- directory: repository,
- email: {
- github: "mock",
- npm: "mock",
- },
- guide: undefined,
- logo: undefined,
- mode: "migrate",
- offline: true,
- owner,
- skipAllContributorsApi: true,
- skipGitHubApi: true,
- title,
- },
- });
-
- expect(mockLogInferredOptions).not.toHaveBeenCalled();
- });
-
- it("uses and logs values when provided on the CLI with auto", async () => {
- mockReadPackageData.mockImplementationOnce(() => ({}));
- mockGetPrefillOrPromptedOption.mockImplementation(
- async ({ getDefaultValue }: GetPrefillOrPromptedOptionOptions) => ({
- value: (await getDefaultValue?.()) ?? "mock",
- }),
- );
-
- const description = "Test description.";
- const owner = "TestOwner";
- const repository = "test-repository";
- const title = "Test Title";
-
- expect(
- await readOptions(
- [
- "--auto",
- "--offline",
- "--description",
- description,
- "--owner",
- owner,
- "--repository",
- repository,
- "--title",
- title,
- ],
- "migrate",
- ),
- ).toStrictEqual({
- cancelled: false,
- octokit: undefined,
- options: {
- ...emptyOptions,
- ...mockOptions,
- access: "public",
- auto: true,
- base: "minimal",
- description,
- directory: repository,
- email: {
- github: "mock",
- npm: "mock",
- },
- guide: undefined,
- logo: undefined,
- mode: "migrate",
- offline: true,
- owner,
- skipAllContributorsApi: true,
- skipGitHubApi: true,
- title,
- },
- });
-
- expect(mockLogInferredOptions.mock.calls).toMatchInlineSnapshot(`
- [
- [
- {
- "access": "public",
- "author": undefined,
- "auto": true,
- "base": "minimal",
- "bin": undefined,
- "description": "Test description.",
- "directory": "test-repository",
- "email": {
- "github": "mock",
- "npm": "mock",
- },
- "excludeAllContributors": undefined,
- "excludeCompliance": undefined,
- "excludeLintESLint": undefined,
- "excludeLintJSDoc": undefined,
- "excludeLintJson": undefined,
- "excludeLintKnip": undefined,
- "excludeLintMd": undefined,
- "excludeLintPackageJson": undefined,
- "excludeLintPackages": undefined,
- "excludeLintPerfectionist": undefined,
- "excludeLintRegexp": undefined,
- "excludeLintSpelling": undefined,
- "excludeLintStrict": undefined,
- "excludeLintYml": undefined,
- "excludeReleases": undefined,
- "excludeRenovate": undefined,
- "excludeTemplatedBy": undefined,
- "excludeTests": undefined,
- "funding": undefined,
- "github": "mock.git",
- "guide": undefined,
- "guideTitle": undefined,
- "logo": undefined,
- "logoAlt": undefined,
- "mode": "migrate",
- "offline": true,
- "owner": "TestOwner",
- "preserveGeneratedFrom": false,
- "repository": "mock.repository",
- "skipAllContributorsApi": true,
- "skipGitHubApi": true,
- "skipInstall": undefined,
- "skipRemoval": undefined,
- "skipRestore": false,
- "skipUninstall": undefined,
- "title": "Test Title",
- },
- ],
- ]
- `);
- });
-});
diff --git a/src/shared/options/readOptions.ts b/src/shared/options/readOptions.ts
deleted file mode 100644
index 181bde21e..000000000
--- a/src/shared/options/readOptions.ts
+++ /dev/null
@@ -1,305 +0,0 @@
-import { parseArgs } from "node:util";
-import { Octokit } from "octokit";
-import { octokitFromAuth } from "octokit-from-auth";
-import { titleCase } from "title-case";
-import { z } from "zod";
-
-import { withSpinner } from "../cli/spinners.js";
-import { Mode, OptionsGuide, PromptedOptions } from "../types.js";
-import { Options, OptionsLogo } from "../types.js";
-import { allArgOptions } from "./args.js";
-import { augmentOptionsWithExcludes } from "./augmentOptionsWithExcludes.js";
-import { createOptionDefaults } from "./createOptionDefaults/index.js";
-import { detectEmailRedundancy } from "./detectEmailRedundancy.js";
-import { ensureRepositoryExists } from "./ensureRepositoryExists.js";
-import { getBase } from "./getBase.js";
-import { getPrefillOrPromptedOption } from "./getPrefillOrPromptedOption.js";
-import { logInferredOptions } from "./logInferredOptions.js";
-import { optionsSchema } from "./optionsSchema.js";
-import { readLogoSizing } from "./readLogoSizing.js";
-
-export interface OctokitAndOptions {
- octokit: Octokit | undefined;
- options: Options;
-}
-
-export interface OptionsParseCancelled {
- cancelled: true;
- error?: string | z.ZodError
+
## Usage
Use it.
@@ -160,6 +161,7 @@ describe("blockREADME", () => {
+
## Usage
Use it.
diff --git a/src/blocks/blockREADME.ts b/src/blocks/blockREADME.ts
index fc386be24..f7991b676 100644
--- a/src/blocks/blockREADME.ts
+++ b/src/blocks/blockREADME.ts
@@ -23,9 +23,10 @@ export const blockREADME = base.createBlock({
const logo =
options.logo &&
- `\n`;
+ `\n\n`;
- const explainer = options.explainer && `\n${options.explainer.join("\n")}`;
+ const explainer =
+ options.explainer && `\n${options.explainer.join("\n")}\n`;
return {
files: {
diff --git a/src/integration.test.ts b/src/integration.test.ts
index 6ee316fda..767b6b8ae 100644
--- a/src/integration.test.ts
+++ b/src/integration.test.ts
@@ -137,20 +137,6 @@ If you're interested in learning more, see the 'getting started' docs on:
",
},
},
- ".vscode": {
- "launch.json": "@@ -13,9 +13,9 @@
- },
- {
- "name": "Debug Program",
- "preLaunchTask": "build",
- - "program": "./bin/index.js",
- + "program": "bin/index.js",
- "request": "launch",
- "skipFiles": ["/**"],
- "type": "node"
- }
- ",
- },
"README.md": "@@ -1,9 +1,9 @@
Create TypeScript App
diff --git a/src/shared/options/createOptionDefaults/index.test.ts b/src/shared/options/createOptionDefaults/index.test.ts
deleted file mode 100644
index efb36a8fc..000000000
--- a/src/shared/options/createOptionDefaults/index.test.ts
+++ /dev/null
@@ -1,218 +0,0 @@
-import { beforeEach, describe, expect, it, vi } from "vitest";
-
-import { createOptionDefaults } from "./index.js";
-
-const mock$ = vi.fn();
-
-vi.mock("execa", () => ({
- get $() {
- return mock$;
- },
-}));
-
-const mockGitUrlParse = vi.fn();
-
-vi.mock("git-url-parse", () => ({
- get default() {
- return mockGitUrlParse;
- },
-}));
-
-const mockNpmUser = vi.fn();
-
-vi.mock("npm-user", () => ({
- get default() {
- return mockNpmUser;
- },
-}));
-
-const mockReadPackageData = vi.fn();
-
-vi.mock("../../packages.js", () => ({
- get readPackageData() {
- return mockReadPackageData;
- },
-}));
-
-const mockReadGitHubEmail = vi.fn();
-
-vi.mock("./readGitHubEmail.js", () => ({
- get readGitHubEmail() {
- return mockReadGitHubEmail;
- },
-}));
-
-describe("createOptionDefaults", () => {
- describe("bin", () => {
- it("returns undefined when package data does not have a bin", async () => {
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().bin();
-
- expect(actual).toBeUndefined();
- });
-
- it("returns the bin when package data has a bin", async () => {
- const bin = "lib/index.js";
-
- mockReadPackageData.mockResolvedValue({ bin });
-
- const actual = await createOptionDefaults().bin();
-
- expect(actual).toBe(bin);
- });
- });
-
- describe("email", () => {
- beforeEach(() => {
- mockNpmUser.mockImplementation((username: string) => ({
- email: `npm-${username}@test.com`,
- }));
- });
-
- it("returns the npm whoami email from npm when only an npm exists", async () => {
- mock$.mockImplementation(([command]: string[]) =>
- command === "npm whoami" ? { stdout: "username" } : undefined,
- );
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "npm-username@test.com",
- npm: "npm-username@test.com",
- });
- });
-
- it("returns the npm whoami email from npm when only a package author email exists", async () => {
- mock$.mockResolvedValue({ stdout: "" });
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({
- author: {
- email: "test@package.com",
- },
- });
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "test@package.com",
- npm: "test@package.com",
- });
- });
-
- it("returns the github email when only a github email exists", async () => {
- mock$.mockResolvedValue({ stdout: "" });
- mockReadPackageData.mockResolvedValueOnce({});
- mockReadGitHubEmail.mockResolvedValueOnce("github@test.com");
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "github@test.com",
- npm: "github@test.com",
- });
- });
-
- it("returns the git user email when only a git user email exists", async () => {
- mock$.mockImplementation(([command]: string[]) =>
- command === "git config --get user.email"
- ? { stdout: "git@test.com" }
- : undefined,
- );
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "git@test.com",
- npm: "git@test.com",
- });
- });
-
- it("returns both the git user email and the npm user email when only those two exist", async () => {
- mock$.mockImplementation(([command]: string[]) => ({
- stdout:
- command === "git config --get user.email"
- ? "git@test.com"
- : "username",
- }));
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "git@test.com",
- npm: "npm-username@test.com",
- });
- });
-
- it("returns all three emails when they all exist", async () => {
- mock$.mockImplementation(([command]: string[]) => ({
- stdout:
- command === "git config --get user.email"
- ? "git@test.com"
- : "username",
- }));
- mockReadGitHubEmail.mockResolvedValueOnce("github@test.com");
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toEqual({
- github: "github@test.com",
- npm: "npm-username@test.com",
- });
- });
-
- it("returns undefined when none of the emails exist", async () => {
- mock$.mockResolvedValue({ stdout: "" });
- mockReadGitHubEmail.mockResolvedValueOnce(undefined);
- mockReadPackageData.mockResolvedValue({});
-
- const actual = await createOptionDefaults().email();
-
- expect(actual).toBeUndefined();
- });
- });
-
- describe("repository", () => {
- it("returns promptedOptions.repository when it exists", async () => {
- const repository = "test-prompted-repository";
- const promptedOptions = { repository };
- const actual = await createOptionDefaults(promptedOptions).repository();
-
- expect(actual).toBe(repository);
- });
-
- it("returns the Git name when it exists and promptedOptions.repository doesn't", async () => {
- const name = "test-git-repository";
- mockGitUrlParse.mockResolvedValueOnce({ name });
-
- const actual = await createOptionDefaults().repository();
-
- expect(actual).toBe(name);
- });
-
- it("returns the package name when it exists and promptedOptions.repository and Git name don't", async () => {
- const name = "test-package-name";
- mockReadPackageData.mockResolvedValueOnce({ name });
-
- const actual = await createOptionDefaults().repository();
-
- expect(actual).toBe(name);
- });
-
- it("returns the directory when it exists and promptedOptions.repository, Git name, and package name don't", async () => {
- mockReadPackageData.mockResolvedValueOnce({});
- const directory = "test-prompted-directory";
- const promptedOptions = { directory };
-
- const actual = await createOptionDefaults(promptedOptions).repository();
-
- expect(actual).toBe(directory);
- });
- });
-});
From c4625dd2858cc76a7cdbfffe442e1de9e417b6c5 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 09:46:40 -0500
Subject: [PATCH 07/22] update integration snapshot for knip fix
---
src/integration.test.ts | 10 ----------
1 file changed, 10 deletions(-)
diff --git a/src/integration.test.ts b/src/integration.test.ts
index 767b6b8ae..d43624e45 100644
--- a/src/integration.test.ts
+++ b/src/integration.test.ts
@@ -176,16 +176,6 @@ If you're interested in learning more, see the 'getting started' docs on:
import eslint from "@eslint/js";
import vitest from "@vitest/eslint-plugin";
import jsdoc from "eslint-plugin-jsdoc";
- ",
- "knip.json": "@@ -1,7 +1,7 @@
- {
- "$schema": "https://unpkg.com/knip@5.41.1/schema.json",
- - "entry": ["src/index.ts", "src/**/*.test.*"],
- + "entry": ["src/index.ts"],
- "ignoreDependencies": ["all-contributors-cli", "cspell-populate-words"],
- "ignoreExportsUsedInFile": { "interface": true, "type": true },
- "project": ["src/**/*.ts"]
- }
",
}
`);
From 04c3c749c14089377aa5cef29dbe109b19474d34 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 09:47:32 -0500
Subject: [PATCH 08/22] update integration snapshot for eslint newline removal
---
eslint.config.js | 1 -
src/integration.test.ts | 11 -----------
2 files changed, 12 deletions(-)
diff --git a/eslint.config.js b/eslint.config.js
index 758d20cbb..055d83f7e 100644
--- a/eslint.config.js
+++ b/eslint.config.js
@@ -7,7 +7,6 @@ If you're interested in learning more, see the 'getting started' docs on:
- ESLint: https://eslint.org
- typescript-eslint: https://typescript-eslint.io
*/
-
import comments from "@eslint-community/eslint-plugin-eslint-comments/configs";
import eslint from "@eslint/js";
import vitest from "@vitest/eslint-plugin";
diff --git a/src/integration.test.ts b/src/integration.test.ts
index d43624e45..7ceab3388 100644
--- a/src/integration.test.ts
+++ b/src/integration.test.ts
@@ -165,17 +165,6 @@ If you're interested in learning more, see the 'getting started' docs on:
-
- ",
- "eslint.config.js": "@@ -6,9 +6,8 @@
- If you're interested in learning more, see the 'getting started' docs on:
- - ESLint: https://eslint.org
- - typescript-eslint: https://typescript-eslint.io
- */
- -
- import comments from "@eslint-community/eslint-plugin-eslint-comments/configs";
- import eslint from "@eslint/js";
- import vitest from "@vitest/eslint-plugin";
- import jsdoc from "eslint-plugin-jsdoc";
",
}
`);
From 82ed2c1e7385d7f7e951ad5c2a9df60702a6bb10 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 09:51:08 -0500
Subject: [PATCH 09/22] Merge branch 'main'
---
eslint.config.js | 1 +
1 file changed, 1 insertion(+)
diff --git a/eslint.config.js b/eslint.config.js
index 055d83f7e..758d20cbb 100644
--- a/eslint.config.js
+++ b/eslint.config.js
@@ -7,6 +7,7 @@ If you're interested in learning more, see the 'getting started' docs on:
- ESLint: https://eslint.org
- typescript-eslint: https://typescript-eslint.io
*/
+
import comments from "@eslint-community/eslint-plugin-eslint-comments/configs";
import eslint from "@eslint/js";
import vitest from "@vitest/eslint-plugin";
From 84227547b652726a7d3fa023eea15f40ad9f8b03 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 09:56:00 -0500
Subject: [PATCH 10/22] fix: blockAllContributors table needs to also disable
cspell
---
README.md | 4 ++--
src/blocks/blockAllContributors.test.ts | 4 +++-
src/blocks/blockAllContributors.ts | 2 ++
src/integration.test.ts | 29 -------------------------
4 files changed, 7 insertions(+), 32 deletions(-)
diff --git a/README.md b/README.md
index e17c9bf57..97fc7ceea 100644
--- a/README.md
+++ b/README.md
@@ -1,8 +1,8 @@
Create TypeScript App
- Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling.
- 🎁
+ Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling.
+ 🎁
diff --git a/src/blocks/blockAllContributors.test.ts b/src/blocks/blockAllContributors.test.ts
index 59151d897..616330758 100644
--- a/src/blocks/blockAllContributors.test.ts
+++ b/src/blocks/blockAllContributors.test.ts
@@ -116,6 +116,7 @@ describe("blockAllContributors", () => {
"sections": [
"## Contributors
+
@@ -130,7 +131,8 @@ describe("blockAllContributors", () => {
- ",
+
+ ",
],
},
"block": [Function],
diff --git a/src/blocks/blockAllContributors.ts b/src/blocks/blockAllContributors.ts
index cc8f02013..6a996940a 100644
--- a/src/blocks/blockAllContributors.ts
+++ b/src/blocks/blockAllContributors.ts
@@ -81,6 +81,7 @@ function printAllContributorsTable(contributors: Contributor[]) {
return [
`## Contributors`,
``,
+ ``,
``,
``,
``,
@@ -102,6 +103,7 @@ function printAllContributorsTable(contributors: Contributor[]) {
``,
``,
``,
+ ``,
].join("\n");
}
diff --git a/src/integration.test.ts b/src/integration.test.ts
index 7ceab3388..2845937fb 100644
--- a/src/integration.test.ts
+++ b/src/integration.test.ts
@@ -137,35 +137,6 @@ If you're interested in learning more, see the 'getting started' docs on:
",
},
},
- "README.md": "@@ -1,9 +1,9 @@
-
Create TypeScript App
-
-
- - Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling.
- - 🎁
- + Quickstart-friendly TypeScript template with comprehensive, configurable, opinionated tooling.
- + 🎁
-
-
-
-
- @@ -64,9 +64,8 @@
- Thanks! 💖
-
- ## Contributors
-
- -
-
-
-
-
- @@ -145,5 +144,4 @@
-
-
-
-
- -
- ",
}
`);
});
From 158010b04ae5dafa418c5b0e4709ae7f7ebab23e Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 09:59:53 -0500
Subject: [PATCH 11/22] add CODECOV_TOKEN to integration test
---
create.config.ts | 56 -----------------------------------------
src/blocks/index.ts | 3 +++
src/integration.test.ts | 20 +++++----------
3 files changed, 9 insertions(+), 70 deletions(-)
delete mode 100644 create.config.ts
diff --git a/create.config.ts b/create.config.ts
deleted file mode 100644
index 01bb5e71c..000000000
--- a/create.config.ts
+++ /dev/null
@@ -1,56 +0,0 @@
-import { createConfig } from "create";
-
-import {
- blockAreTheTypesWrong,
- blockESLint,
- blockTemplatedWith,
- blockTSup,
- presets,
-} from "./src/index.ts";
-
-export default createConfig(presets.everything, {
- addons: [
- blockESLint({
- explanations: [
- `👋 Hi! This ESLint configuration contains a lot more stuff than many repos'!
-You can read from it to see all sorts of linting goodness, but don't worry -
-it's not something you need to exhaustively understand immediately. 💙
-
-If you're interested in learning more, see the 'getting started' docs on:
-- ESLint: https://eslint.org
-- typescript-eslint: https://typescript-eslint.io`,
- ],
- rules: [
- {
- comment:
- "These on-by-default rules work well for this repo if configured",
- entries: {
- "@typescript-eslint/no-unnecessary-condition": [
- "error",
- { allowConstantLoopConditions: true },
- ],
- "@typescript-eslint/prefer-nullish-coalescing": [
- "error",
- { ignorePrimitives: true },
- ],
- "@typescript-eslint/restrict-template-expressions": [
- "error",
- { allowBoolean: true, allowNullish: true, allowNumber: true },
- ],
- "n/no-unsupported-features/node-builtins": [
- "error",
- { allowExperimental: true },
- ],
- },
- },
- ],
- }),
- blockTSup({
- runArgs: ["--version"],
- }),
- ],
- blocks: {
- add: [blockAreTheTypesWrong],
- exclude: [blockTemplatedWith],
- },
-});
diff --git a/src/blocks/index.ts b/src/blocks/index.ts
index efc0fb385..300b4cf0a 100644
--- a/src/blocks/index.ts
+++ b/src/blocks/index.ts
@@ -1,5 +1,6 @@
import { blockAllContributors } from "./blockAllContributors.js";
import { blockAreTheTypesWrong } from "./blockAreTheTypesWrong.js";
+import { blockCodecov } from "./blockCodecov.js";
import { blockContributingDocs } from "./blockContributingDocs.js";
import { blockContributorCovenant } from "./blockContributorCovenant.js";
import { blockCSpell } from "./blockCSpell.js";
@@ -44,6 +45,7 @@ import { blockVSCode } from "./blockVSCode.js";
export const blocks = {
blockAllContributors,
blockAreTheTypesWrong,
+ blockCodecov,
blockContributingDocs,
blockContributorCovenant,
blockCSpell,
@@ -88,6 +90,7 @@ export const blocks = {
export { blockAllContributors } from "./blockAllContributors.js";
export { blockAreTheTypesWrong } from "./blockAreTheTypesWrong.js";
+export { blockCodecov } from "./blockCodecov.js";
export { blockContributingDocs } from "./blockContributingDocs.js";
export { blockContributorCovenant } from "./blockContributorCovenant.js";
export { blockCSpell } from "./blockCSpell.js";
diff --git a/src/integration.test.ts b/src/integration.test.ts
index 2845937fb..1ee3b31b7 100644
--- a/src/integration.test.ts
+++ b/src/integration.test.ts
@@ -8,6 +8,7 @@ import {
base,
BaseOptions,
blockAreTheTypesWrong,
+ blockCodecov,
blockCSpell,
blockESLint,
blockKnip,
@@ -39,6 +40,11 @@ test("Producing the everything preset matches the files in this repository", asy
const created = await producePreset(presets.everything, {
addons: [
+ blockCodecov({
+ env: {
+ CODECOV_TOKEN: "${{ secrets.CODECOV_TOKEN }}",
+ },
+ }),
blockCSpell({
words: [
"Anson",
@@ -105,20 +111,6 @@ If you're interested in learning more, see the 'getting started' docs on:
{
".github": {
"workflows": {
- "ci.yml": "@@ -64,11 +64,9 @@
- steps:
- - uses: actions/checkout@v4
- - uses: ./.github/actions/prepare
- - run: pnpm run test --coverage
- - - env:
- - CODECOV_TOKEN: \${{ secrets.CODECOV_TOKEN }}
- - if: always()
- + - if: always()
- uses: codecov/codecov-action@v3
- type_check:
- name: Type Check
- runs-on: ubuntu-latest
- ",
"release.yml": "@@ -14,13 +14,9 @@
- run: pnpm build
- env:
From c3af52124bb2062fd30520e0d239dcb72b61d388 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 10:05:36 -0500
Subject: [PATCH 12/22] docs: add missing Codecov to Blocks.md
---
docs/Blocks.md | 1 +
1 file changed, 1 insertion(+)
diff --git a/docs/Blocks.md b/docs/Blocks.md
index 6ff111b38..ccb908ffe 100644
--- a/docs/Blocks.md
+++ b/docs/Blocks.md
@@ -12,6 +12,7 @@ This table summarizes each block and which base levels they're included in:
| Contributing Docs | `--exclude-contributing-docs` | ✔️ | ✅ | 💯 |
| Contributor Covenant | `--exclude-contributor-covenant` | ✔️ | ✅ | 💯 |
| CSpell | `--exclude-cspell` | | | 💯 |
+| Codecov | `--exclude-codecov` | | ✅ | 💯 |
| Development Docs | `--exclude-development-docs` | ✔️ | ✅ | 💯 |
| ESLint | `--exclude-eslint` | ✔️ | ✅ | 💯 |
| ESLint Comments Plugin | `--exclude-eslint-comments-plugin` | | | 💯 |
From 1fcff5fda4f6336dd17aa5c9a5fe83acd5f9edf2 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 10:18:56 -0500
Subject: [PATCH 13/22] bump get-github-auth-token to avoid crash
---
pnpm-lock.yaml | 210 ++++++++++++++++++++++++-------------------------
1 file changed, 105 insertions(+), 105 deletions(-)
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index 2487195e8..feefb85ff 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -509,8 +509,8 @@ packages:
cpu: [ppc64]
os: [aix]
- '@esbuild/aix-ppc64@0.23.0':
- resolution: {integrity: sha512-3sG8Zwa5fMcA9bgqB8AfWPQ+HFke6uD3h1s3RIwUNK8EG7a4buxvuFTs3j1IMs2NXAk9F30C/FF4vxRgQCcmoQ==}
+ '@esbuild/aix-ppc64@0.23.1':
+ resolution: {integrity: sha512-6VhYk1diRqrhBAqpJEdjASR/+WVRtfjpqKuNw11cLiaWpAT/Uu+nokB+UJnevzy/P9C/ty6AOe0dwueMrGh/iQ==}
engines: {node: '>=18'}
cpu: [ppc64]
os: [aix]
@@ -527,8 +527,8 @@ packages:
cpu: [arm64]
os: [android]
- '@esbuild/android-arm64@0.23.0':
- resolution: {integrity: sha512-EuHFUYkAVfU4qBdyivULuu03FhJO4IJN9PGuABGrFy4vUuzk91P2d+npxHcFdpUnfYKy0PuV+n6bKIpHOB3prQ==}
+ '@esbuild/android-arm64@0.23.1':
+ resolution: {integrity: sha512-xw50ipykXcLstLeWH7WRdQuysJqejuAGPd30vd1i5zSyKK3WE+ijzHmLKxdiCMtH1pHz78rOg0BKSYOSB/2Khw==}
engines: {node: '>=18'}
cpu: [arm64]
os: [android]
@@ -545,8 +545,8 @@ packages:
cpu: [arm]
os: [android]
- '@esbuild/android-arm@0.23.0':
- resolution: {integrity: sha512-+KuOHTKKyIKgEEqKbGTK8W7mPp+hKinbMBeEnNzjJGyFcWsfrXjSTNluJHCY1RqhxFurdD8uNXQDei7qDlR6+g==}
+ '@esbuild/android-arm@0.23.1':
+ resolution: {integrity: sha512-uz6/tEy2IFm9RYOyvKl88zdzZfwEfKZmnX9Cj1BHjeSGNuGLuMD1kR8y5bteYmwqKm1tj8m4cb/aKEorr6fHWQ==}
engines: {node: '>=18'}
cpu: [arm]
os: [android]
@@ -563,8 +563,8 @@ packages:
cpu: [x64]
os: [android]
- '@esbuild/android-x64@0.23.0':
- resolution: {integrity: sha512-WRrmKidLoKDl56LsbBMhzTTBxrsVwTKdNbKDalbEZr0tcsBgCLbEtoNthOW6PX942YiYq8HzEnb4yWQMLQuipQ==}
+ '@esbuild/android-x64@0.23.1':
+ resolution: {integrity: sha512-nlN9B69St9BwUoB+jkyU090bru8L0NA3yFvAd7k8dNsVH8bi9a8cUAUSEcEEgTp2z3dbEDGJGfP6VUnkQnlReg==}
engines: {node: '>=18'}
cpu: [x64]
os: [android]
@@ -581,8 +581,8 @@ packages:
cpu: [arm64]
os: [darwin]
- '@esbuild/darwin-arm64@0.23.0':
- resolution: {integrity: sha512-YLntie/IdS31H54Ogdn+v50NuoWF5BDkEUFpiOChVa9UnKpftgwzZRrI4J132ETIi+D8n6xh9IviFV3eXdxfow==}
+ '@esbuild/darwin-arm64@0.23.1':
+ resolution: {integrity: sha512-YsS2e3Wtgnw7Wq53XXBLcV6JhRsEq8hkfg91ESVadIrzr9wO6jJDMZnCQbHm1Guc5t/CdDiFSSfWP58FNuvT3Q==}
engines: {node: '>=18'}
cpu: [arm64]
os: [darwin]
@@ -599,8 +599,8 @@ packages:
cpu: [x64]
os: [darwin]
- '@esbuild/darwin-x64@0.23.0':
- resolution: {integrity: sha512-IMQ6eme4AfznElesHUPDZ+teuGwoRmVuuixu7sv92ZkdQcPbsNHzutd+rAfaBKo8YK3IrBEi9SLLKWJdEvJniQ==}
+ '@esbuild/darwin-x64@0.23.1':
+ resolution: {integrity: sha512-aClqdgTDVPSEGgoCS8QDG37Gu8yc9lTHNAQlsztQ6ENetKEO//b8y31MMu2ZaPbn4kVsIABzVLXYLhCGekGDqw==}
engines: {node: '>=18'}
cpu: [x64]
os: [darwin]
@@ -617,8 +617,8 @@ packages:
cpu: [arm64]
os: [freebsd]
- '@esbuild/freebsd-arm64@0.23.0':
- resolution: {integrity: sha512-0muYWCng5vqaxobq6LB3YNtevDFSAZGlgtLoAc81PjUfiFz36n4KMpwhtAd4he8ToSI3TGyuhyx5xmiWNYZFyw==}
+ '@esbuild/freebsd-arm64@0.23.1':
+ resolution: {integrity: sha512-h1k6yS8/pN/NHlMl5+v4XPfikhJulk4G+tKGFIOwURBSFzE8bixw1ebjluLOjfwtLqY0kewfjLSrO6tN2MgIhA==}
engines: {node: '>=18'}
cpu: [arm64]
os: [freebsd]
@@ -635,8 +635,8 @@ packages:
cpu: [x64]
os: [freebsd]
- '@esbuild/freebsd-x64@0.23.0':
- resolution: {integrity: sha512-XKDVu8IsD0/q3foBzsXGt/KjD/yTKBCIwOHE1XwiXmrRwrX6Hbnd5Eqn/WvDekddK21tfszBSrE/WMaZh+1buQ==}
+ '@esbuild/freebsd-x64@0.23.1':
+ resolution: {integrity: sha512-lK1eJeyk1ZX8UklqFd/3A60UuZ/6UVfGT2LuGo3Wp4/z7eRTRYY+0xOu2kpClP+vMTi9wKOfXi2vjUpO1Ro76g==}
engines: {node: '>=18'}
cpu: [x64]
os: [freebsd]
@@ -653,8 +653,8 @@ packages:
cpu: [arm64]
os: [linux]
- '@esbuild/linux-arm64@0.23.0':
- resolution: {integrity: sha512-j1t5iG8jE7BhonbsEg5d9qOYcVZv/Rv6tghaXM/Ug9xahM0nX/H2gfu6X6z11QRTMT6+aywOMA8TDkhPo8aCGw==}
+ '@esbuild/linux-arm64@0.23.1':
+ resolution: {integrity: sha512-/93bf2yxencYDnItMYV/v116zff6UyTjo4EtEQjUBeGiVpMmffDNUyD9UN2zV+V3LRV3/on4xdZ26NKzn6754g==}
engines: {node: '>=18'}
cpu: [arm64]
os: [linux]
@@ -671,8 +671,8 @@ packages:
cpu: [arm]
os: [linux]
- '@esbuild/linux-arm@0.23.0':
- resolution: {integrity: sha512-SEELSTEtOFu5LPykzA395Mc+54RMg1EUgXP+iw2SJ72+ooMwVsgfuwXo5Fn0wXNgWZsTVHwY2cg4Vi/bOD88qw==}
+ '@esbuild/linux-arm@0.23.1':
+ resolution: {integrity: sha512-CXXkzgn+dXAPs3WBwE+Kvnrf4WECwBdfjfeYHpMeVxWE0EceB6vhWGShs6wi0IYEqMSIzdOF1XjQ/Mkm5d7ZdQ==}
engines: {node: '>=18'}
cpu: [arm]
os: [linux]
@@ -689,8 +689,8 @@ packages:
cpu: [ia32]
os: [linux]
- '@esbuild/linux-ia32@0.23.0':
- resolution: {integrity: sha512-P7O5Tkh2NbgIm2R6x1zGJJsnacDzTFcRWZyTTMgFdVit6E98LTxO+v8LCCLWRvPrjdzXHx9FEOA8oAZPyApWUA==}
+ '@esbuild/linux-ia32@0.23.1':
+ resolution: {integrity: sha512-VTN4EuOHwXEkXzX5nTvVY4s7E/Krz7COC8xkftbbKRYAl96vPiUssGkeMELQMOnLOJ8k3BY1+ZY52tttZnHcXQ==}
engines: {node: '>=18'}
cpu: [ia32]
os: [linux]
@@ -707,8 +707,8 @@ packages:
cpu: [loong64]
os: [linux]
- '@esbuild/linux-loong64@0.23.0':
- resolution: {integrity: sha512-InQwepswq6urikQiIC/kkx412fqUZudBO4SYKu0N+tGhXRWUqAx+Q+341tFV6QdBifpjYgUndV1hhMq3WeJi7A==}
+ '@esbuild/linux-loong64@0.23.1':
+ resolution: {integrity: sha512-Vx09LzEoBa5zDnieH8LSMRToj7ir/Jeq0Gu6qJ/1GcBq9GkfoEAoXvLiW1U9J1qE/Y/Oyaq33w5p2ZWrNNHNEw==}
engines: {node: '>=18'}
cpu: [loong64]
os: [linux]
@@ -725,8 +725,8 @@ packages:
cpu: [mips64el]
os: [linux]
- '@esbuild/linux-mips64el@0.23.0':
- resolution: {integrity: sha512-J9rflLtqdYrxHv2FqXE2i1ELgNjT+JFURt/uDMoPQLcjWQA5wDKgQA4t/dTqGa88ZVECKaD0TctwsUfHbVoi4w==}
+ '@esbuild/linux-mips64el@0.23.1':
+ resolution: {integrity: sha512-nrFzzMQ7W4WRLNUOU5dlWAqa6yVeI0P78WKGUo7lg2HShq/yx+UYkeNSE0SSfSure0SqgnsxPvmAUu/vu0E+3Q==}
engines: {node: '>=18'}
cpu: [mips64el]
os: [linux]
@@ -743,8 +743,8 @@ packages:
cpu: [ppc64]
os: [linux]
- '@esbuild/linux-ppc64@0.23.0':
- resolution: {integrity: sha512-cShCXtEOVc5GxU0fM+dsFD10qZ5UpcQ8AM22bYj0u/yaAykWnqXJDpd77ublcX6vdDsWLuweeuSNZk4yUxZwtw==}
+ '@esbuild/linux-ppc64@0.23.1':
+ resolution: {integrity: sha512-dKN8fgVqd0vUIjxuJI6P/9SSSe/mB9rvA98CSH2sJnlZ/OCZWO1DJvxj8jvKTfYUdGfcq2dDxoKaC6bHuTlgcw==}
engines: {node: '>=18'}
cpu: [ppc64]
os: [linux]
@@ -761,8 +761,8 @@ packages:
cpu: [riscv64]
os: [linux]
- '@esbuild/linux-riscv64@0.23.0':
- resolution: {integrity: sha512-HEtaN7Y5UB4tZPeQmgz/UhzoEyYftbMXrBCUjINGjh3uil+rB/QzzpMshz3cNUxqXN7Vr93zzVtpIDL99t9aRw==}
+ '@esbuild/linux-riscv64@0.23.1':
+ resolution: {integrity: sha512-5AV4Pzp80fhHL83JM6LoA6pTQVWgB1HovMBsLQ9OZWLDqVY8MVobBXNSmAJi//Csh6tcY7e7Lny2Hg1tElMjIA==}
engines: {node: '>=18'}
cpu: [riscv64]
os: [linux]
@@ -779,8 +779,8 @@ packages:
cpu: [s390x]
os: [linux]
- '@esbuild/linux-s390x@0.23.0':
- resolution: {integrity: sha512-WDi3+NVAuyjg/Wxi+o5KPqRbZY0QhI9TjrEEm+8dmpY9Xir8+HE/HNx2JoLckhKbFopW0RdO2D72w8trZOV+Wg==}
+ '@esbuild/linux-s390x@0.23.1':
+ resolution: {integrity: sha512-9ygs73tuFCe6f6m/Tb+9LtYxWR4c9yg7zjt2cYkjDbDpV/xVn+68cQxMXCjUpYwEkze2RcU/rMnfIXNRFmSoDw==}
engines: {node: '>=18'}
cpu: [s390x]
os: [linux]
@@ -797,8 +797,8 @@ packages:
cpu: [x64]
os: [linux]
- '@esbuild/linux-x64@0.23.0':
- resolution: {integrity: sha512-a3pMQhUEJkITgAw6e0bWA+F+vFtCciMjW/LPtoj99MhVt+Mfb6bbL9hu2wmTZgNd994qTAEw+U/r6k3qHWWaOQ==}
+ '@esbuild/linux-x64@0.23.1':
+ resolution: {integrity: sha512-EV6+ovTsEXCPAp58g2dD68LxoP/wK5pRvgy0J/HxPGB009omFPv3Yet0HiaqvrIrgPTBuC6wCH1LTOY91EO5hQ==}
engines: {node: '>=18'}
cpu: [x64]
os: [linux]
@@ -815,8 +815,8 @@ packages:
cpu: [x64]
os: [netbsd]
- '@esbuild/netbsd-x64@0.23.0':
- resolution: {integrity: sha512-cRK+YDem7lFTs2Q5nEv/HHc4LnrfBCbH5+JHu6wm2eP+d8OZNoSMYgPZJq78vqQ9g+9+nMuIsAO7skzphRXHyw==}
+ '@esbuild/netbsd-x64@0.23.1':
+ resolution: {integrity: sha512-aevEkCNu7KlPRpYLjwmdcuNz6bDFiE7Z8XC4CPqExjTvrHugh28QzUXVOZtiYghciKUacNktqxdpymplil1beA==}
engines: {node: '>=18'}
cpu: [x64]
os: [netbsd]
@@ -827,8 +827,8 @@ packages:
cpu: [x64]
os: [netbsd]
- '@esbuild/openbsd-arm64@0.23.0':
- resolution: {integrity: sha512-suXjq53gERueVWu0OKxzWqk7NxiUWSUlrxoZK7usiF50C6ipColGR5qie2496iKGYNLhDZkPxBI3erbnYkU0rQ==}
+ '@esbuild/openbsd-arm64@0.23.1':
+ resolution: {integrity: sha512-3x37szhLexNA4bXhLrCC/LImN/YtWis6WXr1VESlfVtVeoFJBRINPJ3f0a/6LV8zpikqoUg4hyXw0sFBt5Cr+Q==}
engines: {node: '>=18'}
cpu: [arm64]
os: [openbsd]
@@ -845,8 +845,8 @@ packages:
cpu: [x64]
os: [openbsd]
- '@esbuild/openbsd-x64@0.23.0':
- resolution: {integrity: sha512-6p3nHpby0DM/v15IFKMjAaayFhqnXV52aEmv1whZHX56pdkK+MEaLoQWj+H42ssFarP1PcomVhbsR4pkz09qBg==}
+ '@esbuild/openbsd-x64@0.23.1':
+ resolution: {integrity: sha512-aY2gMmKmPhxfU+0EdnN+XNtGbjfQgwZj43k8G3fyrDM/UdZww6xrWxmDkuz2eCZchqVeABjV5BpildOrUbBTqA==}
engines: {node: '>=18'}
cpu: [x64]
os: [openbsd]
@@ -863,8 +863,8 @@ packages:
cpu: [x64]
os: [sunos]
- '@esbuild/sunos-x64@0.23.0':
- resolution: {integrity: sha512-BFelBGfrBwk6LVrmFzCq1u1dZbG4zy/Kp93w2+y83Q5UGYF1d8sCzeLI9NXjKyujjBBniQa8R8PzLFAUrSM9OA==}
+ '@esbuild/sunos-x64@0.23.1':
+ resolution: {integrity: sha512-RBRT2gqEl0IKQABT4XTj78tpk9v7ehp+mazn2HbUeZl1YMdaGAQqhapjGTCe7uw7y0frDi4gS0uHzhvpFuI1sA==}
engines: {node: '>=18'}
cpu: [x64]
os: [sunos]
@@ -881,8 +881,8 @@ packages:
cpu: [arm64]
os: [win32]
- '@esbuild/win32-arm64@0.23.0':
- resolution: {integrity: sha512-lY6AC8p4Cnb7xYHuIxQ6iYPe6MfO2CC43XXKo9nBXDb35krYt7KGhQnOkRGar5psxYkircpCqfbNDB4uJbS2jQ==}
+ '@esbuild/win32-arm64@0.23.1':
+ resolution: {integrity: sha512-4O+gPR5rEBe2FpKOVyiJ7wNDPA8nGzDuJ6gN4okSA1gEOYZ67N8JPk58tkWtdtPeLz7lBnY6I5L3jdsr3S+A6A==}
engines: {node: '>=18'}
cpu: [arm64]
os: [win32]
@@ -899,8 +899,8 @@ packages:
cpu: [ia32]
os: [win32]
- '@esbuild/win32-ia32@0.23.0':
- resolution: {integrity: sha512-7L1bHlOTcO4ByvI7OXVI5pNN6HSu6pUQq9yodga8izeuB1KcT2UkHaH6118QJwopExPn0rMHIseCTx1CRo/uNA==}
+ '@esbuild/win32-ia32@0.23.1':
+ resolution: {integrity: sha512-BcaL0Vn6QwCwre3Y717nVHZbAa4UBEigzFm6VdsVdT/MbZ38xoj1X9HPkZhbmaBGUD1W8vxAfffbDe8bA6AKnQ==}
engines: {node: '>=18'}
cpu: [ia32]
os: [win32]
@@ -917,8 +917,8 @@ packages:
cpu: [x64]
os: [win32]
- '@esbuild/win32-x64@0.23.0':
- resolution: {integrity: sha512-Arm+WgUFLUATuoxCJcahGuk6Yj9Pzxd6l11Zb/2aAuv5kWWvvfhLFo2fni4uSK5vzlUdCGZ/BdV5tH8klj8p8g==}
+ '@esbuild/win32-x64@0.23.1':
+ resolution: {integrity: sha512-BHpFFeslkWrXWyUPnbKm+xYYVYruCinGcftSBaa8zoF9hZO4BcSCFUvHVTtzpIY6YzUnYtuEhZ+C9iEXjxnasg==}
engines: {node: '>=18'}
cpu: [x64]
os: [win32]
@@ -2148,8 +2148,8 @@ packages:
engines: {node: '>=12'}
hasBin: true
- esbuild@0.23.0:
- resolution: {integrity: sha512-1lvV17H2bMYda/WaFb2jLPeHU3zml2k4/yagNMG8Q/YtfMjCwEUZa2eXXMgZTVSL5q1n4H7sQ0X6CdJDqqeCFA==}
+ esbuild@0.23.1:
+ resolution: {integrity: sha512-VVNz/9Sa0bs5SELtn3f7qhJCDPCF5oMEl5cO9/SSinpE9hbPVvxbd572HH5AKiP7WD8INO53GgfDDhRjkylHEg==}
engines: {node: '>=18'}
hasBin: true
@@ -2447,8 +2447,8 @@ packages:
resolution: {integrity: sha512-2nk+7SIVb14QrgXFHcm84tD4bKQz0RxPuMT8Ag5KPOq7J5fEmAg0UbXdTOSHqNuHSU28k55qnceesxXRZGzKWA==}
engines: {node: '>=18'}
- get-github-auth-token@0.1.0:
- resolution: {integrity: sha512-ENm+A39AV0X4+Ls1jiCvmqx+C8hSYTv4d5hV9Ks+EL+gx9a0P9pYYpRE1k2ExwRT3EFQabGXF1Rkdp5FIsLkiw==}
+ get-github-auth-token@0.1.1:
+ resolution: {integrity: sha512-SifKYtFehQUVgeEBiTeQMGWeNfNx59gZ7GzYzD6vKCIGDh5YT+Axm5i0VgP0XaD8TeN8ABtp593eX+wSFk8IpQ==}
engines: {node: '>=18'}
get-stdin@9.0.0:
@@ -4683,7 +4683,7 @@ snapshots:
'@esbuild/aix-ppc64@0.19.12':
optional: true
- '@esbuild/aix-ppc64@0.23.0':
+ '@esbuild/aix-ppc64@0.23.1':
optional: true
'@esbuild/aix-ppc64@0.24.0':
@@ -4692,7 +4692,7 @@ snapshots:
'@esbuild/android-arm64@0.19.12':
optional: true
- '@esbuild/android-arm64@0.23.0':
+ '@esbuild/android-arm64@0.23.1':
optional: true
'@esbuild/android-arm64@0.24.0':
@@ -4701,7 +4701,7 @@ snapshots:
'@esbuild/android-arm@0.19.12':
optional: true
- '@esbuild/android-arm@0.23.0':
+ '@esbuild/android-arm@0.23.1':
optional: true
'@esbuild/android-arm@0.24.0':
@@ -4710,7 +4710,7 @@ snapshots:
'@esbuild/android-x64@0.19.12':
optional: true
- '@esbuild/android-x64@0.23.0':
+ '@esbuild/android-x64@0.23.1':
optional: true
'@esbuild/android-x64@0.24.0':
@@ -4719,7 +4719,7 @@ snapshots:
'@esbuild/darwin-arm64@0.19.12':
optional: true
- '@esbuild/darwin-arm64@0.23.0':
+ '@esbuild/darwin-arm64@0.23.1':
optional: true
'@esbuild/darwin-arm64@0.24.0':
@@ -4728,7 +4728,7 @@ snapshots:
'@esbuild/darwin-x64@0.19.12':
optional: true
- '@esbuild/darwin-x64@0.23.0':
+ '@esbuild/darwin-x64@0.23.1':
optional: true
'@esbuild/darwin-x64@0.24.0':
@@ -4737,7 +4737,7 @@ snapshots:
'@esbuild/freebsd-arm64@0.19.12':
optional: true
- '@esbuild/freebsd-arm64@0.23.0':
+ '@esbuild/freebsd-arm64@0.23.1':
optional: true
'@esbuild/freebsd-arm64@0.24.0':
@@ -4746,7 +4746,7 @@ snapshots:
'@esbuild/freebsd-x64@0.19.12':
optional: true
- '@esbuild/freebsd-x64@0.23.0':
+ '@esbuild/freebsd-x64@0.23.1':
optional: true
'@esbuild/freebsd-x64@0.24.0':
@@ -4755,7 +4755,7 @@ snapshots:
'@esbuild/linux-arm64@0.19.12':
optional: true
- '@esbuild/linux-arm64@0.23.0':
+ '@esbuild/linux-arm64@0.23.1':
optional: true
'@esbuild/linux-arm64@0.24.0':
@@ -4764,7 +4764,7 @@ snapshots:
'@esbuild/linux-arm@0.19.12':
optional: true
- '@esbuild/linux-arm@0.23.0':
+ '@esbuild/linux-arm@0.23.1':
optional: true
'@esbuild/linux-arm@0.24.0':
@@ -4773,7 +4773,7 @@ snapshots:
'@esbuild/linux-ia32@0.19.12':
optional: true
- '@esbuild/linux-ia32@0.23.0':
+ '@esbuild/linux-ia32@0.23.1':
optional: true
'@esbuild/linux-ia32@0.24.0':
@@ -4782,7 +4782,7 @@ snapshots:
'@esbuild/linux-loong64@0.19.12':
optional: true
- '@esbuild/linux-loong64@0.23.0':
+ '@esbuild/linux-loong64@0.23.1':
optional: true
'@esbuild/linux-loong64@0.24.0':
@@ -4791,7 +4791,7 @@ snapshots:
'@esbuild/linux-mips64el@0.19.12':
optional: true
- '@esbuild/linux-mips64el@0.23.0':
+ '@esbuild/linux-mips64el@0.23.1':
optional: true
'@esbuild/linux-mips64el@0.24.0':
@@ -4800,7 +4800,7 @@ snapshots:
'@esbuild/linux-ppc64@0.19.12':
optional: true
- '@esbuild/linux-ppc64@0.23.0':
+ '@esbuild/linux-ppc64@0.23.1':
optional: true
'@esbuild/linux-ppc64@0.24.0':
@@ -4809,7 +4809,7 @@ snapshots:
'@esbuild/linux-riscv64@0.19.12':
optional: true
- '@esbuild/linux-riscv64@0.23.0':
+ '@esbuild/linux-riscv64@0.23.1':
optional: true
'@esbuild/linux-riscv64@0.24.0':
@@ -4818,7 +4818,7 @@ snapshots:
'@esbuild/linux-s390x@0.19.12':
optional: true
- '@esbuild/linux-s390x@0.23.0':
+ '@esbuild/linux-s390x@0.23.1':
optional: true
'@esbuild/linux-s390x@0.24.0':
@@ -4827,7 +4827,7 @@ snapshots:
'@esbuild/linux-x64@0.19.12':
optional: true
- '@esbuild/linux-x64@0.23.0':
+ '@esbuild/linux-x64@0.23.1':
optional: true
'@esbuild/linux-x64@0.24.0':
@@ -4836,13 +4836,13 @@ snapshots:
'@esbuild/netbsd-x64@0.19.12':
optional: true
- '@esbuild/netbsd-x64@0.23.0':
+ '@esbuild/netbsd-x64@0.23.1':
optional: true
'@esbuild/netbsd-x64@0.24.0':
optional: true
- '@esbuild/openbsd-arm64@0.23.0':
+ '@esbuild/openbsd-arm64@0.23.1':
optional: true
'@esbuild/openbsd-arm64@0.24.0':
@@ -4851,7 +4851,7 @@ snapshots:
'@esbuild/openbsd-x64@0.19.12':
optional: true
- '@esbuild/openbsd-x64@0.23.0':
+ '@esbuild/openbsd-x64@0.23.1':
optional: true
'@esbuild/openbsd-x64@0.24.0':
@@ -4860,7 +4860,7 @@ snapshots:
'@esbuild/sunos-x64@0.19.12':
optional: true
- '@esbuild/sunos-x64@0.23.0':
+ '@esbuild/sunos-x64@0.23.1':
optional: true
'@esbuild/sunos-x64@0.24.0':
@@ -4869,7 +4869,7 @@ snapshots:
'@esbuild/win32-arm64@0.19.12':
optional: true
- '@esbuild/win32-arm64@0.23.0':
+ '@esbuild/win32-arm64@0.23.1':
optional: true
'@esbuild/win32-arm64@0.24.0':
@@ -4878,7 +4878,7 @@ snapshots:
'@esbuild/win32-ia32@0.19.12':
optional: true
- '@esbuild/win32-ia32@0.23.0':
+ '@esbuild/win32-ia32@0.23.1':
optional: true
'@esbuild/win32-ia32@0.24.0':
@@ -4887,7 +4887,7 @@ snapshots:
'@esbuild/win32-x64@0.19.12':
optional: true
- '@esbuild/win32-x64@0.23.0':
+ '@esbuild/win32-x64@0.23.1':
optional: true
'@esbuild/win32-x64@0.24.0':
@@ -5952,7 +5952,7 @@ snapshots:
chalk: 5.4.1
create-fs: 0.1.0
execa: 9.5.2
- get-github-auth-token: 0.1.0
+ get-github-auth-token: 0.1.1
hash-object: 5.0.1
hosted-git-info: 8.0.2
import-local-or-npx: 0.2.0
@@ -6217,32 +6217,32 @@ snapshots:
'@esbuild/win32-ia32': 0.19.12
'@esbuild/win32-x64': 0.19.12
- esbuild@0.23.0:
+ esbuild@0.23.1:
optionalDependencies:
- '@esbuild/aix-ppc64': 0.23.0
- '@esbuild/android-arm': 0.23.0
- '@esbuild/android-arm64': 0.23.0
- '@esbuild/android-x64': 0.23.0
- '@esbuild/darwin-arm64': 0.23.0
- '@esbuild/darwin-x64': 0.23.0
- '@esbuild/freebsd-arm64': 0.23.0
- '@esbuild/freebsd-x64': 0.23.0
- '@esbuild/linux-arm': 0.23.0
- '@esbuild/linux-arm64': 0.23.0
- '@esbuild/linux-ia32': 0.23.0
- '@esbuild/linux-loong64': 0.23.0
- '@esbuild/linux-mips64el': 0.23.0
- '@esbuild/linux-ppc64': 0.23.0
- '@esbuild/linux-riscv64': 0.23.0
- '@esbuild/linux-s390x': 0.23.0
- '@esbuild/linux-x64': 0.23.0
- '@esbuild/netbsd-x64': 0.23.0
- '@esbuild/openbsd-arm64': 0.23.0
- '@esbuild/openbsd-x64': 0.23.0
- '@esbuild/sunos-x64': 0.23.0
- '@esbuild/win32-arm64': 0.23.0
- '@esbuild/win32-ia32': 0.23.0
- '@esbuild/win32-x64': 0.23.0
+ '@esbuild/aix-ppc64': 0.23.1
+ '@esbuild/android-arm': 0.23.1
+ '@esbuild/android-arm64': 0.23.1
+ '@esbuild/android-x64': 0.23.1
+ '@esbuild/darwin-arm64': 0.23.1
+ '@esbuild/darwin-x64': 0.23.1
+ '@esbuild/freebsd-arm64': 0.23.1
+ '@esbuild/freebsd-x64': 0.23.1
+ '@esbuild/linux-arm': 0.23.1
+ '@esbuild/linux-arm64': 0.23.1
+ '@esbuild/linux-ia32': 0.23.1
+ '@esbuild/linux-loong64': 0.23.1
+ '@esbuild/linux-mips64el': 0.23.1
+ '@esbuild/linux-ppc64': 0.23.1
+ '@esbuild/linux-riscv64': 0.23.1
+ '@esbuild/linux-s390x': 0.23.1
+ '@esbuild/linux-x64': 0.23.1
+ '@esbuild/netbsd-x64': 0.23.1
+ '@esbuild/openbsd-arm64': 0.23.1
+ '@esbuild/openbsd-x64': 0.23.1
+ '@esbuild/sunos-x64': 0.23.1
+ '@esbuild/win32-arm64': 0.23.1
+ '@esbuild/win32-ia32': 0.23.1
+ '@esbuild/win32-x64': 0.23.1
optional: true
esbuild@0.24.0:
@@ -6653,7 +6653,7 @@ snapshots:
get-east-asian-width@1.2.0: {}
- get-github-auth-token@0.1.0: {}
+ get-github-auth-token@0.1.1: {}
get-stdin@9.0.0: {}
@@ -7637,7 +7637,7 @@ snapshots:
octokit-from-auth@0.2.0:
dependencies:
- get-github-auth-token: 0.1.0
+ get-github-auth-token: 0.1.1
octokit: 4.1.0
octokit@4.1.0:
@@ -8447,7 +8447,7 @@ snapshots:
tsx@4.19.2:
dependencies:
- esbuild: 0.23.0
+ esbuild: 0.23.1
get-tsconfig: 4.8.1
optionalDependencies:
fsevents: 2.3.3
From 5f5676a674910dd7eba472a1f3b9be1519aa1a06 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 11:28:56 -0500
Subject: [PATCH 14/22] Switch pre-commit file mode: ... to executable: true
---
package.json | 6 ++--
pnpm-lock.yaml | 58 ++++++++++++++++----------------
src/blocks/blockPrettier.test.ts | 6 ++--
src/blocks/blockPrettier.ts | 2 +-
4 files changed, 36 insertions(+), 36 deletions(-)
diff --git a/package.json b/package.json
index 61d05426c..28aaf7772 100644
--- a/package.json
+++ b/package.json
@@ -38,8 +38,8 @@
},
"dependencies": {
"chalk": "^5.4.1",
- "create": "0.1.0-alpha.13",
- "create-fs": "^0.1.0",
+ "create": "0.1.0-alpha.14",
+ "create-fs": "^0.1.1",
"cspell-populate-words": "^0.3.0",
"execa": "^9.5.2",
"git-remote-origin-url": "^4.0.0",
@@ -78,7 +78,7 @@
"@vitest/eslint-plugin": "1.1.20",
"all-contributors-cli": "6.26.1",
"console-fail-test": "0.5.0",
- "create-testers": "0.1.0-alpha.13",
+ "create-testers": "0.1.0-alpha.14",
"cspell": "^8.17.2",
"eslint": "9.17.0",
"eslint-plugin-jsdoc": "50.6.1",
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index feefb85ff..dabe8f3ee 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -12,11 +12,11 @@ importers:
specifier: ^5.4.1
version: 5.4.1
create:
- specifier: 0.1.0-alpha.13
- version: 0.1.0-alpha.13
+ specifier: 0.1.0-alpha.14
+ version: 0.1.0-alpha.14
create-fs:
- specifier: ^0.1.0
- version: 0.1.0
+ specifier: ^0.1.1
+ version: 0.1.1
cspell-populate-words:
specifier: ^0.3.0
version: 0.3.0
@@ -37,13 +37,13 @@ importers:
version: 1.2.0
input-from-file:
specifier: 0.1.0-alpha.4
- version: 0.1.0-alpha.4(create@0.1.0-alpha.13)
+ version: 0.1.0-alpha.4(create@0.1.0-alpha.14)
input-from-file-json:
specifier: 0.1.0-alpha.4
- version: 0.1.0-alpha.4(create@0.1.0-alpha.13)
+ version: 0.1.0-alpha.4(create@0.1.0-alpha.14)
input-from-script:
specifier: 0.1.0-alpha.4
- version: 0.1.0-alpha.4(create@0.1.0-alpha.13)
+ version: 0.1.0-alpha.4(create@0.1.0-alpha.14)
js-yaml:
specifier: ^4.1.0
version: 4.1.0
@@ -127,8 +127,8 @@ importers:
specifier: 0.5.0
version: 0.5.0
create-testers:
- specifier: 0.1.0-alpha.13
- version: 0.1.0-alpha.13(create-fs@0.1.0)(create@0.1.0-alpha.13)
+ specifier: 0.1.0-alpha.14
+ version: 0.1.0-alpha.14(create-fs@0.1.1)(create@0.1.0-alpha.14)
cspell:
specifier: ^8.17.2
version: 8.17.2
@@ -1919,19 +1919,19 @@ packages:
typescript:
optional: true
- create-fs@0.1.0:
- resolution: {integrity: sha512-IxoO6aFIPs+loAGrBn/Hg8UzQ5pRfjtsNqSW8o/RMtjh0RShZcuKbDYWvepnjTqDAaHdbkaXFY7y86WXqQnzpA==}
+ create-fs@0.1.1:
+ resolution: {integrity: sha512-tZqeDZ7yLLje2meTDL1pR3oBAnSlDd8fNyRUHtPxhGIGUGEoY8A/DHIpfxHE9eEoamA2snDJlVP9HyItfb18gQ==}
engines: {node: '>=18'}
- create-testers@0.1.0-alpha.13:
- resolution: {integrity: sha512-rrCLIajWABQcakAr9eLwIqmzaXVx8zyfE/dXKf2Xj/L4+7m/48JE+4q8gRLh6SYsb4c/SFlaKZ/q50gIWitDcg==}
+ create-testers@0.1.0-alpha.14:
+ resolution: {integrity: sha512-IWK69nhyVEhMTDfmOwAr3Hlj3jtQ8P3mz4ezo77BI0er82KpPaNGuSePbj5N9NWfP5TuJeeFJ+uEDcHb2x7fzA==}
engines: {node: '>=18'}
peerDependencies:
- create: ^0.1.0-alpha.13
+ create: ^0.1.0-alpha.14
create-fs: ^0.1.0
- create@0.1.0-alpha.13:
- resolution: {integrity: sha512-8pX3fXtYzyXcCkMppv7R9hsw0SFbxQxzkEDsjX8WyIQd0jPuSIQMr0ofZEUthJQG63NlJCvXmS2V4H7LQDUpFA==}
+ create@0.1.0-alpha.14:
+ resolution: {integrity: sha512-OExXy9Tn7Nugayz/eKMJV1/oM7J/p2SOi3HK/zEyMXNwF2IwnPiHli6rjyvl5hggU7618cumdD6D5ptJK1Zixw==}
engines: {node: '>=18'}
hasBin: true
@@ -5935,22 +5935,22 @@ snapshots:
optionalDependencies:
typescript: 5.7.2
- create-fs@0.1.0: {}
+ create-fs@0.1.1: {}
- create-testers@0.1.0-alpha.13(create-fs@0.1.0)(create@0.1.0-alpha.13):
+ create-testers@0.1.0-alpha.14(create-fs@0.1.1)(create@0.1.0-alpha.14):
dependencies:
- create: 0.1.0-alpha.13
- create-fs: 0.1.0
+ create: 0.1.0-alpha.14
+ create-fs: 0.1.1
diff: 7.0.0
octokit: 4.1.0
without-undefined-properties: 0.1.1
- create@0.1.0-alpha.13:
+ create@0.1.0-alpha.14:
dependencies:
'@clack/prompts': 0.9.1
cached-factory: 0.1.0
chalk: 5.4.1
- create-fs: 0.1.0
+ create-fs: 0.1.1
execa: 9.5.2
get-github-auth-token: 0.1.1
hash-object: 5.0.1
@@ -6901,20 +6901,20 @@ snapshots:
ini@4.1.1: {}
- input-from-file-json@0.1.0-alpha.4(create@0.1.0-alpha.13):
+ input-from-file-json@0.1.0-alpha.4(create@0.1.0-alpha.14):
dependencies:
- create: 0.1.0-alpha.13
- input-from-file: 0.1.0-alpha.4(create@0.1.0-alpha.13)
+ create: 0.1.0-alpha.14
+ input-from-file: 0.1.0-alpha.4(create@0.1.0-alpha.14)
zod: 3.24.1
- input-from-file@0.1.0-alpha.4(create@0.1.0-alpha.13):
+ input-from-file@0.1.0-alpha.4(create@0.1.0-alpha.14):
dependencies:
- create: 0.1.0-alpha.13
+ create: 0.1.0-alpha.14
zod: 3.24.1
- input-from-script@0.1.0-alpha.4(create@0.1.0-alpha.13):
+ input-from-script@0.1.0-alpha.4(create@0.1.0-alpha.14):
dependencies:
- create: 0.1.0-alpha.13
+ create: 0.1.0-alpha.14
zod: 3.24.1
inquirer@7.3.3:
diff --git a/src/blocks/blockPrettier.test.ts b/src/blocks/blockPrettier.test.ts
index 3bdad764b..7c36aa44a 100644
--- a/src/blocks/blockPrettier.test.ts
+++ b/src/blocks/blockPrettier.test.ts
@@ -94,7 +94,7 @@ describe("blockPrettier", () => {
"npx lint-staged
",
{
- "mode": 33279,
+ "executable": true,
},
],
},
@@ -206,7 +206,7 @@ describe("blockPrettier", () => {
"npx lint-staged
",
{
- "mode": 33279,
+ "executable": true,
},
],
},
@@ -335,7 +335,7 @@ describe("blockPrettier", () => {
"npx lint-staged
",
{
- "mode": 33279,
+ "executable": true,
},
],
},
diff --git a/src/blocks/blockPrettier.ts b/src/blocks/blockPrettier.ts
index ce2d7147a..d54671820 100644
--- a/src/blocks/blockPrettier.ts
+++ b/src/blocks/blockPrettier.ts
@@ -95,7 +95,7 @@ pnpm format --write
files: {
".husky": {
".gitignore": "_\n",
- "pre-commit": ["npx lint-staged\n", { mode: 33279 }],
+ "pre-commit": ["npx lint-staged\n", { executable: true }],
},
".prettierignore": formatIgnoreFile(
["/.husky", "/lib", "/pnpm-lock.yaml", ...ignores].sort(),
From 3717ae87684ee9f3df6276e05604c6dda2e8637c Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 14:24:41 -0500
Subject: [PATCH 15/22] Merge branch 'main'
---
src/base.test.ts | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/src/base.test.ts b/src/base.test.ts
index f69445dfa..9734102e7 100644
--- a/src/base.test.ts
+++ b/src/base.test.ts
@@ -27,7 +27,7 @@ describe("base", () => {
},
explainer: [
`\`create-typescript-app\` is a one-stop-shop solution to set up a new or existing repository with the latest and greatest TypeScript tooling.`,
- `It includes options not just for building and testing but also GitHub repository templates, contributor recognition, automated release management, and more.`,
+ `It includes options not just for building and testing but also automated release management, contributor recognition, GitHub repository settings, and more.`,
],
funding: "JoshuaKGoldberg",
guide: {
From a68b0c5ae8c10a556d90134d2c529c2969d5cf1e Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 14:38:11 -0500
Subject: [PATCH 16/22] Assorted docs touchups
---
README.md | 2 +-
docs/Blocks.md | 8 ++++----
docs/Creation.md | 12 ++++++------
docs/Initialization.md | 10 ++++------
docs/Migration.md | 8 ++++----
src/presets/common.ts | 2 ++
src/presets/everything.ts | 2 --
src/presets/minimal.ts | 4 ++--
8 files changed, 23 insertions(+), 25 deletions(-)
diff --git a/README.md b/README.md
index 006a54bb6..cb7690d91 100644
--- a/README.md
+++ b/README.md
@@ -49,7 +49,7 @@ You can read more about the supported setup modes in their docs pages:
You can read more about `create-typescript-app` and the tooling it supports:
-1. [**Tooling**](./docs/Tooling.md): a breakdown of all the pieces this template can set up.
+1. [**Blocks**](./docs/Blocks.md): a breakdown of all the pieces this template can set up.
2. [**Options**](./docs/Options.md): granular options to customize how the template is run.
3. [**FAQs**](./docs/FAQs.md): frequently asked questions
diff --git a/docs/Blocks.md b/docs/Blocks.md
index ccb908ffe..2b828d9a7 100644
--- a/docs/Blocks.md
+++ b/docs/Blocks.md
@@ -1,6 +1,6 @@
# Blocks
-`create-typescript-app` provides over two dozen pieces of tooling, ranging from code building and formatting to various forms of GitHub repository management.
+`create-typescript-app` provides several dozen pieces of tooling, ranging from code building and formatting to various forms of GitHub repository management.
Each can be individually turned off or on.
This table summarizes each block and which base levels they're included in:
@@ -25,10 +25,10 @@ This table summarizes each block and which base levels they're included in:
| ESLint Perfectionist Plugin | `--exclude-eslint-perfectionist-plugin` | | | 💯 |
| ESLint Regexp Plugin | `--exclude-eslint-regexp-plugin` | | | 💯 |
| ESLint YML Plugin | `--exclude-eslint-yml-plugin` | | | 💯 |
-| Funding | `--exclude-funding` | ✔️ | ✅ | 💯 |
+| Funding | `--exclude-funding` | | ✅ | 💯 |
| GitHub Actions CI | `--exclude-github-actions-ci` | ✔️ | ✅ | 💯 |
| GitHub Issue Templates | `--exclude-github-issue-templates` | ✔️ | ✅ | 💯 |
-| GitHub Issue Templates | `--exclude-github-issue-templates` | ✔️ | ✅ | 💯 |
+| GitHub PR Template | `--exclude-github-pr-template` | ✔️ | ✅ | 💯 |
| Gitignore | `--exclude-gitignore` | ✔️ | ✅ | 💯 |
| Knip | `--exclude-knip` | | | 💯 |
| Markdownlint | `--exclude-markdownlint` | | | 💯 |
@@ -44,7 +44,7 @@ This table summarizes each block and which base levels they're included in:
| README.md | `--exclude-readme-md` | ✔️ | ✅ | 💯 |
| release-it | `--exclude-release-it` | | ✅ | 💯 |
| Renovate | `--exclude-renovate` | | | 💯 |
-| Security Docs | `--exclude-security-docs` | | | 💯 |
+| Security Docs | `--exclude-security-docs` | ✔️ | ✅ | 💯 |
| Templated By Notice | `--exclude-templated-by-notice` | ✔️ | ✅ | 💯 |
| TSup | `--exclude-tsup` | ✔️ | ✅ | 💯 |
| TypeScript | `--exclude-typescript` | ✔️ | ✅ | 💯 |
diff --git a/docs/Creation.md b/docs/Creation.md
index ba1f3ac04..1fa7bfd06 100644
--- a/docs/Creation.md
+++ b/docs/Creation.md
@@ -1,6 +1,6 @@
# Creating from the Terminal
-You can run `npx create typescript-app` in your terminal to interactively create a new repository in a child directory:
+You can run `npx create typescript-app` in your terminal to interactively create a new repository:
```shell
npx create typescript-app
@@ -8,8 +8,8 @@ npx create typescript-app
The creation script will by default:
-1. Create a new directory with the given repository name
-2. Initialize that new directory as a local Git repository
+1. Prompt you for a directory, which template preset to run with, and some starting information
+2. Initialize new directory as a local Git repository
3. Copy the template's files to that directory
4. Create a new repository on GitHub and set it as the local repository's upstream
5. Configure relevant settings on the GitHub repository
@@ -28,13 +28,13 @@ Hooray! 🥳
## Options
-You can explicitly provide some or all of the options the script would prompt for as command-line flags.
+You can customize which pieces of tooling are provided and the options they're created with.
See [Options.md](./Options.md).
-For example, running the creation script and skipping the _"This package was templated with..."_ block:
+For example, skipping the _"This package was templated with..."_ block:
```shell
npx create typescript-app --mode create --exclude-templated-with
```
-See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
+See [Blocks.md](./Blocks.md) for details on the tooling pieces and which presets they're included in.
diff --git a/docs/Initialization.md b/docs/Initialization.md
index 4ec14ead3..2164c75bb 100644
--- a/docs/Initialization.md
+++ b/docs/Initialization.md
@@ -23,15 +23,13 @@ Hooray! 🥳
## Options
-You can explicitly provide some or all of the options the script would prompt for as command-line flags.
+You can customize which pieces of tooling are provided and the options they're created with.
See [Options.md](./Options.md).
-`pnpm run initialize` will set `--mode` to `initialize`.
-
-For example, running the creation script and skipping the _"This package was templated with..."_ block:
+For example, skipping the _"This package was templated with..."_ block:
```shell
-npx create typescript-app --mode create --exclude-templated-with
+npx create typescript-app --exclude-templated-with
```
-See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
+See [Blocks.md](./Blocks.md) for details on the tooling pieces and which presets they're included in.
diff --git a/docs/Migration.md b/docs/Migration.md
index 3526c53ae..e4a01e22c 100644
--- a/docs/Migration.md
+++ b/docs/Migration.md
@@ -41,13 +41,13 @@ Hooray! 🥳
## Options
-You can explicitly provide some or all of the options the script would prompt for as command-line flags.
+You can customize which pieces of tooling are provided and the options they're created with.
See [Options.md](./Options.md).
-For example, running the migration script and skipping the _"This package was templated with..."_ block:
+For example, skipping the _"This package was templated with..."_ block:
```shell
-npx create typescript-app --mode migrate --exclude-templated-with
+npx create typescript-app --exclude-templated-with
```
-See [Tooling.md](./Tooling.md) for details on the tooling pieces and which bases they're included in.
+See [Blocks.md](./Blocks.md) for details on the tooling pieces and which presets they're included in.
diff --git a/src/presets/common.ts b/src/presets/common.ts
index 8211e4304..d499d04f5 100644
--- a/src/presets/common.ts
+++ b/src/presets/common.ts
@@ -1,6 +1,7 @@
import { base } from "../base.js";
import { blockAllContributors } from "../blocks/blockAllContributors.js";
import { blockCodecov } from "../blocks/blockCodecov.js";
+import { blockFunding } from "../blocks/blockFunding.js";
import { blockReleaseIt } from "../blocks/blockReleaseIt.js";
import { blockVitest } from "../blocks/blockVitest.js";
import { presetMinimal } from "./minimal.js";
@@ -15,6 +16,7 @@ export const presetCommon = base.createPreset({
...presetMinimal.blocks,
blockAllContributors,
blockCodecov,
+ blockFunding,
blockReleaseIt,
blockVitest,
],
diff --git a/src/presets/everything.ts b/src/presets/everything.ts
index 5cff5fc53..d5317f1fc 100644
--- a/src/presets/everything.ts
+++ b/src/presets/everything.ts
@@ -19,7 +19,6 @@ import { blockPrettierPluginCurly } from "../blocks/blockPrettierPluginCurly.js"
import { blockPrettierPluginPackageJson } from "../blocks/blockPrettierPluginPackageJson.js";
import { blockPrettierPluginSh } from "../blocks/blockPrettierPluginSh.js";
import { blockRenovate } from "../blocks/blockRenovate.js";
-import { blockSecurityDocs } from "../blocks/blockSecurityDocs.js";
import { blockVSCode } from "../blocks/blockVSCode.js";
import { presetCommon } from "./common.js";
@@ -51,7 +50,6 @@ export const presetEverything = base.createPreset({
blockPrettierPluginPackageJson,
blockPrettierPluginSh,
blockRenovate,
- blockSecurityDocs,
blockVSCode,
],
});
diff --git a/src/presets/minimal.ts b/src/presets/minimal.ts
index 0ff44447f..2106bd10a 100644
--- a/src/presets/minimal.ts
+++ b/src/presets/minimal.ts
@@ -4,7 +4,6 @@ import { blockContributorCovenant } from "../blocks/blockContributorCovenant.js"
import { blockDevelopmentDocs } from "../blocks/blockDevelopmentDocs.js";
import { blockESLint } from "../blocks/blockESLint.js";
import { blockExampleFiles } from "../blocks/blockExampleFiles.js";
-import { blockFunding } from "../blocks/blockFunding.js";
import { blockGitHubActionsCI } from "../blocks/blockGitHubActionsCI.js";
import { blockGitHubApps } from "../blocks/blockGitHubApps.js";
import { blockGitHubIssueTemplates } from "../blocks/blockGitHubIssueTemplates.js";
@@ -18,6 +17,7 @@ import { blockRepositoryBranchRuleset } from "../blocks/blockRepositoryBranchRul
import { blockRepositoryLabels } from "../blocks/blockRepositoryLabels.js";
import { blockRepositorySecrets } from "../blocks/blockRepositorySecrets.js";
import { blockRepositorySettings } from "../blocks/blockRepositorySettings.js";
+import { blockSecurityDocs } from "../blocks/blockSecurityDocs.js";
import { blockTemplatedWith } from "../blocks/blockTemplatedWith.js";
import { blockTSup } from "../blocks/blockTSup.js";
import { blockTypeScript } from "../blocks/blockTypeScript.js";
@@ -34,7 +34,6 @@ export const presetMinimal = base.createPreset({
blockDevelopmentDocs,
blockESLint,
blockExampleFiles,
- blockFunding,
blockGitHubActionsCI,
blockGitHubApps,
blockGitHubIssueTemplates,
@@ -48,6 +47,7 @@ export const presetMinimal = base.createPreset({
blockRepositoryLabels,
blockRepositorySecrets,
blockRepositorySettings,
+ blockSecurityDocs,
blockTemplatedWith,
blockTSup,
blockTypeScript,
From 3b82fed37954175007e1332b8e014ca20127c46a Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 15:10:25 -0500
Subject: [PATCH 17/22] More docs touchups, around blocks and options
---
docs/Options.md | 14 ++------------
src/base.ts | 6 ------
src/docs.test.ts | 4 ++--
3 files changed, 4 insertions(+), 20 deletions(-)
diff --git a/docs/Options.md b/docs/Options.md
index 8e67bdfe4..803c85147 100644
--- a/docs/Options.md
+++ b/docs/Options.md
@@ -25,13 +25,11 @@ Per [`create` > CLI > Template Options](https://www.create.bingo/cli#template-op
### Required Base Options
-🛑 TODO: Mention `--repository` and `--owner` in `create`, and add _"you can't override these"_ issue
-
These options can only be inferred when running on an existing repository.
Each will be prompted for when creating a new repository if not explicitly provided:
-- `--description` _(`string`)_: Sentence case description of the repository
-- `--title` _(`string`)_: Title Case title for the repository
+- `--description` _(`string`)_: 'Sentence case.' description of the repository
+- `--title` _(`string`)_: 'Title Case' title for the repository
For example, pre-populating both required base options:
@@ -51,17 +49,9 @@ They will be inferred from the running user, and if migrating an existing reposi
- `--bin` _(`string`)_: Value to set in `package.json`'s `"bin"` property, per [FAQs > How can I use `bin`?](./FAQs.md#how-can-i-use-bin)
- `--directory` _(`string`)_: Directory to create the repository in (by default, the same name as the repository)
- `--email` _(`string`)_: Email address to be listed as the point of contact in docs and packages (e.g. `example@joshuakgoldberg.com`)
- - Optionally, `--email-github` _(`string`)_ and/or `--email-npm` _(`string`)_ may be provided to use different emails in `.md` files and `package.json`, respectively
- `--funding` _(`string`)_: GitHub organization or username to mention in `funding.yml` (by default, `owner`)
-- `--guide` _(`string`)_: Link to a contribution guide to place at the top of development docs
- - `--guide-title` _(`string`)_: If `--guide` is provided or detected from an existing DEVELOPMENT.md, the text title to place in the guide link
- `--keywords` _(`string[]`)_: Any number of keywords to include in `package.json` (by default, none)
- This can be specified any number of times, like `--keywords apple --keywords "banana cherry"`
-- `--logo` _(`string`)_: Local image file in the repository to display near the top of the README.md
- - `--logo-alt` _(`string`)_: If `--logo` is provided or detected from an existing README.md, alt text that describes the image (will be prompted for if not provided)
- - `--logo-height` _(`number`)_: If `--logo` is provided or detected from an existing README.md, an explicit height style (by default, read from the image, capped to `128`)
- - `--logo-width` _(`number`)_: If `--logo` is provided or detected from an existing README.md, an explicit width style (by default, read from the image, capped to `128`)
-- `--preserve-generated-from` _(`boolean`)_: Whether to keep the GitHub repository _generated from_ notice (by default, `false`)
For example, customizing the npm author and funding source:
diff --git a/src/base.ts b/src/base.ts
index 75626c53d..7185d8427 100644
--- a/src/base.ts
+++ b/src/base.ts
@@ -89,12 +89,6 @@ export const base = createBase({
.describe(
"link to a contribution guide to place at the top of development docs",
),
- hideTemplatedBy: z
- .boolean()
- .optional()
- .describe(
- "whether to hide the 'created by ...' notice at the bottom of the README.md",
- ),
keywords: z
.array(z.string())
.optional()
diff --git a/src/docs.test.ts b/src/docs.test.ts
index 0c1273c02..4fcc0b160 100644
--- a/src/docs.test.ts
+++ b/src/docs.test.ts
@@ -8,8 +8,8 @@ import { blocks, presets } from "./index.js";
const actualLines = await createActualLines();
const expectedLines = await createExpectedLines();
-// This test ensures ensures docs/Blocks.md has a row for each of CTA's options.
-// Each row should include emojis describing which preset(s) include the option.
+// This test ensures ensures docs/Blocks.md has a row for each of CTA's blocks.
+// Each row should include emojis describing which preset(s) include the block.
//
// If this fails, it's likely due to adding, removing, or renaming a block.
// You may need to manually change docs/Blocks.md to match to those changes.
From e8f76aeae050ab362209e254349b8a0f1f0a08e2 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 15:58:16 -0500
Subject: [PATCH 18/22] fix: pipe GH Actions CI jobs addons to ruleset contexts
---
src/blocks/blockGitHubActionsCI.ts | 6 ++++++
1 file changed, 6 insertions(+)
diff --git a/src/blocks/blockGitHubActionsCI.ts b/src/blocks/blockGitHubActionsCI.ts
index 073eba4f5..da2cb94f5 100644
--- a/src/blocks/blockGitHubActionsCI.ts
+++ b/src/blocks/blockGitHubActionsCI.ts
@@ -2,6 +2,7 @@ import jsYaml from "js-yaml";
import { z } from "zod";
import { base } from "../base.js";
+import { blockRepositoryBranchRuleset } from "./blockRepositoryBranchRuleset.js";
import { createMultiWorkflowFile } from "./files/createMultiWorkflowFile.js";
import { createSoloWorkflowFile } from "./files/createSoloWorkflowFile.js";
import { CommandPhase } from "./phases.js";
@@ -43,6 +44,11 @@ export const blockGitHubActionsCI = base.createBlock({
const { jobs } = addons;
return {
+ addons: [
+ blockRepositoryBranchRuleset({
+ requiredStatusChecks: jobs?.map((job) => job.name),
+ }),
+ ],
files: {
".github": {
actions: {
From 24fba7ecb0bb55de3dd21fe6e782c0eb047bb367 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 15:58:32 -0500
Subject: [PATCH 19/22] fix: blockPackageJson correct pnpm install --offline...
ugh.
---
src/blocks/blockPackageJson.ts | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/src/blocks/blockPackageJson.ts b/src/blocks/blockPackageJson.ts
index 3629977c7..9fddaa98d 100644
--- a/src/blocks/blockPackageJson.ts
+++ b/src/blocks/blockPackageJson.ts
@@ -100,7 +100,7 @@ export const blockPackageJson = base.createBlock({
scripts: [
{
commands: [
- offline ? "pnpm install" : "pnpm install --offline",
+ offline ? "pnpm install --offline" : "pnpm install",
...addons.cleanupCommands,
],
phase: CommandPhase.Install,
From 64615992a4c4ec554a315e73dcf2250a2852fe2e Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 16:54:23 -0500
Subject: [PATCH 20/22] Remove other test package.json dependencies in
blockVitest
---
docs/Migration.md | 1 -
knip.json | 6 +++++-
package.json | 1 +
pnpm-lock.yaml | 10 ++++++++++
src/blocks/blockGitHubActionsCI.test.ts | 26 +++++++++++++++++++++++++
src/blocks/blockPackageJson.test.ts | 8 ++++----
src/blocks/blockVitest.test.ts | 13 ++++++++++++-
src/blocks/blockVitest.ts | 8 ++++++++
src/integration.test.ts | 6 +++++-
9 files changed, 71 insertions(+), 8 deletions(-)
diff --git a/docs/Migration.md b/docs/Migration.md
index e4a01e22c..0fa226b46 100644
--- a/docs/Migration.md
+++ b/docs/Migration.md
@@ -17,7 +17,6 @@ The migration script will:
For example, if the repository previously using Jest for testing:
- `eslint-plugin-jest`, `jest`, and other Jest-related packages will be uninstalled
- - 🛑 TODO: Missing in `blockVitest`
- Any Jest config file like `jest.config.js` will be deleted
- `@vitest/eslint-plugin`, `vitest`, and other Vitest-related packages will be installed
- A `vitest.config.ts` file will be created
diff --git a/knip.json b/knip.json
index 16c4fc7e1..55408fe8e 100644
--- a/knip.json
+++ b/knip.json
@@ -1,7 +1,11 @@
{
"$schema": "https://unpkg.com/knip@5.41.1/schema.json",
"entry": ["src/index.ts", "src/**/*.test.*"],
- "ignoreDependencies": ["all-contributors-cli", "cspell-populate-words"],
+ "ignoreDependencies": [
+ "all-contributors-cli",
+ "cspell-populate-words",
+ "remove-dependencies"
+ ],
"ignoreExportsUsedInFile": { "interface": true, "type": true },
"project": ["src/**/*.ts"]
}
diff --git a/package.json b/package.json
index 28aaf7772..2abd69784 100644
--- a/package.json
+++ b/package.json
@@ -56,6 +56,7 @@
"object-strings-deep": "^0.1.1",
"parse-author": "^2.0.0",
"parse-package-name": "^1.0.0",
+ "remove-dependencies": "^0.1.0",
"remove-undefined-objects": "^5.0.0",
"set-github-repository-labels": "^0.1.0",
"sort-package-json": "^2.12.0",
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index dabe8f3ee..9ce5cef77 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -65,6 +65,9 @@ importers:
parse-package-name:
specifier: ^1.0.0
version: 1.0.0
+ remove-dependencies:
+ specifier: ^0.1.0
+ version: 0.1.0
remove-undefined-objects:
specifier: ^5.0.0
version: 5.0.0
@@ -3676,6 +3679,11 @@ packages:
engines: {node: ^18.18.0 || ^20.9.0 || ^22.0.0}
hasBin: true
+ remove-dependencies@0.1.0:
+ resolution: {integrity: sha512-K9PXkqGYkuQbRVf57SI/t3xBhw87pLbvd94zir6v4+ptJQ0TSjLWtd0ZlebtQEOfnxSchz+wgzTvG7KNpLMAbg==}
+ engines: {node: '>=18.3.0'}
+ hasBin: true
+
remove-undefined-objects@5.0.0:
resolution: {integrity: sha512-DE8C17uIWeHaY4SqIkpQpHXm0MIdYHtIqjieWuh0I2PG8YcZRxFE6pqeEhnRetsrQ7Lu9uvSNQkDbg95NLpvnQ==}
engines: {node: '>=18'}
@@ -8054,6 +8062,8 @@ snapshots:
- supports-color
- typescript
+ remove-dependencies@0.1.0: {}
+
remove-undefined-objects@5.0.0: {}
repeat-string@1.6.1: {}
diff --git a/src/blocks/blockGitHubActionsCI.test.ts b/src/blocks/blockGitHubActionsCI.test.ts
index 1cc55e004..2e373921f 100644
--- a/src/blocks/blockGitHubActionsCI.test.ts
+++ b/src/blocks/blockGitHubActionsCI.test.ts
@@ -12,6 +12,14 @@ describe("blockGitHubActionsCI", () => {
expect(creation).toMatchInlineSnapshot(`
{
+ "addons": [
+ {
+ "addons": {
+ "requiredStatusChecks": undefined,
+ },
+ "block": [Function],
+ },
+ ],
"files": {
".github": {
"actions": {
@@ -101,6 +109,14 @@ describe("blockGitHubActionsCI", () => {
expect(creation).toMatchInlineSnapshot(`
{
+ "addons": [
+ {
+ "addons": {
+ "requiredStatusChecks": undefined,
+ },
+ "block": [Function],
+ },
+ ],
"files": {
".github": {
"actions": {
@@ -212,6 +228,16 @@ describe("blockGitHubActionsCI", () => {
expect(creation).toMatchInlineSnapshot(`
{
+ "addons": [
+ {
+ "addons": {
+ "requiredStatusChecks": [
+ "Validate",
+ ],
+ },
+ "block": [Function],
+ },
+ ],
"files": {
".github": {
"actions": {
diff --git a/src/blocks/blockPackageJson.test.ts b/src/blocks/blockPackageJson.test.ts
index 7e9bf3b24..49c954aa7 100644
--- a/src/blocks/blockPackageJson.test.ts
+++ b/src/blocks/blockPackageJson.test.ts
@@ -21,7 +21,7 @@ describe("blockPackageJson", () => {
"scripts": [
{
"commands": [
- "pnpm install --offline",
+ "pnpm install",
],
"phase": 1,
},
@@ -44,7 +44,7 @@ describe("blockPackageJson", () => {
"scripts": [
{
"commands": [
- "pnpm install --offline",
+ "pnpm install",
],
"phase": 1,
},
@@ -81,7 +81,7 @@ describe("blockPackageJson", () => {
"scripts": [
{
"commands": [
- "pnpm install --offline",
+ "pnpm install",
"pnpm dedupe",
],
"phase": 1,
@@ -116,7 +116,7 @@ describe("blockPackageJson", () => {
"scripts": [
{
"commands": [
- "pnpm install --offline",
+ "pnpm install",
"pnpm dedupe",
],
"phase": 1,
diff --git a/src/blocks/blockVitest.test.ts b/src/blocks/blockVitest.test.ts
index e942a6e66..1dd05cb97 100644
--- a/src/blocks/blockVitest.test.ts
+++ b/src/blocks/blockVitest.test.ts
@@ -1,9 +1,13 @@
import { testBlock } from "create-testers";
-import { describe, expect, test } from "vitest";
+import { describe, expect, test, vi } from "vitest";
import { blockVitest } from "./blockVitest.js";
import { optionsBase } from "./options.fakes.js";
+vi.mock("../utils/resolveBin.js", () => ({
+ resolveBin: (bin: string) => `path/to/${bin}`,
+}));
+
describe("blockVitest", () => {
test("without addons or mode", () => {
const creation = testBlock(blockVitest, {
@@ -467,6 +471,13 @@ describe("blockVitest", () => {
",
},
"scripts": [
+ {
+ "commands": [
+ "node path/to/remove-dependencies/bin/index.mjs eslint-plugin-jest eslint-plugin-mocha eslint-plugin-vitest jest mocha",
+ ],
+ "phase": 3,
+ "silent": true,
+ },
{
"commands": [
"rm .mocha* jest.config.* vitest.config.*",
diff --git a/src/blocks/blockVitest.ts b/src/blocks/blockVitest.ts
index b5abbadb0..748671252 100644
--- a/src/blocks/blockVitest.ts
+++ b/src/blocks/blockVitest.ts
@@ -2,6 +2,7 @@ import { z } from "zod";
import { base } from "../base.js";
import { getPackageDependencies } from "../data/packageData.js";
+import { resolveBin } from "../utils/resolveBin.js";
import { blockCSpell } from "./blockCSpell.js";
import { blockDevelopmentDocs } from "./blockDevelopmentDocs.js";
import { blockESLint } from "./blockESLint.js";
@@ -32,6 +33,13 @@ export const blockVitest = base.createBlock({
migrate() {
return {
scripts: [
+ {
+ commands: [
+ `node ${resolveBin("remove-dependencies/bin/index.mjs")} eslint-plugin-jest eslint-plugin-mocha eslint-plugin-vitest jest mocha`,
+ ],
+ phase: CommandPhase.Process,
+ silent: true,
+ },
{
commands: ["rm .mocha* jest.config.* vitest.config.*"],
phase: CommandPhase.Migrations,
diff --git a/src/integration.test.ts b/src/integration.test.ts
index 1ee3b31b7..90992bbf6 100644
--- a/src/integration.test.ts
+++ b/src/integration.test.ts
@@ -85,7 +85,11 @@ If you're interested in learning more, see the 'getting started' docs on:
],
}),
blockKnip({
- ignoreDependencies: ["all-contributors-cli", "cspell-populate-words"],
+ ignoreDependencies: [
+ "all-contributors-cli",
+ "cspell-populate-words",
+ "remove-dependencies",
+ ],
}),
blockTSup({
runArgs: ["--version"],
From a98eec32272f13dc1b48c4880b80e4b69dab3553 Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 17:16:35 -0500
Subject: [PATCH 21/22] fix: silent scripts
---
package.json | 2 +-
pnpm-lock.yaml | 36 ++++++++++-----------
src/blocks/blockCodecov.test.ts | 1 +
src/blocks/blockCodecov.ts | 1 +
src/blocks/blockContributingDocs.test.ts | 1 +
src/blocks/blockContributingDocs.ts | 1 +
src/blocks/blockContributorCovenant.test.ts | 1 +
src/blocks/blockContributorCovenant.ts | 1 +
src/blocks/blockDevelopmentDocs.test.ts | 1 +
src/blocks/blockDevelopmentDocs.ts | 1 +
src/blocks/blockESLint.test.ts | 1 +
src/blocks/blockESLint.ts | 1 +
src/blocks/blockGitHubActionsCI.test.ts | 1 +
src/blocks/blockGitHubActionsCI.ts | 1 +
src/blocks/blockKnip.ts | 1 +
src/blocks/blockPackageJson.test.ts | 1 +
src/blocks/blockPackageJson.ts | 1 +
src/blocks/blockPrettier.test.ts | 1 +
src/blocks/blockPrettier.ts | 1 +
src/blocks/blockTSup.test.ts | 1 +
src/blocks/blockTSup.ts | 1 +
src/blocks/blockVitest.test.ts | 1 +
src/blocks/blockVitest.ts | 1 +
23 files changed, 40 insertions(+), 19 deletions(-)
diff --git a/package.json b/package.json
index 2abd69784..2c7e5fd48 100644
--- a/package.json
+++ b/package.json
@@ -38,7 +38,7 @@
},
"dependencies": {
"chalk": "^5.4.1",
- "create": "0.1.0-alpha.14",
+ "create": "0.1.0-alpha.15",
"create-fs": "^0.1.1",
"cspell-populate-words": "^0.3.0",
"execa": "^9.5.2",
diff --git a/pnpm-lock.yaml b/pnpm-lock.yaml
index 9ce5cef77..9152cbd51 100644
--- a/pnpm-lock.yaml
+++ b/pnpm-lock.yaml
@@ -12,8 +12,8 @@ importers:
specifier: ^5.4.1
version: 5.4.1
create:
- specifier: 0.1.0-alpha.14
- version: 0.1.0-alpha.14
+ specifier: 0.1.0-alpha.15
+ version: 0.1.0-alpha.15
create-fs:
specifier: ^0.1.1
version: 0.1.1
@@ -37,13 +37,13 @@ importers:
version: 1.2.0
input-from-file:
specifier: 0.1.0-alpha.4
- version: 0.1.0-alpha.4(create@0.1.0-alpha.14)
+ version: 0.1.0-alpha.4(create@0.1.0-alpha.15)
input-from-file-json:
specifier: 0.1.0-alpha.4
- version: 0.1.0-alpha.4(create@0.1.0-alpha.14)
+ version: 0.1.0-alpha.4(create@0.1.0-alpha.15)
input-from-script:
specifier: 0.1.0-alpha.4
- version: 0.1.0-alpha.4(create@0.1.0-alpha.14)
+ version: 0.1.0-alpha.4(create@0.1.0-alpha.15)
js-yaml:
specifier: ^4.1.0
version: 4.1.0
@@ -131,7 +131,7 @@ importers:
version: 0.5.0
create-testers:
specifier: 0.1.0-alpha.14
- version: 0.1.0-alpha.14(create-fs@0.1.1)(create@0.1.0-alpha.14)
+ version: 0.1.0-alpha.14(create-fs@0.1.1)(create@0.1.0-alpha.15)
cspell:
specifier: ^8.17.2
version: 8.17.2
@@ -1933,8 +1933,8 @@ packages:
create: ^0.1.0-alpha.14
create-fs: ^0.1.0
- create@0.1.0-alpha.14:
- resolution: {integrity: sha512-OExXy9Tn7Nugayz/eKMJV1/oM7J/p2SOi3HK/zEyMXNwF2IwnPiHli6rjyvl5hggU7618cumdD6D5ptJK1Zixw==}
+ create@0.1.0-alpha.15:
+ resolution: {integrity: sha512-iTBTuTIv+f+nD9mr7hyKrKuPyw46UDDBVz0iU7mCnq5MW/0OUqwm7oTFrQ6GTggDiYiS+7Yz3/AZ92EKKZVadA==}
engines: {node: '>=18'}
hasBin: true
@@ -5945,15 +5945,15 @@ snapshots:
create-fs@0.1.1: {}
- create-testers@0.1.0-alpha.14(create-fs@0.1.1)(create@0.1.0-alpha.14):
+ create-testers@0.1.0-alpha.14(create-fs@0.1.1)(create@0.1.0-alpha.15):
dependencies:
- create: 0.1.0-alpha.14
+ create: 0.1.0-alpha.15
create-fs: 0.1.1
diff: 7.0.0
octokit: 4.1.0
without-undefined-properties: 0.1.1
- create@0.1.0-alpha.14:
+ create@0.1.0-alpha.15:
dependencies:
'@clack/prompts': 0.9.1
cached-factory: 0.1.0
@@ -6909,20 +6909,20 @@ snapshots:
ini@4.1.1: {}
- input-from-file-json@0.1.0-alpha.4(create@0.1.0-alpha.14):
+ input-from-file-json@0.1.0-alpha.4(create@0.1.0-alpha.15):
dependencies:
- create: 0.1.0-alpha.14
- input-from-file: 0.1.0-alpha.4(create@0.1.0-alpha.14)
+ create: 0.1.0-alpha.15
+ input-from-file: 0.1.0-alpha.4(create@0.1.0-alpha.15)
zod: 3.24.1
- input-from-file@0.1.0-alpha.4(create@0.1.0-alpha.14):
+ input-from-file@0.1.0-alpha.4(create@0.1.0-alpha.15):
dependencies:
- create: 0.1.0-alpha.14
+ create: 0.1.0-alpha.15
zod: 3.24.1
- input-from-script@0.1.0-alpha.4(create@0.1.0-alpha.14):
+ input-from-script@0.1.0-alpha.4(create@0.1.0-alpha.15):
dependencies:
- create: 0.1.0-alpha.14
+ create: 0.1.0-alpha.15
zod: 3.24.1
inquirer@7.3.3:
diff --git a/src/blocks/blockCodecov.test.ts b/src/blocks/blockCodecov.test.ts
index 878e424c3..185f91f67 100644
--- a/src/blocks/blockCodecov.test.ts
+++ b/src/blocks/blockCodecov.test.ts
@@ -78,6 +78,7 @@ describe("blockCodecov", () => {
"rm .github/codecov.yml codecov.yml",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockCodecov.ts b/src/blocks/blockCodecov.ts
index 36203aa96..2233296a2 100644
--- a/src/blocks/blockCodecov.ts
+++ b/src/blocks/blockCodecov.ts
@@ -16,6 +16,7 @@ export const blockCodecov = base.createBlock({
{
commands: ["rm .github/codecov.yml codecov.yml"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockContributingDocs.test.ts b/src/blocks/blockContributingDocs.test.ts
index 115ab626e..869d7472d 100644
--- a/src/blocks/blockContributingDocs.test.ts
+++ b/src/blocks/blockContributingDocs.test.ts
@@ -234,6 +234,7 @@ describe("blockContributingDocs", () => {
"rm CONTRIBUTING.md",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockContributingDocs.ts b/src/blocks/blockContributingDocs.ts
index 99e30a767..0e68dc515 100644
--- a/src/blocks/blockContributingDocs.ts
+++ b/src/blocks/blockContributingDocs.ts
@@ -11,6 +11,7 @@ export const blockContributingDocs = base.createBlock({
{
commands: ["rm CONTRIBUTING.md"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockContributorCovenant.test.ts b/src/blocks/blockContributorCovenant.test.ts
index e04d232c0..0904a8921 100644
--- a/src/blocks/blockContributorCovenant.test.ts
+++ b/src/blocks/blockContributorCovenant.test.ts
@@ -304,6 +304,7 @@ describe("blockContributorCovenant", () => {
"rm CODE_OF_CONDUCT.md",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockContributorCovenant.ts b/src/blocks/blockContributorCovenant.ts
index 5254b5026..996ec8833 100644
--- a/src/blocks/blockContributorCovenant.ts
+++ b/src/blocks/blockContributorCovenant.ts
@@ -11,6 +11,7 @@ export const blockContributorCovenant = base.createBlock({
{
commands: ["rm CODE_OF_CONDUCT.md"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockDevelopmentDocs.test.ts b/src/blocks/blockDevelopmentDocs.test.ts
index e07c4c126..659faecea 100644
--- a/src/blocks/blockDevelopmentDocs.test.ts
+++ b/src/blocks/blockDevelopmentDocs.test.ts
@@ -58,6 +58,7 @@ describe("blockDevelopmentDocs", () => {
"rm DEVELOPMENT.md",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockDevelopmentDocs.ts b/src/blocks/blockDevelopmentDocs.ts
index fa3e40756..13dcc8986 100644
--- a/src/blocks/blockDevelopmentDocs.ts
+++ b/src/blocks/blockDevelopmentDocs.ts
@@ -70,6 +70,7 @@ export const blockDevelopmentDocs = base.createBlock({
{
commands: ["rm DEVELOPMENT.md"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockESLint.test.ts b/src/blocks/blockESLint.test.ts
index 1f93d2c89..bc6cbf551 100644
--- a/src/blocks/blockESLint.test.ts
+++ b/src/blocks/blockESLint.test.ts
@@ -262,6 +262,7 @@ describe("blockESLint", () => {
"rm .eslintrc* .eslintignore eslint.config.*",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockESLint.ts b/src/blocks/blockESLint.ts
index f2e83f318..663934fd1 100644
--- a/src/blocks/blockESLint.ts
+++ b/src/blocks/blockESLint.ts
@@ -70,6 +70,7 @@ export const blockESLint = base.createBlock({
{
commands: ["rm .eslintrc* .eslintignore eslint.config.*"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockGitHubActionsCI.test.ts b/src/blocks/blockGitHubActionsCI.test.ts
index 2e373921f..cbf7801f5 100644
--- a/src/blocks/blockGitHubActionsCI.test.ts
+++ b/src/blocks/blockGitHubActionsCI.test.ts
@@ -200,6 +200,7 @@ describe("blockGitHubActionsCI", () => {
"rm -rf .circleci travis.yml",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockGitHubActionsCI.ts b/src/blocks/blockGitHubActionsCI.ts
index da2cb94f5..1da0d2289 100644
--- a/src/blocks/blockGitHubActionsCI.ts
+++ b/src/blocks/blockGitHubActionsCI.ts
@@ -36,6 +36,7 @@ export const blockGitHubActionsCI = base.createBlock({
{
commands: ["rm -rf .circleci travis.yml"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockKnip.ts b/src/blocks/blockKnip.ts
index 411e1708e..82585d554 100644
--- a/src/blocks/blockKnip.ts
+++ b/src/blocks/blockKnip.ts
@@ -23,6 +23,7 @@ export const blockKnip = base.createBlock({
{
commands: ["rm .knip* knip.*"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockPackageJson.test.ts b/src/blocks/blockPackageJson.test.ts
index 49c954aa7..56801d9f2 100644
--- a/src/blocks/blockPackageJson.test.ts
+++ b/src/blocks/blockPackageJson.test.ts
@@ -53,6 +53,7 @@ describe("blockPackageJson", () => {
"rm package-lock.json yarn.lock",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockPackageJson.ts b/src/blocks/blockPackageJson.ts
index 9fddaa98d..6ae6bc46b 100644
--- a/src/blocks/blockPackageJson.ts
+++ b/src/blocks/blockPackageJson.ts
@@ -32,6 +32,7 @@ export const blockPackageJson = base.createBlock({
{
commands: ["rm package-lock.json yarn.lock"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockPrettier.test.ts b/src/blocks/blockPrettier.test.ts
index 7c36aa44a..ea0fb6474 100644
--- a/src/blocks/blockPrettier.test.ts
+++ b/src/blocks/blockPrettier.test.ts
@@ -228,6 +228,7 @@ describe("blockPrettier", () => {
"rm .prettierrc* prettier.config*",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockPrettier.ts b/src/blocks/blockPrettier.ts
index d54671820..e385cd658 100644
--- a/src/blocks/blockPrettier.ts
+++ b/src/blocks/blockPrettier.ts
@@ -34,6 +34,7 @@ export const blockPrettier = base.createBlock({
{
commands: ["rm .prettierrc* prettier.config*"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockTSup.test.ts b/src/blocks/blockTSup.test.ts
index 1cfbe508b..fd7a10225 100644
--- a/src/blocks/blockTSup.test.ts
+++ b/src/blocks/blockTSup.test.ts
@@ -181,6 +181,7 @@ describe("blockTSup", () => {
"rm -rf .babelrc* babel.config.* dist lib",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockTSup.ts b/src/blocks/blockTSup.ts
index 43d418ba1..171bc2e29 100644
--- a/src/blocks/blockTSup.ts
+++ b/src/blocks/blockTSup.ts
@@ -22,6 +22,7 @@ export const blockTSup = base.createBlock({
{
commands: ["rm -rf .babelrc* babel.config.* dist lib"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
diff --git a/src/blocks/blockVitest.test.ts b/src/blocks/blockVitest.test.ts
index 1dd05cb97..fdd06f467 100644
--- a/src/blocks/blockVitest.test.ts
+++ b/src/blocks/blockVitest.test.ts
@@ -483,6 +483,7 @@ describe("blockVitest", () => {
"rm .mocha* jest.config.* vitest.config.*",
],
"phase": 0,
+ "silent": true,
},
],
}
diff --git a/src/blocks/blockVitest.ts b/src/blocks/blockVitest.ts
index 748671252..96d3a7a05 100644
--- a/src/blocks/blockVitest.ts
+++ b/src/blocks/blockVitest.ts
@@ -43,6 +43,7 @@ export const blockVitest = base.createBlock({
{
commands: ["rm .mocha* jest.config.* vitest.config.*"],
phase: CommandPhase.Migrations,
+ silent: true,
},
],
};
From d89754f0bd4367b295da2c1dade013b758f773cf Mon Sep 17 00:00:00 2001
From: Josh Goldberg
Date: Wed, 15 Jan 2025 17:19:18 -0500
Subject: [PATCH 22/22] fix: remove-dependencies/bin/index.js
---
src/blocks/blockVitest.ts | 2 +-
1 file changed, 1 insertion(+), 1 deletion(-)
diff --git a/src/blocks/blockVitest.ts b/src/blocks/blockVitest.ts
index 96d3a7a05..2ef2097df 100644
--- a/src/blocks/blockVitest.ts
+++ b/src/blocks/blockVitest.ts
@@ -35,7 +35,7 @@ export const blockVitest = base.createBlock({
scripts: [
{
commands: [
- `node ${resolveBin("remove-dependencies/bin/index.mjs")} eslint-plugin-jest eslint-plugin-mocha eslint-plugin-vitest jest mocha`,
+ `node ${resolveBin("remove-dependencies/bin/index.js")} eslint-plugin-jest eslint-plugin-mocha eslint-plugin-vitest jest mocha`,
],
phase: CommandPhase.Process,
silent: true,