├── 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 |
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 |
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 |
--------------------------------------------------------------------------------