plugins/power-pages/skills/setup-solution/SKILL.md
Creates a Dataverse publisher and solution, then adds Power Pages site components to the solution for ALM and deployment management. Use when asked to: "create solution", "set up solution", "add to solution", "package site into solution", "create publisher", "solutionize my site", or "set up ALM for my site".
npx skillsauth add microsoft/power-platform-skills setup-solutionInstall 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.
Creates a Dataverse publisher and solution, then adds Power Pages site components. Writes .solution-manifest.json for use by export-solution, import-solution, and setup-pipeline skills.
pac env who returns an environment URL)az account show succeeds)powerpages.config.json exists in the project root (site must be deployed at least once so .powerpages-site/ exists with component records)
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 "."
The helper returns JSON with { exists, deferred, stale, staleness: { reason, detail }, generatedAt, planStatus, ... }. Sync mode (when .solution-manifest.json already exists) may additionally pass --envUrl, --token, --solutionId once Phase 1 has acquired them, but for the initial gate the existence-only check is sufficient.
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: setup-solution: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 (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-solution: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 |
/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.Step 4 — Stale plan. Tell the user:
<!-- gate: setup-solution:0.stale-plan | category=intent | cancel-leaves=nothing -->"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-solution: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.Why this gate exists. Direct invocation of setup-solution builds (or syncs) a solution without consulting the orchestrator's plan. If a plan already exists and recommends a multi-solution split, running this skill standalone may consolidate components into the wrong base solution. If no plan exists yet, plan-alm would have surfaced split recommendations, the asset-size advisory, and missing-component gaps before any solution was created — running setup-solution first burns through those decisions silently. The gate ensures setup-solution runs in the right context, while still leaving an explicit bypass for users who genuinely know they want a one-off solution.
Create all tasks upfront at the start of this phase.
Tasks to create:
Steps:
Run pac env who — extract environmentUrl, organizationId (shown to user for confirmation)
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" --envUrl "{environmentUrl}"
Capture output as JSON; extract .envUrl (store as envUrl) and .token (store as token). If the script exits non-zero, stop and explain what is missing (reference ${CLAUDE_PLUGIN_ROOT}/references/dataverse-prerequisites.md).
Locate powerpages.config.json — read siteName and websiteRecordId
Confirm .powerpages-site/ folder exists (required to find component records)
Check for ALM plan context — look for docs/alm/alm-plan-context.json:
🚦 Gate (plan · setup-solution:1.preloaded): Use pre-loaded plan classifications, or re-discover. No write happens before this choice.
AskUserQuestion:
"An ALM plan was previously generated for this site. It includes a pre-classified list of site settings (keepAsIs, promoteToEnvVar, authNoValue, excluded). Would you like to use those choices, or re-discover and re-classify everything now?"
docs/alm/alm-plan-context.json, store the siteSettings object as preloadedSettings. When Step 5.3 is reached, skip the query and classification logic — use preloadedSettings directly.Detect sync mode — check whether .solution-manifest.json exists in the project root.
If present: read it and verify the solutionId still exists in the target environment via GET {envUrl}/api/data/v9.2/solutions({solutionId})?$select=solutionid,uniquename,version,ismanaged.
syncMode = true and store existingSolution = the manifest contents.🚦 Gate (consent · setup-solution:1.stale-manifest): Manifest references a solution missing from the current env. Start fresh (back up the manifest and create a new solution) or abort.
AskUserQuestion:
"The existing
.solution-manifest.jsonpoints to solution{uniqueName}v{version} which I could not find in the current environment. Would you like to: 1) Start fresh (back up the manifest and create a new solution), 2) Abort so you can investigate?" Proceed only after an explicit choice.
If absent: set syncMode = false — this is a fresh setup.
Report the chosen mode to the user:
syncMode = true: "Found existing solution {uniqueName} v{version}. Running in sync mode — I'll discover the current site inventory, diff against what's already in the solution, and only add missing components."syncMode = false: "No existing solution manifest found. Running a fresh setup — I'll create a publisher and solution, then add all site components."Check for split plan (multi-solution mode) — look for docs/alm/alm-split-plan.json (written by plan-alm Phase 1 Step 10):
proposedSolutions.length > 1, set MULTI_SOLUTION_MODE = true and store the array as PROPOSED_SOLUTIONS.PROPOSED_SOLUTIONS in parallel. The order field is data for downstream pipeline-stage ordering — it does NOT constrain creation order, since each solution is independent (distinct uniqueName, shared publisherId, no inter-solution dependency).AddSolutionComponent calls per solution based on proposedSolutions[i].componentTypes and tableLogicalNames (for Strategy 3).proposedSolutions.length === 1, proceed in single-solution mode (existing flow).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 Pages solution publisher creation Dataverse component types ALM.https://learn.microsoft.com/en-us/power-platform/alm/solution-concepts-alm (and at most one sister page if the search surfaces a relevant new tutorial — e.g. multi-solution layering, managed-properties guidance) in parallel via microsoft_docs_fetch.${CLAUDE_PLUGIN_ROOT}/references/solution-api-patterns.md and flag any divergence (new component types, changed action signatures, deprecated patterns).<!-- gate: setup-solution:2.publisher-prefix | category=consent | cancel-leaves=nothing -->Skip this entire phase when
syncMode = true. UseexistingSolution.publisherandexistingSolution.solutionfrom the manifest instead. Jump to Phase 5.
🚦 Gate (consent · setup-solution:2.publisher-prefix): Publisher prefix is PERMANENT and prefixed to every component logical name. Must be confirmed explicitly. Cancel exits before any publisher/solution write.
Ask user (via AskUserQuestion) for:
contoso) — lowercase letters/numbers only, no spaces. Explain this is permanent and cannot be changed.Contoso) — display namecon) — 2–8 lowercase letters, prefixed to all components. Explain this is permanent and cannot be changed.ContosoSite) — letters/numbers/underscores, no spacesContoso Site) — display name1.0.0.0) — must be major.minor.build.revision formatPresent a confirmation summary of all values and wait for user approval before proceeding.
Key Decision Point: Publisher prefix and publisher unique name are irreversible — pause and explicitly confirm with the user before proceeding.
Skip this entire phase when
syncMode = true. The manifest guarantees the solution exists and we already validated it in Phase 1 Step 6.
Before creating anything, check if publisher and solution already exist:
GET {envUrl}/api/data/v9.2/publishers?$filter=uniquename eq '{publisherUniqueName}'&$select=publisherid,uniquename,customizationprefix
(No dedicated script for publishers — query the OData endpoint directly.)verify-solution-exists.js:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/verify-solution-exists.js" \
--envUrl "{envUrl}" \
--uniqueName "{solutionUniqueName}" \
--token "{token}"
Capture output as JSON; check .found (boolean). If found, also read .solutionId, .version, and .isManaged for display.Report findings to user:
{name} (prefix: {prefix}). Will reuse it."{name} version {version}. Will reuse it and add components."Wait for user confirmation before proceeding.
Skip this entire phase when
syncMode = true. The publisher and solution already exist.Version bump in sync mode: before any add operations in Phase 5, bump the existing solution's patch segment so the post-sync manifest and any subsequent export cleanly supersede the prior version. Use the shared helper — it is the single source of truth for the bump rule (pad-with-zero for missing segments, integer-numeric
1.0.0.9 → 1.0.0.10, reject1.0.0.a, reject more-than-4 segments). The same helper is called fromexport-solutionPhase 4 Step 4.0 — both skills must produce identical bumps for the same input version.node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/bump-solution-version.js" \ --envUrl "{envUrl}" \ --token "{token}" \ --solutionId "{solutionId}" \ --projectRoot "."Capture output as JSON; the helper returns
{ previous, next, bumped: true, manifestUpdated, manifestUpdateReason }. Passing--projectRoot "."lets the helper update.solution-manifest.json'ssolution.version(single-solution) or matchingsolutions[].version(multi-solution) field automatically — without it, the manifest drifts behind every bump. UpdateexistingSolution.solution.versionlocally to.nextso the final manifest write reflects the bump. Do this before Step 5.6's component adds, so the manifest stays consistent if the skill is interrupted midway. Do not inline the PATCH — diverging the rule between this skill andexport-solutionis exactly the bug class the helper exists to prevent.
Refer to ${CLAUDE_PLUGIN_ROOT}/references/solution-api-patterns.md for exact request body templates.
Create publisher (if not existing):
POST {envUrl}/api/data/v9.2/publishers with publisher bodypublisherId from OData-EntityId response header[email protected]. Single serial step, no parallelization.Create solution(s):
Single-solution mode (MULTI_SOLUTION_MODE = false) — call create-solution.js. Omit --token so the helper refreshes via getAuthToken(envUrl) at call time (passing a possibly-stale cached token would surface as a 401 the helper doesn't retry):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-solution.js" \
--envUrl "{envUrl}" \
--uniqueName "{solutionUniqueName}" \
--friendlyName "{solutionFriendlyName}" \
--version "{version}" \
--publisherId "{publisherId}" \
--description "Power Pages solution for {siteName}"
Capture output as JSON; extract .solutionId (store as solutionId). On failure (non-zero exit or created: false): report error, stop.
Multi-solution mode (MULTI_SOLUTION_MODE = true) — call create-solutions-batch.js, which fans out all PROPOSED_SOLUTIONS in parallel via Promise.allSettled (typical 5-6 solution splits complete in ~2s vs ~10s serial). The helper skips isFutureBuffer: true entries automatically (the reserved buffer is created later when the user actually adds new components) and handles 409 races idempotently via verify-solution-exists.js. Write the specs to a tmp JSON file, then invoke:
node -e "require('fs').writeFileSync('./docs/alm/.solutions-batch.json', JSON.stringify({{PROPOSED_SOLUTIONS_AS_SPECS}}))"
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-solutions-batch.js" \
--envUrl "{envUrl}" \
--token "{token}" \
--publisherId "{publisherId}" \
--solutionsFile ./docs/alm/.solutions-batch.json
Where {{PROPOSED_SOLUTIONS_AS_SPECS}} is PROPOSED_SOLUTIONS mapped to { uniqueName, friendlyName: displayName, version: "1.0.0.0", description, isFutureBuffer } per entry (carry the isFutureBuffer flag through so the helper can skip it). Capture the output as JSON; build SOLUTIONS_BY_NAME = { uniqueName → { solutionId, created } } from result.results (entries with skipped: true are not added — Future buffer solutions don't exist in Dataverse yet). If result.failed > 0, surface the per-entry error strings and stop — successfully-created solutions remain in Dataverse and the user can re-run setup-solution in sync mode to recover. Delete the tmp file after the call (./docs/alm/.solutions-batch.json).
Token must be fresh before the batch — create-solutions-batch.js refreshes once at start via getAuthToken(envUrl) if no --token is passed, so prefer omitting --token over passing a stale one.
Report: "Publisher {name} is ready. Created {N} solution(s): {name1}, {name2}, …" (single-solution mode: report just the one).
Refer to ${CLAUDE_PLUGIN_ROOT}/references/solution-api-patterns.md for AddSolutionComponent body templates and powerpagecomponents discovery patterns.
Sync-mode behavior: When
syncMode = true, run the discovery helper with--solutionIdpopulated and use the returnedmissing.*arrays as the candidate set. Everything else in this phase (dynamic component-type lookup in 5.1, categorization in 5.3, OAuth secret conversion in 5.4, env var adoption in 5.4b, orphan ppc adoption in 5.4c, manifest summary in 5.5, bulk add in 5.6) runs the same way, just with a pre-filtered "only things that aren't already in the solution" list. The goal of sync mode is: a user who added a server logic, bot, flow, env var, or page aftersetup-solutionlast ran can re-invoke the skill and get those components adopted without any fresh-setup prompts.Fresh-mode behavior (
syncMode = false): run the full discovery as documented below — every ppc, every site language, every custom table, every publisher-prefix env var becomes a candidate for inclusion.
Do not hardcode component type numbers. Component type codes are environment-specific metadata and vary across tenants. Always resolve them at runtime using discover-component-types.js.
Run discover-component-types.js with the website record ID plus one sample powerpagecomponent ID and one site language ID (obtained from the preliminary discovery queries in Step 5.2 below — run those first if not yet available):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-component-types.js" \
--envUrl "{envUrl}" \
--token "{token}" \
--websiteRecordId "{websiteRecordId}" \
--powerpageComponentId "{anyPowerpageComponentId}" \
--siteLanguageId "{siteLanguageId}"
Capture output as JSON; extract .websiteComponentType, .subComponentType, and .siteLanguageComponentType. Use the JSON values returned by the helper exactly as-is — do not substitute "typical" values from documentation. Observed reference values across tenants include 10426/10427/10428 and 10429/10428/10430, but the actual values vary per environment and must come from this script's runtime query. The three sibling unified entities each have their own componenttype — site language is NOT included by AddRequiredComponents: true on the website and must be added explicitly. See references/solution-api-patterns.md for the full 3-entity model.
If the script reports the website record is not yet in any solution, stop and inform the user that the site must be deployed (via /power-pages:deploy-site) before it can be solutionized. If subComponentType is absent (no sub-components indexed yet), proceed anyway — you will discover all component IDs in Step 5.2.
Run six discovery queries in parallel:
A. Component type labels (for display names):
GET {envUrl}/api/data/v9.2/GlobalOptionSetDefinitions(Name='powerpagecomponenttype')
Build a typeLabel map: { [Value]: Label.UserLocalizedLabel.Label }. Fall back to the static table in ${CLAUDE_PLUGIN_ROOT}/references/solution-api-patterns.md Section 3b if this fails.
B. All Power Pages sub-components for this site:
GET {envUrl}/api/data/v9.2/powerpagecomponents
?$filter=_powerpagesiteid_value eq '{websiteRecordId}'
&$select=powerpagecomponentid,name,powerpagecomponenttype
&$orderby=powerpagecomponenttype
Follow @odata.nextLink pagination. Group by powerpagecomponenttype using typeLabel for display names.
C. Site language records:
GET {envUrl}/api/data/v9.2/powerpagesitelanguages?$filter=_powerpagesiteid_value eq '{websiteRecordId}'&$select=powerpagesitelanguageid,languagecode,displayname
Store all language IDs.
D. Dataverse tables — always discover from the environment, don't rely on a manifest file alone:
.datamodel-manifest.json if present (for the known list of tables created by setup-datamodel)GET {envUrl}/api/data/v9.2/EntityDefinitions?$select=LogicalName,MetadataId,IsManaged,IsCustomEntity
Filter client-side: IsCustomEntity === true && IsManaged === false. Group by publisher prefix (characters before first _). Present only tables whose prefix matches the site publisher — or if no prefix match, present all custom unmanaged tables and let the user decide.
Important note on tables: Dataverse solutions carry schema only — entity definitions, columns, relationships, forms, and views. Table data/records do NOT travel with the solution. If the target environment needs seed/reference data, that requires a separate data migration step.
E. Cloud Flow link components (powerpagecomponenttype 33) — runtime field introspection:
Query the powerpagecomponent records that link this site to Cloud Flows:
GET {envUrl}/api/data/v9.2/powerpagecomponents
?$filter=_powerpagesiteid_value eq '{websiteRecordId}' and powerpagecomponenttype eq 33
&$select=powerpagecomponentid,name
If results are returned, fetch the first record without a $select to discover the workflow lookup field:
GET {envUrl}/api/data/v9.2/powerpagecomponents({firstComponentId})
Scan the response JSON for _*_value keys with non-null GUIDs that do not equal websiteRecordId. The remaining key is the workflow lookup field (e.g., _adx_workflow_value). Re-query all type-33 components with that field in $select to collect all backing workflowId GUIDs. Then resolve each workflow name and status:
GET {envUrl}/api/data/v9.2/workflows({workflowId})?$select=name,workflowid,statecode
Also discover the workflow's component type (for AddSolutionComponent):
GET {envUrl}/api/data/v9.2/solutioncomponents?$filter=objectid eq '{workflowId}'&$select=componenttype&$top=1
Store as workflowComponentType. If the query returns empty (flow not yet in any solution), note it — the backing flow record still exists and can be added.
If type-33 query returns no records, store cloudFlows = [] and skip.
F. Bot Consumer link components (powerpagecomponenttype 27) — runtime field introspection:
Same pattern as Query E. Query type-27 powerpagecomponent records, discover the bot lookup field via introspection on the first record, collect bot GUIDs, resolve bot names via:
GET {envUrl}/api/data/v9.2/bots({botId})?$select=name,botid,statecode
And discover bot component type via solutioncomponents. Store as botComponents. If no type-27 records exist, store botComponents = [] and skip.
G. Connection references used by cloud flows in this solution:
Cloud flows reference connectors via connectionreference records. These records are separate Dataverse entities; if they aren't in the solution, the solution will export cleanly but fail to import in the target environment with a MissingDependency / connection-reference validation error. We must enumerate them here and add them in Step 5.6.
Skip this query if Query E returned cloudFlows = [].
Query connection references owned by this site's publisher:
GET {envUrl}/api/data/v9.2/connectionreferences
?$filter=startswith(connectionreferencelogicalname,'{publisherPrefix}_')
&$select=connectionreferenceid,connectionreferencelogicalname,connectionreferencedisplayname,connectorid
For each cloud flow (from Query E), parse its clientdata JSON (workflows({workflowId})?$select=clientdata) to find which connectionReferenceLogicalNames it uses. Filter the Query G.1 result to just those references — these are the ones that must be in the solution.
Resolve the connection-reference componenttype at runtime — the value is environment-specific (observed values include 10137 and 10160 across tenants; do NOT hardcode):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-component-types.js" \
--envUrl "{envUrl}" --token "{token}" \
--websiteRecordId "{websiteRecordId}" \
--objectIds "{firstConnectionReferenceId}"
Read .resolved[0].componentType and store as connectionReferenceComponentType. If the connection ref is not yet in any solution (resolved[0].componentType === null), it has never been added — fall back to passing one ID per call until one resolves, or query a known sibling connection ref. Without a runtime-resolved value, do not guess.
Store the filtered list as connectionReferences[].
If Query G returns no references (the cloud flows don't use connectors, or the publisher prefix doesn't match — rare), store connectionReferences = [] and skip. Surface a soft warning if cloud flows exist but no matching connection refs were found — the user should verify whether their flows are using connectors that need binding in target envs.
If preloadedSettings is available (user chose "Use pre-loaded choices from plan" in Phase 1 Step 5), skip the classification below — use preloadedSettings.keepAsIs, preloadedSettings.promoteToEnvVar, preloadedSettings.authNoValue, and preloadedSettings.credentialNeedsDecision directly. (Plans generated before 2026-05-08 use the older excluded bucket — treat its contents as credentialNeedsDecision for backward compatibility.)
Otherwise, run the shared classifier — ${CLAUDE_PLUGIN_ROOT}/scripts/lib/classify-site-settings.js — which is the single source of truth for the credential regex + tier mapping shared with plan-alm Phase 1 Step 7. Either invoke the CLI (pipe JSON to stdin) or require() it inline. The output is the same four-bucket shape plan-alm produces:
{
keepAsIs: [{name}], // Tier 3 — added to the solution unchanged
authNoValue: [{name}], // Tier 2b — Authentication/AzureAD with empty value; added as-is, user sets per-env
promoteToEnvVar: [{name, value}], // Tier 2a — Authentication/AzureAD with value; reviewed at Step 5.4.A
credentialNeedsDecision: [{name, value}] // Tier 1 — credential-style names; bulk-with-override prompt at Step 5.4.C
}
Tier definitions (mirroring the regex in classify-site-settings.js):
| Tier | Bucket | Matcher | Handling |
|---|---|---|---|
| 1 — Credential-style | credentialNeedsDecision | CREDENTIAL_REGEX (ConsumerKey\|ConsumerSecret\|ClientId\|ClientSecret\|AppSecret\|AppKey\|ApiKey\|Password, case-insensitive) | Bulk-with-override prompt at Step 5.4.C — auto-classify (Secret/String defaults), all-Secret, all-String, skip-all, or pick-per-credential |
| 2a — Auth config with value | promoteToEnvVar | AUTH_PREFIX_REGEX (Authentication/ or AzureAD/) AND NOT credential AND has a value | Multi-select prompt at Step 5.4.A — which to back with env vars |
| 2b — Auth config, no value | authNoValue | Same prefix, no value | Added to solution as-is with a note (user sets value per env) |
| 3 — All other settings | keepAsIs | Anything else | Included in solution unchanged |
Do NOT inline the regex here — if it's wrong in this skill but right in plan-alm, classifications drift between plan time and execution time. The regex lives in classify-site-settings.js exclusively; both skills require it.
Note on authNoValue settings: These are auth configuration settings where no value has been set in the dev environment. They will be added to the solution as-is. After deploying to each target environment, the correct value should be configured there. Present these in a warning note box during the manifest review (Step 5.5).
Before presenting the final manifest, handle the three non-keepAsIs categories:
A. promoteToEnvVar settings (auth config with values):
🚦 Gate (plan · setup-solution:5.4a.promote): Multi-select over auth settings — which to promote to env vars. Leave others as plain site settings.
Ask via AskUserQuestion with multiSelect: true, listing each promoteToEnvVar setting by name + current value:
"These authentication configuration settings have values set in your dev environment. If any of them should have different values per environment (e.g., feature flags, login modes, AzureAD tenant settings), promote them to environment variables — they'll be tracked in the solution and injected per stage at deploy time. Leave others as plain site settings."
Authentication/Registration/LocalLoginEnabled = true)For each setting the user selects to promote:
Generate the canonical schema name with ${CLAUDE_PLUGIN_ROOT}/scripts/lib/generate-env-var-schema-name.js so it matches what configure-env-variables and deploy-pipeline will expect later:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/generate-env-var-schema-name.js" \
--publisherPrefix "{prefix}" \
--settingName "{settingName}"
Output: { schemaName, sanitized }. The helper is the single source of truth for the canonical rule ({prefix}_{sanitized(settingName)}.toLowerCase()) — do not inline it. setup-solution and configure-env-variables MUST emit identical schema names for the same logical setting; inlining the rule risks divergent outputs.
Create an environmentvariabledefinition using the resolved schema name:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-env-var-definition.js" \
--envUrl "{envUrl}" \
--token "{token}" \
--schemaName "{schemaName from step 1}" \
--displayName "{friendlyName}" \
--type 100000000
Use type 100000000 (String) for auth config settings (not Secret — these are feature flags, not credentials). Capture output as JSON; extract .definitionId and .schemaName.
Record the definitionId for inclusion in the components list (Step 5.6, ComponentType: 380).
Link the site setting to the env var using link-site-setting-to-env-var.js:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/link-site-setting-to-env-var.js" \
--envUrl "{envUrl}" \
--token "{token}" \
--siteSettingId "{settingId}" \
--definitionId "{definitionId}" \
--schemaName "{schemaName}"
Check .ok and .verified are both true.
Settings the user chose NOT to promote move from promoteToEnvVar into keepAsIs — they will be included in the solution as plain site settings.
B. authNoValue settings (auth config, no dev value):
No user decision required. These are automatically included in the solution as-is. At Step 5.5, display them in a warning box:
"The following auth settings have no value set in your dev environment. They will be added to the solution as-is. After deploying to each target environment, verify or set the correct value there."
C. credentialNeedsDecision settings (credential-style — bulk-with-override prompt):
These are credential-style site settings (ConsumerKey / ClientSecret / etc.) that need a decision before going into the solution. Shipping raw values inside the solution zip is a real exposure, so the safe path is to add the site-setting record to the solution and route the value through an environment variable per stage. Asking per credential is too much when N is large (a typical site has 20+ auth-related credentials across multiple OAuth providers), so this step uses a bulk-with-override prompt: one question covers all N credentials, with a per-credential escape hatch for granular control.
Step 5.4.C.1 — Auto-classify by name pattern.
Call autoClassifyCredential(name) from ${CLAUDE_PLUGIN_ROOT}/scripts/lib/classify-site-settings.js for each setting. The helper applies these regexes in order (the single source of truth — do not duplicate them here):
| Default | Matcher in helper | When it fires |
|---|---|---|
| Secret env var (type: 100000005) | CREDENTIAL_SECRET_REGEX (Secret\|Password\|ApiKey\|AppKey) | Names with these substrings — *ClientSecret, *AppSecret, *Password, *ApiKey, *AppKey |
| String env var (type: 100000000) | CREDENTIAL_STRING_REGEX (Id\|ConsumerKey) AND not Secret | Names like *ClientId, *ConsumerKey, *TenantId, *AppId |
| Secret env var (fallback) | (helper's defensive default when neither matches) | Anything else — defensive: credential names are sensitive by default |
The helper returns { default: 'secret' | 'string', reason } for each setting. Group the results into AUTO_CLASSIFY = { secrets: [...], strings: [...] } and show the user a one-line summary: "Auto-classified {N} credential-style settings: {S} as Secret env vars (Key Vault per stage), {T} as String env vars (plain text per stage)."
Step 5.4.C.2 — Bulk prompt.
<!-- gate: setup-solution:5.4c.credentials | category=consent | cancel-leaves=nothing -->🚦 Gate (consent · setup-solution:5.4c.credentials): Bulk credential handling decision — Secret env var (Key Vault per stage), String env var (plain per stage), or skip. Per-credential choice. Determines whether secret values ship in the solution zip.
Ask one AskUserQuestion covering all N credentials:
"{N} credential-style site settings detected (
{firstFew.join(', ')}{N>3?', ...':''}). How should I handle them?Shipping their values inside the solution zip is a real exposure, so the recommended approaches add the site-setting record to the solution and route the value through an environment variable per stage. The actual secret value never ships in the zip — it's set per-environment in
deploymentsettingsjson."
Options:
Branching logic:
AUTO_CLASSIFY.secrets, run the env-var-creation steps below with --type 100000005. For each in AUTO_CLASSIFY.strings, run with --type 100000000. No additional prompts.--type 100000005.--type 100000000.userOptedOutOfSolution bucket. Surface in Step 5.5: "The following credential-style settings were skipped at user request and are NOT in the solution. Configure them manually in each target environment after deployment: {names}."AskUserQuestion (Secret env var / String env var / Skip). The auto-classification informs the per-prompt default but the user can override.Step 5.4.C.3 — Env var creation (shared by Options 1, 2, 3, and 5's non-Skip selections).
For each setting routed to env-var-backed handling:
Generate the canonical schema name with ${CLAUDE_PLUGIN_ROOT}/scripts/lib/generate-env-var-schema-name.js (same helper Step 5.4.A uses — single source of truth so configure-env-variables and deploy-pipeline can reference the same schema names later):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/generate-env-var-schema-name.js" \
--publisherPrefix "{prefix}" \
--settingName "{settingName}"
Create an environmentvariabledefinition using the resolved schema name:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/create-env-var-definition.js" \
--envUrl "{envUrl}" \
--token "{token}" \
--schemaName "{schemaName from step 1}" \
--displayName "{friendlyName}" \
--type "{100000005 for Secret, 100000000 for String}"
For Secret env vars, do NOT pass --defaultValue — the dev value goes into Key Vault per stage, not into the definition. For String env vars, capture the dev value as the default.
Record the definitionId for inclusion in the components list (Step 5.6, ComponentType: 380).
Link the site setting to the env var via link-site-setting-to-env-var.js (same call as Step 5.4.A above).
The site setting itself is added to the solution alongside the env var definition — both are tracked components.
If any single env-var creation fails (token expired mid-loop, schema-name collision, etc.), surface the failure with the setting name + reason and ask the user whether to retry, skip just that setting, or abort the whole bulk operation. Do not silently drop credentials.
Backward compatibility: when reading a preloadedSettings plan generated before 2026-05-08, treat any entries in preloadedSettings.excluded as credentialNeedsDecision and run the bulk-with-override prompt above.
Separately from the OAuth-secret conversion above, other skills (notably setup-auth, add-server-logic, and configure-env-variables) may have previously created environment variable definitions that were never added to a user solution — they land in the Default solution and silently drift. This step discovers and adopts them.
Run the shared discovery helper to get the complete site inventory in one call:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-site-components.js" \
--envUrl "{envUrl}" --token "{token}" \
--siteId "{websiteRecordId}" \
--publisherPrefix "{publisherPrefix}" \
--solutionId "{solutionId}"
Parse stdout as JSON and read missing.envVars — env var definitions whose schemaname starts with the publisher prefix but are not already solutioncomponents of this solution.
For each entry, also query which solution it currently belongs to (so the user can tell Default-only orphans apart from env vars that another user solution intentionally owns):
GET {envUrl}/api/data/v9.2/solutioncomponents
?$filter=objectid eq {definitionId}&$select=_solutionid_value
Then fetch the solution's uniquename for each hit. Build per-env-var tags:
DEFAULT-ONLY — only the Default solution owns it (classic orphan from another skill).IN OTHER SOLUTION: <uniquename> — owned by a user solution; the user may intentionally want it scoped there.If at least one env var has the DEFAULT-ONLY tag, prompt via AskUserQuestion with multiSelect: true:
🚦 Gate (plan · setup-solution:5.4b.orphan-envvars): Adopt env var definitions that match the publisher prefix but aren't yet in this solution.
DEFAULT-ONLYorphans are pre-selected as Recommended; env vars already owned by another user solution are listed but not pre-selected (user opts in only if they intend to move ownership).
"We found env var definitions with your publisher prefix (
{prefix}_) that aren't in {solutionUniqueName} yet. Select the ones you want to include. Definitions only — values stay per-environment and won't travel.
{schemaName}({displayName}) — type {type}, currently in: {tag}- ...
Plus: Include all DEFAULT-ONLY orphans (Recommended) / Skip for now"
Collect selected entries into adoptedEnvVars: [{ definitionId, schemaName, displayName, type }].
If none are selected or the list is empty, adoptedEnvVars stays empty — the skill continues silently.
Why this step exists: before this check, env vars created by other skills were silently excluded from the site's solution and didn't travel to staging/prod. Surfacing them here is the cross-skill safety net required by the ALM-aware-by-default principle in
AGENTS.md.
Symmetric to 5.4b but for powerpagecomponent rows. Catches components on the site that were created by other skills or by pac pages upload-code-site without being wrapped into a user solution. Canonical examples surfaced in 2026-04-22 live validation:
invoice-checker server logic (type 35) — added via /power-pages:add-server-logic in an earlier session, never registered into the user solution.index.html (type 3) — the current SPA entry page refreshed by pac pages upload-code-site; on every rebuild a new index.html record is created but nothing auto-adds it to the user solution.Use the shared discovery helper to collect the orphan list (it already excludes Vite/Rollup bundle chunks — Home-XYZ.js, index-XYZ.css, etc. — so the prompt doesn't drown the user in hash-named noise):
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-site-components.js" \
--envUrl "{envUrl}" --token "{token}" \
--siteId "{websiteRecordId}" \
--publisherPrefix "{publisherPrefix}" \
--solutionId "{solutionId}"
From the JSON output, take missing.powerpagecomponents and partition:
name does NOT match the bundle-chunk regex ([-.][A-Za-z0-9_-]{7,14}\.(js|mjs|cjs|css)(\.map)?$). These are the ones to adopt.For each real-content orphan, also deduplicate by name: if there are multiple index.html rows and one is already in the solution (newer modifiedon), the older orphan is a stale duplicate — exclude it from the adoption prompt and log it as a stale duplicate instead. Rule: keep only the most-recent orphan per (powerpagecomponenttype, name) pair.
Also take missing.siteLanguages — these are powerpagesitelanguage records (componenttype 10428) that exist on the site but aren't in the user solution. They are NOT optional: an imported site without its language records silently fails to render post-auth because powerpagesite.content.defaultlanguage references an ID that doesn't exist in the target env. Include every entry verbatim in the orphan-adoption prompt — there is no bundle-chunk noise to filter for languages — and pre-select them as recommended.
If the real-content orphan list (or missing.siteLanguages) is non-empty, prompt via AskUserQuestion with multiSelect: true:
🚦 Gate (plan · setup-solution:5.4c.orphan-ppcs): Adopt orphan
powerpagecomponentrows (incl.powerpagesitelanguagerecords) that exist on the site but aren't in this solution. Site languages are pre-selected as Recommended because omitting them silently breaks post-auth rendering. Other ppc orphans (e.g.invoice-checkerserver logic) are pre-selected if they appear to be real content; stale build-artifact bundle chunks are filtered out upstream.
"Found {N} site components not yet in {solutionUniqueName}:
{name}(type {type} {typeLabel}) — currently in: {currentSolution}- ...
Plus: Include all orphans (Recommended) / Skip for now"
Collect selections into adoptedPpcs: [{ id, name, type, typeLabel }].
When the user selects, call AddSolutionComponent per entry with AddRequiredComponents: false and the right ComponentType (use the values resolved by discover-component-types.js in Step 5.1 — do not hardcode):
subComponentType for missing.powerpagecomponents entriessiteLanguageComponentType for missing.siteLanguages entriesDo not set DoNotIncludeSubcomponents: true — the Dataverse API rejects that flag for non-Entity root components (HTTP 400 0x80040216) and it's not needed for these unified-entity rows.
If both missing.powerpagecomponents (after filtering) and missing.siteLanguages are empty, the step runs silently.
Why this step exists: before this check, a recurring failure pattern was that
setup-solutionfinished with the user convinced everything was wrapped up, whileinvoice-checker/index.html/ similar site-linked records quietly stayed in theActivesolution and didn't travel to staging/prod. Today's live validation found 1 real orphan (invoice-checker) on SupplierInvoicePortal — adopted via AddSolutionComponent, solution bumped from v1.0.0.1 → v1.0.0.2.
This is the key decision point. Build a full manifest of everything that will be added and present it to the user before writing anything.
If custom tables were discovered, ask via AskUserQuestion with multiSelect: true before showing the final manifest:
{logicalName} ({DisplayName})Present as a structured summary:
Here is everything that will be added to solution "{solutionName}":
WEBSITE & LANGUAGE
✓ Website record: {siteName}
✓ Site language(s): English (en-US)
SITE COMPONENTS ({total} components across {K} types)
✓ Publishing States (2)
✓ Web Pages (10)
✓ Web Files (90) — compiled JS/CSS/HTML assets
✓ Page Templates (5)
✓ Web Templates (13)
✓ Content Snippets (11)
✓ Web Roles (2)
✓ Website Access (6)
✓ Table Permissions (13) — required for Web API authorization in target env
✓ Site Markers (5)
✓ Webpage Rules (2)
SITE SETTINGS (64 included)
✓ Web API settings (14): Webapi/crd50_invoice/enabled, ...
✓ Feature flags (32): CodeSite/Enabled, Search/Enabled, ...
✓ Auth config (18): Authentication/Registration/LocalLoginEnabled, ...
~ OAuth as env vars (3): ids_auth_openauth_microsoft_clientsecret, ... [ENV VAR]
✗ OAuth excluded (5): Authentication/OpenAuth/Facebook/AppSecret, ... [EXCLUDED]
CLOUD FLOWS ({N} linked via powerpagecomponent type 33)
✓ Invoice Approval Flow (workflowId: {guid}, Active)
~ Draft Flow (workflowId: {guid}, Inactive — excluded by default)
BOT CONSUMERS ({N} linked via powerpagecomponent type 27)
✓ Support Bot (botId: {guid}, Active)
DATAVERSE TABLES (schema only — no data)
✓ crd50_invoice (Invoice)
...
ENV VAR DEFINITIONS (componenttype 380)
✓ ids_auth_openauth_microsoft_clientsecret (Secret) [converted from OAuth secret]
✓ crd50_auth_openauth_microsoft_clientsecret (Secret) [ADOPTED — was in Default only]
...
Total to add: ~{N} components
For clarity, use these tags after each env var entry in the manifest:
[converted from OAuth secret] — created in Step 5.4 from a site setting[ADOPTED — was in Default only] — existed before this run; being pulled into the solution in Step 5.4b[ADOPTED ppc — was in Active only] — powerpagecomponent adopted in Step 5.4c (e.g. invoice-checker server logic, real site pages not yet registered)[ADOPTED — also in {otherSolutionName}] — existed in another user solution; being additionally added here (user explicitly opted in)If cloudFlows is non-empty, use AskUserQuestion with multiSelect: true:
{name} ({workflowId})Default: include active flows, exclude inactive ones. If a flow is already in a different solution, warn the user: "This flow is in solution X — adding it here will move it."
If botComponents is non-empty, use AskUserQuestion with multiSelect: true (same pattern).
If both are empty, skip and display (None discovered).
After presenting the manifest summary, add a free-text escape hatch:
<!-- gate: setup-solution:5.5.manifest-confirm | category=plan | cancel-leaves=partial-manifest -->"If you know of cloud flows or bots that should be in this solution but are not shown above, paste their GUIDs here (comma-separated). Leave blank to continue."
🚦 Gate (plan · setup-solution:5.5.manifest-confirm): Final manifest confirmation before any
AddSolutionComponentwrite. Covers tables, flows, bots, env vars, orphan adoption. Cancel here keeps the in-memory manifest but no Dataverse writes happen.
Ask via AskUserQuestion:
"Does this look right? You can proceed, or tell me which categories or tables to exclude."
Options: "Proceed with this selection" / "I want to change something"
Wait for explicit confirmation before Step 5.6.
Build a JSON array of all components to add, then call scripts/lib/add-components-to-solution.js to perform the bulk operation with token refresh and idempotency handling built in.
The components array should be built in this order:
{ componentId: websiteRecordId, componentType: websiteComponentType, addRequired: true, description: "Website: {siteName}" }siteLanguageComponentType (NOT auto-included by AddRequiredComponents)subComponentType
{ componentType: 380 }. Include:
adoptedEnvVars from Step 5.4b (orphans the user chose to include){ componentType: 1, componentId: MetadataId }{ componentId: workflowId, componentType: workflowComponentType } (uses runtime-discovered type){ componentId: botId, componentType: botComponentType } (uses runtime-discovered type){ componentId: connectionReferenceId, componentType: connectionReferenceComponentType, addRequired: false }. Skip if connectionReferences = []. Use the runtime-resolved connectionReferenceComponentType — do not hardcode (observed values across tenants include 10137 and 10160; the value is env-specific). Without these entries, the solution exports cleanly but the target import fails with a MissingDependency error — deploy-pipeline Phase 6.6.1 will surface it as a "missing connection reference" validation failure.{ componentId: ppc.id, componentType: subComponentType, addRequired: false }. Use the subComponentType value resolved by discover-component-types.js in Step 5.1 — do not hardcode. Do not set DoNotIncludeSubcomponents: true — Dataverse rejects that flag on non-Entity components (HTTP 400 0x80040216).Single-solution mode (MULTI_SOLUTION_MODE = false): write the array to a temp file (e.g., C:/Users/{user}/AppData/Local/Temp/components-to-add.json), then run:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/add-components-to-solution.js" \
--envUrl "{envUrl}" \
--componentsFile "C:/Users/{user}/AppData/Local/Temp/components-to-add.json" \
--solutionUniqueName "{solutionUniqueName}"
Multi-solution mode (MULTI_SOLUTION_MODE = true): partition the unified component list across PROPOSED_SOLUTIONS based on each solution's componentTypes (and tableLogicalNames for Strategy 3), then run add-components-to-solution.js once per solution. The per-solution loop SHOULD run serially across solutions (each helper call already batches + refreshes tokens internally; running solutions in parallel multiplies the token-refresh load with no real wall-clock win since the bottleneck is per-component Dataverse calls inside each batch). For each entry in PROPOSED_SOLUTIONS (skip isFutureBuffer: true):
componentTypes array. The mapping from numeric componentType → name is the same one discover-component-types.js and discover-site-components.js use (PPC_TYPE_LABELS). Tables route to the solution whose tableLogicalNames includes the table's logical name (Strategy 3) or to whichever solution claims 'Table' (Strategies 1 and 2).C:/Users/{user}/AppData/Local/Temp/components-{uniqueName}.json), then invoke the helper with all three required flags:
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/add-components-to-solution.js" \
--envUrl "{envUrl}" \
--componentsFile "C:/Users/{user}/AppData/Local/Temp/components-{uniqueName}.json" \
--solutionUniqueName "{proposedSolutions[i].uniqueName}"
Capture the JSON summary keyed by uniqueName, delete the temp file. All three flags are required — omitting --envUrl or --componentsFile (only passing --solutionUniqueName) causes the helper to exit 1 with --envUrl is required / --componentsFile is required. Both must be passed per iteration, even though --envUrl is the same across all iterations of the loop.componentTypes, surface a per-component warning and STOP — the partitioning lost a component. This usually means the split plan dropped a type (regression in compute-split-plan.js); the user needs to re-plan rather than silently leaking components into Default.Use SOLUTIONS_BY_NAME from Phase 4 to resolve each uniqueName → solutionId if the helper's resolution by name isn't sufficient.
The script handles token refresh every 20 calls, treats "already in solution" as success, and outputs a JSON summary { total, success, skipped, failed, failures }. Delete the temp file(s) after completion.
GET {envUrl}/api/data/v9.2/solutioncomponents?$filter=_solutionid_value eq '{solutionId}'&$select=objectid,componenttypewebsiteComponentType) is present2b. Capture the post-setup env var snapshot for the rendered ALM plan. Ensure docs/alm/ exists, then run the discovery helper and write its output to a sidecar marker file (docs/alm/last-env-vars.json). The plan-refresh helper (Phase 7's self-refresh) ingests this sidecar into planData.envVars so the rendered plan's Env Variables tab shows the definitions setup-solution just created/adopted (without it the tab stays empty even after Phase 5.4 / 5.4.C / 5.4b created definitions):
node -e "require('fs').mkdirSync('docs/alm',{recursive:true})"
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/discover-env-var-definitions.js" \
--envUrl "{envUrl}" \
--publisherPrefix "{publisherPrefix}" \
--websiteRecordId "{websiteRecordId}" \
--token "{token}" > docs/alm/last-env-vars.json.tmp \
&& mv docs/alm/last-env-vars.json.tmp docs/alm/last-env-vars.json
The tmp-file write pattern preserves a prior good docs/alm/last-env-vars.json on a transient discovery failure (parallel to the docs/alm/alm-size-estimate.json pattern in plan-alm Phase 1). If the helper exits non-zero, log the stderr and continue — the existing sidecar (or absence of one) is acceptable; the refresh just won't update env vars this run.
The sidecar's shape mirrors what discover-env-var-definitions.js already returns: { envVars: [{ schemaName, type, defaultValue, siteSetting }], count }. Don't transform — the renderer reads these fields directly.
Write .solution-manifest.json to project root (alongside powerpages.config.json):
${CLAUDE_PLUGIN_ROOT}/references/solution-api-patterns.md Section 7cloudFlows array: [{ "workflowId": "...", "name": "...", "status": "active|inactive" }]botComponents array: [{ "botId": "...", "name": "..." }][] = tracked but none selected)In MULTI_SOLUTION_MODE, write manifest v2 with a solutions[] array:
{
"schemaVersion": 2,
"publisher": { "publisherId": "...", "uniqueName": "...", "friendlyName": "...", "customizationPrefix": "..." },
"solutions": [
{
"uniqueName": "IdeaSphere_Core",
"solutionId": "...",
"version": "1.0.0.0",
"order": 1,
"componentTypes": ["Table", "Site Setting", ...],
"components": [ { "componentId": "...", "componentType": 1, "description": "..." } ]
},
{
"uniqueName": "IdeaSphere_WebAssets",
"solutionId": "...",
"version": "1.0.0.0",
"order": 2,
"componentTypes": ["Web File"],
"components": [ ... ]
}
],
"splitStrategy": "strategy-1-layer",
"assetAdvisory": [ /* pass-through from plan context */ ]
}
v1 single-solution manifest stays backward compatible. Readers (export-solution, import-solution, setup-pipeline, deploy-pipeline) check schemaVersion:
schemaVersion absent or 1 → treat as single-solution (existing behavior).schemaVersion: 2 → iterate solutions[] in order.Commit: git add .solution-manifest.json && git commit -m "Add solution manifest for ALM"
Display a summary table:
| Item | Value |
|---|---|
| Publisher | {friendlyName} ({uniqueName}, prefix: {prefix}) |
| Solution | {friendlyName} ({uniqueName}, v{version}) |
| Solution ID | {solutionId} |
| Components added | N |
| Env var definitions added | N (if any OAuth secrets converted) |
| Manifest written | .solution-manifest.json |
If any auth settings were promoted to env vars, confirm that each site setting was automatically linked. Show a brief confirmation:
Auth settings promoted to environment variables:
✓ Authentication/Registration/LocalLoginEnabled → ids_authentication_registration_localloginenabled
✓ Authentication/Registration/AzureADLoginEnabled → ids_authentication_registration_azureadloginenabled
Note: Per-environment values must still be set via configure-env-variables or the Power Pages Management UI.
If any authNoValue settings were included, show a reminder:
Auth settings included without a dev value (configure in each target env after deploy):
⚠ Authentication/OpenAuth/Facebook/AppId
⚠ Authentication/Registration/LoginButtonAuthenticationType
<!-- gate: setup-solution:7.next-step | category=plan | cancel-leaves=nothing -->
🚦 Gate (plan · setup-solution:7.next-step): Routing choice for downstream deployment skill — PP Pipelines, manual export/import, or defer. All Dataverse writes for this skill are already complete; this gate selects what runs next.
Ask what the user wants to do next via AskUserQuestion:
"How would you like to deploy this solution to other environments?"
Options:
If the user selects option 1, immediately invoke /power-pages:setup-pipeline.
If the user selects option 2, immediately invoke /power-pages:export-solution.
If the user selects option 3, show:
/power-pages:setup-pipeline for automated staged deployments/power-pages:export-solution to export a zip for manual import/power-pages:configure-env-variables if environment-specific values need to be set per stageWhen the live site grows beyond what's in this solution — server logic from
add-server-logic, cloud flows fromadd-cloud-flow, env vars fromsetup-authorconfigure-env-variables, new tables fromsetup-datamodel, or new web roles — re-run/power-pages:setup-solution. The skill auto-detects sync mode when.solution-manifest.jsonexists in the project root, runs the discovery pass, diffs the live site against the solution, bumps the version, and adds only the missing components. No need for a separate "add to solution" workflow.
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 "SetupSolution".
node "${CLAUDE_PLUGIN_ROOT}/scripts/lib/refresh-alm-plan-data.js" \
--projectRoot "." \
--phase setup-solution \
--render
The helper resets planData.plannedEnvVarCount to 0 (the planned env vars have either been created or skipped at the user's request) and re-renders docs/alm-plan.html so the Overview stat card and Env Variables tab reflect post-setup state. 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.
AddSolutionComponent returns "already in solution": treat as success (idempotent)| Task subject | activeForm | Description | |---|---|---| | Verify prerequisites | Verifying prerequisites | Confirm PAC CLI auth, acquire Azure CLI token, verify API access, locate powerpages.config.json | | Gather solution configuration | Gathering solution configuration | Collect publisher name, prefix, solution name, version from user — confirm irreversible choices | | Check existing publishers and solutions | Checking existing state | Query Dataverse for existing publisher and solution to avoid duplicate creation | | Create publisher and solution | Creating publisher and solution | POST publisher and solution to Dataverse OData API, capture IDs | | Add site components to solution | Adding site components | Discover website/language/powerpagecomponents/tables/cloud flows (type 33)/bot consumers (type 27) via runtime field introspection; split site settings by category; present full manifest including CLOUD FLOWS and BOT CONSUMERS sections with active/inactive status; get user confirmation; call add-components-to-solution.js for website, site language(s), all confirmed components, tables (ComponentType=1), confirmed cloud flows, and confirmed bot components | | Verify and write manifest | Verifying solution and writing manifest | Confirm components in solution, write .solution-manifest.json, commit | | Present summary | Presenting summary | Show solution details, component count, and next steps |
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.