├── Export2Dataflow.ps1 ├── Export2Dataflow_Logo.png ├── Export2Dataflow_Logo.svg ├── LICENSE ├── Publish2Dataflow_Logo.png ├── Publish2Dataflow_Logo.svg ├── README.md ├── export2dataflow.pbitool.json └── publish2dataflow.pbitool.json /Export2Dataflow.ps1: -------------------------------------------------------------------------------- 1 | <# 2 | .SYNOPSIS 3 | This script exports the Power Query to a Dataflow JSON file or publishes it to the Power BI service. 4 | .DESCRIPTION 5 | This Powershell script can be called as an External Tool from Power BI Desktop, which then extracts the Power Queries via the 6 | TOM (Tabular Object Model) and stores them in the Dataflow JSON format or publishes it to the Power BI service. 7 | .NOTES 8 | File Name : Export2Dataflow.ps1 9 | Date : 10/16/2023 10 | Version : 1.1.1 11 | Author : Marcus Wegener 12 | 13 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 14 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 15 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 16 | IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR 17 | OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 18 | ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 19 | OTHER DEALINGS IN THE SOFTWARE. 20 | .LINK 21 | https://github.com/MarcusWegener/Export2Dataflow 22 | .Parameter Server 23 | The server name and portnumber of the local instance of Analysis Services Tabular for imported/DirectQuery data models 24 | .Parameter Database 25 | The database name of the model hosted in the local instance of Analysis Services Tabular for imported/DirectQuery data models 26 | .Parameter Deployment 27 | Deployment options : 28 | 0 - saves to Dataflow JSON file 29 | 1 - published in Power BI service 30 | #> 31 | 32 | # Below section defines the server, database and deployment based on the input captured from the External tools integration 33 | # This is defined as arguments \"%server%\" and \"%database%\" in the external tools json 34 | Param( 35 | [string] $server = $null, 36 | [string] $database = $null, 37 | [int] $deployment = 0 38 | ) 39 | 40 | Set-StrictMode -Version Latest 41 | 42 | # ================================================================================================================================================= 43 | # DEFINE FUNCTIONS 44 | # ================================================================================================================================================= 45 | 46 | # Function of Marc Lelijveld 47 | # https://github.com/marclelijveld/Power-BI-Automation/blob/master/PowerBI_MoveDataflows.ps1 48 | function _postDataflowDefinition([string] $GroupID, [string]$DataflowDefinition, [string]$NameConflict) { 49 | 50 | $UserAccessToken = Get-PowerBIAccessToken 51 | $bearer = $UserAccessToken.Authorization.ToString() 52 | 53 | $url = [string]::Format("https://api.powerbi.com/v1.0/myorg/groups/{0}/imports?datasetDisplayName=model.json&nameConflict={1}", $GroupID, $NameConflict) 54 | 55 | $boundary = [System.Guid]::NewGuid().ToString("N") 56 | $LF = [System.Environment]::NewLine 57 | 58 | $body = ( 59 | "--$boundary", 60 | "Content-Disposition: form-data; name=`"`"; filename=`"model.json`"", 61 | "Content-Type: application/json$LF", 62 | $DataflowDefinition, 63 | "--$boundary--$LF" 64 | ) -join $LF 65 | 66 | $headers = @{ 67 | 'Authorization' = "$bearer" 68 | 'Content-Type' = "multipart/form-data; boundary=--$boundary" 69 | } 70 | 71 | $postFlow = Invoke-RestMethod -Uri $url -ContentType 'multipart/form-data' -Method POST -Headers $headers -Body ([System.Text.Encoding]::UTF8.GetBytes($body)) 72 | 73 | return $postFlow 74 | } 75 | 76 | # Form 77 | function _showSelectionForm($mode, $workspace, $selectionList){ 78 | 79 | #create GUI 80 | $watermarkText = "" 81 | $labelText = "" 82 | $textBoxWidth = 560 83 | if ($mode -eq "workspace") { 84 | $watermarkText = "Search" 85 | $labelText = "Select a destination:" 86 | } elseif ($mode -eq "dataflow") { 87 | $watermarkText = "name for a new dataflow" 88 | $labelText = "Workspace: " + $workspace + "`n" + "Create a new dataflow or select an existing dataflow:" 89 | $textBoxWidth = 480 90 | } 91 | #create the window settings 92 | $form = New-Object System.Windows.Forms.Form 93 | $form.Text = 'Publish to Power BI' 94 | $form.Size = New-Object System.Drawing.Size(620,400) 95 | $form.StartPosition = 'CenterScreen' 96 | $form.FormBorderStyle = 'FixedSingle' 97 | $form.MaximizeBox = $false 98 | $form.MinimizeBox = $false 99 | $form.BackColor = "White" 100 | $form.ShowIcon = $false 101 | 102 | #create the label settings 103 | $label = New-Object System.Windows.Forms.Label 104 | $label.Location = New-Object System.Drawing.Point(20,10) 105 | $label.Size = New-Object System.Drawing.Size(560,40) 106 | $label.Font = [System.Drawing.Font]::new($label.Font.FontFamily.Name, 10) 107 | $label.Text = $labelText 108 | $form.Controls.Add($label) 109 | 110 | #create Input settings 111 | $textBox = New-Object System.Windows.Forms.TextBox 112 | $textBox.Location = New-Object System.Drawing.Size(20,50) 113 | $textBox.Size = New-Object System.Drawing.Size($textBoxWidth,20) 114 | $textBox.Font = [System.Drawing.Font]::new($textBox.Font.FontFamily.Name, 10) 115 | $textBox.MaxLength = 250 116 | $textBox.ForeColor = 'LightGray' 117 | $textBox.Text = $watermarkText 118 | $form.Controls.Add($textBox) 119 | 120 | if ($mode -eq "dataflow") { 121 | #create the create button settings 122 | $createButton = New-Object System.Windows.Forms.Button 123 | $createButton.Location = New-Object System.Drawing.Point(500,49) 124 | $createButton.Size = New-Object System.Drawing.Size(80,25) 125 | $createButton.Font = [System.Drawing.Font]::new($createButton.Font.FontFamily.Name, 10) 126 | $createButton.Text = 'Create' 127 | $createButton.DialogResult = [System.Windows.Forms.DialogResult]::Yes 128 | $createButton.BackColor = "#f2c811" 129 | $form.AcceptButton = $createButton 130 | $form.Controls.Add($createButton) 131 | } 132 | 133 | #create list Box settings for the selection List 134 | $listBox = New-Object System.Windows.Forms.ListBox 135 | $listBox.Location = New-Object System.Drawing.Point(20,77) 136 | $listBox.Font = [System.Drawing.Font]::new($listBox.Font.FontFamily.Name, 10) 137 | $listBox.Height = 243 138 | $listBox.Width = 560 139 | $listBox.Sorted = $true 140 | $listBox.HorizontalScrollbar = $true 141 | 142 | #add selectionList to the listBox 143 | $listBox.Items.Clear() 144 | if ($selectionList) { 145 | [void] $listBox.Items.AddRange($selectionList) 146 | } 147 | 148 | $form.Controls.Add($listBox) 149 | 150 | #create the select button settings 151 | $selectButton = New-Object System.Windows.Forms.Button 152 | $selectButton.Location = New-Object System.Drawing.Point(400,318) 153 | $selectButton.Size = New-Object System.Drawing.Size(80,27) 154 | $selectButton.Font = [System.Drawing.Font]::new($selectButton.Font.FontFamily.Name, 10) 155 | $selectButton.Text = 'Select' 156 | $selectButton.DialogResult = [System.Windows.Forms.DialogResult]::OK 157 | $selectButton.Enabled = $false 158 | $selectButton.BackColor = "lightgray" 159 | $form.AcceptButton = $selectButton 160 | $form.Controls.Add($selectButton) 161 | 162 | #create the cancel button settings 163 | $cancelButton = New-Object System.Windows.Forms.Button 164 | $cancelButton.Location = New-Object System.Drawing.Point(500,318) 165 | $cancelButton.Size = New-Object System.Drawing.Size(80,27) 166 | $cancelButton.Font = [System.Drawing.Font]::new($cancelButton.Font.FontFamily.Name, 10) 167 | $cancelButton.Text = 'Cancel' 168 | $cancelButton.DialogResult = [System.Windows.Forms.DialogResult]::Cancel 169 | $form.CancelButton = $cancelButton 170 | $form.Controls.Add($cancelButton) 171 | 172 | $form.Topmost = $true 173 | 174 | #behavior if you enter the input textbox 175 | $textBox.Add_Enter({ 176 | if($textBox.Text -eq $watermarkText -and $textBox.ForeColor -eq 'LightGray') 177 | { 178 | #Clear the text 179 | $textBox.Text = "" 180 | $textBox.ForeColor = 'WindowText' 181 | $listBox.ClearSelected() 182 | } 183 | }) 184 | #behavior if you leave the input textbox 185 | $textBox.Add_Leave({ 186 | if($textBox.Text -eq "") 187 | { 188 | #Display the watermark 189 | $textBox.Text = $watermarkText 190 | $textBox.ForeColor = 'LightGray' 191 | if ($mode -eq "workspace") { 192 | $listBox.Items.Clear() 193 | [void] $listBox.Items.AddRange($selectionList) 194 | } 195 | } 196 | }) 197 | 198 | if ($mode -eq "workspace") { 199 | #behavior if you type inside the input textbox 200 | $textBox.Add_TextChanged({ 201 | $textBoxText = [regex]::Escape($textBox.Text) 202 | if ($textBoxText) { 203 | $listBox.Items.Clear() 204 | forEach ($selection in $selectionList) { 205 | if($selection -match $textBoxText){ 206 | [void] $listBox.Items.Add($selection) 207 | } 208 | } 209 | } 210 | }) 211 | } 212 | 213 | $listBox.Add_SelectedIndexChanged({ 214 | if($listBox.SelectedIndex -ge 0){ 215 | $selectButton.Enabled = $true 216 | $selectButton.BackColor = "#f2c811" 217 | } else { 218 | $selectButton.Enabled = $false 219 | $selectButton.BackColor = "lightgray" 220 | } 221 | }) 222 | 223 | $result = $form.ShowDialog() 224 | 225 | if($result -eq [System.Windows.Forms.DialogResult]::OK){ 226 | return $listBox.SelectedItem 227 | } elseif ($result -eq [System.Windows.Forms.DialogResult]::Yes){ 228 | if ($textBox.TextLength -lt 1 -or $textBox.Text -eq $watermarkText) { 229 | return $null 230 | } else { 231 | return $textBox.Text 232 | } 233 | } else { 234 | Write-Host -ForegroundColor Yellow "No " + $mode + " was selected.. Terminating.." 235 | Start-Sleep -Seconds 1.5 236 | exit 237 | } 238 | } 239 | 240 | # ================================================================================================================================================= 241 | # RUN APPLICATION 242 | # ================================================================================================================================================= 243 | 244 | # Write Intro, Server and Database information to screen 245 | Write-Host -ForegroundColor White '========================================================================================================================' 246 | if ($deployment -eq 0) { 247 | Write-Host -ForegroundColor Yellow ' ______ __ ___ ____ __ ____ __ ' 248 | Write-Host -ForegroundColor Yellow ' / ____/_ __ ____ ____ _____ / /_|__ \ / __ \ ____ _ / /_ ____ _ / __// /____ _ __ ' 249 | Write-Host -ForegroundColor Yellow ' / __/ | |/_// __ \ / __ \ / ___// __/__/ / / / / // __ `// __// __ `// /_ / // __ \| | /| / / ' 250 | Write-Host -ForegroundColor Yellow ' / /___ _> < / /_/ // /_/ // / / /_ / __/ / /_/ // /_/ // /_ / /_/ // __// // /_/ /| |/ |/ / ' 251 | Write-Host -ForegroundColor Yellow ' /_____//_/|_|/ .___/ \____//_/ \__//____//_____/ \__,_/ \__/ \__,_//_/ /_/ \____/ |__/|__/ ' 252 | Write-Host -ForegroundColor Yellow ' /_/ ' 253 | } else { 254 | Write-Host -ForegroundColor Yellow ' ____ __ __ _ __ ___ ____ __ ____ __ ' 255 | Write-Host -ForegroundColor Yellow ' / __ \ __ __ / /_ / /(_)_____ / /_ |__ \ / __ \ ____ _ / /_ ____ _ / __// /____ _ __ ' 256 | Write-Host -ForegroundColor Yellow ' / /_/ // / / // __ \ / // // ___// __ \ __/ / / / / // __ `// __// __ `// /_ / // __ \| | /| / / ' 257 | Write-Host -ForegroundColor Yellow ' / ____// /_/ // /_/ // // /(__ )/ / / // __/ / /_/ // /_/ // /_ / /_/ // __// // /_/ /| |/ |/ / ' 258 | Write-Host -ForegroundColor Yellow ' /_/ \__,_//_.___//_//_//____//_/ /_//____//_____/ \__,_/ \__/ \__,_//_/ /_/ \____/ |__/|__/ ' 259 | Write-Host -ForegroundColor Yellow ' ' 260 | } 261 | Write-Host -ForegroundColor Yellow ' By Marcus Wegener v1.1.0 03/28/2021 ' 262 | Write-Host -ForegroundColor White '========================================================================================================================' 263 | Write-Host -ForegroundColor White "Your Power BI Model currently runs with the following connection details:" 264 | Write-Host -ForegroundColor White "Server: " $server 265 | Write-Host -ForegroundColor White "Database: " $database 266 | 267 | # Install latest package (if not already installed) of Microsoft.AnalysisServices.retail.amd64 for current user 268 | Write-Host -ForegroundColor White '========================================================================================================================' 269 | Write-Host -ForegroundColor Gray "Install latest package (if not already installed) of Microsoft.AnalysisServices.retail.amd64 for current user..." 270 | [Net.ServicePointManager]::SecurityProtocol = [Net.SecurityProtocolType]::Tls12 271 | Install-Package Microsoft.AnalysisServices.retail.amd64 -Scope CurrentUser -Source "https://www.nuget.org/api/v2" -SkipDependencies 272 | $installedPackages = Find-Package -Name Microsoft.AnalysisServices.retail.amd64 -Source "$env:USERPROFILE\AppData\Local\PackageManagement\NuGet\Packages\" -AllVersions 273 | $maxVersion = $installedPackages[0].Version 274 | 275 | # Load Assembly Files from insalled package Microsoft.AnalysisServices.retail.amd64 276 | Write-Host -ForegroundColor Gray "Load Assembly Files from insalled package Microsoft.AnalysisServices.retail.amd64..." 277 | $assemblyPathTabular = "$env:USERPROFILE\AppData\Local\PackageManagement\NuGet\Packages\Microsoft.AnalysisServices.retail.amd64.$maxVersion\lib\net45\Microsoft.AnalysisServices.Tabular.dll" 278 | 279 | Write-Host -ForegroundColor White '========================================================================================================================' 280 | 281 | # Connect to Power BI Model 282 | Write-Host -ForegroundColor Gray "Connect to Power BI Model..." 283 | 284 | Add-Type -Path $assemblyPathTabular 285 | 286 | $as = New-Object Microsoft.AnalysisServices.Tabular.Server 287 | $as.Connect($server) 288 | $db = $as.Databases[$database] 289 | $dbModel = $db.Model 290 | 291 | $modelCulture = $dbModel.Culture 292 | $modelModifiedTime = Get-Date($dbModel.ModifiedTime) -format s 293 | 294 | # Create query groups 295 | Write-Host -ForegroundColor Gray "`Create query groups..." 296 | 297 | $queryGroups = @{} 298 | foreach($queryGroup in $dbModel.QueryGroups) { 299 | $groupStructur = $queryGroup.Folder.Split("\") 300 | $groupName = $groupStructur[-1] 301 | $groupParentName = "" 302 | $groupParentId = $null 303 | 304 | if($groupStructur.Length -gt 1) { 305 | $groupParentName = $groupStructur[0..($groupStructur.Length -2)] -join "\" 306 | $groupParentId = $queryGroups[$groupParentName].id 307 | } 308 | 309 | $queryGroups[$queryGroup.Name] = [ordered]@{ 310 | "id" = New-Guid 311 | "name" = $groupName 312 | "description" = $queryGroup.Description 313 | "parentId" = $groupParentId 314 | "order" = $groupStructur.Length 315 | } 316 | } 317 | 318 | $pbiQueryGroups = "" 319 | 320 | if($queryGroups.Count -le 1) { 321 | $pbiQueryGroups = "[" 322 | } 323 | 324 | $pbiQueryGroups += $queryGroups.values | ConvertTo-Json -Compress 325 | 326 | if($queryGroups.Count -le 1) { 327 | $pbiQueryGroups += "]" 328 | } 329 | 330 | 331 | # Create queries 332 | Write-Host -ForegroundColor Gray "`nCreate queries..." 333 | 334 | $queriesMetadata = @{} 335 | $entities = @() 336 | $document = "section Section1;`n" 337 | 338 | 339 | foreach($e in $dbModel.Expressions) { 340 | $queriesMetadata[$e.Name] = [ordered]@{ 341 | queryId = New-Guid 342 | queryName = $e.Name 343 | } 344 | if($e.QueryGroup) { 345 | $queriesMetadata[$e.Name].queryGroupId = $queryGroups[$e.QueryGroup.Folder].id 346 | } 347 | 348 | $tableName = $e.Name 349 | if($tableName -notmatch '^[a-z0-9]+$') { 350 | $tableName = "#"""+$tableName+"""" 351 | } 352 | 353 | $document += "shared " + $tableName + " = " + $e.Expression + ";`n" 354 | } 355 | 356 | foreach($t in $dbModel.Tables) { 357 | if($t.Partitions.SourceType -eq "M") { 358 | $queriesMetadata[$t.Name] = [ordered]@{ 359 | queryId = New-Guid 360 | queryName = $t.Name 361 | "loadEnabled" = $true 362 | } 363 | if($t.Partitions.QueryGroup) { 364 | $queriesMetadata[$t.Name].queryGroupId = $queryGroups[$t.Partitions.QueryGroup.Folder].id 365 | } 366 | 367 | $tableName = $t.Name 368 | if($tableName -notmatch '^[a-z0-9]+$') { 369 | $tableName = "#"""+$tableName+"""" 370 | } 371 | 372 | $document += "shared " + $tableName + " = " + $t.Partitions.Source.Expression + ";`n" 373 | 374 | $attributes = @() 375 | 376 | foreach($c in $t.Columns) { 377 | if ($c.Type -eq "Data") { 378 | $attributes += [ordered]@{ 379 | name = $c.Name 380 | dataType = $c.DataType.ToString().ToLower() 381 | } 382 | } 383 | } 384 | 385 | $entities += [ordered]@{ 386 | '$type' = "LocalEntity" 387 | "name" = $t.Name 388 | "description" = $t.Description 389 | "pbi:refreshPolicy"= [ordered]@{ 390 | '$type' = "FullRefreshPolicy" 391 | "location" = [uri]::EscapeDataString($t.Name) + ".csv" 392 | } 393 | "attributes" = $attributes 394 | } 395 | } 396 | } 397 | 398 | $document = $document.Replace("`n","`r`n") 399 | 400 | Write-Host -ForegroundColor Gray "`nDisconnect from Power BI Model..." 401 | 402 | $as.Disconnect() 403 | 404 | $dataflowName = "" 405 | $conflict = "Ignore" 406 | 407 | if ($deployment -eq 0) { 408 | # Save file dialog 409 | Write-Host -ForegroundColor Gray "Export Dataflow model to file..." 410 | 411 | Add-Type -AssemblyName System.Windows.Forms 412 | 413 | $fileBrowser = New-Object System.Windows.Forms.SaveFileDialog -Property @{ 414 | InitialDirectory = [Environment]::GetFolderPath('Desktop') 415 | Filter = 'JSON Files (*.json)|*.json' 416 | Title = 'Export Dataflow model to file:' 417 | } 418 | 419 | $null = $fileBrowser.ShowDialog() 420 | $fileName = $fileBrowser.FileName 421 | 422 | if(!$fileName){ 423 | Write-Host -ForegroundColor Yellow "No file chosen. Terminating..." 424 | Start-Sleep -Seconds 1.5 425 | exit 426 | } 427 | 428 | $fileNameWithoutPath = Split-path $fileBrowser.FileName -leaf 429 | 430 | 431 | Write-Host -ForegroundColor Gray "$fileName chosen..." 432 | 433 | $dataflowName = [regex]::Match($fileNameWithoutPath, ".+?(?=\.json)").Value 434 | } else { 435 | 436 | Add-Type -AssemblyName System.Windows.Forms 437 | Add-Type -AssemblyName System.Drawing 438 | Add-Type -AssemblyName PresentationCore 439 | Add-Type -AssemblyName PresentationFramework 440 | 441 | [System.Windows.Forms.Application]::EnableVisualStyles() 442 | 443 | #Get Module "MicrosoftPowerBIMgmt" 444 | $moduleName = Get-Module -ListAvailable -Verbose:$false | Where-Object { $_.Name -eq "MicrosoftPowerBIMgmt" } | Select-Object -ExpandProperty Name 445 | if ([string]::IsNullOrEmpty($moduleName)) { 446 | Write-Host -ForegroundColor White "`n========================================================================================================================" 447 | Write-Host -ForegroundColor Gray "Install module MicrosoftPowerBIMgmt..." 448 | Install-Module MicrosoftPowerBIMgmt -SkipPublisherCheck -AllowClobber -Force -Scope CurrentUser 449 | Write-Host -ForegroundColor White "========================================================================================================================" 450 | } 451 | 452 | Write-Host -ForegroundColor Gray "`nConnect to PowerBI service" 453 | Connect-PowerBIServiceAccount 454 | 455 | #Get all Workspaces of connected user 456 | $workspaces = Get-PowerBIWorkspace -All 457 | 458 | $workspaceName = _showSelectionForm -mode "workspace" -workspace $null -selectionList $workspaces.Name 459 | 460 | $selectedWorkspace = $workspaces | Where Name -eq $workspaceName 461 | 462 | $dataflowsInWorkspace = Get-PowerBIDataflow -Workspace $selectedWorkspace 463 | 464 | $dataflowNames = $dataflowsInWorkspace | Select-Object -ExpandProperty Name 465 | 466 | do { 467 | $dataflowName = _showSelectionForm -mode "dataflow" -workspace $workspaceName -selectionList $dataflowNames 468 | 469 | if ($dataflowName -eq $null) { 470 | [void] [System.Windows.MessageBox]::Show("You need to write at least 1 character" 471 | ,"" 472 | ,[System.Windows.MessageBoxButton]::OK 473 | ,[System.Windows.MessageBoxImage]::Exclamation) 474 | } elseif ($dataflowsInWorkspace | Where Name -eq $dataflowName){ 475 | $confirmationResult = [System.Windows.MessageBox]::Show("Are you sure you want to overwrite the dataflow '$dataflowName'?" 476 | ,"Confirm Overwriting dataflow" 477 | ,[System.Windows.MessageBoxButton]::YesNoCancel 478 | ,[System.Windows.MessageBoxImage]::Exclamation) 479 | 480 | switch ($confirmationResult) { 481 | "Yes"{ 482 | Write-Host -ForegroundColor Yellow "Overwriting Dataflow '$dataflowName'" 483 | $conflict = "Overwrite" 484 | }"No"{ 485 | Write-Host "Not overwriting Dataflow '$dataflowName'.. Navigating back to Dataflow-Selection" 486 | $dataflowName = $null 487 | }"Cancel"{ 488 | Write-Host -ForegroundColor Yellow "Cancel-Button clicked.. Terminating.." 489 | Start-Sleep -Seconds 1 490 | exit 491 | } 492 | } 493 | } 494 | } while ($dataflowName -eq $null) 495 | 496 | Write-Host -ForegroundColor Gray $dataflowName 497 | 498 | } 499 | 500 | Write-Host -ForegroundColor White '========================================================================================================================' 501 | 502 | $json = [ordered]@{ 503 | "name" = $dataflowName 504 | "description" = "" 505 | "version" = "1.0" 506 | "culture" = $modelCulture 507 | "modifiedTime" = $modelModifiedTime 508 | "pbi:mashup" = [ordered]@{ 509 | "fastCombine" = $false 510 | "allowNativeQueries" = $false 511 | "skipAutomaticHeaderAndTypeDetection" = $false 512 | "queriesMetadata" = $queriesMetadata 513 | "document" = $document 514 | } 515 | "annotations" = @( 516 | [ordered]@{ 517 | "name" = "pbi:QueryGroups" 518 | "value" = $pbiQueryGroups 519 | } 520 | ) 521 | "entities" = $entities 522 | } 523 | 524 | 525 | if ($deployment -eq 0) { 526 | Write-Host -ForegroundColor Gray "`nWrite into JSON file..." 527 | 528 | $json | ConvertTo-Json -Depth 10 -Compress | Out-File -Encoding UTF8 "$fileName" 529 | 530 | Start-Process -FilePath C:\Windows\explorer.exe -ArgumentList "/select, ""$fileName""" 531 | 532 | Write-Host -ForegroundColor Gray "Dataflow model file ($fileName) created..." 533 | } else { 534 | Write-Host -ForegroundColor Gray "`nGenerate JSON..." 535 | 536 | $dataflowJSON = $json | ConvertTo-Json -Depth 10 -Compress 537 | 538 | $newDataFlow = _postDataflowDefinition -GroupID $selectedWorkspace.Id -DataflowDefinition $dataflowJSON -NameConflict $conflict 539 | 540 | Write-Host -ForegroundColor White ( [string]::Format("New dataflow with id '{0}' created in workspace '{1}'", $newDataFlow.id, $selectedWorkspace.Name )) 541 | } 542 | Write-Host -ForegroundColor White '========================================================================================================================' 543 | Start-Sleep -Seconds 1.5 -------------------------------------------------------------------------------- /Export2Dataflow_Logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MarcusWegener/Export2Dataflow/15b1f4aa1f2b4d11daefae198743b25b37751bb9/Export2Dataflow_Logo.png -------------------------------------------------------------------------------- /Export2Dataflow_Logo.svg: -------------------------------------------------------------------------------- 1 | 2 | 20 | 22 | 43 | 45 | 46 | 48 | image/svg+xml 49 | 51 | 52 | 53 | 54 | 55 | 59 | 61 | 68 | 71 | 74 | 78 | 82 | 86 | 91 | 92 | 97 | 100 | 105 | 109 | 114 | 115 | 116 | 119 | 137 | 145 | 146 | 149 | 156 | 163 | 170 | 174 | 178 | 179 | 180 | 181 | 182 | -------------------------------------------------------------------------------- /LICENSE: -------------------------------------------------------------------------------- 1 | This is free and unencumbered software released into the public domain. 2 | 3 | Anyone is free to copy, modify, publish, use, compile, sell, or 4 | distribute this software, either in source code form or as a compiled 5 | binary, for any purpose, commercial or non-commercial, and by any 6 | means. 7 | 8 | In jurisdictions that recognize copyright laws, the author or authors 9 | of this software dedicate any and all copyright interest in the 10 | software to the public domain. We make this dedication for the benefit 11 | of the public at large and to the detriment of our heirs and 12 | successors. We intend this dedication to be an overt act of 13 | relinquishment in perpetuity of all present and future rights to this 14 | software under copyright law. 15 | 16 | THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, 17 | EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF 18 | MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. 19 | IN NO EVENT SHALL THE AUTHORS BE LIABLE FOR ANY CLAIM, DAMAGES OR 20 | OTHER LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, 21 | ARISING FROM, OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR 22 | OTHER DEALINGS IN THE SOFTWARE. 23 | 24 | For more information, please refer to 25 | -------------------------------------------------------------------------------- /Publish2Dataflow_Logo.png: -------------------------------------------------------------------------------- https://raw.githubusercontent.com/MarcusWegener/Export2Dataflow/15b1f4aa1f2b4d11daefae198743b25b37751bb9/Publish2Dataflow_Logo.png -------------------------------------------------------------------------------- /Publish2Dataflow_Logo.svg: -------------------------------------------------------------------------------- 1 | 2 | 20 | 22 | 43 | 45 | 46 | 48 | image/svg+xml 49 | 51 | 52 | 53 | 54 | 55 | 59 | 61 | 68 | 71 | 74 | 79 | 84 | 89 | 95 | 96 | 102 | 105 | 111 | 116 | 122 | 123 | 124 | 126 | 144 | 152 | 153 | 156 | 163 | 170 | 177 | 182 | 187 | 188 | 189 | 190 | 191 | -------------------------------------------------------------------------------- /README.md: -------------------------------------------------------------------------------- 1 | # Export2Dataflow 2 | This repository contains everything needed to run the Export2Dataflow PowerShell script as an External Tool in the Power BI Desktop and export the Power Query to a Dataflow JSON file or publishes it to the Power BI service. 3 | 4 | ## Disclaimer 5 | Please know, that everything I have created and shared on the blogpost and on GitHub is based on best effort. No rights can be derived, as well as I am not liable for the use or misuse of the solution or possible damage resulting from this. Use of the solutions and execution of the scripts is all on your own risk and your own responsibility. 6 | 7 | ## Getting Started 8 | ### Download everything you need from this repository. 9 | * External Tool integration file: _export2dataflow.pbitool.json_ 10 | * External Tool integration file: _publish2dataflow.pbitool.json_ 11 | * The PowerShell script: _Export2Dataflow.ps1_ 12 | 13 | ### Copy the External Tool integration file 14 | The External Tool integration file is needed to get the button in the Power BI ribbon. In order to achieve this, you need to copy the _export2dataflow.pbitool.json_ and _publish2dataflow.pbitool.json_ file in the External Tools folder. For me this location was: 15 | _C:\Program Files (x86)\Common Files\Microsoft Shared\Power BI Desktop\External Tools_ 16 | 17 | While copying the files, it can be that Windows asks you to login with admin privileges before you can continue. This is mandatory to copy the files. If you cannot do this yourself, please contact your administrator. 18 | 19 | ### Copy the PowerShell script 20 | Create a subfolder _Export2Dataflow_ in the _C:\Program Files\\_ folder and copy the PowerShell script _Export2Dataflow.ps1_ file into it. 21 | 22 | Identical to the previous step, Windows may ask you to authenticate with administrator privileges before you can proceed. 23 | 24 | ### Restart Power BI Desktop 25 | You have applied all required steps by now. The new buttons will appear in the Power BI Desktop top ribbon for External Tools. In case you had Power BI running already, please restart Power BI desktop first. 26 | 27 | Did something not workout as expected for you, kindly check the FAQ to see if your question is already listed there. If not, please let me know. 28 | 29 | ## Usage 30 | I want to shortly describe how this tool works and what you can expect. 31 | 1. Open a Power BI Desktop file whose Power Query transformations you want to export to a Dataflow. 32 | 2. In the Top Ribbon under External Tools, click _Export to Dataflow_. 33 | 3. After the click a PowerShell window opens which converts the Power Query transformations into the Dataflow JSON format. 34 | During the first execution it may be necessary to agree to the installation of _nuget_ and the _package source_ for the installation of the latest Microsoft.AnalysisServices.Tabular.dll version. 35 | 4. Then the script asks for the location where the Dataflow JSON file should be exported. 36 | 5. The exported Dataflow JSON file can then be uploaded to the Power BI Portal as Dataflow. To do this, create a new dataflow in the workspace with the Import Datamodel option. 37 | 38 | ## Author 39 | Marcus Wegener 40 | 41 | [Website](https://thinkbi.de) - 42 | [twitter](https://twitter.com/PowerBIler) - 43 | [LinkedIn](https://www.linkedin.com/in/marcuswegener/) - 44 | [Xing](https://www.xing.com/profile/Marcus_Wegener3/cv) 45 | 46 | ## Acknowledgements 47 | Marc Lelijveld, for his tutorial on creating external tools for Power BI and his GitHub repository [External-Tools-Model-Documentation](https://github.com/marclelijveld/External-Tools-Model-Documentation), from which I was able to adopt a lot. 48 | 49 | [Website](https://data-marc.com/) - 50 | [Blogpost](https://data-marc.com/2020/07/28/external-tools-document-your-power-bi-model/) - 51 | [twitter](https://twitter.com/PowerBIler) - 52 | [LinkedIn](https://www.linkedin.com/in/marclelijveld/) - 53 | [Github](https://github.com/marclelijveld/External-Tools-Model-Documentation) 54 | 55 | Julian Kaiser, for reviewing, testing and adding some features to the script. 56 | 57 | [LinkedIn](https://www.linkedin.com/in/julian-kaiser-5b849519a/) 58 | 59 | ## FAQ 60 | ### The Document model button does not appear in Power BI Desktop 61 | There are a few things that can cause this issue. There are a few things you can check up front: 62 | 63 | 1. Do you run the latest version of Power BI Desktop? If not, please update first. 64 | 2. Do you have other External Tools, such as DAX Studio, Tabular Editor or ALM Toolkit successfully running? 65 | 3. Did you put the export2dataflow.pbitool.json file in the correct location? Please check the blogpost to find the correct location. 66 | 67 | If still nothing happens, or you don't have the External Tools folder on your PC, I advise you to install any of the above mentioned external tools that will generate this location for you during installation. 68 | 69 | ### The buttons in the External Tools section in Power BI Desktop are greyed out 70 | External tools require Enhanced Meta Data to be enabled. You can enable this in the preview features of Power BI Desktop. 71 | 72 | 1. Open Power BI Desktop 73 | 2. Go to File 74 | 3. Options & Settings 75 | 4. Options 76 | 5. In the Global Settings go to Preview Features 77 | 6. Close all Power BI Desktop instances and re-open Power BI. 78 | 79 | ### I clicked the button, saw the PowerShell window flickering on the screen, but nothing happened 80 | Most likely, this is caused by the execution policies for PowerShell configured on the computer. As far as I know, this is not something I can change in my script, but has everything to do with the setup of your PC or how your company configured it. This results in the fact that PowerShell.exe application did start, but it did not execute the script, because this was prevented by the policy. The solution to fix this, is changing the execution policy (which is a register thing on your computer. Please know, that this is all at your own risk! I cannot take any responsibility for this. 81 | 82 | The following steps might help you 83 | 84 | 1. Open PowerShell.exe manually 85 | 2. Execute Get-ExecutionPolicy -List 86 | 3. As a result, you see the current configuration of your execution policies. Most probably everything is set to undefined at this moment. 87 | 4. The easiest way to fix this, is by changing the execution policy for the current user. The ensures that this does not happen again. We need to do this by setting it to Unrestricted by executing this task: Set-ExecutionPolicy Unrestricted -Scope CurrentUser (all on your own responsibility and risk!) 88 | 5. Confirm that you want to change this. 89 | 6. Close PowerShell and Power BI Desktop 90 | Try again if it works now. More information about Execution Policies can be found [in this article](https://winaero.com/change-powershell-execution-policy-windows-10/) 91 | 92 | ## License 93 | All code in this repository is licenses as specified by the [LICENSE](https://github.com/MarcusWegener/Export2Dataflow/blob/main/LICENSE) file. -------------------------------------------------------------------------------- /export2dataflow.pbitool.json: -------------------------------------------------------------------------------- 1 | { 2 | "version": "1.1.0", 3 | "name": "Export to Dataflow", 4 | "description": "Export to Dataflow", 5 | "path": "C:\\Windows\\System32\\WindowsPowerShell\\v1.0\\powershell.exe", 6 | "arguments": "C:\\'Program Files'\\Export2Dataflow\\Export2Dataflow.ps1 \"%server%\" \"%database%\" 0", 7 | "iconData": "" 8 | } 9 | -------------------------------------------------------------------------------- /publish2dataflow.pbitool.json: -------------------------------------------------------------------------------- 1 | { 2 | "version": "1.1.0", 3 | "name": "Publish to Dataflow", 4 | "description": "Publish to Dataflow", 5 | "path": "C:\\Windows\\System32\\WindowsPowerShell\\v1.0\\powershell.exe", 6 | "arguments": "C:\\'Program Files'\\Export2Dataflow\\Export2Dataflow.ps1 \"%server%\" \"%database%\" 1", 7 | "iconData": "" 8 | } 9 | --------------------------------------------------------------------------------