| name | semantic-model-authoring |
| description | Develops and manages Power BI semantic models across Desktop, PBIP projects, and Fabric Service. Handles: (1) creating new models (Import, DirectQuery, Direct Lake), (2) editing existing models (e.g. measures, tables, columns, relationships), (3) deploying models to Fabric workspaces, (4) working with PBIP project files, (5) refreshing semantic models, (6) configuring data sources and permissions, (7) DAX performance optimization. Supports both Power BI Desktop and Fabric Service development workflows. For read-only DAX queries, use `semantic-model-consumption`. Does NOT handle report layout/visual authoring, workspace administration, or RLS/OLS role membership management. Triggers: "create semantic model", "edit semantic model", "add a DAX measure to semantic model", "refresh semantic model", "set semantic model permissions", "Prepare semantic model for AI/Copilot".
|
Update Check — ONCE PER SESSION (mandatory)
The first time this skill is used in a session, run the check-updates skill before proceeding.
- GitHub Copilot CLI / VS Code: invoke the
check-updates skill.
- Claude Code / Cowork / Cursor / Windsurf / Codex: compare local vs remote package.json version.
- Skip if the check was already performed earlier in this session.
CRITICAL NOTES
- To find the workspace details (including its ID) from workspace name: list all workspaces and, then, use JMESPath filtering
- To find the item details (including its ID) from workspace ID, item type, and item name: list all items of that type in that workspace and, then, use JMESPath filtering
- Always consider the Tool selection priority when choosing which tool to use for each operation. Do not default to TMDL edits or
az rest if MCP is available and connected to the target model.
Power BI Semantic Model Authoring — CLI Skill
Workflow Selector
Use this decision tree to route to the correct workflow based on user intent:
Table of Contents
Load these references on demand when a workflow step requires them. Do not load all at once.
| Topic | Reference | When to load |
|---|
| Modeling Best Practices | modeling-guidelines.md | Before creating or editing any model |
| Naming Conventions | naming-conventions.md | When naming or renaming tables, columns, measures |
| Direct Lake Modeling | direct-lake-guidelines.md | When model connects to OneLake |
| TMDL Editing | tmdl-guidelines.md | Before generating or editing any TMDL file |
| PBIP Projects | pbip.md | When working with PBIP folders |
| DAX Language | dax-guidelines.md | When writing or reviewing any DAX code |
| DAX Queries & Metadata Discovery | semantic-model-consumption | Read-only DAX queries; use for post-creation validation |
| DAX Performance Decision Guide | dax-perf-decision-guide.md | Start here when optimizing DAX |
| DAX Performance Pattern Catalog | dax-perf-patterns.md | Load on demand after the decision guide identifies candidate patterns |
| Semantic Model AI Readiness | semantic-model-ai-readiness.md | When preparing a model for Copilot or Data Agents |
| Semantic Model REST API | semantic-model-rest-api.md | When using az rest for TMDL CRUD, refresh, parameters, permissions, or property retrieval |
| Connection Binding | connection-binding.md | When binding/unbinding a semantic model to a Fabric data connection (gateway, cloud, VNet, automatic, none) |
| Finding Workspaces/Items | COMMON-CLI.md | When resolving workspace/item IDs |
| Fabric Control-Plane API | COMMON-CLI.md | When using az rest patterns, LRO, pagination |
| Authentication | COMMON-CLI.md | When authenticating with az login |
| Authentication & Token Acquisition | COMMON-CORE.md § Authentication & Token Acquisition | Wrong audience = 401; read before any auth issue |
| Core Control-Plane REST APIs | COMMON-CORE.md § Core Control-Plane REST APIs | Includes pagination, LRO polling, and rate-limiting patterns |
| Definition Envelope | ITEM-DEFINITIONS-CORE.md | When building TMDL definition payloads |
| Examples | Examples | Reference end-to-end walkthroughs. |
Tool Selection Priority
Priority order (highest first):
-
Tier 1 — powerbi-modeling-mcp MCP is registered -> Use MCP for authoring (new or edit) operations against the model from any source: Power BI Desktop, Fabric workspace, or local PBIP folder. MCP is the most reliable and full-featured way to edit semantic models, with immediate effect on the live model and no risk of TMDL desync.
Important: In case of dynamic search tools is available (e.g. tool_search_tool_regex) search for an available MCP server matching the pattern powerbi-modeling-mcp.
This includes BOTH writes AND reads/inspection.
- To inspect or verify changes -> use the corresponding MCP operations (List / Get).
- Anti-pattern: opening,
view-ing, glob-ing or otherwise reading TMDL files (*.tmdl) while MCP is connected. The MCP-loaded model is the source of truth - the on-disk TMDL is stale. The only exceptions is when the user explicitly asks to work with the TMDL files.
-
Tier 2 — MCP not registered + PBIP folder or Fabric workspace -> Edit TMDL files directly. Load tmdl-guidelines.md and pbip.md. When the source is a Fabric workspace, use az rest to round-trip the TMDL (load semantic-model-rest-api.md): getDefinition -> edit TMDL locally -> updateDefinition.
Fallback — none of the above available (e.g., Power BI Desktop with no PBIP and no MCP) -> STOP. The agent cannot author the model in this configuration. Instruct the user to either (a) install and register the powerbi-modeling-mcp MCP server, or (b) save the PBIX as a PBIP project, then restart the workflow.
All workflows below are tool-agnostic. Workflow steps describe the intent (connect, create, edit, save, deploy, refresh). The tool used to perform each step is determined here. Always select the highest-priority tool available for the current environment; do not mix tools when a higher-priority option works. Some workflows OVERRIDE this default priority, always check the workflow's own tool-selection rules before defaulting to Tier 1.
Connecting to a Semantic Model
A semantic model can live in three locations. Resolve the connection per Tool Selection Priority:
- Power BI Desktop: Locate the running Power BI Desktop instance and connect to its local model.
- Fabric workspace: First, find the workspace and semantic model using the Finding Workspaces and Items pattern: list workspaces to resolve the workspace ID by name, then list items of type
SemanticModel in that workspace to resolve the model ID by name. Then connect to the model (live) or export its TMDL definition for local editing.
- PBIP project: Connect to the
[Name].SemanticModel/definition folder. Load pbip.md to understand the PBIP folder structure - only load the [Name].SemanticModel/definition folder that includes the TMDL code.
Saving Changes to a Semantic Model
How changes are persisted depends on where the model lives and which tool tier (per Tool Selection Priority) is in use:
Live connection (Tier 1 - MCP against Desktop or Fabric workspace):
- Changes are applied immediately as each operation executes against the live model. No explicit save step is needed.
- PBIP project (live via MCP): Serialize the model back to the
[Name].SemanticModel/definition folder at the end of the session. If the PBIP folder does not exist yet, follow Export to PBIP to create the full structure first.
Local TMDL editing (Tier 2 - direct file edits or az rest round-trip):
- PBIP project: Changes are already written to the TMDL files during editing. No additional save step is needed.
- Fabric workspace: Changes were made to local TMDL files exported from the service. Re-deploy the model (load semantic-model-rest-api.md for the
updateDefinition flow) to push changes back to the workspace.
Workflow: Create new Semantic Model
When this applies: User asks to create a new semantic model from scratch.
Steps:
- Gather requirements - interview the user until both reach a shared understanding of: purpose of the model, data source connection details and schemas, and key business entities/facts. If data source information is not available, STOP and use
ask_user. Do not guess or fabricate.
- Determine storage mode - data source is Fabric OneLake -> Direct Lake; otherwise default to Import. Only use DirectQuery when the user explicitly asks for it.
- Design star schema - identify fact and dimension tables and relationship keys.
- If fact table includes date field(s), create a separate date dimension table and link it to the fact with a relationship. If not explicitly requested, use PowerQuery/M partition instead of DAX calculated table.
- Load applicable guidelines - modeling-guidelines.md always; direct-lake-guidelines.md if Direct Lake.
- Build - create an empty database (compatibility level 1702+), then for each table follow the execution order from Modify an Existing Model (partitions -> columns -> relationships -> measures). Storage-mode specifics:
- Import / DirectQuery - create M parameters for the data source (
Server, Database, ...) and reference them in partition M code; ensure proper dataType and sourceColumn mapping on columns.
- Direct Lake - create a shared named expression for the Direct Lake connection using the
AzureStorage.DataLake connector; use EntityPartitionSource with directLake mode mapped to the lakehouse table columns.
- Deploy or save - Fabric workspace available -> Deploy to Fabric; otherwise -> Export to PBIP. See Saving Changes to a Semantic Model.
- Validate - run Validation Checklist.
Workflow: Modify an Existing Model
When this applies: User asks to add/edit/remove measures, tables, columns, relationships, write DAX code, refactor with UDFs, or edit TMDL directly.
Steps:
- Connect & discover - per Connecting to a Semantic Model. List tables, relationships, existing measures, and identify storage mode (it dictates which guidelines apply).
- Load applicable guidelines - modeling-guidelines.md always; direct-lake-guidelines.md if Direct Lake; tmdl-guidelines.md when editing TMDL directly; dax-guidelines.md for any DAX changes (includes UDF refactoring).
- Plan changes - identify exactly what to add, modify, or remove. Check for naming conflicts and duplicates.
- Execute in correct order:
- Adding tables - partitions -> columns -> relationships -> measures.
- Adding relationships - ensure key columns exist on both sides with matching data types;
- Adding measures - verify referenced columns/tables exist;
- Save & validate - per Saving Changes to a Semantic Model and Validation Checklist.
Workflow: Optimize DAX Performance
When this applies: User asks to improve DAX query performance, diagnose slow measures, or optimize calculations.
Hard requirement: Requires a trace-capable client (MCP preferred))
Load dax-perf-decision-guide.md first and follow the framework defined there. Load dax-perf-patterns.md only when applying candidate optimization patterns. The framework includes:
- Tier model for categorizing optimization effort
- Trace diagnostics to identify bottlenecks
- Pattern catalog with candidate optimization techniques to test and validate
Workflow: Analyze Best Practices
When this applies: User asks to review, audit, or analyze a semantic model against best practices.
Steps:
- Connect & inventory - per Connecting to a Semantic Model. Capture all tables, columns, relationships, measures, and storage mode.
- Load applicable guidelines - modeling-guidelines.md always; direct-lake-guidelines.md if Direct Lake; naming-conventions.md when assessing naming; dax-guidelines.md when assessing DAX.
- Evaluate - compare the model against the loaded guidelines (star schema, naming, relationship cardinality and cross-filter, explicit measures with
formatString, column data types and sourceColumn, hidden FK columns, calculated-column-vs-measure choices, Direct Lake constraints, etc.).
- Present findings grouped by severity (critical, recommended, optional). For each item state the rule violated and the proposed fix. Wait for user approval.
- Apply approved fixes via Modify an Existing Model.
- Save & validate - per Saving Changes to a Semantic Model and Validation Checklist.
Workflow: Semantic Model AI Readiness
When this applies: User asks to make a semantic model ready for Microsoft Fabric Copilot, a Power BI Data Agent, or any conversational BI experience. Triggers include "Copilot readiness", "AI readiness", "prep for AI", "prepare model for Copilot".
Load semantic-model-ai-readiness.md before starting.
Steps:
- Confirm scope & gather context - via
ask_user, confirm consumption mode (reports only / conversational BI / both) and model stability per the When to Apply section. Collect business context (process, key metrics, common natural-language questions, vocabulary). Do not invent.
- Connect & inventory - per Connecting to a Semantic Model. Capture model contents and the source location (PBIP / Fabric workspace / Desktop-only).
- Evaluate & route - walk the Readiness Checklist in order; for each gap, classify the fix per Editing Capability by Source and Tier (MCP-editable TOM metadata vs PBIP-only artifact).
- Present findings grouped by severity, each tagged with routing (agent-applicable vs user-action-required). Wait for approval.
- Apply approved changes - TOM metadata via Modify an Existing Model; PBIP-only artifacts via direct file edits or Fabric
getDefinition/updateDefinition round-trip; Desktop-only PBIX -> instruct user.
- Save, validate, recommend live testing - per Saving Changes to a Semantic Model and Validation Checklist; advise the user to test representative natural-language prompts in Copilot or the Data Agent and iterate.
Workflow: Export to PBIP
When this applies: User asks to export or save a semantic model to a PBIP project folder, or there is no Fabric workspace available to deploy to (e.g., after building a model in-memory).
Key fact: Exporting a model only produces the TMDL definition files. It does NOT create the surrounding PBIP folder structure (Report folder, definition.pbism, definition.pbir, .pbip entry point). The agent must scaffold these before exporting, otherwise the result cannot be opened in Power BI Desktop.
Load pbip.md before starting and follow the PBIP folder structure defined there.
Steps:
- Determine target - via
ask_user, get the target folder path and the semantic model name. If only a folder is provided, use the model's database name as the semantic model folder name.
- Scaffold the PBIP structure - per pbip.md, ensure
<Name>.SemanticModel/ (with definition/ and definition.pbism), <Name>.Report/ (with definition/ and definition.pbir using a byPath reference), and <Name>.pbip exist. Create any missing piece.
- Export TMDL into
<Name>.SemanticModel/definition/, per Tool Selection Priority:
- Tier 1 (MCP) - use the MCP export/save operation against the live model.
- Tier 2 (Fabric workspace, no MCP) - call
getDefinition (load semantic-model-rest-api.md) and write the returned parts.
- Local TMDL files already on disk - copy or move them into the
definition/ folder.
- Validate - confirm the
definition/ folder contains at minimum model.tmdl and table .tmdl files; confirm definition.pbism, <Name>.Report/definition.pbir (with correct byPath to ../<Name>.SemanticModel), and <Name>.pbip exist and reference each other correctly.
Workflow: Deploy to Fabric
When this applies: User asks to deploy or publish a semantic model to a Fabric workspace.
Hard rule — this workflow OVERRIDES the default Tool Selection Priority. Do not default to MCP just because it is available. The deployment path is determined by the source of the model, not by which tools are connected. If the source is PBIP/TMDL files on disk, you MUST use the Fabric REST API even when an MCP session is active.
Decision tree (pick exactly one — top-down, first match wins):
- Are there PBIP / TMDL files on disk that need to be deployed?
-> YES — use Fabric REST API. Call
az rest with createItemWithDefinition (new model) or updateDefinition (existing model). Load semantic-model-rest-api.md.
- Rationale: deploying TMDL files directly via the Fabric API is more reliable, faster, and avoids unnecessarily loading the model into MCP only to push it back out.
- Do NOT open the PBIP in MCP first and then deploy via MCP. That is an explicit anti-pattern for this workflow.
- Is the model already loaded in a live MCP session (e.g., just built in-memory, or currently being edited via MCP) with no PBIP/TMDL files involved?
-> Use MCP
database_operations Deploy with the target workspace and semantic model name.
- Is the model live in Power BI Desktop with no PBIP saved?
-> Use MCP Deploy if MCP is connected to Desktop. If MCP is not available, instruct the user to save as PBIP first, then restart this workflow at step 1.
Verify deployment succeeded by listing workspace items of type SemanticModel.
Workflow: Refresh Semantic Model
When this applies: User asks to refresh data in a semantic model.
Refresh is only possible when working against a live model in Desktop or Fabric Service. If working with local TMDL files, deploy the model first.
Trigger a refresh per Tool Selection Priority:
- Power BI Desktop: Tier 1 (MCP) only — use the MCP Refresh operation.
- Fabric Service: Tier 1 (MCP Refresh operation) or fallback to the Power BI Enhanced Refresh API (load semantic-model-rest-api.md).
If the refresh fails with a credential error, stop immediately and instruct the user to configure the data source connections manually in Power BI Service. Do not attempt to retry or work around credential errors programmatically.
Workflow: Manage Semantic Model in Fabric
When this applies: User asks to configure data sources, update parameters, or manage permissions for a semantic model in Fabric Service.
Hard rule — this workflow OVERRIDES the default Tool Selection Priority. Do not default to MCP (Tier 1) prefer using az rest and REST APIs.
Data Sources & Parameters
Get/update data sources and parameters via Power BI REST API. Load semantic-model-rest-api.md.
Permissions
List/grant/update dataset user permissions via Power BI REST API. Load semantic-model-rest-api.md.
Connection Binding
Load connection-binding.md and follow it. The reference covers prerequisites, the bindConnection endpoint, the discover -> match -> bind -> validate steps, all connectivityType values, the unbind pattern, and troubleshooting.
Key rules (full details in the reference):
- Use the Fabric Bind Semantic Model Connection REST API (supersedes the legacy Power BI
BindToGateway).
- One bind request per data source reference - the API does not support bulk binding.
- Discover the model's data source references via
List Item Connections, then match connectionDetails against List Connections to find the target id. Create a connection first if no match exists.
- Validate by re-listing item connections and triggering a refresh.
Validation Checklist
Run after any model creation or modification:
Always (works with PBIP, Desktop, and Fabric Service):
- Check the PBIP structure - if the model is sourced from a PBIP folder, ensure the folder structure and files are correct (see pbip.md).
- Verify against modeling guidelines - re-check every change against modeling-guidelines.md (and direct-lake-guidelines.md for Direct Lake models).
Only when connected to an Analysis Services database (Power BI Desktop or Fabric Service):
- Test new measures - for each new measure, run a simple DAX query to validate it returns expected results (e.g.,
EVALUATE { [Measure Name] }). Skip this step when working with local TMDL/PBIP files only.
- Test table refresh - when new tables were created, trigger a refresh to verify that partitions, data source expressions, and column mappings are correct. A failed refresh typically indicates mismatched
sourceColumn names, invalid M expressions, or incorrect Direct Lake entity references. Skip this step when working with local TMDL/PBIP files only.
If any check fails, fix the issue and re-run validation.
Must/Prefer/Avoid
MUST
- Understand the data source schema before starting - analyze source tables, columns, and data types before designing or modifying the model.
- Follow modeling guidelines - load modeling-guidelines.md before creating or editing any model; apply star schema design, naming conventions, and column/measure rules
- Follow Tool Selection Priority - always pick the highest-priority tool tier available for the current environment; do not mix tiers when a higher-priority option works
PREFER
- Star schema over snowflake or flat tables - denormalized dimensions with single-column relationship keys
- Consistency with existing model patterns - when editing an existing model, match its naming conventions and structure rather than imposing new ones
- TMDL format over TMSL - text-based, diff-friendly, preferred for Fabric
- Cross-reference
semantic-model-consumption for post-creation validation — run DAX queries to verify measures, relationships, and data.
AVOID
- Hardcoded workspace/item IDs - resolve dynamically via API
- Reading TMDL files when MCP is connected -
view/glob on *.tmdl while a Tier 1 MCP session is live is an anti-pattern (see Tool Selection Priority).
- Hand-authoring TMDL files when MCP is registered - using
create/edit/file-write tools to scaffold model.tmdl, database.tmdl, relationships.tmdl, or tables/*.tmdl is a Tier 1 anti-pattern, including for brand-new models. Build via MCP operations (database_operations, table_operations, column_operations, relationship_operations, measure_operations, partition_operations, named_expression_operations) and serialize via ExportToTmdlFolder only at the end.
DENY
- Manage RLS/OLS role membership - do not propose REST calls,
az rest URLs, MCP operations, or TMDL changes to add/remove users or groups from a security role. Refuse the request as out-of-scope here and redirect the user to the Power BI portal.
Examples
Scope note — examples use az rest for discovery to resolve ID's and discover Fabric metadata (see COMMON-CLI.md § Finding Workspaces and Items). Authoring of the semantic model definition is routed through Tool Selection Priority: Tier 1 MCP powerbi-modeling-mcp when available, Tier 2 TMDL editing via getDefinition / updateDefinition otherwise.
Example 1: Modify an Existing Semantic Model
Prompt: "Create base measures for all aggregable columns in the semantic model Sales in workspace Marketing."
Agent response - follows Workflow: Modify an Existing Model.
- Discover IDs via
az rest.
- Connect to the model per Connecting to a Semantic Model and Tool Selection Priority. With
powerbi-modeling-mcp registered, connect MCP directly to the Fabric workspace model (Tier 1). Otherwise fall back to Tier 2 (getDefinition -> edit TMDL locally).
- Inspect & plan - list tables and columns via the active tool tier; identify aggregable columns (numeric, not foreign keys, not hidden surrogate IDs) and decide on
SUM / AVERAGE / MIN / MAX per column following modeling-guidelines.md and naming-conventions.md. Load dax-guidelines.md before writing DAX.
- Add measures per Workflow: Modify an Existing Model.
- Tier 1 (MCP): call tool create for each new measure with
expression, formatString, and target table. Do not hand-author TMDL.
- Tier 2 (no MCP): edit the table's
.tmdl file directly and round-trip via updateDefinition REST API.
- Save & validate per Saving Changes to a Semantic Model.
Example 2: Create a New Semantic Model from a Fabric Lakehouse
Prompt: "Create a new Power BI semantic model in workspace Marketing, using the SalesLakehouse in the same workspace as the data source."
Agent response - follows Workflow: Create new Semantic Model.
- Discover workspace + lakehouse IDs via
az rest.
- Discover lakehouse schema - list tables and columns via the Lakehouse OneLake APIs or the SQL Endpoint of the lakehouse. Capture table names, column names, and data types - these drive semantic model table, column definitions and partition metadata.
- Select tool tier per Tool Selection Priority: prefer Tier 1 (
powerbi-modeling-mcp) for the full build. Tier 2 (PBIP/TMDL scaffolding then deploy) only if MCP is not registered.
- Build the model following Workflow: Create new Semantic Model - data source is OneLake so the storage mode is Direct Lake (load direct-lake-guidelines.md and modeling-guidelines.md). Design a star schema from the lakehouse tables, then:
- Create the empty database.
- Create one shared named expression for the lakehouse connection.
- For each table: add
EntityPartitionSource (mode directLake) -> columns mapped to lakehouse columns -> relationships -> measures.
- Deploy per Workflow: Deploy to Fabric:
- Validate per the Validation Checklist - confirm the model appears in the workspace and trigger an initial refresh to verify Direct Lake column mappings resolve.
Example 3: Bind a Semantic Model to a Fabric Connection
Prompt: "Bind semantic model Sales in workspace Marketing to the sql-prod connection."
Agent response - follows Workflow: Manage Semantic Model in Fabric § Connection Binding. No MCP - this workflow overrides Tool Selection Priority and is REST-only.
-
Discover workspace + model IDs via az rest.
-
Discover the target connection in Fabric by matching the server name:
SERVER="sql-prod"
CONN_ID=$(az rest --method get --resource "https://api.fabric.microsoft.com" \
--url "https://api.fabric.microsoft.com/v1/connections" \
--query "value[?connectionDetails.path | contains(@, '$SERVER')] | [0].id" -o tsv)
If no connection matches, stop and instruct the user to create one first per connection-binding.md.
-
Execute the bind per Workflow: Manage Semantic Model in Fabric § Connection Binding - load connection-binding.md and follow the discover -> match -> bind -> validate flow:
- List the model's data source references (
List Item Connections).
- For each reference whose
connectionDetails matches $CONN_ID, call the Fabric bindConnection endpoint one request per data source reference (no bulk binding).
-
Validate - re-list item connections to confirm the binding, prompt the user to trigger a refresh per Workflow: Refresh Semantic Model. Credential errors -> stop and direct the user to the Service portal (per TROUBLESHOOTING).
TROUBLESHOOTING
| Symptom | Fix |
|---|
| MCP connection failure | Fall back to TMDL editing (see Tool Selection Priority). Inform the user about the fallback. |
| TMDL validation errors | Read error details, fix syntax, re-validate. Load tmdl-guidelines.md. |
403 Forbidden / identity None | User needs Contributor+ role - stop immediately. Do not retry. |
401 Unauthorized | Wrong --resource audience or missing permissions to the item. Check semantic-model-rest-api.md. |
202 Accepted but no result | Poll LRO to completion. |
| Parts missing after updateDefinition | Must include ALL parts - modified + unmodified. |
| Refresh credential error | Direct user to configure in Service portal. Do not retry. |
| DAX errors in measures | Check column/table name references (case-sensitive). Verify referenced objects exist. |
| Deployment failure | Check workspace permissions, model compatibility level, and Direct Lake expression source references. |
| Missing data source | Verify M parameters or named expressions are correctly defined. |