Agent Skill · Microsoft Power Apps

plan-alm

Creates an ALM (Application Lifecycle Management) plan for deploying a Power Pages site across environments. Gathers your promotion strategy, target environments, and approval requirements upfront, generates a visual HTML plan document for review, then — after your approval — executes the plan by calling setup-solution, setup-pipeline, export-solution, and deploy-pipeline (or import-solution) in sequence. Use when asked to: "plan my alm", "set up alm", "create deployment plan", "plan my deployments", "help me deploy to multiple environments", "set up promotion strategy", "create cicd plan", "plan site promotion", "help me go to production", "set up pipeline for my site".

Provider: Microsoft Power Apps Path in repo: plugins/power-pages/skills/plan-alm/SKILL.md

Skill body

Plugin check: Run node "${CLAUDE_PLUGIN_ROOT}/scripts/check-version.js" — if it outputs a message, show it to the user before proceeding.

plan-alm

An 8-phase orchestrator that gathers ALM strategy from the user, generates an HTML deployment plan, gets approval, then executes the plan by calling existing skills in sequence.

Overview

This skill detects the current project state (existing solution, pipeline), asks targeted questions about the desired promotion strategy (Power Platform Pipelines or Manual export/import), generates a visual docs/alm-plan.html, gets user approval, and then invokes setup-solution, setup-pipeline (or export-solution), and deploy-pipeline (or import-solution) in the correct order.

Do NOT create tasks at the start — strategy is unknown until Phase 2 completes. Create all tasks in Phase 3 once the strategy is determined.


Phase 1 — Detect Project State

Do NOT create tasks yet. Use natural language progress reporting only during this phase.

Steps:

  1. Detect prior ALM deferral for this project. Before any discovery work, check whether the project root contains a .alm-deferred marker file. The marker is written by users who explicitly opted ALM-skill validators out of “missing artifacts” warnings (e.g. “this site is handled separately” or “ni-dev — no ALM”). If a user is now invoking plan-alm, we should surface that the marker is present and ask what to do, rather than silently proceeding (which would build a plan the user previously decided not to maintain) or silently removing the marker (which would re-enable nags on every other ALM skill).

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/check-alm-plan.js" --projectRoot "."
    

    🚦 Gate (progress · plan-alm:1.deferral): .alm-deferred marker present — continue and remove, continue and keep marker, or cancel. Determines whether downstream ALM skills resume gate enforcement.

    The helper returns { deferred, deferral, ... }. If deferred === true, read the deferral reason (deferral.reason or the raw marker text) and ask via AskUserQuestion:

    “This project has an .alm-deferred marker — {reason}. ALM was previously deferred here, so the other ALM skills (setup-solution, setup-pipeline, deploy-pipeline, …) skip their plan-completeness checks for this project. How would you like to proceed?”

    Question Header Options
    How would you like to proceed? ALM deferral marker Continue planning and remove the marker (Recommended), Continue planning but keep the marker (record deferral context in plan), Cancel
    • Continue and remove marker (Recommended) → delete .alm-deferred (the user is re-engaging with ALM). Set DEFERRAL_CLEARED = true and proceed to step 1.
    • Continue and keep marker → set DEFERRAL_PRESERVED = true and DEFERRAL_REASON = {reason}. Proceed to step 1. Surface a one-line note in the Phase 1 step 9 user report (e.g. “Note: .alm-deferred is preserved — other ALM skills will continue to skip plan-completeness checks for this project.”) so the user remembers the marker remains in effect after planning.
    • Cancel → exit cleanly (don’t touch the marker).

    If deferred === false, skip this step silently and proceed to step 1.

  2. Resolve the site identity from the local project. ALM skills are normally invoked from a site-root directory where pac pages download-code-site (or a create-site scaffold followed by a deploy) has written .powerpages-site/website.yml. That YAML file is the source of truth for websiteRecordId and siteName.

    Resolution order (first match wins):

    1. .powerpages-site/website.yml (preferred, present for every deployed site) — read with the Read tool and extract:
      • id field → websiteRecordId
      • name field → siteName (the file uses short keys; it is name:, not adx_name:)
    2. powerpages.config.json (fallback, used during plugin development from this repo root or for sites scaffolded but not yet deployed) — read siteName and websiteRecordId.

    If neither is found, stop with:

    “No Power Pages site found in the current directory. Run this skill from your site project root (where .powerpages-site/ exists after pac pages download-code-site). If you haven’t created the site yet, run /power-pages:create-site first.”

    environmentUrl is always re-confirmed from pac env who in step 4 — it does not need to come from either source.

  3. Check for .solution-manifest.json in the project root:
    • Store SOLUTION_DONE = true if found, false otherwise
    • If found, read solution.uniqueName and store as SOLUTION_UNIQUE_NAME
  4. Check for docs/alm/last-pipeline.json in the project root:
    • Store PIPELINE_DONE = true if found, false otherwise
    • If found, read pipelineName and stages[] for later use
  5. Run silently:
    pac env who
    

    Capture the Environment URL and display name. Store as DEV_ENV_URL and DEV_ENV_NAME.

  6. Run silently:
    pac env list --output json 2>/dev/null
    

    Store output as ENV_LIST for pre-filling environment URLs in Phase 2.

  7. Acquire dev environment token (silently):
    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/verify-alm-prerequisites.js" --envUrl "{DEV_ENV_URL}"
    

    Store .token as DEV_TOKEN and .userId as userId. If this fails (auth error), set DEV_TOKEN = null and continue — contents discovery will be skipped gracefully.

  8. Discover and classify site settings (if DEV_TOKEN is available and websiteRecordId is known):

    Use Node.js https module to query. Paginate via @odata.nextLink — sites with > 500 settings would otherwise silently truncate, dropping tier classifications and underreporting plannedEnvVarCount. Send Prefer: odata.maxpagesize=5000 so Dataverse emits the continuation link, then loop until exhausted:

    GET {DEV_ENV_URL}/api/data/v9.2/mspp_sitesettings?$filter=_mspp_websiteid_value eq '{websiteRecordId}'&$select=mspp_name,mspp_value&$top=5000
    Authorization: Bearer {DEV_TOKEN}
    Prefer: odata.maxpagesize=5000
    OData-MaxVersion: 4.0
    OData-Version: 4.0
    Accept: application/json
    

    On each response, append value[] to the running array. If @odata.nextLink is present, GET that URL with the same headers (no need to re-add the filter — the nextLink already encodes the query). Stop when the response has no @odata.nextLink. Cap at 100 iterations for safety.

    Classify the returned settings using ${CLAUDE_PLUGIN_ROOT}/scripts/lib/classify-site-settings.js — the single source of truth for the credential regex and tier mapping shared with setup-solution Phase 5. Either pipe the JSON array of {name, value} rows into the script’s stdin (CLI mode) or require() it inline:

    echo '<JSON array of {name,value}>' \
      | node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/classify-site-settings.js"
    

    Output (the four-bucket shape that downstream phases + setup-solution consume directly):

    SITE_SETTINGS_DATA = {
      keepAsIs: [{name}],                          // regular settings (Tier 3 — Search/Bootstrap/WebApi/feature flags)
      authNoValue: [{name}],                       // Authentication/* or AzureAD/* with empty value (Tier 2b — added as-is, set in target env)
      promoteToEnvVar: [{name, value}],            // Authentication/* or AzureAD/* with value (Tier 2a — setup-solution offers env-var promotion)
      credentialNeedsDecision: [{name, value}]     // ConsumerKey/ConsumerSecret/ClientId/ClientSecret/AppSecret/AppKey/ApiKey/Password (Tier 1 — bulk-with-override prompt in setup-solution Phase 5.4.C)
    }
    

    Tier semantics in plain English (so reviewers reading the plan know what each bucket implies):

    • Tier 1 (credentialNeedsDecision) — credential-style names. Setup-solution Phase 5.4.C runs a single bulk prompt: auto-classify by name (Secret-typed env var for *Secret/*Password/*ApiKey/*AppKey; String-typed for *Id/*ConsumerKey), all-as-Secret, all-as-String, skip-all, or pick-per-credential.
    • Tier 2a (promoteToEnvVar) — auth config with a dev value. Setup-solution Phase 5.4.A asks which to back with env vars so each stage can use different values.
    • Tier 2b (authNoValue) — auth config with no dev value yet. Added to the solution as-is; user sets the value in each target env after deployment.
    • Tier 3 (keepAsIs) — everything else. Added unchanged.

    If the OData query fails or the helper errors out, set SITE_SETTINGS_DATA = null and continue — the plan still renders, it just can’t break down site settings by tier.

  9. Build SOLUTION_CONTENTS_DATA:
    {
      tables: solutionManifest?.components?.tables || [],     // from .solution-manifest.json if SOLUTION_DONE
      botComponents: solutionManifest?.botComponents || [],   // from manifest if available
      siteSettings: SITE_SETTINGS_DATA                        // from step 7, or null
    }
    

    If SOLUTION_DONE = false and manifest is absent, tables and botComponents will be empty arrays — the plan will show a note that they will be discovered during setup-solution.

  10. Report to user:
    Found: **{siteName}** on `{devEnvUrl}`.
    Solution: {✓ already set up ({solutionUniqueName}) / ✗ not yet}.
    Pipeline: {✓ already set up ({pipelineName}) / ✗ not yet}.
    Site settings: {N total — K regular (keep as-is), P auth settings to review for env var, A auth settings (no dev value), C credential-style settings (setup-solution will prompt per credential) / unable to query}.
    
  11. Estimate solution size and evaluate the split decision tree. First ensure the ALM artifacts directory exists (all .alm-* and last-* artifacts live under docs/alm/ to keep the project root uncluttered):
    node -e "require('fs').mkdirSync('docs/alm',{recursive:true})"
    

    Run the estimate helper to classify the site across size, component count, schema heaviness, web file aggregate, and env var count. Use the tmp-file write pattern — if the estimator fails, a prior good docs/alm/alm-size-estimate.json is preserved instead of being overwritten with an empty/partial file. When SOLUTION_DONE = true (a .solution-manifest.json exists), pass --solutionId {solutionId} so the env var count is scoped to the target solution — without it, the estimator falls back to a publisher-prefix tenant-wide query and overcounts whenever the prefix is shared across projects (the common new_ / cr5fe_ regression):

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/estimate-solution-size.js" \
      --envUrl "{DEV_ENV_URL}" --websiteRecordId "{websiteRecordId}" \
      --publisherPrefix "{publisherPrefix}" --siteName "{siteName}" \
      {if SOLUTION_DONE: --solutionId "{solutionManifest.solution.solutionId}"} \
      --projectRoot "." \
      --datamodelManifest "./.datamodel-manifest.json" > ./docs/alm/alm-size-estimate.json.tmp \
      && mv ./docs/alm/alm-size-estimate.json.tmp ./docs/alm/alm-size-estimate.json
    

    When SOLUTION_DONE = false, omit --solutionId; the estimator’s output will include envVarCountScope: "publisher-prefix" to signal the wider scope, and the renderer surfaces this caveat in the Env Variables tab so reviewers know the number reflects the tenant view, not a specific solution. --projectRoot "." enables the disk cross-check — the estimator walks the local build output (dist/, public-output/, build/, .output/) and surfaces webFilesDiskMeasuredMB. When that number is much larger than the Dataverse-measured webFilesAggregateMB, the estimator flips truncationSuspected: true with a warning — file-typed columns whose bytes aren’t returned by $select=content are the usual cause and the plan should trust the disk number. Then run the decision tree (same tmp-file pattern):

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/compute-split-plan.js" \
      --estimate ./docs/alm/alm-size-estimate.json \
      --projectRoot "." \
      --siteName "{siteName}" \
      --publisherPrefix "{publisherPrefix}" > ./docs/alm/alm-split-plan.json.tmp \
      && mv ./docs/alm/alm-split-plan.json.tmp ./docs/alm/alm-split-plan.json
    

    If either command exits non-zero, stop and report the stderr message to the user. Do not proceed to Q1b in Phase 2 without a valid split plan. Store the output as SPLIT_PLAN. Fields to read: splitStrategy, proposedSolutions[], appliedStrategies[], assetAdvisory, sizeAnalysis, recommendations[].

    If SPLIT_PLAN.proposedSolutions.length > 1, set RECOMMEND_SPLIT = true. Otherwise false.

    Report to the user:

    Estimated size: {totalSizeMB} MB — components: {count} — tier: {overall tier}.
    Decision tree result: {splitStrategy} → {N} solutions recommended.
    Asset advisory: {K} files flagged for Azure Blob externalization.
    

10b. Enumerate environment variable definitions (runs whenever DEV_TOKEN is available — the size estimator gives a count but not per-variable metadata).

The renderer's Env Variables tab needs schema name, type, default value, and bound site setting per definition. Without this step, the tab can only show a count-summary note while the size signal and the warning quote a number — three views that don't fully agree. Running this query produces the row-level data so the table renders properly.

Pass `--solutionId` when `SOLUTION_DONE = true` so the returned envVars[] is scoped to the target solution. The helper paginates correctly (Prefer: odata.maxpagesize + @odata.nextLink) regardless of scope — the difference is just which env vars are returned.

```bash
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-env-var-definitions.js" \
  --envUrl "{DEV_ENV_URL}" --token "{DEV_TOKEN}" \
  --publisherPrefix "{publisherPrefix}" \
  --websiteRecordId "{websiteRecordId}" \
  {if SOLUTION_DONE: --solutionId "{solutionManifest.solution.solutionId}"} > ./docs/alm/alm-env-vars.json.tmp \
  && mv ./docs/alm/alm-env-vars.json.tmp ./docs/alm/alm-env-vars.json
```

Read the JSON: `{ envVars: [{ schemaName, type, defaultValue, siteSetting }], count, scope }`. The `scope` field is `'solution'` when `--solutionId` was passed and the solution had env var defs, `'publisher-prefix'` otherwise, `'none'` when the helper short-circuited (no prefix or auth lapse). Store `envVars` as `ENV_VARS_DETAILS` and `scope` as `ENV_VARS_SCOPE` for use when building `planData.envVars` in Phase 3 (pass through unchanged).

The helper degrades gracefully (returns `{ envVars: [], count: 0, scope: 'none' }`) when the publisher prefix is unknown, the token has expired, or the query errors. In those cases the renderer falls back to the size estimator's count via `sizeAnalysis.envVarCount.value` (commit `8cbc39a`) — `ENV_VARS_DETAILS = []` is acceptable.

> **Why scoping matters here**: without `--solutionId`, the helper filters env var defs by publisher prefix tenant-wide. For tenants with a generic prefix (`new_`, `cr5fe_`), this returns env vars from unrelated projects and inflates the count + the `envVars[]` table the renderer draws. The plan looks correct ("12 env vars detected") but is actually showing rows from someone else's project. With `--solutionId`, the helper intersects against `solutioncomponents.componenttype=380` for the target solution — only env vars actually owned by the plan's solution.

**Skip rule**: if `DEV_TOKEN` is null (auth was unavailable in step 6), skip this step and set `ENV_VARS_DETAILS = []`. The renderer's count-summary fallback covers this case.
  1. Pre-plan completeness check (only runs when SOLUTION_DONE = true).

    Before the user approves a plan, verify the existing solution already covers everything on the live site. Components created after the last /power-pages:setup-solution run (server logic from add-server-logic, flows from add-cloud-flow, env vars from configure-env-variables or setup-auth) are silently excluded from any plan built on top of a stale solution.

    Run the shared discovery helper against the source environment:

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-site-components.js" \
      --envUrl "{envUrl}" --token "{token}" \
      --siteId "{websiteRecordId from powerpages.config.json}" \
      --publisherPrefix "{solutionManifest.publisher.prefix}" \
      --solutionId "{solutionManifest.solution.solutionId}"
    

    Parse stdout and evaluate missing.*:

    • All missing.* arrays empty → report “Solution contents match the site — proceeding with fresh plan inputs.” Continue to Phase 2.
    • Any non-empty missing.* array → report a compact summary:

      “Your solution is missing {N} component(s) that exist on the site:

      • {X} site components (e.g. {first 3 names})
      • {L} site languages (powerpagesitelanguage — required; without these the target site silently fails to render post-auth)
      • {Y} cloud flows
      • {Z} environment variable definitions
      • {W} custom tables

      A plan built now will ignore these components. How would you like to proceed?”

      Always render the site languages line when missing.siteLanguages.length > 0, even when other categories are zero — this gap was a recurring silent-failure mode before discover-site-components started enumerating powerpagesitelanguages. See references/solution-api-patterns.md for the 3-entity model.

      🚦 Gate (progress · plan-alm:1.completeness): Completeness check found gaps vs live site. Sync first, plan with gaps recorded, or cancel.

      Ask via AskUserQuestion:

      Question Header Options
      Run /power-pages:setup-solution in sync mode to adopt the missing components before planning? Completeness Check Yes — sync first (Recommended), No — plan with current solution contents, Cancel
      • Yes, sync first (Recommended): invoke /power-pages:setup-solution (auto-detects the existing manifest and enters sync mode). After it completes, re-run the discovery helper; if missing.* is now empty proceed to Phase 2, otherwise repeat the prompt.
      • No, plan with current contents: store the gap summary as KNOWN_GAPS so Phase 3 can surface it in the plan HTML’s Risks section, then continue.
      • Cancel: stop the skill so the user can investigate.

    Why this exists: the same check runs at export (export-solution Phase 2.5) and deploy (deploy-pipeline Phase 3.5). Adding it here catches gaps at the earliest possible gate — before the user invests time reviewing a plan built on stale inputs. See AGENTS.md → ALM-aware by default.

    Skip when SOLUTION_DONE = false: if there is no manifest yet, there is nothing to be stale against — Phase 2 Q1 will handle first-time solution setup.

  2. Run host resolution (PP Pipelines path only — runs after the completeness check).

    Skip rule: if PIPELINE_DONE = true, skip this step entirely — the host info comes from docs/alm/last-pipeline.json. Only fresh-pipeline projects need resolution.

    Acquire a BAP-audience access token (the BAP API uses a different audience than Dataverse):

    az account get-access-token --resource "https://service.powerapps.com/" --query accessToken -o tsv
    

    Capture the output as BAP_TOKEN. If acquisition fails, set HOST_RESOLUTION = { status: 'DetectionFailed', error: '<stderr>' } and skip the detect call.

    Run the detect-only wrapper. Use the same tmp-file-then-mv pattern as Phase 1 step 10 so a prior good docs/alm/alm-host-resolution.json is preserved if the script fails mid-write. Pass --skus Production,Sandbox,Trial so trial-license and developer tenants see their eligible envs in the env-first menu (the helper’s default is Production,Sandbox; we widen to include Trial here because plan-alm’s NoHost branch always offers an existing-env install path that Trial envs can take, even though Trial envs cannot use the create-new fast-path):

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/ensure-pipelines-host-detect.js" \
      --envUrl "{DEV_ENV_URL}" --token "{DEV_TOKEN}" --userId "{userId}" \
      --bapToken "{BAP_TOKEN}" \
      --projectRoot "." \
      --cacheMaxAgeHours 24 \
      --skus Production,Sandbox,Trial > ./docs/alm/alm-host-resolution.json.tmp \
      && mv ./docs/alm/alm-host-resolution.json.tmp ./docs/alm/alm-host-resolution.json
    

    Note: ensure-pipelines-host-detect.js is a detection-only wrapper the ensure-pipelines-host skill exposes for orchestrators. It runs Phases 1.0 (cache fast-path) + 2 (resolution order including tenant-wide enumeration) + 5 (verify if a host is found) of that workflow, but never enters Phase 3 (decision tree) or Phase 4 (provisioning). Output matches the docs/alm/last-host-check.json schemaVersion 2 with actionTaken: "none" always.

    Failure handling: if the detection script exits non-zero, set HOST_RESOLUTION = { status: 'DetectionFailed', error: '<stderr>' } and continue. Phase 2 Q4 falls back to today’s “enter URL manually” branch.

    On success, parse docs/alm/alm-host-resolution.json and store as HOST_RESOLUTION (mapping the wrapper’s field names into the plan-alm shape):

    HOST_RESOLUTION = {
      status: parsed.resolutionStatus,                  // one of: AvailableUsingCustomHost | AvailableUsingCustomHostByAdminDefault | AvailableUsingPlatformHost | AvailableUnboundCustomHost | MultipleUnboundCustomHosts | PlatformHostExistsUnbound | CannotRedirect | NoHost | OrgSettingStale | PermissionDenied
      finalHostEnvUrl: parsed.finalHostEnvUrl,          // string | null
      finalHostEnvId: parsed.finalHostEnvId,            // string | null
      hostType: parsed.isPlatformHost ? 'platform' : (parsed.finalHostEnvUrl ? 'custom' : null),
      pipelinesSolutionVersion: parsed.pipelinesSolutionVersion,  // string | null
      candidates: parsed.candidates                     // { existingCustomHosts[], existingPlatformHost, eligibleForAppInstall[], inaccessibleEnvs[] }
    }
    

    Report a single line:

    Pipeline host: {finalHostEnvUrl} ({status})
    

    or, when no URL is set yet:

    Pipeline host: will be ensured during setup-pipeline ({status})
    

Phase 2 — Gather ALM Strategy

Ask questions in sequence. Solution is always Q1 — it is the prerequisite for all other steps. Branch after Q2 based on promotion strategy selection.

Q1 — Solution Setup (always asked first)

If SOLUTION_DONE = true (manifest found in Phase 1):

🚦 Gate (plan · plan-alm:2.q1-existing): Existing solution found — reuse it (skip setup-solution) or create new (run setup-solution).

Ask via AskUserQuestion:

“A Dataverse solution is already configured for this site: {SOLUTION_UNIQUE_NAME}. Use this existing solution?”

Options:

  1. Yes, use the existing solutionsetup-solution will be skipped in the plan
  2. No, create a new solution — set SOLUTION_DONE = false; setup-solution will run

If SOLUTION_DONE = false (no manifest found):

Tell the user (not via AskUserQuestion — informational only):

“No Dataverse solution is set up for this site yet. setup-solution will be the first step in your plan. The publisher prefix you choose during setup is irreversible — choose carefully.”

🚦 Gate (plan · plan-alm:2.q1-fresh): No existing solution — include setup-solution in plan, or accept a user-supplied unique name.

Ask via AskUserQuestion:

“Ready to include solution setup in the plan?”

Options:

  1. Yes, include solution setup — continue
  2. I already have a solution (enter name) — accept free-text solution unique name, set SOLUTION_DONE = true, SOLUTION_UNIQUE_NAME = user input

Q1b — Split Recommendation (only if RECOMMEND_SPLIT = true)

🚦 Gate (plan · plan-alm:2.q1b-split): Follow recommended split strategy, override to single, accept Asset Advisory first, or show migration guidance.

The decision tree from Phase 1 Step 10 recommended splitting into multiple solutions. Ask via AskUserQuestion:

“Based on the site size and component analysis, the recommended approach is {splitStrategy} — {N} solutions instead of one. Do you want to follow this recommendation?”

Options:

  1. Use the recommended split — proceed with proposedSolutions[] from the decision tree. setup-solution will create all N solutions.
  2. Keep as a single solution anyway — override to single. Record override reason; setup-solution creates one solution with all components.
  3. Accept Asset Advisory first (only offered if assetAdvisory.candidates.length > 0) — user commits to externalizing the flagged assets. Recompute size excluding those files, re-run the decision tree, present the new recommendation.
  4. Show me migration guidance (only offered if an existing .solution-manifest.json is found and does not match the recommendation) — produce docs/alm-migration-plan.md and exit. Do not execute.

If option 1: continue with proposedSolutions.

If option 2 — Keep as a single solution anyway: this overrides a data-driven recommendation that’s frequently right. Before honoring the override, re-surface the tier signals so the user is making an informed choice, not a one-click dismissal. Read from SPLIT_PLAN.sizeAnalysis:

   You're about to override a {splitStrategy} recommendation. Before doing that, here's what the estimator measured:

   • Total size:           {totalSizeMB.value} MB   (tier: {totalSizeMB.tier} — threshold {thresholds.maxSolutionSizeMB} MB)
   • Component count:      {componentCount.value}   (tier: {componentCount.tier} — threshold {thresholds.maxComponentCount})
   • Schema attributes:    {schemaAttrCount.value}  (tier: {schemaAttrCount.tier} — threshold {thresholds.maxSchemaAttrs})
   • Web files aggregate:  {webFilesAggregateMB.value} MB  (tier: {webFilesAggregateMB.tier} — threshold {thresholds.maxAggregateWebFilesMB} MB)
   • Env var definitions:  {envVarCount.value}      (tier: {envVarCount.tier})

   {if SPLIT_PLAN.truncationSuspected === true:
     ⚠ The estimator flagged its inputs as possibly truncated:
       {SPLIT_PLAN.truncationWarnings.join('\n       ')}
     The numbers above could be UNDER-counted. Investigate before overriding.
   }

   Solutions exceeding the platform thresholds frequently fail to import (timeouts, OOM, partial state). A single-solution plan that lands in the red tier is the most common cause of "the deploy hung overnight" reports. Recovering means splitting after the fact, which is harder than splitting upfront.

🚦 Gate (consent · plan-alm:2.q1b-override): Override the data-driven split recommendation to keep as single solution. Free-text overrideReason follows on Yes.

Then ask via AskUserQuestion:

Question Header Options
Still want to keep as a single solution? Override confirmation No — use the recommended {splitStrategy} split (Recommended), Yes — override anyway and note the reason, Cancel — re-think the strategy

Why the friction: in field-reported sessions, “keep as single anyway” was a one-click override and turned out to be the single most common path to a wrong recommendation. The re-confirmation isn’t there to talk the user out of it — it’s there to make sure the override is informed and the reason gets recorded for audit. Override-with-recorded-reason is fully respected; the gate only blocks the silent click-through.

If option 3: subtract advisory candidate sizes from the estimate, re-run compute-split-plan.js, re-present. If option 4: write docs/alm-migration-plan.md (see the spec doc solution-splitting-logic.md §7), commit it, mark plan as Deferred, exit.


Q2 — Strategy Selection (always asked)

🚦 Gate (plan · plan-alm:2.q2-strategy): Pick promotion strategy — PP Pipelines, manual export/import, existing pipeline, or help-me-decide. Branches the rest of the plan.

Ask via AskUserQuestion:

“How do you want to promote your solution between environments?”

Options:

  1. Power Platform Pipelines — Microsoft’s native CI/CD, managed deployments, approval gates
  2. Manual export/import — export a zip from dev and import directly to each target environment
  3. I already have a pipeline set up — run a deployment now
  4. Help me decide — show a quick comparison

If option 4 selected: Explain:

“Power Platform Pipelines is recommended for teams and multiple environments — it provides automated promotion, approval gates, and deployment history in one place. Manual export/import is simpler for one-off migrations or when you only need to deploy once. For ongoing CI/CD, choose Power Platform Pipelines.”

Then re-ask Q2 with only options 1–3.

If option 3 selected: Read docs/alm/last-pipeline.json, confirm pipeline name and stages, then skip to Phase 3 (generate plan) with strategy = pp-pipelines, PIPELINE_DONE = true.


PP Pipelines Path — Q3 through Q6

🚦 Gate (plan · plan-alm:2.q3-stages): Pick how many deployment stages — Staging only / +Production / Production directly / Custom.

Q3: Ask via AskUserQuestion:

“How many deployment stages do you want in this pipeline?”

Options:

  1. Staging only — Dev → Staging (I’ll add Production later)
  2. Staging + Production — Dev → Staging → Production (full promotion chain)
  3. Production directly — Dev → Production only (bypass staging)
  4. Custom — I’ll describe my own stage layout

If option 4: accept free-text description (via “Other”) and build a stage list from the response.

Store stages as PP_STAGES (array of { label, envUrl, envName, type }). Dev is always the source.

For each stage, populate envName from ENV_LIST (gathered in Phase 1 Step 5 via pac env list --output json). Match by URL origin (lowercase, trailing slash stripped, path/query ignored) and copy the entry’s DisplayName (or displayName) into envName. When no match is found — usually because the user pasted a custom URL via “Other” — leave envName unset; the renderer falls back to showing the URL alone in the stage card. The renderer puts envName between the stage label and the URL (e.g. Staging / Supplier Portal Staging / https://orgd6a9894f.crm5.dynamics.com/) so reviewers recognize the env at a glance and the URL stays available as a one-click jump-to-env. Set type: "source" for the dev/source stage and type: "target" for every downstream stage so the renderer applies the active-stage styling correctly.

Q4 (host environment — branches on HOST_RESOLUTION.status from Phase 1 step 12):

This question consumes HOST_RESOLUTION populated by the new detect-only wrapper run in Phase 1 step 12. Each branch sets HOST_ENV_URL (which feeds the rest of plan-alm) and may also set the auxiliary flags CHOSEN_ENV_URL, WILL_PROVISION_PLATFORM, WILL_PROVISION_CUSTOM, WILL_USE_PPAC, WILL_ENSURE_HOST, and USER_CHOSE_DEFER_TO_SETUP_PIPELINE. Defaults: HOST_ENV_URL = HOST_RESOLUTION.finalHostEnvUrl, all flags false / null.

Why the NoHost branch presents the env-first menu here instead of deferring to ensure-pipelines-host Phase 3.C: the original design asked a yes/no “we’ll provision new — continue?” question in plan-alm and let 3.C surface the env-first choice at execution time. In practice the agent treated the plan-alm yes-confirmation as authorization to skip 3.C entirely (or to skip 4.A’s pre-call gate), and users hit 4.A → 409 trial-license errors when an existing env install (4.B) would have been a clean path. Surfacing the env-first menu here — once, at planning time, when the user has full context — eliminates the ambiguity. ensure-pipelines-host then trusts CHOSEN_ENV_URL and skips its own 3.C menu (see ensure-pipelines-host Phase 3 skip rule).

status Q4 prompt Result
AvailableUsingCustomHost, AvailableUsingCustomHostByAdminDefault, AvailableUsingPlatformHost “Detected host {finalHostEnvUrl} (Pipelines v{pipelinesSolutionVersion}). Use this host?” Options: 1. Yes, use this / 2. Use a different host environment (Other) Y → HOST_ENV_URL = HOST_RESOLUTION.finalHostEnvUrl. N → fall back to today’s “enter different URL” branch (free-text via Other).
AvailableUnboundCustomHost “Existing Custom Host {displayName} ({finalHostEnvUrl}) found in tenant — not yet bound to dev env. setup-pipeline will reuse it (recommended; avoids duplicates). Use this host?” Options: 1. Yes, use this / 2. Use a different host environment (Other) Y → HOST_ENV_URL = HOST_RESOLUTION.finalHostEnvUrl, WILL_ENSURE_HOST = true. N → fall back to “enter different URL”.
MultipleUnboundCustomHosts “{N} Custom Hosts found in tenant. Which one should setup-pipeline use?” Options: enumerate HOST_RESOLUTION.candidates.existingCustomHosts[] (up to 3) by display name + URL, plus “Other” for a custom URL, plus “Decide later — setup-pipeline will ask”. Picked candidate → HOST_ENV_URL = candidate.instanceApiUrl, WILL_ENSURE_HOST = true. Decide-later → HOST_ENV_URL = null, WILL_ENSURE_HOST = true, USER_CHOSE_DEFER_TO_SETUP_PIPELINE = true.
PlatformHostExistsUnbound “Tenant Platform Host {finalHostEnvUrl} exists. Use it (no admin role required) or create a new Custom Host?” Options: 1. Use Platform Host / 2. Create new Custom Host / 3. Cancel 1 → HOST_ENV_URL = HOST_RESOLUTION.finalHostEnvUrl, WILL_ENSURE_HOST = true. 2 → HOST_ENV_URL = null, WILL_PROVISION_CUSTOM = true, WILL_ENSURE_HOST = true. 3 → exit.
NoHost Host-type prompt — same shape as ensure-pipelines-host Phase 3.C so the user makes the host choice once, here, instead of being asked again at execution time. Present: “No Pipelines host bound to {devEnvUrl}. Which environment should host Pipelines? Pipelines lives in one env per tenant; pipelines, stages, and run history are stored there. Source envs deploy through it.” Top-level options: 1. “Provision a Platform Host (recommended) — Microsoft-managed Dataverse env auto-provisioned in your tenant home geo. Pipelines app pre-installed. Idempotent. ~3–5 min.” 2. “Set up a Custom Host — Pipelines lives in a Dataverse env you control. We’ll ask whether to use an existing env or create a brand-new dedicated one.” 3. “Open PPAC and create one manually (admin fallback).” 4. “Switch to Manual export/import strategy.” 5. “Cancel.” When the user picks Option 2, surface the Custom-Host sub-prompt: build the eligible-env list from HOST_RESOLUTION.candidates.eligibleForAppInstall[] with role labels (dev env, source env, staging env, production env) per origin match; cap the visible list at 5 envs with role-aware ranking (see “Eligible-env presentation cap” below). Sub-options: a. Each visible env (display name + URL + role labels) labeled “Install Pipelines app on this env” — sandbox-sku envs add a “(Sandbox — confirmation gate)” suffix; append “Other (paste URL)” as the last per-env entry. b. “Create a brand-new dedicated env (D365_ProjectHost template, ~5–10 min, requires Power Platform admin).” c. “Back — return to host-type menu.” When the eligible list is empty, drop sub-option a and present only b / c. Option 1 (Platform Host) → HOST_ENV_URL = null, WILL_PROVISION_PLATFORM = true, WILL_ENSURE_HOST = true. Option 2 → sub-prompt; sub-option a picked env → HOST_ENV_URL = picked.instanceApiUrl, CHOSEN_ENV_URL = picked.instanceApiUrl, WILL_ENSURE_HOST = true (Sandbox confirmation gate, if applicable, must be passed before this resolution stands; “Other (paste URL)” → ask for the env URL via free-text, then proceed as a picked eligible env); sub-option bHOST_ENV_URL = null, WILL_PROVISION_CUSTOM = true, WILL_ENSURE_HOST = true; sub-option c → re-show top-level menu. Option 3 (PPAC) → HOST_ENV_URL = null, WILL_USE_PPAC = true, WILL_ENSURE_HOST = true. Option 4 (Manual strategy) → restart Phase 2 with STRATEGY = manual. Option 5 → exit.
CannotRedirect Block. Show the org-setting vs tenant-default mismatch error from HOST_RESOLUTION.candidates/warnings and stop the skill — only a Power Platform admin can resolve. Exit with the specific error.
OrgSettingStale, PermissionDenied, DetectionFailed Surface the error; ask the user to enter the host URL manually with pac env list pre-fill (today’s fallback). Pre-fill options from ENV_LIST (up to 3 known environment URLs) plus “Other” for a custom URL; pre-fill first option from docs/alm/last-pipeline.json if present. HOST_ENV_URL = user-supplied.

Store the resulting HOST_ENV_URL for use by the rest of plan-alm. The auxiliary flags CHOSEN_ENV_URL, WILL_PROVISION_PLATFORM, WILL_PROVISION_CUSTOM, WILL_USE_PPAC, WILL_ENSURE_HOST, and USER_CHOSE_DEFER_TO_SETUP_PIPELINE feed the planData hostResolution block in Phase 3 and the inline summary in Phase 4. ensure-pipelines-host reads chosenEnvUrl, willProvisionPlatform, willProvisionCustom, and willUsePpac from that block to bypass its own Phase 3.C menu when the user has already made the choice here.

Eligible-env presentation cap (used by the NoHost row above and by MultipleUnboundCustomHosts). When the eligible list runs long, build the visible options as follows so the prompt stays scannable:

  1. Always-visible role-labeled envs first. Include any eligible env carrying a dev env, source env, staging env, or production env label (matched by URL origin against devEnvUrl and against PP_STAGES[].envUrl). These are the project’s own envs and are nearly always the right pick. Dedupe by origin.
  2. Fill remaining slots up to 5 from the rest of the eligible list, in the order returned by list-tenant-envs.js (name-hint pattern pipeline|deploy|host|alm|cicd|govern → admin-perms → recency).
  3. Append “Other (paste URL)” as the last per-env entry inside option 1’s nested list — escape hatch for envs that didn’t make the cap.
  4. When eligible.length > 5, suffix option 1’s headline with: ` Showing top 5 of {N}; the remaining {N-5} eligible env(s) can be reached via the “Other (paste URL)” entry. When eligible.length <= 5`, no suffix (all envs visible inline).
  5. When the user picks “Other (paste URL)”, pre-fill the URL input with ENV_LIST (the pac env list --output json output gathered in Phase 1) so they can paste-or-pick from the full inventory rather than typing a URL by hand.

The same cap policy applies to ensure-pipelines-host Phase 3.C — see that skill’s Step 3a for the same rules. Keep the two implementations consistent so users see the same prompt shape regardless of whether they enter via plan-alm or directly via setup-pipeline → ensure-pipelines-host.

🚦 Gate (plan · plan-alm:2.q5-approval): Pick approval mode — required per stage / staging auto + prod required / no gates.

Q5: Ask via AskUserQuestion:

“Should deployments require approval before each stage?”

Options:

  1. Required before each stage (Recommended for production)
  2. Staging auto-approve, production requires approval
  3. No approval gates — deploy automatically

Store as PP_APPROVAL_MODE.

Note: PP Pipelines always exports as a managed solution to target environments. Set EXPORT_TYPE = "managed" automatically — no question needed.

Q6 (auto-detect, no question): Resolve HAS_ENV_VARS in this order:

  1. If ENV_VARS_DETAILS.length > 0 (Phase 1 Step 10b returned per-variable rows): HAS_ENV_VARS = true. This is the most accurate signal — we’ve enumerated the definitions directly.
  2. Else if SPLIT_PLAN.sizeAnalysis.envVarCount.value > 0 (the size estimator counted definitions but the enumerator didn’t return rows — usually because the publisher prefix or token was missing): HAS_ENV_VARS = true. The plan’s count-summary fallback will explain to the user that per-variable details will be enumerated later.
  3. Else if SOLUTION_DONE = true and .solution-manifest.json lists components with componenttype 380: HAS_ENV_VARS = true. (Stale-manifest fallback — should rarely fire now that 10b is in place.)
  4. Otherwise: HAS_ENV_VARS = false. Variables will be discovered during setup-solution.

When HAS_ENV_VARS = true, the plan notes that deploy-pipeline will prompt for per-stage env var values (see Phase 3 risks population).


Manual Path — Q3 through Q6

🚦 Gate (plan · plan-alm:2.q3-manual): Pick how many target environments for manual export/import path.

Q3: Ask via AskUserQuestion:

“How many target environments do you need to deploy to?”

Options:

  1. One target (e.g. Production)
  2. Two targets (e.g. Staging then Production)
  3. Dev only — not deploying yet

Store as MANUAL_TARGET_COUNT.

If option 3: set MANUAL_TARGET_COUNT = 0. Proceed to Q5.

🚦 Gate (plan · plan-alm:2.q4-manual-target): Pick the URL for each manual target env. Fires PER TARGET in the MANUAL_TARGET_COUNT loop. Two targets (Staging + Production) = two separate prompts. Each prompt pre-fills from pac env list and accepts a different URL. Do NOT collect all target URLs in a single multi-input prompt — each target is a distinct decision (different audiences, different env characteristics, possibly different SKUs).

Q4 (one per stage): For each target environment needed, ask via AskUserQuestion:

“What is the URL for target environment {N}?”

Pre-fill from ENV_LIST: show up to 3 known environment URLs from pac env list as options, plus “Enter a different URL” as the last option.

Store target URLs as MANUAL_TARGETS (array).

🚦 Gate (consent · plan-alm:2.q5-manual-type): Managed vs Unmanaged export — irreversible choice for the produced zip.

Q5: Ask via AskUserQuestion:

“How should the solution be exported?”

Options:

  1. Managed — for staging/production (cannot edit in target)
  2. Unmanaged — for dev-to-dev (editable in target)

Store as EXPORT_TYPE.

🚦 Gate (plan · plan-alm:2.q6-manual-checkpoint): Pause between export and import for review, or proceed automatically.

Q6: Ask via AskUserQuestion:

“Do you want a checkpoint pause between export and import for review?”

Options:

  1. Yes — pause after export so I can review the zip before importing
  2. No — proceed automatically

Store as MANUAL_CHECKPOINT (true or false).

Q6 (auto-detect, no question): Same as PP Pipelines Q6 — check for env var definitions.


Phase 3 — Generate HTML Plan

Now create all tasks — strategy is known.

Task creation

For PP Pipelines path, create these tasks (in order):

# Subject activeForm Description
1 Generate ALM plan Generating ALM plan Build planData, render docs/alm-plan.html
2 Approve ALM plan Awaiting plan approval Present inline summary, get user confirmation
3 Setup solution Setting up solution Invoke setup-solution skill (conditional)
4 Setup pipeline Setting up pipeline Invoke setup-pipeline skill (conditional)
5..N Deploy to {stageName} Deploying to {stageName} Invoke deploy-pipeline skill for this stage — one task per target stage
5..N+1 Activate site in {stageName} Activating site in {stageName} Check activation status; if Pending/null: pac env select --environment "{stage.targetEnvironmentUrl}" then invoke activate-site — one task per target stage
5..N+2 Test site in {stageName} Testing site in {stageName} Invoke /power-pages:test-site against the activated URL; capture pass/fail counts; non-blocking — one task per target stage
N+3 Finalize Finalizing Update HTML status, commit, run skill tracking

Create one Deploy to {stageName} + Activate site in {stageName} + Test site in {stageName} task triplet for each target stage in PP_STAGES (e.g. Staging, Production).

For Manual path, create:

# Subject activeForm Description
1 Generate ALM plan Generating ALM plan Build planData, render docs/alm-plan.html
2 Approve ALM plan Awaiting plan approval Present inline summary, get user confirmation
3 Setup solution Setting up solution Invoke setup-solution skill (conditional)
4 Export solution Exporting solution Invoke export-solution skill
5..N Import to {targetLabel} Importing solution Switch PAC CLI context, invoke import-solution (one task per target)
N+1 Activate site in {targetLabel} Activating site Check activation status, invoke activate-site if not yet provisioned (one task per target, optional)
N+2 Finalize Finalizing Update HTML status, commit, run skill tracking

If SOLUTION_DONE = true, add (will skip — already set up) to the setup-solution task description. If PIPELINE_DONE = true (PP path), add (will skip — already set up) to the setup-pipeline task description.

Activation steps (PP path): Create a separate “Activate site in {stageName}” task for every target stage. After each deploy-pipeline invocation succeeds, the activation task for that stage runs immediately — do not wait until all stages are deployed. The planData steps array must include one "Deploy to {stageName}" + one "Activate site in {stageName}" + one "Test site in {stageName}" triplet per target stage. Activation and testing happen after every stage deployment — not just Production.

Test steps (PP path): Create a separate “Test site in {stageName}” task for every target stage. After each activation completes (or is skipped), the test task runs immediately and is non-blockingtest-site writes docs/alm/last-test-site.json, plan-alm ingests it into validationRuns[stageName], and the rendered HTML’s Validation tab gets a per-stage sub-tab with categorized findings. Failures do not abort the plan.

Activation steps (Manual path): For the Manual path, create one “Activate site in {targetLabel}” task per target environment. These run after the corresponding import completes. The Manual path does not include automatic test-site invocations — site testing is left to the user after manual deployment.

Mark task 1 (“Generate ALM plan”) as in_progress.

Build planData

Build a planData object with all gathered strategy inputs:

{
  "SITE_NAME": "{siteName}",
  "GENERATED_AT": "{ISO timestamp}",
  "STRATEGY": "pp-pipelines | manual",
  "EXPORT_TYPE": "managed | unmanaged",   // PP Pipelines path: always "managed"
  "APPROVAL_MODE": "{approvalMode description}",
  "HAS_ENV_VARS": true | false,
  "SOLUTION_DONE": true | false,
  "PIPELINE_DONE": true | false,
  "PLAN_STATUS": "Draft",
  "LAST_INVOCATION_AT": null,
  "APPROVED_BY": "",
  "APPROVAL_DATE": "",
  "stages": [
    { "label": "Dev", "envUrl": "{devEnvUrl}", "type": "source" },
    { "label": "Staging", "envUrl": "{stagingUrl}", "type": "target" },
    { "label": "Production", "envUrl": "{prodUrl}", "type": "target" }
  ],
  "steps": [
    { "name": "Setup solution", "status": "pending", "skip": false },
    { "name": "Setup pipeline", "status": "pending", "skip": false },
    { "name": "Deploy via pipeline to Staging", "status": "pending", "skip": false },
    { "name": "Activate site in Staging", "status": "pending", "skip": false },
    { "name": "Test site in Staging", "status": "pending", "skip": false },
    { "name": "Deploy via pipeline to Production", "status": "pending", "skip": false },
    { "name": "Activate site in Production", "status": "pending", "skip": false },
    { "name": "Test site in Production", "status": "pending", "skip": false }
  ],
  "validationRuns": {
    "Staging": null,
    "Production": null
  },
  "pipelineMeta": null,                   // populated when docs/alm/last-pipeline.json exists  see "pipelineMeta block" below
  "risks": [
    { "type": "info", "message": "..." }
  ],
  "solutionContents": {
    "tables": ["{table1}", "{table2}"],
    "botComponents": [{ "name": "..." }],
    "siteSettings": {
      "keepAsIs": [{ "name": "..." }],
      "promoteToEnvVar": [{ "name": "...", "value": "..." }],
      "credentialNeedsDecision": [{ "name": "..." }]
    }
  },

  // --- v2 fields from the split decision tree (Phase 1 Step 10) ---
  "sizeAnalysis": { /* tier-classified signals from SPLIT_PLAN.sizeAnalysis */ },
  "assetAdvisory": { /* candidates + recommendation from SPLIT_PLAN.assetAdvisory */ },
  "splitStrategy": "single | strategy-1-layer | strategy-2-change-frequency | strategy-3-schema-segmentation | strategy-4-config-isolation",
  "appliedStrategies": ["strategy-1-layer"],            // may include "composite-sub-partition" when a Layer-split child still exceeded the size or count cap and was sub-divided
  "compositeSubPartitioned": false,                     // mirrors SPLIT_PLAN.compositeSubPartitioned  true when the renderer's strategy rationale should mention sub-partitioning
  "proposedSolutions": [ /* from SPLIT_PLAN.proposedSolutions  ALWAYS at least 1 entry */ ],
  "recommendations": [ /* from SPLIT_PLAN.recommendations */ ],
  "envVars": [ /* from ENV_VARS_DETAILS (Phase 1 Step 10b)  { schemaName, type, defaultValue, siteSetting } per definition; empty array when DEV_TOKEN unavailable */ ],
  "plannedEnvVarCount": 0,                // sum of SITE_SETTINGS_DATA.promoteToEnvVar.length + credentialNeedsDecision.length  env vars setup-solution will offer to create. Renderer shows this as "N planned" alongside the existing-count stat so a fresh project (envVars: []) doesn't look like nothing is happening when the risks list says auth settings will be promoted. Reset to 0 by refresh-alm-plan-data setup-solution phase.
  "breakdown": { /* bytes-per-category from the estimate */ },
  "estimationMethod": "metadata-based",
  "estimationAccuracyPct": 15,
  "truncationSuspected": false,             // true when the estimator's truncation canary fired  surfaced as a red banner on the rendered plan and gates the "keep as single solution anyway" override in Phase 2 Q1b
  "truncationWarnings": [],                 // specific category-level reasons the canary fired (e.g. "Dataverse reports 6050 powerpagecomponent rows but discovery returned 500")
  "webFilesDiskMeasuredMB": null,           // disk-measured total when `--projectRoot` was passed to estimate-solution-size.js and a build-output dir (dist/public-output/build/.output) was found; null otherwise. Renderer surfaces this under the Web Files signal when it disagrees materially with `webFilesAggregateMB`  useful when file-typed columns hold bytes that $select=content can't return.
  "webFilesDiskMeasuredPath": null,         // the build-output directory that produced webFilesDiskMeasuredMB; null when not measured.
  "webFileSampleSize": 0,                   // how many web-file rows the stratified sampler actually read for size measurement. Compared with webFileCount this tells reviewers how aggressively the aggregate was extrapolated.

  // --- Raw discovery snapshot  embedded verbatim for diagnosability ---
  // The mapped fields above (sizeAnalysis, splitStrategy, hostResolution, …)
  // are derived from these. When a rendered plan looks wrong, this block is
  // what a reviewer (or another agent) needs to diagnose without re-running
  // discovery. Never reference these fields from the renderer's display
  // paths  keep them as a sealed diagnostic envelope.
  "rawDiscovery": {
    "estimate": { /* full output of estimate-solution-size.js, verbatim */ },
    "splitPlan": { /* full output of compute-split-plan.js, verbatim */ },
    "hostResolution": { /* full output of ensure-pipelines-host-detect.js (PP path only) or null */ }
  },

  // --- v3 fields from the host resolution (Phase 1 Step 12)  PP Pipelines path only ---
  "hostResolution": {
    "status": "AvailableUsingCustomHost | AvailableUsingCustomHostByAdminDefault | AvailableUsingPlatformHost | AvailableUnboundCustomHost | MultipleUnboundCustomHosts | PlatformHostExistsUnbound | CannotRedirect | NoHost | OrgSettingStale | PermissionDenied | DetectionFailed",
    "hostEnvUrl": "https://pascalepipelineshost.crm.dynamics.com" | null,
    "hostEnvId": "0817fd3d-a664-e99a-a758-dd9dc03ceb01" | null,
    "hostType": "custom | platform | null",
    "pipelinesSolutionVersion": "9.x.y.z" | null,
    "candidatesCount": 0,
    "willEnsureDuringExecution": true | false,
    "willProvisionPlatform": true | false,
    "willProvisionCustom": true | false,
    "willUsePpac": true | false,
    "chosenEnvUrl": "https://orgc4f78248.crm5.dynamics.com/" | null,
    "userChoseDeferToSetupPipeline": false
  }
}

solutionContents is populated from SOLUTION_CONTENTS_DATA built in Phase 1. If discovery was unavailable, pass null — the renderer will show a fallback note.

plannedEnvVarCount is computed from SITE_SETTINGS_DATA (Phase 1 Step 7): (SITE_SETTINGS_DATA.promoteToEnvVar?.length || 0) + (SITE_SETTINGS_DATA.credentialNeedsDecision?.length || 0). When SITE_SETTINGS_DATA is null (Step 7 query failed), set plannedEnvVarCount = 0. The renderer reads this alongside envVars.length (existing) and sizeAnalysis.envVarCount.value (size-estimator’s count) to produce a “N today / +M planned” display in the Overview stat card and Size Analysis signal.

validationRuns block (PP Pipelines path only — initialize one entry per target stage with value null; populated during Phase 7 Step C by ingesting docs/alm/last-test-site.json after each stage’s test run). The full categorized test report drives the new Validation tab in the rendered HTML. Shape per stage:

{
  "validationRuns": {
    "Staging": {
      "url": "https://example.powerappsportals.com",
      "runAt": "2026-04-27T15:00:00.000Z",
      "durationSec": 120,
      "runOutcome": "passed | passed-with-warnings | failed",
      "summary": {
        "critical": 0, "high": 1, "medium": 0, "low": 2,
        "total": 3, "automated": 2, "manual": 1,
        "passed": 2, "failed": 1, "skipped": 0
      },
      "categories": [
        {
          "id": "site-load",
          "name": "Site Load",
          "icon": "📦",
          "tests": [
            {
              "id": "t01",
              "name": "Homepage returns 200 OK",
              "severity": "critical",
              "type": "automated",
              "status": "passed",
              "description": "...",
              "steps": ["GET /", "Expect 200"],
              "expected": "200 OK",
              "actual": "200 OK",
              "validates": "Site activation"
            }
          ]
        }
      ]
    },
    "Production": null
  }
}

The shape is identical to docs/alm/last-test-site.json written by test-site Phase 6.7a — plan-alm reads that file verbatim and assigns it to validationRuns[stageName]. The renderer maps runOutcome to green / yellow / red Outcome badges and produces a sub-tab per stage on the Validation tab. For the Manual path, omit validationRuns from planData.

pipelineMeta block (PP Pipelines path only — read from docs/alm/last-pipeline.json and docs/alm/last-deploy.json at planData-build time. null on fresh plans where no pipeline is configured yet; populated after setup-pipeline and refreshed after each deploy-pipeline run via the post-deploy re-render in Phase 7). Highlights the pipeline that is actually moving configurations for this project. Shape:

{
  "pipelineMeta": {
    "isActive": true,
    "pipelineId": "2b8b5de8-8f43-f111-bec7-6045bd569497",
    "pipelineName": "BYOC Supplier Portal Pipeline",
    "reusedByWiring": null,
    "lastDeploy": {
      "status": "Succeeded",
      "stageName": "Deploy to Staging",
      "deployedAt": "2026-04-29T08:42:00.000Z",
      "artifactVersion": "1.0.0.2",
      "componentCount": 118
    }
  }
}

How to populate. During Phase 3 planData build, read both files (Node.js inline) and inject:

node -e "
const fs = require('fs');
const meta = { isActive: false, pipelineId: null, pipelineName: null, reusedByWiring: null, lastDeploy: null };
try {
  const lp = JSON.parse(fs.readFileSync('docs/alm/last-pipeline.json','utf8'));
  meta.isActive = true;
  meta.pipelineId = lp.pipelineId || null;
  meta.pipelineName = lp.pipelineName || null;
  meta.reusedByWiring = lp.reusedByWiring || null;
} catch {}
try {
  const ld = JSON.parse(fs.readFileSync('docs/alm/last-deploy.json','utf8'));
  meta.lastDeploy = {
    status: ld.status, stageName: ld.stageName, deployedAt: ld.deployedAt,
    artifactVersion: ld.artifactVersion, componentCount: ld.componentCount,
  };
} catch {}
process.stdout.write(JSON.stringify(meta));
"

Embed the result as planData.pipelineMeta.

v2 fields (sizeAnalysis, assetAdvisory, splitStrategy, appliedStrategies, compositeSubPartitioned, proposedSolutions, recommendations, breakdown) come straight from SPLIT_PLAN computed in Phase 1 Step 10, mutated by Q1b user choices. Pass them through unchanged to the renderer. envVars comes from ENV_VARS_DETAILS populated in Phase 1 Step 10b — pass it through unchanged. When the array is empty, the renderer’s count-aware fallback uses sizeAnalysis.envVarCount.value so the Env Variables tab still shows a count-summary note.

Disk-measurement cross-check (webFilesDiskMeasuredMB, webFilesDiskMeasuredPath, webFileSampleSize) — hoist these from the estimator output (rawDiscovery.estimate.webFilesDiskMeasuredMB, etc.) onto the top level of planData. They’re only populated when --projectRoot "." was passed to estimate-solution-size.js AND a build-output directory was detected. The renderer surfaces webFilesDiskMeasuredMB next to the Dataverse-measured Web Files signal when the two numbers disagree materially (the same condition that flips truncationSuspected), so reviewers can see at a glance which number to trust. Pass null for all three when the estimator didn’t measure disk.

Truncation canary (truncationSuspected, truncationWarnings) — pass through verbatim from SPLIT_PLAN.truncationSuspected / SPLIT_PLAN.truncationWarnings (which compute-split-plan.js itself reads from the estimate). When truncationSuspected === true, the rendered plan shows a red banner above the size analysis, and the Phase 2 Q1b “keep as single anyway” override is gated by an extra confirmation step (see Phase 2 Q1b). Common causes: Dataverse pagination regression in estimate-solution-size.js OR a web-file undercount (file-typed columns whose bytes aren’t returned via $select=content) — the disk cross-check (when enabled) flags the latter explicitly.

rawDiscovery block — embed the full estimator + split-plan + host-resolution outputs verbatim, before any field mapping. This is a diagnostic envelope: in general, the renderer must not reference its contents from display paths. Narrow exception: the renderer’s webFilesDiskMeasured* / webFileSampleSize read paths fall back to rawDiscovery.estimate.* if the top-level hoist is missing, so older plan files (written before the hoist was specified) still render the disk-compare note. The right fix when you see that fallback fire is to re-build planData with the hoist populated, not to keep the fallback path active. Otherwise the principle stands: rawDiscovery exists so a reviewer (or a future agent debugging a wrong plan) can read docs/.alm-plan-data.json and see what the discovery scripts actually produced, side-by-side with the mapped fields the renderer used. Read each source file once and assign:

planData.rawDiscovery = {
  estimate: readJson('./docs/alm/alm-size-estimate.json'),
  splitPlan: readJson('./docs/alm/alm-split-plan.json'),
  hostResolution: PIPELINE_DONE
    ? null                                       // host info comes from .last-pipeline.json in that path
    : readJson('./docs/alm/alm-host-resolution.json'),  // PP path with detection; null for Manual path
};

Use null for any source that was skipped (Manual path doesn’t run host resolution; pre-pipeline projects don’t have a manifest). Do not redact or summarize — the value of the snapshot is that it’s the raw input.

proposedSolutions[] is never empty. Even when splitStrategy === "single" and the decision tree recommends one solution, compute-split-plan.js returns one entry describing the base solution (uniqueName, displayName, sizeMB, componentCount, componentTypes). Pass that through. The renderer drives the Solutions tab off this array — leaving it empty produces an unhelpful “structure will be determined” placeholder. If you find yourself with proposedSolutions = [] because compute-split-plan wasn’t run, synthesize a single base entry from solutionContents.solution / data.SITE_NAME / componentCount / totalSizeMB rather than passing through an empty array. The renderer has a safety-net synthesizer for this case but the right fix is upstream — populate it explicitly.

hostResolution block (PP Pipelines path only — omit for Manual path). Built from HOST_RESOLUTION (Phase 1 step 12) plus the auxiliary flags set by Phase 2 Q4:

Populate risks based on gathered data:

Write planData to docs/.alm-plan-data.json (create docs/ if it doesn’t exist).

Render the HTML plan

node "${CLAUDE_PLUGIN_ROOT}/skills/plan-alm/scripts/render-alm-plan.js" \
  --output "<projectRoot>/docs/alm-plan.html" \
  --data "<projectRoot>/docs/.alm-plan-data.json"

Keep docs/.alm-plan-data.json on disk. Phases 5 / 6 / 7 / 8 read this file to refresh hostResolution, pipelineMeta, validationRuns, risks, and the plan footer after each run step, then re-render docs/alm-plan.html so the rendered tabs reflect actual run state (not the pre-run plan). The file is also what check-alm-plan.js reads for the Phase 0 ALM-plan gate in setup-pipeline / deploy-pipeline / setup-solution / export-solution / import-solution / configure-env-variables — deleting it makes every downstream skill think no plan exists. Earlier guidance to delete this file after the initial render was incorrect and caused the Pipelines tab + risks list to stay frozen at pre-run state for the lifetime of the plan.

Write docs/alm/alm-plan-context.json (persists so setup-solution can read it):

{
  "generatedAt": "{ISO timestamp}",
  "siteName": "{siteName}",
  "siteSettings": {
    "keepAsIs": [{name}],
    "authNoValue": [{name}],
    "promoteToEnvVar": [{name, value}],
    "excluded": [{name}]
  }
}

This file is intentionally NOT deleted — setup-solution and other skills read it to skip re-discovery.

Open the HTML plan in the user’s default browser

The inline Markdown summary presented in Phase 4 is intentionally compact — reviewers need to see the full rendered plan (size gauge, signal cards, per-solution breakdown, asset advisory, pipeline stages) before giving informed approval. Launch docs/alm-plan.html in the default browser before the approval prompt so the user can scan the full plan while reading the CLI summary.

Why no Node wrapper. The earlier node -e "spawn('powershell.exe', [...])" chain hits the agent’s sandbox classifier (Node spawning a process that spawns another process is the textbook pattern the classifier blocks). The agent should use the OS-native shell tool directly — no Node detour, no child_process, no detached subprocess.

Step 1. Print the absolute file:// URL prominently first. This is the user’s reliable fallback if the launcher gets blocked:

node -e "process.stdout.write('Plan URL: file:///' + require('path').resolve('docs/alm-plan.html').replace(/\\\\/g, '/') + '\n')"

(Single Node call, no spawn, never blocked. Output: Plan URL: file:///C:/Projects/.../docs/alm-plan.html.)

Step 2. Launch the browser via the OS-native shell. Pick the shell tool the agent has available:

Step 3. Report the URL to the user. After the launch attempt, surface the file:// URL the agent printed in Step 1, so the user always has a clickable backup:

“Opened docs/alm-plan.html in your browser. If it didn’t open automatically, use this link: file:///C:/Projects/.../docs/alm-plan.html. Review it, then answer the approval prompt below.”

If the launch silently fails (sandboxed terminal, SSH session, headless environment), do not retry, do not block, do not loop. Step 1’s printed URL is the contract — the user can click or paste it themselves. Continue to Phase 4 and rely on the CLI summary as backup.

Mark task 1 as completed.


Phase 4 — Present Plan and Get Approval

Mark task 2 (“Approve ALM plan”) as in_progress.

Present a concise inline Markdown summary:

## ALM Plan: {siteName}

**Strategy:** {PP Pipelines / Manual export/import}
**Stages:** {Dev} → {Staging} → {Production (if applicable)}
**Approval gates:** {description from PP_APPROVAL_MODE, or "N/A — manual path"}
**Solution export:** {Managed / Unmanaged}
**Pipeline host:** {hostEnvUrl} ({status}) — *(PP Pipelines path only; when `WILL_ENSURE_HOST = true`, render as `Will be ensured during setup-pipeline ({status})` instead)*

**Steps that will run:**
- [ ] Setup solution {(SKIP — already set up) if SOLUTION_DONE}
- [ ] Setup pipeline {(SKIP — already set up) if PIPELINE_DONE} {(PP path only)}
- [ ] Export solution {(manual path only)}
- [ ] Import to {targetLabel} × {N} {(manual path only)}
- [ ] Deploy via pipeline {(PP path only)}

Full plan written to: docs/alm-plan.html

🚦 Gate (plan · plan-alm:4.approve): The capstone — user approves the rendered HTML plan before plan-alm dispatches any downstream skill. Save-for-later writes the plan but exits without execution.

Ask via AskUserQuestion:

“Does this ALM plan look correct?”

Options:

  1. Approve and execute the plan
  2. Save plan but execute manually later
  3. I want to change something — go back to questions

Capturing the approver (both options 1 and 2):

Capture the name silently using git, falling back to the OS user:

node -e "const {execSync}=require('child_process');let n='';try{n=execSync('git config user.name',{encoding:'utf8'}).trim();}catch{};if(!n){n=process.env.USER||process.env.USERNAME||'';}process.stdout.write(n);"

Store the output as APPROVER. If APPROVER is empty (no git config, no USER env var), ask via AskUserQuestion:

“Who is approving this plan? (needed for the audit trail in docs/alm-plan.html)”

Options: 1. {current system user from whoami} · 2. Other (enter name)

Once APPROVER is known, use Edit to replace the empty/placeholder value in docs/alm-plan.html:

Both spans are guaranteed to exist in the template — there is exactly one of each in the “Execution Checklist” tab footer.


Phase 5 — Execute: setup-solution (conditional)

If SOLUTION_DONE = true: Mark the “Setup solution” task as completed with description “Skipped — solution already configured”. Update the HTML checklist step for “Setup solution” to status-skipped via Edit tool. Skip to Phase 6.

If SOLUTION_DONE = false: Mark the “Setup solution” task as in_progress. Update the HTML checklist step to status-in-progress via Edit tool.

Invoke the skill:

/power-pages:setup-solution

After completion: mark the task as completed. Update the HTML checklist step to status-completed via Edit tool.


Phase 6 — Execute: setup-pipeline OR export-solution

PP Pipelines path

If PIPELINE_DONE = true: Mark the “Setup pipeline” task as completed with description “Skipped — pipeline already configured”. Update HTML checklist step to status-skipped. Skip to Phase 7.

If PIPELINE_DONE = false: Mark the “Setup pipeline” task as in_progress. Update HTML checklist step to status-in-progress.

Invoke the skill:

/power-pages:setup-pipeline

After completion: mark task as completed. Update HTML checklist step to status-completed. Then run the post-run plan refresh — this is not optional; without it the Pipelines tab stays at “Will be ensured during setup-pipeline” and the risks list keeps surfacing the pre-run NoHost warning even though the host now exists:

node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
  --projectRoot "." \
  --phase setup-pipeline \
  --render

This (a) reads docs/alm/last-host-check.json and rewrites planData.hostResolution to the post-run state (status flips from NoHost to AvailableUsingCustomHost, all the willEnsure* / willProvision* / chosenEnvUrl flags clear), (b) reads docs/alm/last-pipeline.json and populates planData.pipelineMeta with the actual pipeline name + ID + host URL + stages (no lastDeploy yet — that fills in after Phase 7 Step A), (c) drops resolved entries from planData.risks (NoHost / Unbound / Platform-Host warnings), then (d) re-renders docs/alm-plan.html. If the helper exits with ok:false, surface the reason — the most likely cause is that docs/.alm-plan-data.json is missing (Phase 3 must have written it; the file should never be deleted between phases).

Manual path

Mark the “Export solution” task as in_progress. Update HTML checklist step to status-in-progress.

Invoke the skill:

/power-pages:export-solution

After completion: mark task as completed. Update HTML checklist step to status-completed.

Refresh the plan after export. Run the helper (export-solution self-refreshes too — this is belt-and-suspenders):

node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
  --projectRoot "." \
  --phase export-solution \
  --render

The helper re-renders docs/alm-plan.html so the Export-step status update is visible to reviewers immediately.

🚦 Gate (progress · plan-alm:7.manual-checkpoint): Manual path pause between export and import — user reviews zip before import proceeds. Defer-now exits cleanly leaving the zip.

If MANUAL_CHECKPOINT = true: Ask via AskUserQuestion:

“Export complete. Review the solution zip at {zipPath} before importing. Ready to proceed with import?”

Options:

  1. Yes, proceed with import
  2. Stop here — I’ll import manually later

If option 2: update HTML plan footer to “Approved — Deferred (paused after export)”. Commit docs/alm-plan.html. Exit.


Phase 7 — Execute: Deploy

PP Pipelines path

For each target stage in PP_STAGES (e.g. Staging, then Production), run this loop:

Step A — Deploy: Mark the “Deploy to {stageName}” task as in_progress. Update HTML checklist step to status-in-progress.

Invoke the skill:

/power-pages:deploy-pipeline

Halt-on-failure check (Step A.1). After deploy-pipeline returns, read docs/alm/last-deploy.json to see what actually happened. Do NOT unconditionally mark the deploy task completeddeploy-pipeline can halt at many points (validation failure, Phase 3.6 batch-validation-failed, blocked-attachments cancel, user-cancel at Phase 6.0 consent gate, mid-deploy AttachmentBlocked, etc.) and each writes a marker with a status field. Proceeding to Step B / the next stage on a failed deploy wastes time and produces confusing output (activate-site against a target that didn’t receive the solution).

node -e "try { const d = require('./docs/alm/last-deploy.json'); const gaps = Array.isArray(d.knownGaps) ? d.knownGaps : []; process.stdout.write(JSON.stringify({status: d.status, stageName: d.stageName, knownGaps: gaps, knownGapsCount: gaps.length})) } catch (e) { process.stdout.write(JSON.stringify({status: 'NoMarker', knownGaps: [], knownGapsCount: 0, error: e.message})) }"

The extractor returns knownGaps: [] when the field is absent or malformed — the agent branches on knownGapsCount > 0 rather than null/undefined checks, which avoids the ambiguity between “field missing” and “field present but empty array”.

Branch on the parsed status. Check knownGaps presence first because Phase 3.6.5’s “deploy succeeded subset” path writes status: "Succeeded" alongside a populated knownGaps[] — a naive status === "Succeeded" branch would miss the gap signal:

🚦 Gate (plan · plan-alm:7.deploy-failure): deploy-pipeline halted before completing the deploy to {stageName}. Caller decides: retry this stage (re-invoke deploy-pipeline — Dataverse import idempotency makes already-succeeded solutions in a multi-solution loop a no-op), skip this stage and continue to the next (advanced — leaves a stage gap; the rendered plan will show it), or exit the orchestration. Cancel exits the loop without touching subsequent stages.

Use AskUserQuestion:

Question Header Options
Deploy to {stageName} did not complete ({status}). What now? Deploy failure Retry this stage, Skip to next stage (advanced — leaves a gap), Exit orchestration

Refresh the plan after each stage’s deploy. Run the helper (regardless of success/failure — the refresh ingests both outcomes):

node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
  --projectRoot "." \
  --phase deploy-pipeline \
  --render

The helper reads docs/alm/last-deploy.json, populates planData.pipelineMeta.lastDeploy (status, stageName, deployedAt, artifactVersion, componentCount, activationStatus, siteUrl), and re-renders docs/alm-plan.html. The Pipelines tab now shows the actual pipeline name + ACTIVE chip + last-run footer (Succeeded / Failed status + version + component count + stage label). Cheap; runs once per stage in the deploy loop. The setStepStatus cross-cutting behavior already maps failure-status markers to failed step status so the checklist surfaces what actually happened — Step A.1’s halt-on-failure gate just adds the user-facing prompt that was previously missing.

Step B — Activate (immediately after deploy for this stage, only when Step A succeeded or completed-with-gaps): Mark the “Activate site in {stageName}” task as in_progress. Update HTML checklist step to status-in-progress.

Read docs/alm/last-deploy.json to check whether activation already happened inside deploy-pipeline:

node -e "const d=require('./docs/alm/last-deploy.json'); process.stdout.write(JSON.stringify({activationStatus: d.activationStatus, siteUrl: d.siteUrl}))"

Step C — Test site (immediately after activate for this stage): Mark the “Test site in {stageName}” task as in_progress. Update HTML checklist step to status-in-progress.

Determine the URL to test:

If a URL is available, invoke the skill (forwarding the URL as the argument):

/power-pages:test-site --siteUrl {activatedUrl}

When test-site completes, ingest its docs/alm/last-test-site.json marker into validationRuns[stageName] and re-render via the same refresh helper used by other phases:

node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
  --projectRoot "." \
  --phase test-site \
  --stageName "{stageName}" \
  --render

The helper reads docs/alm/last-test-site.json, populates planData.validationRuns[{stageName}] with the test outcome (runOutcome, summary counts, categories), and re-renders docs/alm-plan.html so the Validation tab updates immediately.

runOutcome is set by test-site itself (see Phase 6.7a in the test-site skill): "failed" when any critical/high failure exists, "passed-with-warnings" for any non-critical failure or console errors, "passed" otherwise. Trust the value as written — do not re-compute.

Decision rule (non-blocking): Regardless of runOutcome, mark the task completed and continue to the next stage. Failures are diagnostic, not gating — the plan does not abort.

Update the HTML checklist step:

Checklist substep rendering (the renderer handles this automatically once validationRuns is populated and the planData re-rendered): every Test site in {stageName} step gets an inline substep showing the test-result badge (PASSED / WARNINGS / FAILED), the tested URL, the pass / fail / skip summary line, and a “View details →” link that jumps to the Validation tab. Every Deploy via pipeline to {stageName} and Activate site in {stageName} step also gets a Target: <envUrl> substep so reviewers see the target env without leaving the Execution tab. The renderer derives env URLs from data.stages[].envUrl (matched by trailing stage label) — keep stage labels consistent across data.stages and data.steps.

After handling activation and testing, switch PAC CLI back to the dev environment:

pac env select --environment "{devEnvUrl}"

Then repeat Step A + B + C for the next stage (if any).

Manual path (one import per target environment)

For each entry in MANUAL_TARGETS:

  1. Mark the “Import to {targetLabel}” task as in_progress. Update the corresponding HTML checklist step to status-in-progress.

  2. Switch the PAC CLI context to the target environment:
    pac env select --environment "{targetEnvUrl}"
    
  3. Invoke the skill:
    /power-pages:import-solution
    
  4. After completion: mark the task as completed. Update the HTML checklist step to status-completed.

  5. Refresh the plan after the import. Run the helper (import-solution self-refreshes too — this is belt-and-suspenders):

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
      --projectRoot "." \
      --phase import-solution \
      --stageName "{targetLabel}" \
      --render
    

    {targetLabel} is the current iteration’s target (e.g. Staging, Production). The helper reads docs/alm/last-import.json, writes a per-target entry into planData.manualImports[targetLabel] (status, version, component count, failures), and re-renders docs/alm-plan.html. The matching Import to {targetLabel} checklist step picks up an IMPORTED (or FAILED) badge with import details inline — same idiom as the test-site validation substep on PP-path Test steps. Always pass --stageName from the per-target loop so the helper doesn’t have to fall back to URL matching.

  6. Activate site in {targetLabel} (optional) — mark the “Activate site in {targetLabel}” task as in_progress. Update HTML checklist step to status-in-progress.

    PAC CLI is already pointing to the target environment from step 2. Run the activation check:

    node "${CLAUDE_PLUGIN_ROOT}/scripts/check-activation-status.js" --projectRoot "."
    
    • activated: true: Site is already live. Mark task as completed. Update checklist step to status-completed.
    • activated: false: Invoke /power-pages:activate-site. After completion, mark task as completed. Update checklist step to status-completed.
    • error: Mark task as skipped. Note error in summary.
  7. Refresh the plan after activation. Run the helper (activate-site self-refreshes too — this is belt-and-suspenders):

    node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
      --projectRoot "." \
      --phase activate-site \
      --render
    

After all imports: switch PAC CLI back to the dev environment:

pac env select --environment "{devEnvUrl}"

Phase 8 — Finalize

Mark the “Finalize” task as in_progress.

8.1 Update HTML plan status

Run the post-run plan refresh in finalize mode and re-render. This sets planData.PLAN_STATUS = "Completed" and produces the final HTML with all post-run state (latest hostResolution, pipelineMeta + lastDeploy, validationRuns, status footer) consistent across every tab:

node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
  --projectRoot "." \
  --phase finalize \
  --render

If you also need to surface a timestamp in the footer (e.g. plan completion time), apply that via Edit tool after the re-render — the renderer doesn’t currently emit a completion timestamp.

8.2 Run skill tracking

Reference: ${CLAUDE_PLUGIN_ROOT}/references/skill-tracking-reference.md

node "${CLAUDE_PLUGIN_ROOT}/scripts/update-skill-tracking.js" \
  --projectRoot "." \
  --skillName "PlanAlm" \
  --authoringTool "ClaudeCode"

8.3 Commit

git add docs/alm-plan.html && git commit -m "Add ALM plan for {siteName}"

8.4 Present final summary

Display a summary:

## ALM Complete: {siteName}

**Strategy used:** {PP Pipelines / Manual export/import}
**Skills invoked:** {comma-separated list of skills that ran}

**Artifacts created:**
- docs/alm-plan.html — ALM plan document
- .solution-manifest.json — Solution configuration {(if newly created)}
- docs/alm/last-pipeline.json — Pipeline configuration {(PP path only, if newly created)}
- docs/alm/last-deploy.json — Last deployment record {(PP path only)}
- {solutionName}_{managed|unmanaged}.zip — Solution package {(manual path only)}

**Site activation:** {
  PP path: "Activation status per stage is in docs/alm/last-deploy.json and each deploy history file."
  Manual path: list each target env and its activation status (Activated / Pending)
}

Mark the “Finalize” task as completed.


Progress Tracking Table

Task subject activeForm Description
Generate ALM plan Generating ALM plan Gather strategy inputs, build planData, render docs/alm-plan.html
Approve ALM plan Awaiting plan approval Present inline summary + HTML plan path, get user confirmation
Setup solution Setting up solution Invoke setup-solution skill (skip if .solution-manifest.json exists)
Setup pipeline Setting up pipeline Invoke setup-pipeline skill — PP Pipelines path only (skip if docs/alm/last-pipeline.json exists). May delegate to ensure-pipelines-host internally to resolve or provision the host environment when hostResolution.willEnsureDuringExecution is true; that delegation is transparent to plan-alm and is not a separate top-level task.
Export solution Exporting solution Invoke export-solution skill — Manual path only
Deploy to {stageName} Deploying to {stageName} Invoke deploy-pipeline skill — PP Pipelines path, one task per target stage
Activate site in {stageName} Activating site in {stageName} Check activation status + invoke activate-site immediately after each stage deploys — one task per target stage
Test site in {stageName} Testing site in {stageName} Invoke /power-pages:test-site against the activated URL; capture pass/fail counts; non-blocking
Import to {targetEnv} Importing solution Switch PAC CLI context, invoke import-solution — Manual path, one task per target
Activate site in {targetEnv} Activating site Check activation status + invoke activate-site if needed — Manual path, one task per target
Finalize Finalizing Update HTML plan status, commit, run skill tracking, present summary

Key Decision Points (Wait for User)

  1. Phase 2, Q1: Solution setup — confirm existing or include setup-solution in plan
  2. Phase 2, Q2: Promotion strategy — PP Pipelines, Manual, or already set up
  3. Phase 2, Q3–Q6 (PP path): Stage count, host env, approval gates (managed auto-set) Phase 2, Q3–Q6 (Manual path): Target count, target env URLs, export type, checkpoint pause
  4. Phase 4: Plan approval — execute, defer, or revise
  5. Phase 6, Manual: Checkpoint pause after export (if Q6 = Yes)
  6. Phase 7 (delegated): Each invoked skill has its own approval gates

Error Handling

Skill frontmatter

user-invocable: true argument-hint: Optional: 'pipelines' or 'manual' to skip strategy selection allowed-tools: Read, Write, Edit, Bash, Glob, Grep, TaskCreate, TaskUpdate, TaskList, AskUserQuestion model: opus