Skip to content

Expand TypeScript brownfield workspace validation#17510

Merged
sebastienros merged 2 commits into
mainfrom
sebastienros/sebros-expand-typescript-brownfield-work
May 26, 2026
Merged

Expand TypeScript brownfield workspace validation#17510
sebastienros merged 2 commits into
mainfrom
sebastienros/sebros-expand-typescript-brownfield-work

Conversation

@sebastienros
Copy link
Copy Markdown
Contributor

Description

TypeScript brownfield AppHosts need to work in realistic workspace layouts without taking over scripts that users already own. This change preserves existing root aspire:* scripts during TypeScript brownfield init, adds missing AppHost delegate scripts only when safe, and warns when a delegate script was intentionally left untouched.

It also locks down the package-manager boundary for TypeScript AppHosts: AppHost package-manager markers take precedence inside the AppHost package, while guest apps can keep their own package manager and lockfile. The new coverage exercises workspace-subdirectory brownfield init, script preservation, and mixed AppHost/guest package-manager scenarios.

User-facing usage

In a brownfield TypeScript app under a workspace subdirectory, users can run:

aspire init --language typescript --non-interactive

The generated AppHost stays under aspire-apphost, existing app scripts such as dev, build, preview, and existing aspire:start values are preserved, and missing root delegates such as aspire:build and aspire:dev are added to call the nested AppHost package.

Fixes: #17036

Validation:

  • ./restore.sh
  • git diff --check
  • dotnet test --project tests/Aspire.Cli.Tests/Aspire.Cli.Tests.csproj --no-launch-profile -- --filter-class "*.ScaffoldingServiceTests" --filter-class "*.PackageJsonMergerTests" --filter-class "*.TypeScriptAppHostToolchainResolverTests" --filter-not-trait "quarantined=true" --filter-not-trait "outerloop=true"
  • Targeted TypeScript E2E tests were attempted locally, but this environment installed GA CLI 13.3.5 instead of the workspace build and failed on older apphost.ts output before exercising this branch.

Checklist

  • Is this feature complete?
    • Yes. Ready to ship.
    • No. Follow-up changes expected.
  • Are you including unit tests for the changes and scenario tests if relevant?
    • Yes
    • No
  • Did you add public API?
    • Yes
      • If yes, did you have an API Review for it?
        • Yes
        • No
      • Did you add <remarks /> and <code /> elements on your triple slash comments?
        • Yes
        • No
    • No
  • Does the change make any security assumptions or guarantees?
    • Yes
      • If yes, have you done a threat model and had a security review?
        • Yes
        • No
    • No

Preserve existing Aspire package scripts during TypeScript brownfield init and add coverage for workspace subdirectory and package-manager boundary scenarios.

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
Copilot AI review requested due to automatic review settings May 26, 2026 19:17
@github-actions
Copy link
Copy Markdown
Contributor

github-actions Bot commented May 26, 2026

🚀 Dogfood this PR with:

⚠️ WARNING: Do not do this without first carefully reviewing the code of this PR to satisfy yourself it is safe.

curl -fsSL https://raw.githubusercontent.com/microsoft/aspire/main/eng/scripts/get-aspire-cli-pr.sh | bash -s -- 17510

Or

  • Run remotely in PowerShell:
iex "& { $(irm https://raw.githubusercontent.com/microsoft/aspire/main/eng/scripts/get-aspire-cli-pr.ps1) } 17510"

Copy link
Copy Markdown
Contributor

Copilot AI left a comment

Choose a reason for hiding this comment

The reason will be displayed to describe this comment to others. Learn more.

Pull request overview

This PR expands TypeScript “brownfield” workspace validation for Aspire CLI initialization and toolchain resolution. It aims to avoid overwriting user-owned package.json scripts during TypeScript brownfield init, while still adding safe delegate scripts and tightening package-manager/toolchain selection behavior in workspace layouts.

Changes:

  • Preserve existing aspire:* scripts (and only add missing delegates) when augmenting a TypeScript brownfield package.json.
  • Update package.json merge behavior so existing aspire:-prefixed scripts are preserved rather than overwritten.
  • Add/adjust unit and E2E coverage for workspace-subdirectory brownfield init and AppHost-vs-parent toolchain precedence.

Reviewed changes

Copilot reviewed 6 out of 6 changed files in this pull request and generated 1 comment.

Show a summary per file
File Description
src/Aspire.Cli/Scaffolding/ScaffoldingService.cs Adds logic to only add missing root delegate scripts and warn when existing ones are preserved.
src/Aspire.Cli/Scaffolding/PackageJsonMerger.cs Changes merge semantics to preserve existing aspire:-prefixed scripts; updates docs accordingly.
tests/Aspire.Cli.Tests/Scaffolding/ScaffoldingServiceTests.cs Adds tests for delegate-script addition and preservation behavior.
tests/Aspire.Cli.Tests/Scaffolding/PackageJsonMergerTests.cs Updates expectations to validate preservation of existing aspire: scripts.
tests/Aspire.Cli.Tests/Projects/TypeScriptAppHostToolchainResolverTests.cs Adds coverage for AppHost vs parent toolchain marker/lockfile precedence.
tests/Aspire.Cli.EndToEnd.Tests/TypeScriptPolyglotTests.cs Adds E2E coverage for mixed AppHost/guest toolchains and workspace-subdirectory brownfield init behavior.

Comment thread src/Aspire.Cli/Scaffolding/ScaffoldingService.cs Outdated
Resolve brownfield root delegate scripts from the nested AppHost package boundary so generated delegates match the AppHost package manager.

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
@sebastienros sebastienros added the area-polyglot Issues related to polyglot apphosts label May 26, 2026
@sebastienros sebastienros added this to the 13.4 milestone May 26, 2026
@github-actions
Copy link
Copy Markdown
Contributor

CLI E2E Tests unknown — 107 passed, 0 failed, 3 unknown (commit 2656563)

View all recordings
Status Test Recording
AddPackageInteractiveWhileAppHostRunningDetached ▶️ View recording
AddPackageWhileAppHostRunningDetached ▶️ View recording
AgentCommands_AllHelpOutputs_AreCorrect ▶️ View recording
AgentInitCommand_DefaultSelection_InstallsDefaultSkills ▶️ View recording
AgentInitCommand_MigratesDeprecatedConfig ▶️ View recording
AgentMcpListStructuredLogsReturnsLogsFromStarterApp ▶️ View recording
AgentMcpListStructuredLogsReturnsLogsFromStarterApp_DevLocalhost ▶️ View recording
AgentMcpListStructuredLogsReturnsLogsFromStarterApp_Isolated ▶️ View recording
AllPublishMethodsBuildDockerImages ▶️ View recording
AspireAddPackageVersionToDirectoryPackagesProps ▶️ View recording
AspireInitSingleFileAppHostRunsViaDotnetRunAppHost ▶️ View recording
AspireInitWithExistingAppHostDirRecreatesMissingNuGetConfigAndPreservesFiles ▶️ View recording
AspireInitWithSolutionFileGeneratesAppHostThatBuildsAgainstChannelHive ▶️ View recording
AspireStartUpdatesStaleTypeScriptAppHostPath ▶️ View recording
AspireUpdateRemovesAppHostPackageVersionFromDirectoryPackagesProps ▶️ View recording
AspireUpdateRemovesOrphanAppHostPackageVersionWhenSdkAlreadyCurrent ▶️ View recording
Banner_DisplayedOnFirstRun ▶️ View recording
Banner_DisplayedWithExplicitFlag ▶️ View recording
Banner_NotDisplayedWithNoLogoFlag ▶️ View recording
CertificatesClean_RemovesCertificates ▶️ View recording
CertificatesTrust_WithNoCert_CreatesAndTrustsCertificate ▶️ View recording
CertificatesTrust_WithUntrustedCert_TrustsCertificate ▶️ View recording
ConfigSetGet_CreatesNestedJsonFormat ▶️ View recording
CreateAndRunAspireStarterProject ▶️ View recording
CreateAndRunAspireStarterProjectWithBundle ▶️ View recording
CreateAndRunEmptyAppHostProject ▶️ View recording
CreateAndRunJavaEmptyAppHostProject ▶️ View recording
CreateAndRunJsReactProject ▶️ View recording
CreateAndRunPythonReactProject ▶️ View recording
CreateAndRunTypeScriptEmptyAppHostProject ▶️ View recording
CreateAndRunTypeScriptStarterProject ▶️ View recording
CreateJavaAppHostWithViteApp ▶️ View recording
CreateTypeScriptAppHostWithViteApp_AllowsGuestAppPackageManagerToDiffer ▶️ View recording
CreateTypeScriptAppHostWithViteApp_UsesConfiguredToolchain ▶️ View recording
DashboardRunWithAgentMcpListTracesReturnsNoTraces ▶️ View recording
DashboardRunWithAgentMcpListTracesReturnsNoTraces_DevLocalhost ▶️ View recording
DashboardRunWithOtelTracesReturnsNoTraces ▶️ View recording
DashboardRunWithOtelTracesReturnsNoTraces_DevLocalhost ▶️ View recording
DeployK8sBasicApiService ▶️ View recording
DeployK8sWithExternalHelmChart ▶️ View recording
DeployK8sWithGarnet ▶️ View recording
DeployK8sWithMongoDB ▶️ View recording
DeployK8sWithMySql ▶️ View recording
DeployK8sWithPostgres ▶️ View recording
DeployK8sWithRabbitMQ ▶️ View recording
DeployK8sWithRedis ▶️ View recording
DeployK8sWithSqlServer ▶️ View recording
DeployK8sWithValkey ▶️ View recording
DeployTypeScriptAppToKubernetes ▶️ View recording
DescribeCommandResolvesReplicaNames ▶️ View recording
DescribeCommandShowsRunningResources ▶️ View recording
DetachFormatJsonProducesValidJson ▶️ View recording
DetachFormatJsonProducesValidJsonWhenRestartingExistingInstance ▶️ View recording
DoListStepsShowsPipelineSteps ▶️ View recording
DocsCommand_RendersInteractiveMarkdownFromLocalSource ▶️ View recording
DoctorCommand_DetectsDeprecatedAgentConfig ▶️ View recording
DoctorCommand_TypeScriptAppHostReportsMissingConfiguredToolchain ▶️ View recording
DoctorCommand_WithSslCertDir_ShowsTrusted ▶️ View recording
DoctorCommand_WithoutSslCertDir_ShowsPartiallyTrusted ▶️ View recording
GatewayWithoutExternalEndpoint_FailsPublishWithGuidance ▶️ View recording
GeneratedAspireDevScript_StartsWatchMode_WithConfiguredToolchain ▶️ View recording
GlobalMigration_HandlesCommentsAndTrailingCommas ▶️ View recording
GlobalMigration_HandlesMalformedLegacyJson ▶️ View recording
GlobalMigration_PreservesAllValueTypes ▶️ View recording
GlobalMigration_SkipsWhenNewConfigExists ▶️ View recording
GlobalSettings_MigratedFromLegacyFormat ▶️ View recording
IngressWithoutExternalEndpoint_FailsPublishWithGuidance ▶️ View recording
InitTypeScriptAppHost_AugmentsExistingViteRepoInWorkspaceSubdirectory ▶️ View recording
InteractiveCSharpInitCreatesExpectedFiles ▶️ View recording
InvalidAppHostPathWithComments_IsHealedOnRun ▶️ View recording
JavaScriptHostingApisRunFromTypeScriptAppHost ▶️ View recording
LatestCliCanStartStableChannelAppHost ▶️ View recording
LatestCliCanStartStableChannelTypeScriptAppHost ▶️ View recording
LegacySettingsMigration_AdjustsRelativeAppHostPath ▶️ View recording
LogLevelTrace_ProducesTraceEntriesInCliLogFile ▶️ View recording
LogsCommandShowsResourceLogs ▶️ View recording
OtelLogsReturnsStructuredLogsFromStarterApp ▶️ View recording
OtelLogsReturnsStructuredLogsFromStarterAppIsolated ▶️ View recording
PsCommandListsRunningAppHost ▶️ View recording
PsFormatJsonOutputsOnlyJsonToStdout ▶️ View recording
PublishJavaScriptPatternsGeneratesExpectedDockerComposeArtifacts ▶️ View recording
PublishWithConfigureEnvFileUpdatesEnvOutput ▶️ View recording
PublishWithDockerComposeServiceCallbackSucceeds ▶️ View recording
PublishWithoutOutputPathUsesAppHostDirectoryDefault ▶️ View recording
ResourceCommand_FailedExecution_DisplaysAppHostLogPathAndLogContainsEntries ▶️ View recording
ResourceCommand_FailsWhenInteractionServiceIsRequired ▶️ View recording
ResourceCommand_SetAndDeleteParameterUpdatesDescribeOutput ▶️ View recording
RestoreGeneratesSdkFiles ▶️ View recording
RestoreGeneratesSdkFiles_WithConfiguredToolchain ▶️ View recording
RestoreRefreshesGeneratedSdkAfterAddingIntegration ▶️ View recording
RestoreSupportsConfigOnlyHelperPackageAndCrossPackageTypes ▶️ View recording
RunFromParentDirectory_UsesExistingConfigNearAppHost ▶️ View recording
RunReportsSyntaxErrorsForDotNetAppHost ▶️ View recording
RunReportsSyntaxErrorsForTypeScriptAppHost ▶️ View recording
SecretCrudOnDotNetAppHost ▶️ View recording
SecretCrudOnTypeScriptAppHost ▶️ View recording
StagingChannel_ConfigureAndVerifySettings_ThenSwitchChannels ▶️ View recording
StartAndWaitForTypeScriptSqlServerAppHostWithNativeAssets ▶️ View recording
StartReportsSyntaxErrorsForDotNetAppHost ▶️ View recording
StartReportsSyntaxErrorsForTypeScriptAppHost ▶️ View recording
StopAllAppHostsFromAppHostDirectory ▶️ View recording
StopJavaPolyglotAppHostUsingApphostDirectory ▶️ View recording
StopNonInteractiveSingleAppHost ▶️ View recording
StopTypeScriptPolyglotAppHostUsingApphostDirectory ▶️ View recording
StopWithNoRunningAppHostExitsSuccessfully ▶️ View recording
UnAwaitedChainsCompileWithAutoResolvePromises ▶️ View recording
UpdateProjectChannelToStable_CSharpEmptyAppHost_PreservesAspireConfigChannel ▶️ View recording
UpdateProjectChannelToStable_CSharpSingleFileInit_PreservesAspireConfigChannel ▶️ View recording
UpdateProjectChannelToStable_TypeScriptSingleFileInit_PreservesAspireConfigChannel ▶️ View recording
UpdateProjectChannelToStable_TypeScript_PreviewsStablePackagesAndPreservesChannel ▶️ View recording

📹 Recordings uploaded automatically from CI run #26470329909

@sebastienros sebastienros merged commit f7f4393 into main May 26, 2026
613 of 619 checks passed
@sebastienros sebastienros deleted the sebastienros/sebros-expand-typescript-brownfield-work branch May 26, 2026 23:01
aspire-repo-bot Bot added a commit to microsoft/aspire.dev that referenced this pull request May 26, 2026
…script preservation

Documents the behavior introduced in microsoft/aspire#17510:
- Workspace subdirectory layout when running aspire init from a subdirectory
- Root package.json delegate scripts (aspire:start, aspire:build, aspire:dev)
- Script preservation: existing aspire:-prefixed scripts are not overwritten
- Package-manager isolation between the AppHost and guest packages

Co-authored-by: Copilot <223556219+Copilot@users.noreply.github.com>
@aspire-repo-bot
Copy link
Copy Markdown
Contributor

Pull request created: #1080

Generated by PR Documentation Check

@aspire-repo-bot
Copy link
Copy Markdown
Contributor

📝 Documentation has been drafted in microsoft/aspire.dev#1080 targeting release/13.4.

Updated src/frontend/src/content/docs/get-started/add-aspire-existing-app.mdx to document the TypeScript brownfield workspace improvements from this PR. Added sections covering: workspace subdirectory layout when aspire init is run from inside a subdirectory (with --non-interactive), root package.json delegate scripts (aspire:start, aspire:build, aspire:dev), preservation of existing aspire:-prefixed scripts with a warning, and package-manager isolation between the AppHost and guest apps.

Note

This draft PR needs human review before merging.

@IEvangelist
Copy link
Copy Markdown
Member

PR Testing Report

PR Information

CLI Version Verification

  • Expected Commit: 2656563
  • Installed Version: 13.4.0-pr.17510.g26565635
  • Status: Verified

Changes Analyzed

Files Changed

  • src/Aspire.Cli/Scaffolding/PackageJsonMerger.cs - Modified
  • src/Aspire.Cli/Scaffolding/ScaffoldingService.cs - Modified
  • tests/Aspire.Cli.EndToEnd.Tests/TypeScriptPolyglotTests.cs - Modified
  • tests/Aspire.Cli.Tests/Projects/TypeScriptAppHostToolchainResolverTests.cs - Modified
  • tests/Aspire.Cli.Tests/Scaffolding/PackageJsonMergerTests.cs - Modified
  • tests/Aspire.Cli.Tests/Scaffolding/ScaffoldingServiceTests.cs - Modified

Change Categories

  • CLI changes detected
  • Hosting integration changes
  • Dashboard changes
  • Template changes
  • Client/Component changes
  • Test changes detected

Test Scenarios Executed

Scenario 1: Version verification

Objective: Verify dogfood CLI matches the PR head commit.
Coverage Type: Happy path
Status: Passed

Steps:

  1. Installed PR dogfood CLI into an isolated temp directory.
  2. Ran the installed CLI binary with --version.
  3. Compared output with the PR head commit short SHA.

Evidence:

  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\install-output.txt
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\version.txt

Observations:

  • Installed version: 13.4.0-pr.17510.g26565635
  • Expected commit: 2656563

Scenario 2: Brownfield workspace script preservation

Objective: Verify TypeScript brownfield init preserves custom aspire:* scripts and adds missing delegates.
Coverage Type: Happy path + unhappy path
Status: Passed

Steps:

  1. Created an npm workspace with a Vite TypeScript app under packages/brownfield.
  2. Added a custom aspire:start script before initialization.
  3. Ran TypeScript aspire init non-interactively.
  4. Read package.json after init and checked generated root delegates.

Evidence:

  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-2-brownfield-preserve\npm-create-vite.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-2-brownfield-preserve\packages\brownfield\aspire-init.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-2-brownfield-preserve\packages\brownfield\package.json

Observations:

  • Custom aspire:start was preserved.
  • Generated aspire:build: npm --prefix aspire-apphost run aspire:build
  • Generated aspire:dev: npm --prefix aspire-apphost run aspire:dev
  • Init output included the preservation warning.

Expected Unhappy-Path Outcome: Existing custom aspire:start must not be overwritten; init should warn and continue safely.


Scenario 3: AppHost toolchain delegate selection

Objective: Verify brownfield root delegate scripts use the selected pnpm AppHost toolchain.
Coverage Type: Boundary
Status: Passed

Steps:

  1. Created a Vite TypeScript brownfield app.
  2. Set the app package manager to pnpm and installed dependencies.
  3. Ran TypeScript aspire init non-interactively.
  4. Checked generated aspire:* delegate scripts.

Evidence:

  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-3-pnpm-delegates\npm-create-vite.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-3-pnpm-delegates\brownfield\pnpm-install-brownfield.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-3-pnpm-delegates\brownfield\aspire-init.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-3-pnpm-delegates\brownfield\package.json

Observations:

  • aspire:start = pnpm --dir aspire-apphost run aspire:start
  • aspire:build = pnpm --dir aspire-apphost run aspire:build
  • aspire:dev = pnpm --dir aspire-apphost run aspire:dev

Scenario 4: Mixed package-manager guest app

Objective: Verify a pnpm TypeScript AppHost can restore/type-check with an npm Vite guest app.
Coverage Type: Happy path
Status: Passed

Steps:

  1. Initialized a TypeScript AppHost.
  2. Switched the AppHost package manager to pnpm and installed dependencies.
  3. Created a Vite TypeScript guest app using npm.
  4. Added Aspire.Hosting.JavaScript from the PR hive.
  5. Added addViteApp to apphost.mts.
  6. Ran aspire restore and AppHost type-check.

Evidence:

  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\aspire-init-typescript-apphost.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\pnpm-install-apphost.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\npm-create-vite-guest.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\viteapp\npm-install-guest.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\aspire-add-javascript.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\aspire-restore.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\pnpm-typecheck-apphost.log
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\pnpm-lock.yaml
  • C:\Users\dapine\AppData\Local\Temp\aspire-pr-test-ccaddc94f5624f12a3060bdf28f4f4e3\scenario-4-mixed-package-managers\viteapp\package-lock.json

Observations:

  • AppHost restore completed successfully with pnpm.
  • Guest app kept npm lock file.
  • AppHost type-check completed successfully.

Summary

Scenario Status Notes
Version verification Passed -
Brownfield workspace script preservation Passed -
AppHost toolchain delegate selection Passed -
Mixed package-manager guest app Passed -

Overall Result

PR VERIFIED

Recommendations

  • No issues found in the tested TypeScript brownfield workspace scenarios.

Sign up for free to join this conversation on GitHub. Already have an account? Sign in to comment

Labels

area-polyglot Issues related to polyglot apphosts

Projects

None yet

Development

Successfully merging this pull request may close these issues.

Expand TypeScript brownfield and workspace validation

4 participants