setup-pipeline
Sets up a Power Platform Pipeline for automated Power Pages deployments. Power Platform Pipelines is Microsoft's native CI/CD tool built into the Power Platform — no external infrastructure required. Use when asked to: "set up ci/cd", "create pipeline", "setup pipeline", "set up power platform pipelines", "create power pipelines", "automate deployments", "set up automated deployment", "create deployment pipeline", "use power pipelines". Also handles: "set up github actions" or "set up azure devops pipeline" (shows coming-soon guidance for those platforms).
Skill body
Plugin check: Run
node "${CLAUDE_PLUGIN_ROOT}/scripts/check-version.js"— if it outputs a message, show it to the user before proceeding.
setup-pipeline
Sets up a Power Platform Pipeline for automated Power Pages solution deployments. Creates the pipeline configuration directly in Dataverse using the PP Pipelines OData API — no YAML files, no external CI/CD infrastructure needed.
GitHub Actions and Azure DevOps Pipeline options are shown in the platform menu as coming soon.
Refer to
${CLAUDE_PLUGIN_ROOT}/references/cicd-pipeline-patterns.mdfor all HAR-confirmed API patterns used in this skill.
Prerequisites
powerpages.config.jsonexists in the project root.solution-manifest.jsonexists (solution must be created first viasetup-solution)- Azure CLI logged in (
az account showsucceeds) - PAC CLI logged in (
pac env whosucceeds) - A Power Platform environment with Pipelines package installed (the “host” environment)
Phases
Phase 0 — ALM plan gate
plan-almis the front door. When the user expresses an ALM intent (promote / ship / deploy / set up CI-CD / move to staging / push to prod), the orchestrator (/power-pages:plan-alm) should run first. This Phase 0 enforces that and is meant to fail closed when there’s no plan, not to be a one-time check the user can dismiss forever.
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:
“No ALM plan exists for this project.
/power-pages:plan-almbuilds one — it detects the project state, asks about your promotion strategy (PP Pipelines vs Manual export/import), and orchestrates the right skills (including this one) in the right order. Want me to run plan-alm now?”
🚦 Gate (intent · setup-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 know what I’m doing), Cancel |
- Yes (Recommended) → invoke
/power-pages:plan-alm. plan-alm’s Phase 7 dispatches back into this skill at the appropriate stage. - Continue without a plan → set
BYPASSED_PLAN_GATE = trueand proceed to Phase 1. - Cancel → exit cleanly.
Step 4 — Stale plan. Tell the user:
“ALM plan exists from
{generatedAt}but the source solution has been modified since (at{solution.modifiedon}). Components may have changed. Re-runningplan-almwill refresh the analysis and the rendered HTML.”
🚦 Gate (intent · setup-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 |
- Refresh (Recommended) → invoke
/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). - Continue → set
STALE_PLAN_ACK = trueand proceed to Phase 1. - Cancel → exit cleanly.
Why this gate exists. Direct invocation of this skill bypasses the orchestrator’s pre-deploy completeness check, host-resolution decision, deployment-strategy selection, and rendered HTML plan. Users who run setup-pipeline directly often miss components that should have been added to the solution, miss the asset advisory for large web files, or build a pipeline against the wrong host environment. The gate ensures plan-alm either ran (so all of those decisions are surfaced and recorded) or the user explicitly chose to bypass it.
Phase 1 — Detect Project Context
Create all tasks upfront at the start of this phase.
Tasks to create:
- “Detect project context”
- “Select CI/CD platform”
- “Confirm pipeline configuration”
- “Run preflight checks”
- “Create deployment environments”
- “Create pipeline and stages”
- “Verify and write artifacts”
Steps:
- Read project context using
detect-project-context.js:node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/detect-project-context.js"Capture output as JSON; extract
.siteName(store assiteName),.websiteRecordId,.environmentUrl(store asdevEnvUrl), and.solutionManifest(store assolutionManifest). IfsiteNameis absent (nopowerpages.config.json), stop and advise running/power-pages:create-sitefirst. IfsolutionManifestis null (no.solution-manifest.json), stop and advise running/power-pages:setup-solutionfirst.Manifest version check:
- If
solutionManifest.schemaVersion === 2(multi-solution layout), setMULTI_SOLUTION_MODE = trueand storesolutionManifest.solutions[]asSOLUTIONS_LIST. See Phase 6b — a SINGLE pipeline ships all solutions through per-solution stage runs (the pre-v1.3.x “one pipeline per solution” layout was reverted because it cluttered the Pipelines UI). - If
schemaVersionis absent or1(single solution), readsolutionManifest.solution.uniqueNameandsolutionManifest.solution.solutionId. One pipeline will be created (existing flow).
- If
- Run
verify-alm-prerequisites.jsto confirm PAC CLI auth, acquire a token, and verify API access:node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/verify-alm-prerequisites.js" --envUrl "{devEnvUrl}"Capture output as JSON; extract
.envUrl(use to confirmdevEnvUrl) and.token(store asDEV_TOKEN). - Run silently:
pac env list --output json 2>/dev/nullStore output as
ENV_LIST. -
Resolve the Pipelines host via
ensure-pipelines-host-detect.js(the same flow/power-pages:ensure-pipelines-hostruns internally — it reads any cacheddocs/alm/last-host-check.json, then walks the resolution order: org-setting binding → BAP env GET → tenant default custom host → tenant-wide enumeration. Read-only; never prompts the user):BAP_TOKEN=$(az account get-access-token --resource "https://service.powerapps.com/" --query accessToken -o tsv) node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/ensure-pipelines-host-detect.js" \ --envUrl "{devEnvUrl}" \ --token "{DEV_TOKEN}" \ --userId "{userId}" \ --bapToken "{BAP_TOKEN}" \ --projectRoot "."Capture stdout as JSON:
const hostResult = JSON.parse(output). ReadhostResult.resolutionStatus,hostResult.finalHostEnvUrl,hostResult.ready.Branch on
resolutionStatus:AvailableUsingPlatformHost/AvailableUsingCustomHost/AvailableUsingCustomHostByAdminDefault— host is already established andready: true. StoreHOST_ENV_URL = hostResult.finalHostEnvUrland continue. Phase 3 confirms with the user.AvailableUnboundCustomHost/MultipleUnboundCustomHosts/PlatformHostExistsUnbound/NoHost— no host bound to the dev env. Delegate to/power-pages:ensure-pipelines-hostso the user can reuse an existing host or provision a new Custom Host (D365_ProjectHosttemplate). Tell the user: “No Pipelines host bound to{devEnvUrl}. Invoking/power-pages:ensure-pipelines-hostto set one up — it will run a tenant-wide search for existing hosts and offer to provision a new Custom Host if none are found.” After the sub-skill completes, re-readdocs/alm/last-host-check.json; captureHOST_ENV_URL = finalHostEnvUrlonly if the new marker hasready: true. If the user cancelled the sub-skill, stop this skill — no pipeline can be created without a host.CannotRedirect— stop with the specific tenant-misconfiguration error fromhostResult.warnings[0]. Tell the user: “This tenant’sDefaultCustomPipelinesHostEnvForTenantsetting and the source env’sProjectHostEnvironmentIdorg setting disagree — only a Power Platform admin can resolve.”OrgSettingStale— stop and surface the warning: “ProjectHostEnvironmentIdon{devEnvUrl}points at a host env that is no longer visible (deleted, disabled, or you lack access). Clear the org setting via PPAC or contact the env owner.”PermissionDenied— stop and surface the warning: “Caller lacks BAP read access on the env{devEnvUrl}is bound to. Contact the host env owner for at leastDeployment Pipeline Useraccess.”
Why this replaces the old
discover-pipelines-host.jscall: that helper only checked the tenant-levelDefaultCustomPipelinesHostEnvForTenantsetting (one of four resolution signals).ensure-pipelines-host-detect.jswalks the full resolution order the Power Apps UI uses (mirrorsProjectHostProvider.tsx), so we agree with the UI in every case — including the previously-undetectedAvailableUnboundCustomHostcase where a Custom Host exists in the tenant but the source env hasn’t been bound yet. Seereferences/cicd-pipeline-patterns.mdfor the full state matrix. -
Check for existing
docs/alm/last-pipeline.json. If found, read its contents. - Report findings: “Project:
{siteName}. Solution:{uniqueName}. Dev env:{devEnvUrl}. Host env:{HOST_ENV_URL ?? 'pending — will be ensured next'}({hostResult.resolutionStatus}). Existing pipeline: found/not found.”
🚦 Gate (plan · setup-pipeline:1.existing-pipeline): Existing
docs/alm/last-pipeline.jsonfound — overwrite, review first, or cancel. No Dataverse write yet.
If an existing docs/alm/last-pipeline.json is found, ask via AskUserQuestion:
“A pipeline configuration already exists for
{pipelineName}(created {createdAt}). How would you like to proceed?
- Overwrite — create a new pipeline, replacing the marker
- Review existing setup first, then decide
- Cancel”
- If Review: display the existing
docs/alm/last-pipeline.jsoncontents, then ask again with the same 3 options. - If Cancel: stop the skill and inform the user no changes were made.
- If Overwrite: proceed.
Phase 1.5 — Ground in current Pipelines documentation
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.
- Run
microsoft_docs_searchwith the query:Power Platform Pipelines setup OData API host environment deploymentenvironments. - Fetch
https://learn.microsoft.com/en-us/power-platform/alm/pipelines(and at most one sister page on host setup or pipeline creation) in parallel viamicrosoft_docs_fetch. - Extract a one-paragraph summary of what Microsoft Learn currently says about Pipelines host resolution,
deploymentenvironments/deploymentpipelines/deploymentstagesschema, and pipeline lifecycle. Compare against${CLAUDE_PLUGIN_ROOT}/references/cicd-pipeline-patterns.mdand flag any divergence (new fields, deprecated APIs, changed validation status codes). - Use the summary to inform Phase 2+ decisions. Do not silently change skill behavior — surface any divergence to the user as a soft warning before Phase 5 (Register Environments with the Pipelines Host).
Phase 2 — Select CI/CD Platform
🚦 Gate (plan · setup-pipeline:2.platform): Pick CI/CD platform — PP Pipelines (full) vs GitHub Actions / ADO (coming soon stubs).
Ask user via AskUserQuestion:
“Which CI/CD platform do you want to use?
- Power Platform Pipelines — Microsoft’s native deployment pipeline. No external infrastructure needed. (Recommended)
- GitHub Actions — Coming soon
- Azure DevOps Pipeline — Coming soon”
If the user passed power-platform, github, or ado as an argument, skip this question and use the provided value.
Store the selection as PLATFORM.
If github or ado selected → display the Coming Soon path and stop.
Power Platform Pipelines Path
Phase 3 — Confirm Pipeline Configuration
Before asking any questions, assemble what was auto-detected:
| Setting | Auto-detected value |
|---|---|
| Site name | {siteName} from powerpages.config.json |
| Solution unique name | {uniqueName} from .solution-manifest.json |
| Dev environment URL | {devEnvUrl} from pac env who |
| Host environment URL | {HOST_ENV_URL} from ensure-pipelines-host-detect.js (resolved in Phase 1 step 4) |
| BAP environment ID (dev) | From pac env list |
🚦 Gate (plan · setup-pipeline:3.config): Confirm auto-detected pipeline configuration — pipeline name, host env, target envs. Cancel exits before any Dataverse write to the host.
Ask user via AskUserQuestion with pre-filled values:
“I’ve gathered the following pipeline configuration. Please confirm or correct:
- Pipeline name:
{siteName} Pipeline(can change)- Source (Dev) environment:
{devEnvUrl}- Host environment (where Pipelines is installed):
{HOST_ENV_URL}(resolved in Phase 1 — should always be present at this point;ensure-pipelines-hostwould have stopped the skill otherwise)- Solution to deploy:
{uniqueName}- Target environments: How many? (Dev → Staging / Dev → Staging → Production)”
Collect from user:
PIPELINE_NAME(default:{siteName} Pipeline)HOST_ENV_URL(confirm — already resolved in Phase 1; user can override only if they want to point at a different host they administer, in which case re-run/power-pages:ensure-pipelines-hostfirst to validate it)- Target environment count and URLs (
STAGING_ENV_URL,PROD_ENV_URLif applicable) - BAP environment IDs for each target (from
pac env list— pre-fill if found, otherwise ask)
Store HOST_TOKEN by running:
az account get-access-token --resource "{hostEnvOrigin}" --query accessToken -o tsv
Present a final confirmation summary and ask user to approve before proceeding.
Phase 4 — Preflight Checks
Use Node.js https module for all Dataverse calls (curl has encoding issues on Windows).
4.1 Verify host environment has Pipelines installed:
GET {hostEnvUrl}/api/data/v9.1/deploymentpipelines?$top=0
Authorization: Bearer {HOST_TOKEN}
If response is 404 or returns an “unknown entity” error, stop and inform the user: “The selected host environment does not have Power Platform Pipelines installed. Please select a different environment or install the Pipelines package.”
4.2 Verify solution exists in dev environment using verify-solution-exists.js:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/verify-solution-exists.js" \
--envUrl "{devEnvUrl}" \
--uniqueName "{uniqueName}" \
--token "{DEV_TOKEN}"
Capture output as JSON; check .found. If false: warn the user — the solution must be exported from dev before it can be deployed.
4.3 Check for existing pipeline with same name:
GET {hostEnvUrl}/api/data/v9.1/deploymentpipelines?$filter=name eq '{PIPELINE_NAME}'&$select=deploymentpipelineid&$top=1
Authorization: Bearer {HOST_TOKEN}
If found: ask via AskUserQuestion whether to use the existing pipeline ID or create a new one with a different name.
4.4 Check blockedattachments on source + all target envs:
Power Pages code sites include .js files in their compiled output. If .js is in the env’s blockedattachments setting, pac pages upload-code-site (on the source) and deploy-pipeline (on targets) will both fail with AttachmentBlocked. Run this on the source env and on every target env:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/fix-blocked-attachments.js" \
--envUrl "{envUrl}" \
--extensions js \
--dry-run
If wasBlocked is non-empty for any env, inform the user:
”
.jsfiles are blocked in{envUrl}. This will cause upload/deployment failures for Power Pages code sites. Remove the block? This modifies an environment-level security setting.”
🚦 Gate (consent · setup-pipeline:4.4.blocked-attachments): Modify env-level
blockedattachmentssecurity setting (tenant-wide impact). Affects all users of the env, not just this skill. Reversible from PPAC. Fires PER ENV that has blocks. Phase 4.4 checks source + every target env; if M envs out of N have.js(or other media extensions) on the blocklist, the gate fires M times — once per env. Each env has its own security setting and its own group of affected makers. Yes for source does NOT cover staging; yes for staging does NOT cover production. Do NOT batch consent across envs.
Ask via AskUserQuestion: 1. Yes, remove block (recommended) / 2. Skip (I’ll fix manually).
If approved, re-run without --dry-run to apply the change. If the user declines, record it as a warning — they’ll need to fix it manually before deployment succeeds.
Report preflight results. If any critical check failed, stop with clear instructions. If warnings only, ask user to confirm before proceeding.
Phase 5 — Register Environments with the Pipelines Host
Register each environment (source + targets) with the Pipelines host by creating a deploymentenvironments row in the host’s Dataverse. This is a metadata-only registration — the row is a pointer to an existing BAP environment, not a provisioning call. The environments themselves must already exist in BAP. The host validates that the referenced env is reachable and the caller has the right access (validationstatus flips Pending → Succeeded). Process source env first, then targets.
Use create-deployment-environment.js for each environment (dev source + each target):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-deployment-environment.js" \
--hostEnvUrl "{HOST_ENV_URL}" \
--token "{HOST_TOKEN}" \
--name "{siteName} {label}" \
--bapEnvId "{BAP_ENV_GUID}" \
--environmentType 200000000 \
[--environmentUrl "{environmentUrl}"]
Required args (per scripts/lib/create-deployment-environment.js):
--bapEnvId— the BAP environment GUID for the env being added. Resolve viapac env list(columnEnvironment ID) orpac env whofor the current source env. NOT the org/Dataverse URL.--environmentType—200000000for the dev/source env,200000001for each target env.--environmentUrlis optional and only echoed back into the output marker; it is not posted to Dataverse.
Capture stdout as JSON: const envResult = JSON.parse(output).
Store envResult.deploymentEnvironmentId as SOURCE_DEPLOYMENT_ENV_ID (for the dev source env) or append to TARGET_DEPLOYMENT_ENV_IDs (for each target). Also retain the bapEnvId value used for each call — Phase 5a’s force-link auto-fix needs it if creation lands in a Failed state.
Note: The script POSTs to
deploymentenvironmentswith unprefixed fields (name,environmentid,environmenttype), extracts thedeploymentenvironmentidGUID from theOData-EntityIdheader, then pollsvalidationstatusevery 3 seconds (max 20 attempts) until status200000001(Succeeded) or200000002(Failed). On failure the script writes the error details to stderr and exits 1 — stop and report the error to the user. (The earliermsdyn_-prefixed field shape and192350001/192350002status codes were from an early-preview HAR; the shipped Pipelines schema rejectsmsdyn_-prefixed properties and uses the2000000XXcodes.)
On failure: stop with the error — deployment environment creation is mandatory.
5a — Detect “already associated with another pipelines host” (Pattern 15)
If the script’s stderr (case-insensitively) contains any of these substrings, the BAP env is currently stamped to a different Pipelines host:
already associated with another pipelines hostassociated with another pipelines hostenvironment is already linked to a different hostenvironment is already bound tolinked to another hostclaimed by another host
Match all of these case-insensitively (String.prototype.toLowerCase() before .includes()) so backend wording drift between Pipelines package versions doesn’t silently break detection. If none match but the script exited with the underlying Dataverse error code 0x80048d18 (or a wrapped errormessage containing that hex code), treat it as the same pattern — that’s the stable signal even when the message wording shifts.
🚦 Gate (consent · setup-pipeline:5a.pattern-15): Target env stamped to a different Pipelines host. Offer force-link as documented auto-fix — DESTRUCTIVE: previous host loses pipeline access for this env. Cancel here exits setup-pipeline cleanly. Fires PER ENV that triggers Pattern 15. Phase 5 loops over source + each target env when registering with the host; if two target envs both turn out to be stamped to different hosts, this gate fires twice — once per env. Do NOT batch the consent across envs; the destructive blast radius is per-env (each env carries its own previous-host stamp and its own group of makers losing access).
This is Pattern 15 in ${CLAUDE_PLUGIN_ROOT}/references/deployment-error-catalog.md. Do NOT silently retry. Surface the raw errormessage to the user verbatim and offer the documented auto-fix via AskUserQuestion:
question: "<envLabel> is already linked to a different Pipelines host. The /power-pages:force-link-environment skill can take over the association (DESTRUCTIVE to the previous host — makers there lose pipeline access for this env). Run it now?"
header: "Force Link?"
options:
- "Run /power-pages:force-link-environment now (Recommended)" — auto-fix per the deployment error catalog
- "Cancel setup-pipeline" — investigate the previous host first
Important guardrails:
- Never invoke
/power-pages:force-link-environmentwithout explicit user consent through this prompt — the action is reversible only by performing Force Link again from the previous host. - If the user picks “Run …”, invoke
/power-pages:force-link-environmentwith--host <HOST_ENV_URL>and--dev-env <bapEnvId>(the BAP env GUID captured for this env in Phase 5 — see the “Also retain thebapEnvIdvalue” note above) so the sub-skill skips its own host/env prompts. - When that sub-skill returns success, re-attempt just the failing environment by re-running
create-deployment-environment.jswith the same args — do NOT restart Phase 5 wholesale. The create script is idempotent: it short-circuits viafindExistingByBapIdfor envs already created (they returnreused: true), and the previously-failing env will now resolve to Succeeded because the host stamp has moved. - If the user picks “Cancel”, stop the pipeline setup and recommend
/power-pages:ensure-pipelines-host detect-onlyto inspect the current host bindings before retrying.
For any other create-deployment-environment failure, fall through to the generic “stop with the error” path above.
Report progress for each environment as validation completes.
Phase 6 — Create Pipeline, Associate Source, Create Stages
Use create-deployment-pipeline.js to create the pipeline, associate the source environment, and create all stage records in one call:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-deployment-pipeline.js" \
--hostEnvUrl "{HOST_ENV_URL}" \
--token "{HOST_TOKEN}" \
--pipelineName "{PIPELINE_NAME}" \
--description "Power Pages deployment pipeline for {siteName}" \
--sourceDeploymentEnvironmentId "{SOURCE_DEPLOYMENT_ENV_ID}" \
--stagesJson '[{"name":"Deploy to {targetLabel}","targetDeploymentEnvironmentId":"{TARGET_DEPLOYMENT_ENV_ID}","order":1}]'
Capture stdout as JSON: const pipelineResult = JSON.parse(output).
Extract:
pipelineResult.pipelineId→ store asPIPELINE_IDpipelineResult.stages→ array of{ stageId, name, targetDeploymentEnvironmentId }
What the script does internally (uses the unprefixed field schema — the earlier
msdyn_-prefixed body was rejected by the shipped Pipelines schema; see the comment block at the top ofcreate-deployment-pipeline.jsfor the full migration map):
- POSTs
{ name, description }todeploymentpipelines(v9.1) — extractsdeploymentpipelineidfromOData-EntityIdheader- POSTs a relative-path
@odata.idbody todeploymentpipelines({pipelineId})/deploymentpipeline_deploymentenvironment/$refto associate the source environment (HAR-confirmed — no leading/or full URL)- For each stage: POSTs
{ name, [email protected], [email protected] }todeploymentstages— extractsdeploymentstagesidfromOData-EntityIdheader
On failure: the script writes the error to stderr and exits 1 — stop and report the error to the user.
Phase 6b — Multi-solution deploymentOrder (only if MULTI_SOLUTION_MODE = true)
Design note (updated v1.3.x): A single Power Platform Pipeline can deploy multiple solutions through separate stage runs — each run just specifies a different
artifactname+solutionidon the samedeploymentstagesrecord. Creating one pipeline per solution was wasteful and cluttered the Pipelines UI. We now create ONE pipeline + one stage per target env, and record the per-solution deployment order indocs/alm/last-pipeline.json.deploy-pipelinethen loops over the order, creating a stage run per solution against the same stage.
When the manifest is schemaVersion: 2, do not call create-deployment-pipeline.js multiple times. Instead:
- Call
create-deployment-pipeline.jsonce with:pipelineName = "{siteName}-Pipeline"(e.g.IdeaSphere-Pipeline).descriptionlisting the solutions that will deploy through it (e.g."Deploys IdeaSphere_Core → IdeaSphere_WebAssets → IdeaSphere_Future in order").- One
deploymentstagesrecord per target environment (not per solution).
- Build the
deploymentOrderarray fromSOLUTIONS_LISTsorted byorder. Each entry has{ solutionUniqueName, solutionId, order }. Skip entries whereisFutureBuffer: trueANDcomponents.length === 0— an empty Future solution has nothing to deploy; it’s created bysetup-solutionbut does not participate in the deployment loop until it has content. Keep it in the order array withstatus: "SkippedEmpty"so the renderer can show the intent. - Collect the single
pipelineIdand itsstages[]. PersistdeploymentOrdertodocs/alm/last-pipeline.json(see Phase 7).
Phase 7 — Verify, Write Artifacts, Commit
7.1 Verify pipeline was created:
GET {hostEnvUrl}/api/data/v9.1/deploymentpipelines({PIPELINE_ID})?$select=name,statecode
Authorization: Bearer {HOST_TOKEN}
Confirm statecode = 0 (Active). If the query fails, report as “verification inconclusive — pipeline may still be valid”.
7.2 Write docs/alm/last-pipeline.json (create the docs/alm/ directory first if missing — node -e "require('fs').mkdirSync('docs/alm',{recursive:true})"):
{
"pipelineId": "{PIPELINE_ID}",
"pipelineName": "{PIPELINE_NAME}",
"hostEnvUrl": "{HOST_ENV_URL}",
"sourceDeploymentEnvironmentId": "{SOURCE_DEPLOYMENT_ENV_ID}",
"sourceEnvironmentUrl": "{devEnvUrl}",
"solutionName": "{uniqueName}",
"createdAt": "{ISO timestamp}",
"stages": [
{
"stageId": "{deploymentstagesid}",
"name": "Deploy to {targetLabel}",
"rank": 1,
"targetDeploymentEnvironmentId": "{TARGET_DEPLOYMENT_ENV_ID}",
"targetEnvironmentUrl": "{targetEnvUrl}"
}
]
}
Multi-solution marker (manifest v2): When MULTI_SOLUTION_MODE = true, docs/alm/last-pipeline.json uses schemaVersion: 3 with a single pipeline and a deploymentOrder[] describing which solutions deploy through it, in what order:
{
"schemaVersion": 3,
"pipelineId": "...",
"pipelineName": "IdeaSphere-Pipeline",
"hostEnvUrl": "{HOST_ENV_URL}",
"sourceDeploymentEnvironmentId": "{SOURCE_DEPLOYMENT_ENV_ID}",
"sourceEnvironmentUrl": "{devEnvUrl}",
"createdAt": "{ISO timestamp}",
"stages": [
{
"stageId": "...",
"name": "Deploy to Staging",
"rank": 1,
"targetDeploymentEnvironmentId": "...",
"targetEnvironmentUrl": "https://staging.crm.dynamics.com"
}
],
"deploymentOrder": [
{ "solutionUniqueName": "IdeaSphere_Core", "solutionId": "...", "order": 1 },
{ "solutionUniqueName": "IdeaSphere_WebAssets", "solutionId": "...", "order": 2 },
{ "solutionUniqueName": "IdeaSphere_Future", "solutionId": "...", "order": 3, "status": "SkippedEmpty", "isFutureBuffer": true }
]
}
Migration note: Earlier versions of this skill used
schemaVersion: 2with apipelines[]array (one Dataverse pipeline record per solution). Projects pinned to v2 continue to work with the olddeploy-pipelineMULTI_PIPELINE_MODE path; the v3 format should be used for all new setups. When re-runningsetup-pipelineon a v2 project, ask viaAskUserQuestionwhether to migrate (delete the N-1 extra pipelines and collapse to a single one) or keep the legacy layout.
7.3 Write (or re-render) docs/pipeline-setup.md (create docs/ directory if needed).
Contents:
- Pipeline Created — name, host env URL, pipeline ID
- Environments configured — source + each target with their deployment environment IDs
- Solutions in deployment order (multi-solution mode only) — for each entry in
solutionManifest.solutions[], list{uniqueName, version, componentCount}. ReadcomponentCountfrom each entry’scomponents.lengthif the manifest tracks it, otherwise from a live Dataverse query (solutioncomponents?$filter=_solutionid_value eq '{solutionId}' and componenttype ne 380&$count=true) — DO NOT hard-code or carry forward a stale count from a prior invocation. - How to trigger a deployment — Run
/power-pages:deploy-pipelineor open Power Platform make.powerapps.com → Solutions → Pipelines - Approval gates (if applicable) — How to configure in Power Platform Admin Center
- Troubleshooting — Common validation errors and how to resolve them
Sync-mode re-render: when
setup-pipelineis invoked on a project wheredocs/alm/last-pipeline.jsonALREADY exists (re-run afterconfigure-env-variables,setup-solutionsync, or a follow-up env-var addition that bumped component counts), regenerate this file in full from current Dataverse state — do not patch in place. Validated failure: a Citizens portalpipeline-setup.mdshowed Foundation = 13 components while Dataverse had 15 afterconfigure-env-variablesadded 2 env var definitions to that solution; the markdown never updated. The simplest safe behavior is “always re-render in Phase 7.3”, because the operation reads current state directly and the file has no user-editable sections worth preserving.
7.4 Commit:
git add docs/alm/last-pipeline.json docs/pipeline-setup.md
git commit -m "Add Power Platform Pipeline configuration for {siteName}"
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 "SetupPipeline".
7.5b Refresh the ALM plan (if one exists):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
--projectRoot "." \
--phase setup-pipeline \
--render
The helper reads docs/alm/last-host-check.json + docs/alm/last-pipeline.json, refreshes planData.hostResolution and planData.pipelineMeta, drops pre-setup “no host detected” risks, and re-renders docs/alm-plan.html. When docs/.alm-plan-data.json is absent (standalone invocation, not via plan-alm), the helper returns ok:false as a soft no-op — safe to run unconditionally.
7.6 Present summary:
| Resource | ID / URL |
|---|---|
| Pipeline | {PIPELINE_NAME} ({PIPELINE_ID}) |
| Host environment | {HOST_ENV_URL} |
| Source deployment env | {SOURCE_DEPLOYMENT_ENV_ID} |
| Stage: {name} | {stageId} → {targetEnvUrl} |
Files written:
docs/alm/last-pipeline.json— pipeline configuration markerdocs/pipeline-setup.md— setup documentation
Next step:
Run
/power-pages:deploy-pipelineto trigger your first deployment run.
Coming Soon Path
If GitHub Actions or Azure DevOps was selected:
Inform the user:
“GitHub Actions and Azure DevOps Pipeline support are coming soon for this skill.
For now, you have two options:
- Use Power Platform Pipelines — select option 1 to set up Microsoft’s native deployment pipeline (recommended)
- Exit — I’ll set up GitHub Actions / Azure DevOps manually using the documentation”
🚦 Gate (plan · setup-pipeline:coming-soon.exit): User selected GitHub/ADO (coming-soon stubs) — offer to switch back to PP Pipelines or exit cleanly.
Ask via AskUserQuestion:
- Switch to Power Platform Pipelines — go back to Phase 2
- Exit — I’ll set up manually
If GitHub/ADO passed as argument: display above message and exit gracefully.
Key Decision Points (Wait for User)
- Phase 1: Existing pipeline file — overwrite, review, or cancel (only if
docs/alm/last-pipeline.jsonfound) - Phase 2: Platform selection (Power Platform Pipelines / GitHub coming soon / ADO coming soon)
- Phase 3: Confirm pipeline configuration — pipeline name, host env URL, target environments
- Phase 4: Preflight warnings — proceed or cancel
- Phase 3: Parameter confirmation before pipeline creation
Error Handling
- No
powerpages.config.json: stop, advise/power-pages:create-site - No
.solution-manifest.json: stop, advise/power-pages:setup-solution RetrieveSettingreturns empty: ask user for host environment URL manually- Deployment environment
statecode = 1with non-nullerrormessage(validation failed): stop with error details - Pipeline
$refcall fails: stop — this association is required before stages can be created - Stage creation fails: record failure, continue with remaining stages — partial success is valid
Progress Tracking Table
| Task subject | activeForm | Description |
|---|---|---|
| Detect project context | Detecting project context | Read powerpages.config.json and .solution-manifest.json; run pac env who and pac env list; call RetrieveSetting to find host env; check for existing docs/alm/last-pipeline.json |
| Select CI/CD platform | Selecting CI/CD platform | Ask user: Power Platform Pipelines (full) or GitHub/ADO (coming soon) |
| Confirm pipeline configuration | Confirming pipeline configuration | Pre-fill pipeline name, source env, host env, solution name from auto-detected values; ask for target environments; get user confirmation |
| Run preflight checks | Running preflight checks | Verify host env has Pipelines installed; verify solution exists in dev env; check for pipeline name conflict |
| Create deployment environments | Creating deployment environments | POST deploymentenvironments for source + each target; poll validationstatus for each until Succeeded |
| Create pipeline and stages | Creating pipeline and stages | POST deploymentpipelines; $ref associate source env; POST deploymentstages for each target (linked via previousdeploymentstageid) |
| Verify and write artifacts | Verifying and writing artifacts | Query pipeline to confirm active; write docs/alm/last-pipeline.json; write docs/pipeline-setup.md; commit; present summary with next steps |