plugins/power-pages/skills/deploy-pipeline/SKILL.md
Triggers a Power Platform Pipeline deployment run for a Power Pages solution. Selects a target stage, validates the package, optionally configures deployment settings (environment variables, connection references), then deploys and polls for completion. Use when asked to: "deploy pipeline", "run pipeline", "trigger deployment", "deploy to staging", "deploy to production", "run power platform pipeline", "deploy solution via pipeline", "promote solution", "push to staging", "push to production".
npx skillsauth add microsoft/power-platform-skills deploy-pipelineInstall this skill globally with one command. Works with Claude Code, Cursor, and Windsurf.
3 of 9 scanners reported clean
Some scanners were skipped, did not run, or reported a non-clean status. Review each row below.
Plugin check: Run
node "${CLAUDE_PLUGIN_ROOT}/scripts/check-version.js"— if it outputs a message, show it to the user before proceeding.
Triggers a Power Platform Pipeline deployment run. Reads the existing pipeline configuration from docs/alm/last-pipeline.json, selects a target stage, validates the solution package, and deploys it to the target environment.
Prerequisite: Run
/power-pages:setup-pipelinefirst to create the pipeline configuration.
Refer to
${CLAUDE_PLUGIN_ROOT}/references/cicd-pipeline-patterns.mdfor all HAR-confirmed API patterns used in this skill.
Important: The source (dev) environment must have a Power Platform Pipelines host environment configured. This is set in Power Platform Admin Center (Environments → select env → Pipelines) or via the tenant-level
DefaultCustomPipelinesHostEnvForTenantsetting. Without this configuration,pac pipeline deploywill fail. Thesetup-pipelineskill creates the pipeline definition in the host; this admin step connects the dev environment to that host.
docs/alm/last-pipeline.json exists (created by setup-pipeline).solution-manifest.json existsaz account show succeeds)pac env who succeeds)
plan-almis the front door. When the user expresses an ALM intent (promote / ship / deploy / move to staging / push to prod / release this version), the orchestrator (/power-pages:plan-alm) should run first. Direct invocation ofdeploy-pipelinebypasses the orchestrator's pre-plan completeness check, env-var resolution per stage, activation steps, and validation runs. This gate makes that bypass explicit.
Skip rule. If this skill was invoked as part of an active plan-alm orchestration, skip Phase 0 entirely and proceed to Phase 1. The gate helper exposes this via its inExecution block — pass through silently to Phase 1 when:
inExecution.status === "active"
The helper computes this from docs/.alm-plan-data.json — PLAN_STATUS === "In Execution" AND LAST_INVOCATION_AT within the last 60 minutes. check-alm-plan.js refreshes LAST_INVOCATION_AT automatically on every invocation that finds the plan in execution, so each in-chain skill keeps the chain alive for the next one — even multi-hour deploys (deploy-pipeline alone can take 60 min per stage) survive the window without the chain incorrectly de-classifying. Stalled chains (no heartbeat for > 60 min) reclassify as stale-heartbeat and Phase 0 gates fire normally so an abandoned plan doesn't silently bypass user confirmation.
When inExecution.status is anything other than "active" ("not-running", "stale-heartbeat", "no-plan"), run the Phase 0 gate flow below. Branch on the remaining helper fields:
Step 1 — Run the gate helper.
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/check-alm-plan.js" \
--projectRoot "." \
--envUrl "{devEnvUrl}" \
--token "{token}" \
--solutionId "{solutionId from .solution-manifest.json, if available}"
The helper returns JSON with { exists, stale, staleness: { reason, detail }, generatedAt, planStatus, ... }. The freshness check requires env credentials + solutionId; without those the helper does an existence-only check.
Step 2 — Branch on the result.
| Result | Behavior |
|---|---|
| deferred: true | The user has explicitly deferred ALM for this project (.alm-deferred marker present). Pass through silently to Phase 1 — do not nag. |
| exists: false | The user hasn't run plan-alm yet. See Step 3. |
| exists: true, stale: false | Plan is current. Pass through silently to Phase 1. |
| exists: true, stale: true (reason: solution-modified) | The solution changed after the plan was generated. See Step 4. |
Step 3 — No plan. Tell the user:
<!-- gate: deploy-pipeline:0.no-plan | category=intent | cancel-leaves=nothing -->"No ALM plan exists for this project.
/power-pages:plan-almbuilds one — it detects the project state, asks about your promotion strategy, and orchestrates this skill in the right order alongside setup-solution / setup-pipeline / activate-site / test-site. Want me to run plan-alm now?"
🚦 Gate (intent · deploy-pipeline:0.no-plan): Fail-closed entry gate when
check-alm-plan.jsreturnsexists:false. Helper-script-backed.
AskUserQuestion:
| Question | Header | Options |
|---|---|---|
| Run /power-pages:plan-alm first? | ALM plan gate | Yes — run /power-pages:plan-alm now (Recommended), Continue without a plan (advanced — I just want to deploy), Cancel |
/power-pages:plan-alm. plan-alm's Phase 7 dispatches back into this skill at the appropriate stage.BYPASSED_PLAN_GATE = true and proceed to Phase 1. The deploy will still work, but env-var per-stage values, activation, and post-deploy validation aren't orchestrated.Step 4 — Stale plan. Tell the user:
<!-- gate: deploy-pipeline:0.stale-plan | category=intent | cancel-leaves=nothing -->"ALM plan exists from
{generatedAt}but the source solution has been modified since (at{solution.modifiedon}). The plan's component count, size analysis, and split decisions may be outdated. Re-runningplan-almwill refresh the analysis."
🚦 Gate (intent · deploy-pipeline:0.stale-plan): Fail-closed entry gate when
check-alm-plan.jsreturnsstale:true. Helper-script-backed.
AskUserQuestion:
| Question | Header | Options | |---|---|---| | Refresh the plan first? | ALM plan freshness | Refresh — re-run /power-pages:plan-alm (Recommended), Continue with the existing plan, Cancel |
/power-pages:plan-alm. After completion, re-run the Phase 0 helper once to confirm freshness; if still stale, surface the detail and proceed to Phase 1 anyway (don't infinite-loop).STALE_PLAN_ACK = true and proceed to Phase 1.Relationship to Phase 3.5 (pre-deploy completeness check). Phase 3.5 (later in this skill) catches solution gaps right before deploy. Phase 0 catches the bigger miss: the user who never ran the orchestrator at all and is about to push a half-baked deploy through. The two are complementary.
Create all tasks upfront at the start of this phase.
Tasks to create:
MULTI_RUN_MODE this becomes a single parallel batch (Phase 3.6) covering all N non-skipped solutions; in single-solution / legacy v2 mode it runs per-iteration inline in Phase 4Steps:
Run verify-alm-prerequisites.js to confirm PAC CLI auth, acquire a token, and verify API access:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/verify-alm-prerequisites.js" --require-manifest
Capture output as JSON; extract .envUrl (store as devEnvUrl) and .token (store as DEV_TOKEN). If the script exits non-zero, stop and surface the error — it will indicate whether az login, pac auth, or WhoAmI failed.
Run detect-project-context.js to read project config and solution manifest:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/detect-project-context.js"
Capture output as JSON; extract .solutionManifest (store as solutionManifest), .siteName (store as siteName), and .websiteRecordId. If solutionManifest is null, continue — the manifest is not strictly required at this step (solution info will come from docs/alm/last-pipeline.json).
Locate docs/alm/last-pipeline.json — if not found, stop and advise running /power-pages:setup-pipeline first.
Manifest version check:
schemaVersion === 3, set MULTI_RUN_MODE = true and store deploymentOrder[] as DEPLOYMENT_ORDER. There is a single pipeline with a single set of stages; multi-solution is expressed via N stage runs against the same stage, one per solution in order. This is the current recommended layout.schemaVersion === 2 (legacy), set MULTI_PIPELINE_MODE = true and store pipelines[] as PIPELINES_LIST. The skill falls back to the older "loop over N separate deploymentpipelines records" behavior. Advise the user to re-run setup-pipeline to migrate to v3.pipelineId, pipelineName, hostEnvUrl, sourceDeploymentEnvironmentId, solutionName, stages[] (single-solution mode — existing behavior).In MULTI_RUN_MODE, resolve solutionName + solutionId per iteration of DEPLOYMENT_ORDER. Entries where status === "SkippedEmpty" (typically the {Prefix}_Future buffer) are short-circuited — no stage run is created for them. The single pipelineId / hostEnvUrl / sourceDeploymentEnvironmentId apply to every run.
In MULTI_PIPELINE_MODE (legacy v2), resolve solutionName per pipeline in the loop (not globally). All pipelines share the same hostEnvUrl and sourceDeploymentEnvironmentId.
Acquire host environment token:
az account get-access-token --resource "{hostEnvOrigin}" --query accessToken -o tsv 2>/dev/null
Where hostEnvOrigin = scheme + host of hostEnvUrl. Store as HOST_TOKEN. If acquisition fails, stop with instructions to check Azure CLI auth.
If solutionManifest is available, read solutionManifest.solution.solutionId and solutionManifest.solution.uniqueName from the detected context. Otherwise, use solutionName from docs/alm/last-pipeline.json.
Report: "Pipeline: {pipelineName}. Solution: {solutionName}. Available stages: {stage names}."
Reference:
${CLAUDE_PLUGIN_ROOT}/references/alm-docs-grounding.md
Cap this step at ~30 seconds. If MCP search / fetch errors out, log a one-line note and continue — this skill must remain runnable offline.
microsoft_docs_search with the query: Power Platform Pipelines stage run validation ValidatePackageAsync DeployPackageAsync approval.https://learn.microsoft.com/en-us/power-platform/alm/pipelines (and at most one sister page on stage runs, validation, or approval gates) in parallel via microsoft_docs_fetch.deploymentsettingsjson overrides. Compare against ${CLAUDE_PLUGIN_ROOT}/references/cicd-pipeline-patterns.md and flag any divergence (new status codes, changed stagerunstatus terminal values, new approval-gate API).If the user passed a stage name or environment label as an argument (e.g., staging), match it against stages in docs/alm/last-pipeline.json and skip this question.
🚦 Gate (plan · deploy-pipeline:2.stage): Pick target stage — Staging / Production / etc. Wrong stage selection here is the biggest single failure mode of this skill.
Otherwise, ask via AskUserQuestion:
"Which environment do you want to deploy to? {numbered list of stages from docs/alm/last-pipeline.json, e.g.:
- Deploy to Staging → {stagingEnvUrl}
- Deploy to Production → {prodEnvUrl}}"
Store selected stage as SELECTED_STAGE (with stageId, name, targetDeploymentEnvironmentId, targetEnvironmentUrl).
Design rationale — the deploy loop is serial by design. When
DEPLOYMENT_ORDERhas N entries (e.g.Core → WebAssets → Futurefor a 2-solution split with a future buffer), the loop runs one solution at a time, inorderascending, halt-on-first-failure. This is intentional. Four constraints stack up and make parallelDeployPackageAsynccalls actively harmful, not just non-beneficial:
- Dataverse import lock at the target env.
ImportSolutionAsynctakes an env-level lock — only one solution can actively import at a time per environment. Even if the skill fired NDeployPackageAsynccalls in parallel, the host would queue them and run them serially anyway. The wall-clock win for parallel deploys is effectively zero.- Inter-split dependencies in 3 of 4 split strategies. Change Frequency (
Foundation → Integration → Config → Content), Schema (Domain_1..N → Site), and Layer (Core → WebAssets, where WebAssets ppc rows reference thepowerpagesiterecord in Core) all encode strict ordering. Re-ordering breaks the import — a flow that references a table not yet in the target fails withMissingDependency.- Per-iteration consent gates. Phase 6.0 (
deploy-pipeline:6.0.final-consent) fires before EVERYDeployPackageAsync. The Phase 5 env-var prompt fires per iteration too. Parallel execution would require either batching the gates (explicitly forbidden — see the per-iteration callout below) or running concurrentAskUserQuestions, neither of which the harness supports.- Clean failure handling. Serial + halt-on-first-failure means
docs/alm/last-deploy.jsonrecords per-solutionstatuscleanly. On retry the loop iterates from the start; Dataverse's same-version idempotency turns already-landed solutions into no-ops.Do not "optimize" this loop by wrapping iterations in
Promise.all/Promise.allSettled/await Promise.race. The validation phase (Phase 3.6) IS parallelized —ValidatePackageAsyncdoes NOT take the import lock — but the deploy phase is intentionally serial. If a future Dataverse release removes the env-level import lock, revisit this rationale; until then it is load-bearing, not an oversight.
In MULTI_RUN_MODE (v3 — recommended): The selected stage is looked up once from the single stages[] array. The skill then loops over DEPLOYMENT_ORDER in order, creating one stage run per solution against the same stageId:
create-stage-run + ValidatePackageAsync + poll-validation-status for every non-skipped solution in parallel. Halts the entire deploy if any solution fails validation. Stores the per-solution stageRunId in VALIDATED_STAGE_RUNS so the serial deploy loop can reuse them.DEPLOYMENT_ORDER where status !== "SkippedEmpty": resolve its solutionUniqueName + solutionId, retrieve its stageRunId from VALIDATED_STAGE_RUNS[solutionUniqueName], set ARTIFACT_SOLUTION_NAME / ARTIFACT_SOLUTION_ID / STAGE_RUN_ID, then run Phases 4.4 (fetch deployment notes) → 5 (configure) → 6.0 (consent gate fires every iteration) → 6.1 (deploy) → 6.2 (poll) against the same pipeline. Phase 4.1–4.3 (create stage run + validate + poll-validation) are skipped — Phase 3.6 already did the work in parallel.docs/alm/last-deploy.json at the end summarizing all runs for the selected stage. Record per-solution status (Succeeded / Failed / NotAttempted / SkippedEmpty) plus the shared pipelineId.⚠ Per-iteration gate firing — non-negotiable. Inside the loop, the full Phase 3 → 3.5 → 4 → 5 → 6.0 → 6.1 → 6.2 → 7 sequence runs for each solution. Do NOT batch validation across solutions, do NOT batch the Phase 6.0 consent gate, and do NOT treat any upstream answer (Phase 2 stage selection,
--stageargument, the previous iteration's "Deploy now") as covering subsequent iterations. The Phase 6.0 gate firesNtimes forNnon-skipped solutions inDEPLOYMENT_ORDER. If you find yourself proceeding from iteration 1's success directly to iteration 2'sDeployPackageAsyncwithout a fresh Phase 6.0 prompt, you have skipped the gate.
In MULTI_PIPELINE_MODE (v2 — legacy): The selected stage label (e.g., "Staging") is matched against each pipeline's stages[] — each pipeline has its own stageId for the same target environment. All subsequent phases (validate, deploy, poll) are looped over PIPELINES_LIST in order:
pipelines[i].stageId where stage label matches SELECTED_STAGE.name, pipelines[i].solutionName, etc.docs/alm/last-deploy.json at the end summarizing all pipeline runs for this stage. Record per-pipeline status (Succeeded / Failed / NotAttempted) so a retry can tell which ones still need to run.⚠ Per-iteration gate firing also applies here. Same rule as MULTI_RUN_MODE: each pipeline in the loop gets its own Phase 4 / 5 / 6.0 / 6.1 / 6.2 sequence. The Phase 6.0 consent gate fires once per pipeline. Do NOT batch.
Partial-deploy risk. When the loop halts (e.g.,
Coresucceeded,WebAssetsfailed), the target environment is left in a mixed state — there is no automatic rollback of solutions that already imported. The per-solution (v3) or per-pipeline (v2)statusindocs/alm/last-deploy.jsonis the source of truth for what landed. When the user re-runsdeploy-pipelineafter fixing the failure, the loop iterates all entries again from the start; rely on the solution-import idempotency (same version = no-op) rather than skipping. Warn the user of this before starting a multi-solution deploy to production.
Check docs/alm/last-deploy.json — if the last deployment to this stage failed, warn the user:
"The last deployment to
{stageName}had status: Failed. Would you like to retry? 1. Yes, retry / 2. No, cancel"
Power Pages code-site solutions almost always contain .js bundle chunks (Vite/Rollup output) as Web File components. If the target env's blockedattachments setting includes .js, ImportSolutionAsync will reject every web-file write — typically 50-75 minutes into an import for sites with thousands of bundle chunks (real-world: a Content solution failed at 3,909 rejected .js files on Staging after the same issue had already been fixed on Dev). The reactive Phase 7.6 handler will detect this and offer unblock-and-retry, but the user has already burned an hour. This pre-flight catches it in 10 seconds.
Skip rule. This check is for Power Pages projects only. Skip when powerpages.config.json has no websiteRecordId (non-Power-Pages ALM run — pure data-model solution, etc.). Skip when the user's plan/manifest indicates no solution being deployed has Web File componentType (rare in code-site projects but possible for back-end-only solutions).
Detection signal. In MULTI_RUN_MODE / MULTI_PIPELINE_MODE: read .solution-manifest.json and check whether any entry in solutions[] has componentTypes including "Web File". In single-solution mode: assume true for any Power Pages project (the umbrella solution carries web files).
Steps:
Switch PAC CLI context to the target environment so fix-blocked-attachments.js queries the right env:
pac env select --environment "{SELECTED_STAGE.targetEnvironmentUrl}"
Run the helper in dry-run mode to detect the current state:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/fix-blocked-attachments.js" \
--envUrl "{SELECTED_STAGE.targetEnvironmentUrl}" \
--extensions js,css \
--dry-run
Capture the output as JSON. Inspect wasBlocked[]:
wasBlocked: [] → target env doesn't block the relevant extensions. Switch PAC CLI back to source (pac env select --environment "{sourceEnvUrl}") and proceed to Phase 3. No prompt, no noise.🚦 Gate (consent · deploy-pipeline:2.5.blocked-attachments): Pre-flight — modify target env's
blockedattachmentssecurity setting (tenant-wide impact). Reversible from PPAC. Skipping costs 50–75 min of wasted import.
wasBlocked: ["js"] or includes other media-relevant extensions → the deployment WILL fail mid-import. Prompt the user immediately via AskUserQuestion (do NOT bury this in chat — it MUST gate Phase 3 progression):
Pre-flight detected an issue. The target environment
{targetEnvName}currently blocks file types that this solution needs:{wasBlocked.join(', ')}. Power Pages code sites ship.jsbundle chunks as web files — if you proceed without unblocking, the deployment will run for ~50-75 minutes and then fail (the failure is recoverable via Phase 7.6's retry path, but the wasted time is not). The block can be removed in 10 seconds.Note: this modifies an environment-level security setting that affects all users of
{targetEnvName}. Reversible from PPAC → Environments →{targetEnvName}→ Settings → Product → Features → Blocked Attachments.
| Question | Header | Options |
|---|---|---|
| Allow removing the block on {wasBlocked.join(', ')} for the {targetEnvName} environment so the deployment can proceed? | Unblock attachments | Yes — unblock these types and continue, No — proceed anyway (Phase 7.6 will catch the failure after deploy and prompt again), Cancel deploy |
Branch on the answer:
Yes — unblock and continue: invoke fix-blocked-attachments.js without --dry-run (same --extensions). Read the result and confirm changed: true + removed[] is non-empty. Switch PAC CLI back to source (pac env select --environment "{sourceEnvUrl}"). Proceed to Phase 3. Record the unblock action in the eventual docs/alm/last-deploy.json preflightActions[] block so the deploy summary has an audit trail.
No — proceed anyway: leave the setting unchanged. Switch PAC CLI back to source. Proceed to Phase 3. Tell the user clearly: "Continuing without unblocking. If the deployment fails on AttachmentBlocked, Phase 7.6 will offer the same unblock prompt — but you'll have spent ~50-75 minutes getting there."
Cancel deploy: stop cleanly. Do not modify any environment setting. Do not create the stage run. Tell the user how to re-invoke when ready.
Always switch PAC CLI back to the source environment before exiting this phase. Subsequent phases assume PAC points at the source unless they explicitly switch to the target.
Why pre-flight + reactive both exist. The reactive Phase 7.6 handler stays in place because (a) the pre-flight only inspects the env-level
blockedattachmentssetting — there are other rare blocked-attachment causes (per-table file column attachment policies) that only surface during the actual import; (b) the env's blocklist could change between pre-flight and import (rare but possible if another admin edits it concurrently); (c) backward compatibility with existing flows where the user invokeddeploy-pipelinedirectly and skipped Phase 2.5 via legacy SKILL.md. The two paths are complementary, not redundant.
Call RetrieveDeploymentPipelineInfo to get the authoritative source environment ID and available solution artifacts:
GET {hostEnvUrl}/api/data/v9.1/RetrieveDeploymentPipelineInfo(DeploymentPipelineId={pipelineId},SourceEnvironmentId='{BAP_SOURCE_ENV_ID}',ArtifactName='{solutionName}')
Authorization: Bearer {HOST_TOKEN}
OData-MaxVersion: 4.0
OData-Version: 4.0
Accept: application/json
Where BAP_SOURCE_ENV_ID = the BAP GUID of the dev environment (from pac env list, stored in docs/alm/last-pipeline.json or available from pac env who).
Extract:
SourceDeploymentEnvironmentId — use as the devdeploymentenvironment binding in the stage run. Store as sourceDeploymentEnvironmentId.StageRunsDetails[].DeploymentStage — confirms available stages and their IDsEnableAIDeploymentNotes — store as AI_NOTES_ENABLED (bool)Use solutionId from .solution-manifest.json as ARTIFACT_SOLUTION_ID and uniqueName as ARTIFACT_SOLUTION_NAME.
If
RetrieveDeploymentPipelineInforeturns 404 (older Pipelines package): use the navigation property to find the source deployment environment:GET {hostEnvUrl}/api/data/v9.1/deploymentpipelines({pipelineId})/deploymentpipeline_deploymentenvironment?$select=deploymentenvironmentid,name,environmenttypeFilter for
environmenttype = 200000000to get the source record. Usedeploymentenvironmentidas thesourceDeploymentEnvironmentId. For the artifact/solution list, usesourceDeploymentEnvironmentIdfromdocs/alm/last-pipeline.jsonandsolutionNamefrom.solution-manifest.jsonas fallbacks. Set a flagVALIDATE_PACKAGE_UNAVAILABLE = trueto skip Phase 4.2–4.3 and use the PAC CLI path in Phase 6.
A pipeline's ValidatePackageAsync confirms the solution zip is importable on the target, but it does not tell you whether the solution zip itself covers every component that exists on the source site. Components added after setup-solution last ran (server logic, cloud flows, bots, env vars, etc.) can be silently left behind.
Run the shared site-inventory helper against the source (dev) environment:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-site-components.js" \
--envUrl "{devEnvUrl}" --token "{DEV_TOKEN}" \
--siteId "{websiteRecordId from .solution-manifest.json}" \
--publisherPrefix "{publisherPrefix from .solution-manifest.json}" \
--solutionId "{solutionId from .solution-manifest.json}"
Parse stdout and evaluate missing.*. Before doing anything else, capture the pre-sync state so a post-sync re-confirmation gate can show what changed:
PRE_SYNC_VERSION = solutionManifest.solution.version // from .solution-manifest.json read in Phase 1
PRE_SYNC_MISSING = { siteComponents, siteLanguages, cloudFlows, envVarDefinitions, customTables, ... } // from the discovery stdout above
Then:
missing.* empty → proceed to Phase 4.🚦 Gate (progress · deploy-pipeline:3.5.completeness): Source solution incomplete vs live site. Sync first, deploy anyway (gap ships), or cancel.
Any non-empty → report a short summary ("Solution is missing {N} components"). Ask via AskUserQuestion:
"The source solution appears incomplete relative to the live site. What would you like to do?
- Run
/power-pages:setup-solutionnow (sync mode) — adopts missing components and bumps the version, then re-confirm with you before deploying (Recommended)- Deploy anyway — the missing components will not reach the target
- Cancel — I'll investigate first"
Option 1 — Sync first, then re-confirm before deploy:
/power-pages:setup-solution (auto-detects the existing manifest, enters sync mode, adopts missing components, bumps the version). Wait for completion. setup-solution's final refresh step writes LAST_SYNC_AT into docs/.alm-plan-data.json so subsequent check-alm-plan.js calls do NOT falsely flag the plan as stale just because the sync bumped solutions.modifiedon past GENERATED_AT — the freshness reference becomes max(GENERATED_AT, LAST_SYNC_AT)..solution-manifest.json and capture POST_SYNC_VERSION = solutionManifest.solution.version.missing.* remain non-empty, repeat the Phase 3.5 prompt above.NEWLY_ADOPTED as a per-category set difference between PRE_SYNC_MISSING and the second discovery run's missing.* (the items that disappeared are what setup-solution just adopted into the solution). Total count = sum of all category lengths.🚦 Gate (progress · deploy-pipeline:3.5.post-sync): Post-sync re-confirm. Solution version bumped + components adopted — user inspects delta before deploy proceeds.
Re-confirm with the user before proceeding to Phase 4 — the solution about to ship is now different from what the user originally saw when they started the deploy. Use AskUserQuestion:
"Sync complete.
{solutionUniqueName} is now v{POST_SYNC_VERSION} (was v{PRE_SYNC_VERSION}) with {NEWLY_ADOPTED.total} newly-adopted components:
- {first 3-5 names by category — prefer high-signal categories: cloud flows, server logic, env var definitions, then site components}
- {if more remain:
+ {N} more across {category list}}About to deploy this updated solution to {SELECTED_STAGE.name} ({targetEnvUrl}).
Continue with the deployment?"
| Question | Header | Options | |---|---|---| | Continue with the deployment? | Post-sync approval | Yes — deploy v{POST_SYNC_VERSION} to {SELECTED_STAGE.name} (Recommended), Pause — I want to review the new solution contents first, Cancel — abort the deploy |
/power-pages:deploy-pipeline when you're ready to ship v{POST_SYNC_VERSION} to {SELECTED_STAGE.name}.") so the user can inspect the synced manifest / Dataverse state and resume manually. Do not write docs/alm/last-deploy.json — no deployment happened. Skip the skill-tracking call too.Option 2 — record the deliberate gap in docs/alm/last-deploy.json under a knownGaps field so the audit trail is preserved.
Option 3 — stop.
Why the post-sync gate exists: this skill is the gate that promotes a solution to staging or production — the moment of staging promotion is the last place to catch surprises. When sync mode runs mid-deploy, it produces a different solution version than the one the user had in mind when they invoked the skill. Re-confirming after sync gives the user an explicit chance to inspect the version bump and the list of newly-adopted components before they reach the target environment. The Phase 3.5 trigger is intentional; the post-sync re-confirmation is the safety on top of it. Same principle applies when this skill is invoked from
plan-almorchestration —plan-alm's plan approval (Phase 4) covers the pre-sync state; this gate covers the delta introduced by mid-deploy sync.
Why Phase 3.5 exists in the first place: the ALM-aware-by-default rule in
AGENTS.mdrequires the completeness check at every gate where a solution leaves its source environment.
MULTI_RUN_MODE only)Skip this entire phase when
MULTI_RUN_MODE = false(single-solution mode, or legacyMULTI_PIPELINE_MODEv2). Single-solution and v2 each create + validate exactly one stage run inside Phase 4 below — there's nothing to parallelize. Resume at Phase 4.
This phase compresses the per-solution create-stage-run → ValidatePackageAsync → poll-validation-status chain by running all N solutions concurrently. ValidatePackageAsync does not acquire the env-level import lock that pins DeployPackageAsync to serial execution — it's a structural check on the solution package and per-stage-run state, so the platform happily processes N validations in parallel. For a typical 5-solution split with 60–180s validations, this drops the validation phase from N × ~120s (≈10 min) to roughly the slowest single validation (≈3 min).
The deploy phase (6.1 / 6.2) remains strictly serial — see the Design rationale callout in Phase 2.
3.6.1 Build the input file.
Filter DEPLOYMENT_ORDER down to entries that need validation:
status !== "SkippedEmpty" AND isFutureBuffer !== true.SkippedEmpty and isFutureBuffer: true entries — the Future buffer is a reserved 0-component placeholder and there's nothing to validate.Resolve each remaining entry's solutionId from .solution-manifest.json (schemaVersion: 2 solutions[]) if not already on the deployment-order entry. Write to a tmp file:
node -e "require('fs').writeFileSync('./docs/alm/.validation-batch.json', JSON.stringify({{VALIDATION_SPECS}}))"
Where {{VALIDATION_SPECS}} is the array [{ solutionUniqueName, solutionId }, …].
3.6.2 Run the batch validator.
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/validate-stage-runs-batch.js" \
--hostEnvUrl "{hostEnvUrl}" \
--token "{HOST_TOKEN}" \
--pipelineId "{pipelineId}" \
--stageId "{SELECTED_STAGE.stageId}" \
--sourceDeploymentEnvironmentId "{sourceDeploymentEnvironmentId}" \
--solutionsFile ./docs/alm/.validation-batch.json
Capture stdout as JSON: const batch = JSON.parse(output). Delete the tmp file (./docs/alm/.validation-batch.json) — it's transient. Build VALIDATED_STAGE_RUNS = { [solutionUniqueName]: { stageRunId, validationResults } } from batch.results.
If
VALIDATE_PACKAGE_UNAVAILABLEpropagates (any per-solutionerrormatchesValidatePackageAsync not available on this Pipelines package): setVALIDATE_PACKAGE_UNAVAILABLE = trueglobally, skip the rest of Phase 3.6 (the stage runs created so far stay around — they're harmless validated-but-not-deployed records), and proceed to Phase 4 in single-solution-fallback shape, which routes each iteration through thepac pipeline deployCLI fallback in Phase 6. This is the same code path the older Pipelines package versions take.
3.6.3 Branch on the batch outcome.
| batch.allPassed | batch.pendingApproval | batch.failed + batch.timedOut | Behavior |
|---|---|---|---|
| true | 0 | 0 | All validations passed. Report a single line: "Validated {N} solution(s) in parallel — all passed." Proceed to Phase 4. |
| false | > 0 | 0 | One or more validations are awaiting approval. See 3.6.4 (Pending Approval batch handling) below. |
| false | — | > 0 | One or more validations failed or timed out. See 3.6.5 (Halt on batch validation failure) below. |
3.6.4 Pending Approval batch handling.
<!-- gate: deploy-pipeline:3.6.batch-pending-approval | category=pause | cancel-leaves=external-state-pending -->🚦 Gate (pause · deploy-pipeline:3.6.batch-pending-approval): External wait — one or more solutions hit
stagerunstatus=200000005(Pending Approval) during parallel validation. User approves all in PPAC, then we re-poll. Cancel leaves N validated-but-pending stage runs on the host (the user can either approve them later and re-invoke, or cancel them in PPAC). Fires once per batch — not once per pending solution.
Surface the affected solutions in a single message (not per-solution) and pause. Use AskUserQuestion:
"Validation for
{batch.pendingApproval}of{batch.total}solution(s) is awaiting approval before it can complete: {bulleted list ofresult.solutionUniqueNamewherestatus === 'PendingApproval'}Approve all of them in Power Platform:
make.powerapps.com→ Solutions → Pipelines → for each stage run listed above → Approve. Then return here.The other
{batch.succeeded}solution(s) already validated successfully — they'll deploy after you approve."| Question | Header | Options | |---|---|---| | Approvals complete? | Batch validation approval | Yes — I approved all of them; re-poll, No — cancel the deploy |
Yes: re-run validate-stage-runs-batch.js in --rePoll mode. The helper accepts existing stage run IDs and skips the create-stage-run + ValidatePackageAsync calls — it just runs the poll-and-probe pattern against each stageRunId. After user approval, the stage run transitions 200000005 (PendingApproval) → 200000006 (Validating) → 200000007 (ValidationSucceeded); the helper's probe correctly distinguishes "still pending" (the user clicked Yes prematurely) from "real timeout" so the agent doesn't have to interpret a generic timeout error.
Build a tmp file with only the previously-pending entries (carry stageRunId from the original batch result):
node -e "require('fs').writeFileSync('./docs/alm/.repoll-batch.json', JSON.stringify({{PENDING_SPECS_WITH_STAGERUNIDS}}))"
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/validate-stage-runs-batch.js" \
--hostEnvUrl "{hostEnvUrl}" \
--token "{HOST_TOKEN}" \
--rePoll \
--solutionsFile ./docs/alm/.repoll-batch.json
Where {{PENDING_SPECS_WITH_STAGERUNIDS}} is the array [{ solutionUniqueName, solutionId, stageRunId }, …] filtered to entries with status === 'PendingApproval' from the original batch.
Capture stdout as JSON; delete the tmp file (./docs/alm/.repoll-batch.json). Merge the updated outcomes into VALIDATED_STAGE_RUNS keyed by solutionUniqueName. Then branch on the rePoll batch's tally:
Failed / Timeout / Error → fall through to 3.6.5 (treat as batch validation failure).No: stop cleanly. The validated stage runs and the pending stage runs remain on the host; re-invoking the skill picks up where this left off (the user can approve and re-run).
3.6.5 Halt on batch validation failure.
<!-- gate: deploy-pipeline:3.6.batch-validation-failed | category=plan | cancel-leaves=validated-stage-run -->🚦 Gate (plan · deploy-pipeline:3.6.batch-validation-failed): One or more solutions failed validation in the parallel batch. Surface per-solution
validationResultsfor the failing entries, and let the user decide whether to abort the entire deploy or proceed with only the succeeded solutions (advanced — leaves a known dependency gap on the target). Cancel leaves N validated stage runs on the host; the user can clean them up in PPAC or re-invoke after fixing the source.
Surface the failing entries with their validationResults (which is the double-encoded JSON string from the Dataverse validationresults field — JSON.parse it twice when displaying to extract SolutionValidationResults[].Message). Use AskUserQuestion:
"
{batch.failed + batch.timedOut}of{batch.total}solution(s) failed parallel validation:{for each failing result: a short block with
solutionUniqueName,status, topMessagefromvalidationResults}The other
{batch.succeeded}solution(s) passed and have validated stage runs ready to deploy. Deploying only the succeeded subset risks leaving a dependency gap on the target — e.g. shipping_Contentwithout its prerequisite_Foundationproduces broken site behavior. Recommended: abort, fix the source, re-invoke.What would you like to do?"
| Question | Header | Options | |---|---|---| | Next step? | Batch validation failed | Abort the entire deploy (Recommended), Deploy only the succeeded subset (advanced — accept the gap), Cancel and investigate |
docs/alm/last-deploy.json with status: "ValidationFailed" per failing solution and status: "NotAttempted" for the rest, so the next invocation can see what happened. Do not call DeployPackageAsync.DEPLOY_SUBSET_ACK = true and filter DEPLOYMENT_ORDER down to just the entries with status === 'Succeeded' in VALIDATED_STAGE_RUNS. Record the skipped entries in docs/alm/last-deploy.json knownGaps. Proceed to Phase 4. Never offer this option as the default — it's a foot-gun and the recommended path is always to abort.3.6.6 Persist the batch summary.
Append a batchValidation object to the in-memory state that will be written to docs/alm/last-deploy.json in Phase 7. Source most fields directly from the batch JSON returned by validate-stage-runs-batch.js:
{
"totalSolutions": <batch.total>,
"succeeded": <batch.succeeded>,
"failed": <batch.failed>,
"pendingApproval": <batch.pendingApproval>,
"timedOut": <batch.timedOut>,
"elapsedSeconds": <batch.elapsedSeconds>,
"perSolutionStageRunIds": { "<solutionUniqueName>": "<stageRunId>", ... }
}
Build perSolutionStageRunIds by reducing batch.results into { [r.solutionUniqueName]: r.stageRunId } for every entry where stageRunId is non-null (including failed-but-stage-run-was-created entries — preserves the deep-link to PPAC for debugging). elapsedSeconds comes from the helper directly (it wall-clock-measures the fan-out internally — no need to wrap the bash invocation in a timer).
This gives the next invocation (and any audit consumer) a record of the parallel-validation outcome distinct from the serial deploy outcomes. refresh-alm-plan-data.js's deploy-pipeline phase ingests this block into planData.pipelineMeta.lastDeploy.batchValidation so the rendered ALM plan can show the parallel-validation timing.
In
MULTI_RUN_MODE, Phase 3.6 already created the stage run + ran validation for the current iteration's solution in parallel with its siblings. Inside the per-iteration loop, skip Steps 4.1–4.3 entirely: retrieveSTAGE_RUN_IDfromVALIDATED_STAGE_RUNS[solutionUniqueName].stageRunIdandvalidationResultsfrom the same record. Jump directly to Step 4.4 (fetch deployment notes) and then Phase 5.In single-solution mode and legacy
MULTI_PIPELINE_MODE(v2), run Steps 4.1–4.4 inline as documented below.
Use Node.js https module for all Dataverse calls (curl has encoding issues on Windows).
4.1 Create stage run using create-stage-run.js:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-stage-run.js" \
--hostEnvUrl "{hostEnvUrl}" \
--token "{HOST_TOKEN}" \
--pipelineId "{pipelineId}" \
--stageId "{SELECTED_STAGE.stageId}" \
--sourceDeploymentEnvironmentId "{sourceDeploymentEnvironmentId}" \
--solutionId "{ARTIFACT_SOLUTION_ID}" \
--artifactName "{ARTIFACT_SOLUTION_NAME}"
Capture stdout as JSON: const result = JSON.parse(output). Extract result.stageRunId and store as STAGE_RUN_ID.
If the script exits non-zero, surface the error — likely a pipeline configuration issue (400) or a conflict (409). Both include the Dataverse error body in the message.
Note on field bindings: The script uses the v9.2 API and
msdyn_prefixed nav properties ([email protected],[email protected],[email protected]). These are the HAR-confirmed names for the current Pipelines package. Older package versions used different field names (e.g.,deploymentstageid); the script handles both 201 (JSON body) and 204 (OData-EntityId header) response codes.
Store as STAGE_RUN_ID.
4.2 Trigger package validation (returns 204 — not 200):
POST {hostEnvUrl}/api/data/v9.0/ValidatePackageAsync
Content-Type: application/json
Authorization: Bearer {HOST_TOKEN}
{"StageRunId": "{STAGE_RUN_ID}"}
Treat HTTP 204 as success.
If
ValidatePackageAsyncreturns 404: this Pipelines package version doesn't support the direct OData validation API. SetVALIDATE_PACKAGE_UNAVAILABLE = true. Skip Phase 4.2–4.3 and proceed directly to Phase 5 (deployment settings), then use thepac pipeline deployCLI fallback in Phase 6.
4.3 Poll validation using poll-validation-status.js:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/poll-validation-status.js" \
--hostEnvUrl "{hostEnvUrl}" \
--token "{HOST_TOKEN}" \
--stageRunId "{STAGE_RUN_ID}" \
--intervalMs 5000 \
--maxAttempts 36
Capture stdout as JSON: const result = JSON.parse(output). On non-zero exit, the error message will include validation failure details or a timeout message.
The script polls msdyn_operation on the stage run record. While it equals 200000201 the validation is still in progress; once it changes the script checks msdyn_stagerunstatus — if 200000003 (Failed) it throws with msdyn_validationresults; otherwise it returns { stageRunId, validationResults, stageRunStatus }.
Terminal validation values:
stageRunStatus 200000007 (Validation Succeeded) → proceed to Phase 5200000003 (Failed) — stop, display the validationresults from the error messageImportant:
validationresultsis a double-encoded JSON string — callJSON.parse()on it twice to get the object. The object has shape:{ ValidationStatus, SolutionValidationResults: [{ SolutionValidationResultType, Message, ErrorCode }], SolutionDetails, MissingDependencies }.
Surface any SolutionValidationResults entries to the user as warnings. Pay special attention to:
ErrorCode: -2147188672 — managed/unmanaged conflict: "The solution is already installed as unmanaged but this package is managed." The user must uninstall the existing solution from the target environment first, then retry.🚦 Gate (pause · deploy-pipeline:4.pending-approval): External wait — PP Pipelines
stagerunstatus=200000005(Pending Approval). User must approve in PPAC. Cancel leaves the run pending on the host. In MULTI_RUN_MODE / MULTI_PIPELINE_MODE this gate fires per loop iteration that hits Pending Approval — not once per skill invocation.
If stageRunStatus = 200000005 (Pending Approval): inform the user they need to approve in Power Platform (make.powerapps.com → Solutions → Pipelines → find this run → Approve). Ask via AskUserQuestion: "Have you approved the validation? 1. Yes, continue / 2. No, cancel"
4.4 Fetch AI-generated deployment notes (if AI_NOTES_ENABLED = true):
GET {hostEnvUrl}/api/data/v9.0/deploymentstageruns({STAGE_RUN_ID})?$select=aigenerateddeploymentnotes,deploymentstagerunid
Authorization: Bearer {HOST_TOKEN}
Store aigenerateddeploymentnotes as AI_DEPLOY_NOTES.
5.0a Check for deployment-settings.json:
Check if deployment-settings.json exists in the project root:
If it exists: Read it and show a summary of configured stages and env var counts. Say: "Found existing deployment-settings.json with {N} stages configured." (Count top-level keys as stage names; count EnvironmentVariables entries per stage.)
If it does NOT exist AND there are env var definitions in the solution manifest (from solutionManifest.envVars[] if available, or from the query in 5.1): Generate a template file and inform the user. Template structure:
{
"{stageName}": {
"EnvironmentVariables": [
{ "SchemaName": "{envVarSchemaName}", "Value": "" }
],
"ConnectionReferences": []
}
}
Use the stage name from SELECTED_STAGE.name and env var schema names from .solution-manifest.json (if available) or from the 5.1 query. Write to deployment-settings.json at the project root. Say: "Generated deployment-settings.json template. Fill in values before deploying, or provide them now when prompted."
Note: If env vars are not yet known at this point (5.0a runs before 5.1), generate the template file after 5.1 completes and the env vars are discovered — then inform the user before continuing to the prompt in 5.1.
If it does NOT exist AND there are no env vars in the solution: Note "No env var overrides needed" and skip.
5.0b Surface the file path:
Always display the resolved path {projectRoot}/deployment-settings.json so the user knows where to find it, whether it was just created or already existed.
5.1 Discover env var definitions in the solution and resolve per-stage values:
Query the solution components in the source environment to find all env var definitions (componenttype 380):
GET {sourceEnvUrl}/api/data/v9.2/solutioncomponents?$filter=_solutionid_value eq '{solutionId}' and componenttype eq 380&$select=objectid
Authorization: Bearer {SOURCE_TOKEN}
For each objectid, fetch the schema name:
GET {sourceEnvUrl}/api/data/v9.2/environmentvariabledefinitions({objectid})?$select=schemaname,displayname,type,defaultvalue
This gives you SOLUTION_ENV_VARS — the list of env vars that will travel to the target.
Read deployment-settings.json (if it exists in the project root) and look up the selected stage name to get pre-configured values:
const stageSettings = deploymentSettings?.stages?.[selectedStageName] || {};
const preconfigured = stageSettings.EnvironmentVariables || []; // [{ SchemaName, Value }]
Identify unconfigured env vars — those in SOLUTION_ENV_VARS that have no entry in preconfigured:
const unconfigured = SOLUTION_ENV_VARS.filter(v =>
!preconfigured.find(p => p.SchemaName === v.schemaname)
);
<!-- gate: deploy-pipeline:5.env-vars | category=plan | cancel-leaves=nothing -->
🚦 Gate (plan · deploy-pipeline:5.env-vars): Unconfigured env vars per stage — user supplies values or skips (uses default). Without values, runtime reads default which may be dev-only. In MULTI_RUN_MODE / MULTI_PIPELINE_MODE this gate fires per loop iteration that has unconfigured env vars — values supplied for iteration 1 do NOT carry to iteration 2.
If there are unconfigured env vars, present them to the user via AskUserQuestion:
"This solution has {N} environment variable(s) with no value configured for {stageName}. Enter the value for each (leave blank to use the default, or skip if not applicable):
{schemaname}({displayname}) — default:{defaultvalue ?? 'none'}- ..."
Collect responses and merge with preconfigured to form the final ENV_VAR_OVERRIDES array. Offer to save the values back to deployment-settings.json for future runs:
"Save these values to
deployment-settings.jsonfor future deployments to {stageName}?
- Yes — save for next time
- No — use once only"
If Yes: write/update deployment-settings.json with the collected values under stages.{stageName}.EnvironmentVariables.
If all env vars are pre-configured (or there are none): skip the prompt, use preconfigured directly.
5.1b Pre-PATCH validation of deployment-settings.json Secret references.
Why this gate exists. The Power Platform Pipelines handler validates the
deploymentsettingsjsonPATCH at import time, AFTER the stage run has been queued and potentially after a long wait behind serialized imports. A bad Secret reference value — placeholder syntax like@KeyVault(vaultName=...;secretName=...), raw secret values committed to the file, malformed URIs — fails the import with "ImportAsHolding failed: The value provided as a secret reference does not match a valid secret reference format." Live evidence: a 4h41m queue wait + fail in a recent session. Catching the bad reference here turns hours of wasted compute into a sub-second hard stop with a precise remediation pointer.
Skip this step entirely when ENV_VAR_OVERRIDES is empty AND deployment-settings.json is absent — there's nothing to validate.
Otherwise, run the shared helper:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/validate-deployment-settings.js" \
--settingsFile "./deployment-settings.json" \
--envUrl "{sourceEnvUrl}" \
--stageLabel "{SELECTED_STAGE.name}"
Capture stdout as JSON. The helper classifies each EnvironmentVariables[] entry by valueFormat (kv-uri / kv-resource-id / kv-placeholder / empty / plain-text / invalid-uri) and status (valid / invalid / unknown-type / skipped). When --envUrl is provided (as above), Secret-type entries are validated against the canonical Key Vault reference formats; other types are structurally validated.
Branch on summary:
summary.invalid === 0 → all entries valid, proceed to Phase 5.2.
summary.invalid > 0 → STOP. Display the invalid findings[] to the user with each entry's schemaName, valueFormat, value, and message. Then surface a remediation message:
"Pre-deploy validation found
{summary.invalid}invalid env var value(s) indeployment-settings.json. The Power Platform Pipelines handler would reject these during import (potentially after a long queue wait). Fix the file before re-running this skill.Canonical formats for Secret env vars:
- Key Vault Secret Identifier URI:
https://<vault>.vault.azure.net/secrets/<name>[/<version>]- Azure resource ID:
/subscriptions/<sub>/resourceGroups/<rg>/providers/Microsoft.KeyVault/vaults/<vault>/secrets/<name>- Empty (use the env var definition's default)
See
configure-env-variablesPhase 3.B andadd-server-logicPhase 7.2a for the end-to-end flow (vault selection → secret storage → URI handoff)."
Do NOT prompt for "proceed anyway" — there's no partial-validity case where forcing the bad PATCH leads to a successful import. The deploy will fail; only the wait time changes.
summary.invalid === 0 && summary.['unknown-type'] > 0 → log a one-line warning about the schemas whose types couldn't be looked up (probably auth or transient errors) and proceed. The downstream handler will still reject obvious garbage; the pre-PATCH gate is a fast-path, not the only line of defense.
5.2 PATCH stage run with artifact version, deployment notes, and environment variables (always run):
First, determine the current solution version in the source (dev) environment — this must match exactly:
GET {sourceEnvUrl}/api/data/v9.0/solutions?$filter=uniquename eq '{SOLUTION_NAME}'&$select=version
Authorization: Bearer {SOURCE_TOKEN}
Use the returned version as artifactdevcurrentversion. Do NOT use the version from .solution-manifest.json — that may be stale.
For artifactversion, increment the patch number of the source version (e.g., 1.0.0.2 → 1.0.0.3). This must be strictly greater than the version already deployed in the target stage. If deploying to the same stage multiple times, check docs/alm/last-deploy.json for the last artifactVersion and use a higher value.
Then PATCH (include deploymentsettingsjson only if ENV_VAR_OVERRIDES is non-empty):
PATCH {hostEnvUrl}/api/data/v9.0/deploymentstageruns({STAGE_RUN_ID})
Content-Type: application/json
Authorization: Bearer {HOST_TOKEN}
{
"artifactdevcurrentversion": "{current version from source env — must match exactly}",
"artifactversion": "{new version — must be > current version in target stage}",
"deploymentnotes": "{AI_DEPLOY_NOTES if available, otherwise a brief description of what is being deployed}",
"deploymentsettingsjson": "{JSON.stringify({ EnvironmentVariables: ENV_VAR_OVERRIDES, ConnectionReferences: [] })}"
}
The deploymentsettingsjson value must be a JSON-encoded string (double-serialized):
const deploymentsettingsjson = JSON.stringify({
EnvironmentVariables: ENV_VAR_OVERRIDES,
ConnectionReferences: stageSettings.ConnectionReferences || [],
});
If ENV_VAR_OVERRIDES is empty and there are no connection references, omit deploymentsettingsjson entirely.
Response is HTTP 204. If the PATCH fails with a version conflict error, check both version values and retry.
Caveat —
deploymentsettingsjsondoes NOT always writeenvironmentvariablevaluesrecords on the target post-deploy. Observed live: a Power Pages site with env var definitions that are not yet linked to anmspp_sitesettingrecord can complete a successful deploy with a populateddeploymentsettingsjson, and the target environment'senvironmentvariablevaluestable still ends up empty for those entries. The Power Platform Pipelines handler appears to write values only for definitions that have an existing site-setting binding (or another consumer the platform recognizes). This may be a platform bug or by-design pending a separate Power Pages Management UI step. If you ship Secret env vars or any other type that the user expects to land in the target beforesetup-auth/configure-env-variablesruns there, verify in Phase 7 below thatenvironmentvariablevaluesrecords exist on the target after the deploy completes, and surface a clear prompt if they don't. Workaround: invokeconfigure-env-variables(or runlink-site-setting-to-env-var.jsper definition) on the target after the deploy to create the values explicitly. Track this in the next PR if it becomes a recurring blocker.
🚦 Gate (final · deploy-pipeline:6.0.final-consent): Last-call before each
DeployPackageAsync/pac pipeline deploycall. Validation already passed. Non-transactional import — partial failure leaves whatever already imported on the target.⚠ Fires PER LOOP ITERATION in MULTI_RUN_MODE / MULTI_PIPELINE_MODE. Three solutions in
deploymentOrder→ three Phase 6.0 prompts (one before each iteration'sDeployPackageAsync). The upstream Phase 2 stage selection — whether via interactiveAskUserQuestionor via the--stageargument — covers only stage choice, NOT individual deploy authorization. Never batch the deploy loop. Skipping the per-iteration prompt and proceeding straight from Phase 5 → Phase 6.1 for solutions 2..N is the documented strategy violation this gate exists to prevent.
6.0 Final deploy consent gate. Before kicking off each deployment — whether via DeployPackageAsync (6.1) or the pac pipeline deploy PAC CLI fallback, and whether this is the first solution or the Nth in MULTI_RUN_MODE — confirm with the user explicitly. The earlier gates (Phase 2 stage pick, Phase 2.5 unblock, Phase 3.5 completeness, Phase 5 env var values) each cover their own decision, but none of them is a per-iteration "ready to ship this one?" prompt and the deploy itself is not transactional — partial failures leave whatever has already imported on the target. This gate makes every production-promotion moment explicit. Use AskUserQuestion:
"Ready to deploy
{ARTIFACT_SOLUTION_NAME}(v{newVersion}) to{SELECTED_STAGE.name}({targetEnvUrl})?This will run for ~3–60 minutes depending on solution size. The import is not transactional — if it fails partway, whatever already imported stays on the target and a manual cleanup (or re-deploy with a higher artifact version) is required to recover.
docs/alm/last-deploy.jsonis written regardless of outcome.
- Deploy now — POST
DeployPackageAsync(or runpac pipeline deployfor the CLI fallback) and poll until terminal- Cancel — leave the stage run in its current state (validated but not deployed); re-run
/power-pages:deploy-pipelinelater when ready"
Branch on the answer:
DeployPackageAsync or pac pipeline deploy. Tell the user: "Cancelled. The stage run {STAGE_RUN_ID} is validated but not deployed. Re-invoke /power-pages:deploy-pipeline to resume. If the source solution version changes before you re-invoke, the stage run's artifactversion field will need a fresh PATCH (Phase 5.2) before deploying — re-running the skill handles that automatically." Do not write a docs/alm/last-deploy.json on cancel — there's no deploy outcome to record.For Production targets specifically, treat this gate as the single most important prompt in the skill — the cancellation cost (stage run already validated) is small compared to a wrong-stage import.
If
ValidatePackageAsyncwas unavailable (VALIDATE_PACKAGE_UNAVAILABLE = true): use the PAC CLI as the primary deployment mechanism instead of 6.1. The same Phase 6.0 consent gate above gates this path too — do not runpac pipeline deployuntil the user has answered "Deploy now":Ask user for
currentVersion(pre-fill from.solution-manifest.jsonsolution.versionif available) andnewVersion(suggest incrementing the patch number, e.g.1.0.0.0→1.0.0.1).pac pipeline deploy \ --environment "{devEnvUrl}" \ --solutionName "{ARTIFACT_SOLUTION_NAME}" \ --stageId "{SELECTED_STAGE.stageId}" \ --currentVersion "{currentSolutionVersion}" \ --newVersion "{newVersion}" \ --waitIf the CLI returns "Resource not found for the segment 'deploymentenvironments'": the dev environment is not connected to a Pipelines host. Advise the user to configure the host in Power Platform Admin Center (Environments → select env → Pipelines), then retry.
If CLI succeeds: parse the output for stage run status, write
docs/alm/last-deploy.jsonwithstatus: "Succeeded"(or the parsed status), and skip theDeployPackageAsynccall and polling in 6.1–6.2.
6.1 Trigger deployment (skip if VALIDATE_PACKAGE_UNAVAILABLE = true — use PAC CLI path above). Only run after the Phase 6.0 consent gate returned "Deploy now":
POST {hostEnvUrl}/api/data/v9.0/DeployPackageAsync
Content-Type: application/json
Authorization: Bearer {HOST_TOKEN}
{"StageRunId": "{STAGE_RUN_ID}"}
Note:
DeployPackageAsyncalso returns 404 on older Pipelines package versions. If this occurs, use thepac pipeline deployCLI path above.
6.2 Poll stagerunstatus until terminal using poll-deployment-status.js:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/poll-deployment-status.js" \
--hostEnvUrl "{hostEnvUrl}" \
--token "{HOST_TOKEN}" \
--stageRunId "{STAGE_RUN_ID}" \
--intervalMs 8000 \
--maxAttempts 75
Capture stdout as JSON: const result = JSON.parse(output).
The script polls msdyn_stagerunstatus until a terminal state:
result.status === 'Succeeded' → proceed to Phase 7result.status === 'Awaiting' → approval gate (see below); do NOT treat as error200000003 (Failed) or 200000004 (Canceled) — error message includes msdyn_errordetailssuboperation field (not polled by the script but visible in Power Platform) shows progress detail:
200000100 = None (starting/finishing)200000105 = Deploying Artifact (actively installing solution)🚦 Gate (pause · deploy-pipeline:6.pending-approval): External wait — PP Pipelines
stagerunstatus=200000005mid-deploy. User approves in PPAC. Cancel here PATCHesiscanceled: trueon the run. In MULTI_RUN_MODE / MULTI_PIPELINE_MODE this gate fires per loop iteration that hits Pending Approval.
Approval gate handling: If result.status === 'Awaiting' (msdyn_stagerunstatus = 200000005):
make.powerapps.com → Solutions → Pipelines → find deployment for {STAGE_RUN_ID} → Approve."AskUserQuestion: "Have you approved the deployment? 1. Yes, I approved it — continue polling / 2. Cancel deployment"poll-deployment-status.js to continue polling.PATCH {hostEnvUrl}/api/data/v9.0/deploymentstageruns({STAGE_RUN_ID})
{"iscanceled": true}
Then record status as "Canceled".Token refresh: After every 10 poll cycles (~80 seconds), refresh HOST_TOKEN via az account get-access-token and pass the updated token in a fresh poll-deployment-status.js invocation with reduced --maxAttempts.
Report deployment progress updates as polling continues.
7.1 Determine final status string:
200000002 → "Succeeded"200000003 → "Failed"200000004 → "Canceled"200000005 → "PendingApproval" (if user cancelled waiting)"Unknown"7.2 Post-deployment warnings (only if deployment Succeeded):
Using solutionManifest captured in Phase 1 from detect-project-context.js, check for components that require manual follow-up in the target environment.
Connection reference warning — if solutionManifest.cloudFlows is present and non-empty:
⚠️ Connection references may need binding This solution includes cloud flow(s). If those flows use connection references (e.g. Dataverse, SharePoint), they must be bound to live connections in the target environment or the flows will remain disabled.
To bind: Power Automate → target environment → each flow → Edit → bind connections.
If solutionManifest.cloudFlows is absent or empty, skip this warning entirely.
Bot republish warning — if solutionManifest.botComponents is present and non-empty:
⚠️ Bot republish required This solution includes a Copilot Studio bot. After deployment, the bot must be republished in the target environment to complete provisioning.
To republish: Power Pages Management → target environment → Edit site → Copilot → republish.
If solutionManifest.botComponents is absent or empty, skip this warning entirely.
These warnings are informational only — do not block the summary or use AskUserQuestion.
7.3 Write docs/alm/last-deploy.json (create the docs/alm/ directory first if missing — node -e "require('fs').mkdirSync('docs/alm',{recursive:true})"):
{
"pipelineId": "{pipelineId}",
"pipelineName": "{pipelineName}",
"stageId": "{SELECTED_STAGE.stageId}",
"stageRunId": "{STAGE_RUN_ID}",
"stageName": "{SELECTED_STAGE.name}",
"solutionName": "{ARTIFACT_SOLUTION_NAME}",
"solutionId": "{ARTIFACT_SOLUTION_ID}",
"status": "{final status string}",
"deployedAt": "{ISO timestamp}",
"hostEnvUrl": "{hostEnvUrl}",
"targetEnvironmentUrl": "{SELECTED_STAGE.targetEnvironmentUrl}",
"artifactVersion": "{artifactVersion from Phase 5.2 PATCH}",
"deployHistoryFile": "docs/deploy-history/{YYYY-MM-DD}-{stageName}-{artifactVersion}.html",
"activationStatus": null,
"siteUrl": null
}
activationStatus and siteUrl start as null and are patched at the end of Phase 7.7 once the activation outcome is known.
Where {YYYY-MM-DD} is the date portion of deployedAt and {stageName} is the stage name with spaces replaced by hyphens (lowercased), e.g. 2026-04-06-staging-1.0.0.3.md.
Retry attempts go as PEER entries in
runs[], not nested underruns[i].lastAttempt. InMULTI_RUN_MODE(multi-solution) the marker carries aruns[]array — one entry per solution per attempt. When the user retries a failed solution (e.g. after fixing an env var value, re-running deploy-pipeline against the same stage), the retry is a NEW peer entry inruns[]with its ownartifactVersionandattemptedAt. Do NOT nest the retry asruns[i].lastAttempt: {...}under the original failed entry — that mixes the schema and makes audit traversal ambiguous (was 1.0.0.4 the deployed version or just the latest attempt?). The flat peer-array shape lets every consumer (refresh-alm-plan-data, the rendered Pipelines tab, the deploy history HTML) walkruns[]once and see the full timeline.Marker writers in this skill should append, not mutate. Each retry of a failed solution writes a new
runs[]entry with status='Succeeded' (or 'Failed' on persistent failure) and the post-bumpartifactVersion. The original failed entry stays untouched as history.
7.4 Write deployment history entry (HTML):
Compute the history filename: {YYYY-MM-DD}-{stageName}-{artifactVersion}.html (same derivation as docs/alm/last-deploy.json's deployHistoryFile field — replace spaces with hyphens, lowercase stage name).
Create docs/deploy-history/ if it does not already exist:
mkdir -p docs/deploy-history
Read the template at ${CLAUDE_PLUGIN_ROOT}/skills/deploy-pipeline/assets/deploy-history-template.html and replace the following __PLACEHOLDER__ tokens:
Overview tab:
| Placeholder | Value |
|---|---|
| __SOLUTION_FRIENDLY_NAME__ | Solution friendly name (from .solution-manifest.json) or {solutionUniqueName} |
| __SOLUTION_NAME__ | {ARTIFACT_SOLUTION_NAME} |
| __STAGE_NAME__ | {SELECTED_STAGE.name} |
| __TARGET_ENV_URL__ | {SELECTED_STAGE.targetEnvironmentUrl} |
| __STAGE_RUN_ID__ | {STAGE_RUN_ID} |
| __PIPELINE_NAME__ | {pipelineName} |
| __DEPLOYED_AT__ | {deployedAt ISO string} |
| __ARTIFACT_VERSION__ | {artifactVersion from Phase 5.2} |
| __PREV_ARTIFACT_VERSION__ | {artifactDevCurrentVersion} — the version that was in dev before this deploy |
| __STATUS_CLASS__ | succeeded / failed / pending-approval |
| __STATUS_ICON__ | ✓ for Succeeded, ✗ for Failed, ⏳ for PendingApproval |
| __STATUS_LABEL__ | Succeeded / Failed / Pending Approval |
| __ACTIVATION_SECTION__ | Initially '' — replaced in Phase 7.7 once activation outcome is known |
Solution tab — read .solution-manifest.json to build these sections:
| Placeholder | Value |
|---|---|
| __SOLUTION_META_ROWS__ | <tr> rows for: Friendly Name, Unique Name, Version (new → previous), Type (Managed/Unmanaged), Publisher, Total Components. Source: manifest + validationResults.SolutionDetails from Phase 6. |
| __VALIDATION_SECTION__ | If validation passed: <div class="note-box succeeded"><span class="validation-badge passed">✓ Validation Passed</span> — No missing dependencies.</div>. If failed or deps present: <div class="note-box warning"> listing each missing dependency name. |
| __SOLUTION_CONTENTS_SECTION__ | Build from .solution-manifest.json: a <div class="contents-grid"> with two <div class="contents-card"> blocks — Dataverse Tables (as <span class="table-chip"> per table) and Bot Components (comma-separated names). Below the grid, add a <div class="note-box neutral"> with: {totalAdded} components added to solution (from components.totalAdded). If manifest is unavailable, show a neutral note. |
Config & Notes tab:
| Placeholder | Value |
|---|---|
| __ENV_VARS_SECTION__ | If ENV_VAR_OVERRIDES was non-empty: a <div class="card"><h3>Environment Variable Overrides</h3> table with schema name + override value columns. Otherwise: <div class="note-box neutral">No environment variable overrides applied.</div> |
| __DEPLOYMENT_NOTES_SECTION__ | If AI_DEPLOY_NOTES is available: <div class="card"><h3>AI Deployment Notes</h3><p>…</p></div>. Otherwise: '' |
| __POST_DEPLOY_WARNINGS__ | One <div class="note-box warning"> per post-deploy warning (connection refs, bot republish). Empty string if none. |
Write the rendered HTML to docs/deploy-history/{filename}.html.
Then add to the staging area:
git add docs/alm/last-deploy.json docs/deploy-history/{filename}.html
git commit -m "Deploy {solutionUniqueName} v{artifactVersion} to {stageName} ({status})"
If git is not initialized in the project root (i.e., git rev-parse --git-dir fails), skip the commit silently.
7.5 Record skill usage:
Reference:
${CLAUDE_PLUGIN_ROOT}/references/skill-tracking-reference.md
Follow the skill tracking instructions in the reference to record this skill's usage. Use --skillName "DeployPipeline".
7.5b Refresh the ALM plan (if one exists):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
--projectRoot "." \
--phase deploy-pipeline \
--render
The helper reads the docs/alm/last-deploy.json you just wrote, ingests it into planData.pipelineMeta.lastDeploy, drops any pre-deploy "host not yet provisioned" risks, and re-renders docs/alm-plan.html so the Pipelines tab shows the actual run state (status, version, component count, activation, site URL). When docs/.alm-plan-data.json is absent (the skill was invoked standalone, not via plan-alm), the helper returns ok:false as a soft no-op — safe to run unconditionally.
This step is what makes the rendered plan stay current after a direct /power-pages:deploy-pipeline invocation. plan-alm Phase 7 also runs the same refresh as belt-and-suspenders; running it twice is idempotent (same input → same output).
7.6 Present summary:
If Succeeded:
✓ Deployment succeeded
Solution: {solutionName}
Stage: {stageName}
Target: {targetEnvironmentUrl}
Completed at: {deployedAt}
Stage run ID: {STAGE_RUN_ID}
Site URL: {siteUrl from 7.7, or "— activation pending" if not yet activated, or "— checking…" before 7.7 runs}
If Failed:
✗ Deployment failed
Stage run ID: {STAGE_RUN_ID}
Status: Failed
To investigate: open Power Platform make.powerapps.com → Solutions → Pipelines
and find the failed run for details on what caused the failure.
7.6.1 Diagnose the failure. Before asking the user what to do, query the stage run record for error details so we can offer a targeted remediation prompt:
GET {hostEnvUrl}/api/data/v9.1/deploymentstageruns({STAGE_RUN_ID})?$select=errordetails,validationresults,stagerunstatus
Authorization: Bearer {HOST_TOKEN}
Parse errordetails and validationresults as JSON / text. Check for these patterns (case-insensitive):
| Pattern in error text | Diagnosis | Remediation prompt |
|---|---|---|
| AttachmentBlocked OR -2147188706 OR not a valid type OR \.js.*blocked | Blocked attachments — the target env's blockedattachments setting rejects file types in the solution | See 7.6.2 below |
| secret reference does not match a valid OR ImportAsHolding failed.*secret reference OR KeyVault.*format | Invalid Secret reference — deployment-settings.json carries a Secret value in a non-canonical format (e.g. @KeyVault(vaultName=...;secretName=...), raw secret text, malformed URI) | See 7.6.4 below |
| MissingDependency OR Missing dependency | Missing solution dependencies in target | Surface the dependencies; recommend the user install them and retry |
| (anything else) | Unknown failure | Fall through to the generic retry/exit prompt |
7.6.2 Blocked-attachment remediation (gated). Only run when the diagnostic in 7.6.1 matched the blocked-attachment pattern. Never modify the env's blockedattachments setting without an explicit AskUserQuestion gate. This is a tenant-wide security setting; auto-modifying it would silently weaken security posture across other apps in the same env.
Switch PAC CLI to the target environment so the helper queries the right env's settings, then identify which file types are blocked. By default the helper checks js; pass --extensions if the error mentions other types (e.g. js,css):
pac env select --environment "{TARGET_ENV_URL}"
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/fix-blocked-attachments.js" \
--envUrl "{TARGET_ENV_URL}" \
--extensions js \
--dry-run
Read the output JSON. The fields you care about: wasBlocked[] (the extensions currently blocked on the env that were on your --extensions list — these are what would be removed) and unchanged[] (extensions you asked to remove that aren't actually blocked, no-op).
If wasBlocked is empty, the failure is not caused by a blocked attachment on this list — fall back to 7.6.3's generic retry prompt and surface the raw error text from errordetails.
Tell the user explicitly:
"The deployment failed because the target environment
{targetEnvName}blocks file types that this solution needs:{wasBlocked.join(', ')}. This is an environment-level security setting that affects all users of that env. To proceed, the block needs to be removed for these specific types. The change is reversible from the Power Platform Admin Center → Environments →{targetEnvName}→ Settings → Product → Features → Blocked Attachments."
🚦 Gate (consent · deploy-pipeline:7.6.2.blocked-attachments): Reactive
AttachmentBlockedremediation — modify env-levelblockedattachmentssetting (tenant-wide impact). Reversible from PPAC. Fires PER FAILURE that matches the AttachmentBlocked pattern. In MULTI_RUN_MODE this is rare — Phase 2.5 pre-flight typically catches the block before the loop starts — but if iteration N still fails withAttachmentBlockedafter a successful Phase 2.5, the gate fires again for that failure. Each failure is its own consent decision (different solution, possibly different blocked extensions). Do NOT carry consent across failures.
Invoke AskUserQuestion (do NOT bury this in chat — the user must answer before any change happens):
| Question | Header | Options |
|---|---|---|
| Allow removing the block on {wasBlocked.join(', ')} for the {targetEnvName} environment so the deployment can retry? This modifies a tenant-level security setting. | Unblock attachments | Yes — unblock these types and retry, No — leave settings unchanged (I'll investigate manually), Cancel |
Branch on the answer:
fix-blocked-attachments.js without --dry-run (with the same --extensions), then call RetryFailedDeploymentAsync and resume polling from Phase 6.2. Surface the change in the deploy summary so the user has a clear audit record. Switch PAC CLI back to the source env after the retry resolves so subsequent skill steps run against the source by default."To unblock manually: open Power Platform Admin Center → Environments →
{targetEnvName}→ Settings → Product → Features → Blocked Attachments → remove{blockedTypesPresent.join(', ')}→ Save. Then re-run/power-pages:deploy-pipeline."
7.6.3 Generic retry/exit prompt (when 7.6.1 didn't match a known pattern, or the user declined the targeted remediation):
<!-- gate: deploy-pipeline:7.6.3.retry-exit | category=plan | cancel-leaves=validated-stage-run -->🚦 Gate (plan · deploy-pipeline:7.6.3.retry-exit): Failed deploy with no known pattern matched — call RetryFailedDeploymentAsync or exit for manual investigation.
Ask via AskUserQuestion:
"The deployment failed. What would you like to do?
- Retry — call
RetryFailedDeploymentAsyncto retry the same stage run- Exit — I'll investigate manually"
If Retry: call:
POST {hostEnvUrl}/api/data/v9.1/RetryFailedDeploymentAsync
Content-Type: application/json
Authorization: Bearer {HOST_TOKEN}
{"StageRunId": "{STAGE_RUN_ID}"}
Then resume polling from Phase 6.2.
If Exit: stop and present the failure summary above.
7.6.4 Invalid Secret reference remediation (gated). Only run when the diagnostic in 7.6.1 matched the Secret-reference pattern. This is the strip-and-retry path described in the original Citizens-portal validation report: when the import rejects a deployment-settings.json Secret value, replace it in-place with "" (which Dataverse interprets as "use the env var definition's default") and retry the deploy. The file mutation is persisted so the next run doesn't re-ship the broken value.
This is a backstop. The pre-write validator in configure-env-variables Phase 6.1 and the pre-PATCH validator in this skill's Phase 5.1b should catch most invalid Secret values upstream. Phase 7.6.4 covers the cases where:
deployment-settings.json after configure-env-variables ran.status: "unknown-type" for an entry whose Dataverse type couldn't be resolved at PATCH time, but the host validator rejected at import time.Steps:
Identify which schema(s) have the bad value. Run validate-deployment-settings.js against the current file (same helper as Phase 5.1b) to surface every invalid entry:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/validate-deployment-settings.js" \
--settingsFile "./deployment-settings.json" \
--envUrl "{sourceEnvUrl}" \
--stageLabel "{SELECTED_STAGE.name}"
Capture stdout as JSON; collect findings[] where status === "invalid" AND valueFormat is one of kv-placeholder / plain-text / invalid-uri. The collected schemaName list is the set of values to strip.
If validate-deployment-settings.js finds no invalid entries (the host error referenced a Secret format but the local file looks fine — possible mid-air edit, or the error matched the pattern but was about a different field), fall through to 7.6.3 (generic retry/exit prompt).
Otherwise, prompt the user for consent.
<!-- gate: deploy-pipeline:7.6.4.strip-secret-values | category=consent | cancel-leaves=invalid-secret-in-file -->🚦 Gate (consent · deploy-pipeline:7.6.4.strip-secret-values): Strip invalid Secret-reference values from
deployment-settings.jsonand retry the deploy. The bad values are replaced with empty strings (use definition default) and the file mutation is persisted — the next run won't re-ship the broken values. Cancel leaves the file as-is so the user can fix it manually with canonical Key Vault URIs.
Use AskUserQuestion:
| Question | Header | Options |
|---|---|---|
| The deploy failed because deployment-settings.json carries Secret values in an invalid format: {list of schema names}. Strip those values (set to "" — use definition default) and retry the deploy? The file will be updated; you can supply canonical Key Vault URIs later via /power-pages:configure-env-variables. | Strip invalid Secret values | Yes — strip and retry (Recommended), No — exit so I can fix deployment-settings.json manually with canonical URIs |
Branch on the answer:
Yes — strip and retry: invoke the file-mutation helper, then re-PATCH the stage run with the corrected deploymentsettingsjson, then call RetryFailedDeploymentAsync and resume polling from Phase 6.2.
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/strip-invalid-secret-values.js" \
--settingsFile "./deployment-settings.json" \
--schemaNames "{comma-separated schema names from step 1}" \
--stageLabel "{SELECTED_STAGE.name}"
Confirm stripped.length > 0 and capture the list for the audit summary (Phase 7.5b refresh ingests it into the rendered plan). Re-build ENV_VAR_OVERRIDES from the now-corrected deployment-settings.json (re-run the same logic as Phase 5.1's "Read deployment-settings.json" step) and PATCH the stage run again:
PATCH {hostEnvUrl}/api/data/v9.0/deploymentstageruns({STAGE_RUN_ID})
Content-Type: application/json
Authorization: Bearer {HOST_TOKEN}
{
"deploymentsettingsjson": "{JSON.stringify({ EnvironmentVariables: ENV_VAR_OVERRIDES, ConnectionReferences: ... })}"
}
Then retry the import:
POST {hostEnvUrl}/api/data/v9.1/RetryFailedDeploymentAsync
{"StageRunId": "{STAGE_RUN_ID}"}
Resume polling from Phase 6.2. Record the strip action in docs/alm/last-deploy.json secretRemediation[] so the deploy summary shows which schemas were stripped + what their previous values were.
No — exit: stop with a remediation pointer. Tell the user the canonical Secret-reference formats and point at /power-pages:configure-env-variables for guided remediation:
"To fix manually: open
deployment-settings.jsonand replace the invalid Secret values with one of:
- Key Vault Secret Identifier URI:
https://<vault>.vault.azure.net/secrets/<name>[/<version>]- Azure resource ID:
/subscriptions/<sub>/resourceGroups/<rg>/providers/Microsoft.KeyVault/vaults/<vault>/secrets/<name>- Empty string (use the env var definition's default)
Then re-run
/power-pages:deploy-pipeline. Or invoke/power-pages:configure-env-variablesfor a guided edit that re-runs Phase 6.1's pre-write validation."
7.6.5 Verify environmentvariablevalues landed on the target (only if deployment Succeeded AND ENV_VAR_OVERRIDES was non-empty AND deploymentsettingsjson was PATCHed in Phase 5.2):
Even when the stage run reports success, the Power Platform Pipelines handler does not always write environmentvariablevalues records for every env var definition in deploymentsettingsjson — definitions that aren't bound to an mspp_sitesetting (or another consumer the platform recognizes) can land as zero-value on the target. See the caveat note above Phase 6 for the live evidence. This step closes the gap by querying the target post-deploy and surfacing any missed landings.
Use the shared helper scripts/lib/verify-env-var-values.js. It reads schema names + per-stage expected values from deployment-settings.json and returns a structured JSON result per schema (landed / missing-value-record / missing-definition / value-mismatch / query-error):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/verify-env-var-values.js" \
--envUrl "{targetEnvUrl}" \
--settingsFile "./deployment-settings.json" \
--stageLabel "{SELECTED_STAGE.name}"
Capture stdout as JSON: const verifyResult = JSON.parse(output). The helper acquires its own token via Azure CLI scoped to {targetEnvUrl} — you do NOT need to switch PAC CLI for this call (the helper is fully read-only and bypasses PAC entirely).
Branch on verifyResult.summary:
summary.missing === 0 && summary.mismatched === 0 → all env var values landed cleanly. No warning. Proceed.
summary.missing > 0 OR summary.mismatched > 0 → log a structured warning into docs/alm/last-deploy.json under a new envVarLandingWarnings[] array. Build the array from the non-landed entries in verifyResult.results — preserve schemaName, status, and (when present) expected / value fields. Then surface to the user via the deploy summary:
"
{summary.missing + summary.mismatched}environment variable value(s) fromdeploymentsettingsjsondid not land cleanly on{targetEnvName}. Most likely cause formissing-value-record: the definition isn't yet linked to a site setting on the target. Most likely cause forvalue-mismatch: the per-stage value indeployment-settings.jsondiffers from what's now in the target'senvironmentvariablevaluestable (someone may have edited it post-deploy via PPAC). Run/power-pages:configure-env-variablesagainst{targetEnvName}to reconcile, or set values via Power Platform Admin Center → Solutions → Environment Variables."
summary.error > 0 (auth, transient, 5xx) → log a one-line note: "Env var landing check was inconclusive ({summary.error} schema queries errored). Verify manually in PPAC." Do NOT block the deploy summary on inconclusive results — the deploy itself succeeded.
Why the helper instead of inline OData? The earlier inline-prose version was brittle: each agent run reconstructed the queries by hand, PAC CLI env-switching was manual, and the result-shape was never structured. The helper has 21 tests covering all the result-status branches (
landed/missing-value-record/missing-definition/value-mismatch/query-error), readsdeployment-settings.jsondirectly to avoid drift between caller and helper, and produces a stable JSON contract callers can persist intolast-deploy.jsonwithout re-parsing prose.
7.7 Check site activation (only if deployment Succeeded and this is a Power Pages project):
Trigger rule (updated 2026-04-22): run this check whenever the source project's
powerpages.config.jsonhas awebsiteRecordId— i.e. this project is a Power Pages site project. The earlier rule ("only if the solution we just deployed contains a website componentType 10374") misses a real-world case: the site may pre-exist on the target and the solution we deployed may only contain tables/flows/bots. A Power Pages project's post-deploy is always incomplete if its site isn't activated on the target, regardless of which specific solution was shipped this time.
Read websiteRecordId from powerpages.config.json. If the field is absent, skip the rest of 7.7 (this is a non-Power-Pages ALM run — e.g. a pure data-model solution).
Optional secondary check: query the source solution for a website componentType 10374 only to log whether the site was included in this specific solution — informational, not gating:
GET {sourceEnvUrl}/api/data/v9.2/solutioncomponents?$filter=_solutionid_value eq '{solutionId}' and componenttype eq 10374&$select=objectid
Authorization: Bearer {SOURCE_TOKEN}
If found, temporarily switch PAC CLI to the target environment so check-activation-status.js queries the correct env:
pac env select --environment "{SELECTED_STAGE.targetEnvironmentUrl}"
Run the activation check:
node "${CLAUDE_PLUGIN_ROOT}/scripts/check-activation-status.js" --projectRoot "."
Then switch PAC CLI back to the source (dev) environment regardless of the result:
pac env select --environment "{sourceEnvUrl}"
Evaluate the result and take action based on the outcome. In all cases, after the outcome is resolved, update docs/alm/last-deploy.json and the deploy history file (described below).
activated: true: Site is already live. Set ACTIVATION_OUTCOME = { status: "Activated", siteUrl: "{result.websiteUrl}" }.🚦 Gate (plan · deploy-pipeline:7.7.activate): Site deployed to target but not activated. Offer to invoke activate-site or defer.
activated: false: Ask the user via AskUserQuestion:
| Question | Header | Options |
|---|---|---|
| The Power Pages site was deployed to {SELECTED_STAGE.targetEnvironmentUrl} but is not yet activated (provisioned). Activate it now to make it publicly accessible. | Activate Site | Yes, activate now (Recommended), No, I'll activate later |
/power-pages:activate-site. The activate-site skill handles subdomain selection, confirmation, and provisioning. After it completes, set ACTIVATION_OUTCOME = { status: "Activated", siteUrl: "{site URL from activate-site}" }.ACTIVATION_OUTCOME = { status: "Pending", siteUrl: null }.error present: Set ACTIVATION_OUTCOME = null — skip the update steps below silently.
After activation outcome is resolved, patch docs/alm/last-deploy.json (in-place Edit) with the actual values:
"activationStatus": "{ACTIVATION_OUTCOME.status}" (or keep null if ACTIVATION_OUTCOME is null)"siteUrl": "{ACTIVATION_OUTCOME.siteUrl}" (or keep null)Then update the deploy history HTML file (in-place Edit) — replace __ACTIVATION_SECTION__ with the appropriate HTML:
status: "Activated":
<div class="card">
<h2>Site Activation</h2>
<table><tbody>
<tr><td class="label-col">Status</td><td style="color:var(--succeeded);font-weight:600;">✓ Activated</td></tr>
<tr><td class="label-col">Site URL</td><td><a href="__SITE_URL__" style="color:var(--accent);">__SITE_URL__</a></td></tr>
</tbody></table>
</div>
(Replace __SITE_URL__ with ACTIVATION_OUTCOME.siteUrl)
status: "Pending":
<div class="note-box neutral">
<strong>Site activation pending.</strong> The solution was deployed but the site has not yet been provisioned in this environment. Run <code>/power-pages:activate-site</code> (with PAC CLI authenticated to the target environment) to activate it.
</div>
If ACTIVATION_OUTCOME is null (error during check), leave the __ACTIVATION_SECTION__ placeholder as an empty string (strip it from the file).
7.8 Detect and guide cloud flow registration (only if deployment Succeeded):
Query the solution components on the host environment for cloud flows (componenttype 29 = Workflow):
GET {hostEnvUrl}/api/data/v9.2/solutioncomponents?$filter=solutionid eq '{ARTIFACT_SOLUTION_ID}' and componenttype eq 29&$select=objectid,componenttype
Authorization: Bearer {HOST_TOKEN}
If no results: Skip this step entirely — the solution contains no cloud flows.
If results found:
Count the flows: store N = number of results.
For each objectid, attempt to resolve the flow name by querying workflows({objectid})?$select=name on the host environment. If any query fails or returns no name, fall back to displaying the raw object ID.
Inform the user:
"The solution contains {N} cloud flow(s). After deployment, cloud flows must be registered with the Power Pages site in the target environment to function correctly.
Flows detected: {bulleted list of flow names or IDs}
To register: open Power Pages → select the target environment → open your site → Set up → Cloud flows → register each flow listed above."
🚦 Gate (plan · deploy-pipeline:7.cloud-flow-register): Cloud flows in deployed solution need manual registration in target env. Acknowledge or defer (non-blocking).
Ask via AskUserQuestion:
| Question | Header | Options | |---|---|---| | Have you registered the cloud flow(s) in the target environment? | Cloud Flow Registration | Flows registered — continue, I'll register them later |
Non-blocking: regardless of the answer, continue with the summary step (Phase 7.6). Record the cloud flow registration status in the summary table:
Note: Skipped or deferred registration does not indicate a failed deployment. It only affects live site functionality for pages that call registered flows.
Yes — unblock / No — proceed anyway / Cancel deploy). Only fires when the target env's blockedattachments setting blocks extensions the solution needs (typically .js for code sites). Modifies a tenant-wide security setting; explicit consent required.MULTI_RUN_MODE only): Batch validation outcome — two possible gates fire here:
last-deploy.json knownGaps).MULTI_RUN_MODE this is handled by Phase 3.6 instead)deployment-settings.json surfaced upfront (5.0a: show summary or generate template; 5.0b: display file path). Env var values — always shown if the solution contains env var definitions with no pre-configured value for the selected stage; offer to save values for future runsDeploy now / Cancel. Fires immediately before DeployPackageAsync (or the pac pipeline deploy fallback). Makes the production-promotion moment explicit; without it, Phase 5 → Phase 6.1 could fire without a final confirmation when validation passes cleanly. Treat as the single most important prompt for Production targets.Yes — unblock and retry / No — leave settings unchanged / Cancel). Only fires when the deploy failed with AttachmentBlocked and pre-flight (Phase 2.5) was skipped or declined. Same security-setting consent as Phase 2.5.activationStatus, siteUrl) is written back to docs/alm/last-deploy.json and the deploy history HTML.docs/alm/last-pipeline.json: stop, advise /power-pages:setup-pipelineaz login instructionsRetrieveDeploymentPipelineInfo fails: use sourceDeploymentEnvironmentId from docs/alm/last-pipeline.json as fallback; warn that artifact list could not be retrieved and ask user to confirm solutionValidatePackageAsync fails: report error — usually means the solution is not ready to deploystagerunstatus = 200000003 (Failed): stop with validation details — user must resolve issues before retrying (new stage run required)stagerunstatus = 200000003 (Failed): offer retry via RetryFailedDeploymentAsync (POST /api/data/v9.1/RetryFailedDeploymentAsync {"StageRunId": "..."}) before stoppingDeployPackageAsync call fails: report error and stop| Task subject | activeForm | Description |
|---|---|---|
| Verify prerequisites | Verifying prerequisites | Run verify-alm-prerequisites.js (--require-manifest) for PAC/az/WhoAmI; run detect-project-context.js for solutionManifest/siteName; read docs/alm/last-pipeline.json for pipelineId/stages; acquire host env token |
| Select target stage | Selecting target stage | Show available stages from docs/alm/last-pipeline.json; ask user to select target; warn if last deploy to this stage failed |
| Resolve pipeline info | Resolving pipeline info | Call RetrieveDeploymentPipelineInfo (v9.1) to get SourceDeploymentEnvironmentId and DeployableArtifacts; match solution |
| Validate package | Validating package | MULTI_RUN_MODE: run Phase 3.6 once (parallel batch) — validate-stage-runs-batch.js fans out create-stage-run + ValidatePackageAsync + poll-validation-status for all non-skipped solutions concurrently; halts the deploy on any failure or pending-approval batch; persists per-solution stageRunIds for the serial deploy loop to reuse. Single-solution / legacy v2: Phase 4 inline — POST deploymentstageruns (→ 201 or 204+header); POST ValidatePackageAsync top-level action (204); poll stagerunstatus until not 200000006; JSON.parse validationresults twice; fetch aigenerateddeploymentnotes; PATCH artifactversion + deploymentnotes + deploymentsettingsjson (from deployment-settings.json) |
| Configure deployment settings | Configuring deployment settings | Check/display deployment-settings.json (5.0a: read or generate template; 5.0b: surface path); query solution for env var definitions (componenttype 380); diff against deployment-settings.json for selected stage; prompt user for any unconfigured values; offer to save back to deployment-settings.json; PATCH deploymentsettingsjson on stage run |
| Deploy and monitor | Deploying and monitoring | POST DeployPackageAsync top-level action (204); poll via filter GET (10s) until stagerunstatus terminal; handle approval gates (cancel via PATCH iscanceled=true); offer RetryFailedDeploymentAsync on failure; refresh token every 10 cycles |
| Write deployment record | Writing deployment record | Write docs/alm/last-deploy.json (with artifactVersion + deployHistoryFile fields); write docs/deploy-history/{date}-{stage}-{version}.md; git add + commit history file; run skill tracking; if Succeeded: show connection reference warning (if solutionManifest.cloudFlows non-empty) and bot republish warning (if solutionManifest.botComponents non-empty); present summary; if Succeeded and Power Pages components present: switch PAC to target, run check-activation-status.js, switch back, ask user to activate if not yet provisioned; if Succeeded and cloud flow components present (componenttype 29): query solutioncomponents, resolve flow names, inform user, ask AskUserQuestion (non-blocking), note registration status in summary |
tools
Configure the Canvas Authoring MCP server for the current coauthoring session. USE WHEN "configure MCP", "set up MCP server", "MCP not working", "connect Canvas Apps MCP", "canvas-authoring not available", "MCP not configured", "set up canvas apps". DO NOT USE WHEN prerequisites are missing — direct the user to install .NET 10 SDK first.
development
Use when the user asks to "set up authentication", "add login", "add logout", "add sign in", "enable auth", "add role-based access", "add authorization", "protect routes", "configure identity provider", "configure Entra ID", "configure Entra External ID", "configure OpenID Connect", "add OIDC", "set up SAML", "set up WS-Federation", "set up local login", "add username password", "add Facebook login", "add Google sign in", "add Microsoft Account", "set up invitation login", or otherwise wants to set up authentication (login/logout) and role-based authorization for their Power Pages code site using any supported identity provider (Microsoft Entra ID, Entra External ID, OpenID Connect, SAML2, WS-Federation, local authentication, Microsoft Account, Facebook, or Google).
development
Creates, updates, and deploys Power Apps generative pages for model-driven apps using React v17, TypeScript, and Fluent UI V9. Orchestrates specialist agents for planning, entity creation, and code generation. Use it when user asks to build, retrieve, or update a page in an existing Microsoft Power Apps model-driven app. Use it when user mentions "generative page", "page in a model-driven", or "genux".
development
Creates a new Power Pages code site (SPA) using React, Angular, Vue, or Astro. Guides through the full process from initial concept to deployed site: requirements discovery, scaffolding, component planning, design, implementation, validation, and deployment. Use when the user wants to create, build, or scaffold a new Power Pages website or portal.