Many people use Python with notebooks, so let’s take a look at one possible way to enable a Jupyter external tool for Power BI Desktop. The following stepwise approach begins with simply opening Jupyter. It then progresses to creating and opening a notebook that includes Power BI’s server and database arguments. Finally, it works its way toward downloading a notebook definition contained in a GitHub gist and connects to Power BI’s tabular model to start to make this approach more useful.
This post continues a series of posts related to Python and Power BI. The first three parts of this blog series introduced some possible uses for Python connected to a Power BI model, how to setup a basic Python external tool, and how to both use it with a virtual environment and connect to the Tabular Object Model.
Open Jupyter: Foundational But Not Useful
I use the Anaconda distribution in these examples, and let’s consider how to simply start and open a notebook (vanilla Jupyter and nothing with JupyterLab). We only need a pbitool.json file at this stage because the goal is simply launch Jupyter–not even a notebook file. It’s not a useful example but provides a foundation on which we’ll build.
As discussed in Part 2, pbitool.json files define Power BI Desktop’s external tools. If you look at the Jupyter shortcut, it provides the inspiration for how to define Power BI’s external tool.

The JSON in this gist would need to be saved as a [tool].pbitool.json file and placed in C:\Program Files (x86)\Common Files\Microsoft Shared\Power BI Desktop\External Tools.
Note the path is for Python.exe, and arguments load the default Python environment and run the jupyter-notebook-script.py script to launch Jupyter.

There’s no value in simply opening Jupyter though. At this stage, it’s nothing more than a replacement for a shortcut. The real value is integrating Power BI Desktop more closely with the notebook.
Create and Open a Notebook (Hardcoded)
Taking it a step further, let’s create an empty notebook and write the currently open Power BI Desktop connection information to cells in the notebook. In this case, we need a Python script along with a new [tool].pbitool.json file.
The [tool].pbitool.json file has a few differences from the previous version. Most notably, it executes a Python script that I’ve named powerbinotebook.py and placed in my C:\Temp directory. You’ll also see the server and database arguments that Power BI Desktop uses to pass the current local connection (server:port and database GUID).
C:/Temp/powerbinotebook.py \”%server%\” \”%database%\”
The rest of the arguments start the default Python environment and launch Jupyter, but it’s important that the powerbinotebook.py script runs before Jupyter launches. Why is that?
The Python script defines a nearly empty notebook for Jupyter and then launches it in Jupyter. Pay close attention to notebook_core, which contains the JSON definition for the notebook and writes sys.argv[1] and sys.argv[2] into the notebook for the Power BI server and database respectively. Each cell in the cells array will appear as a separate cell in the sample notebook. As required by the pbitool.json definition above, the Python script is located in C:\Temp\powerbinotebook.py, and you could obviously name and save your own to your location of choice.

The following video shows the basic script execution, where the Python script creates the notebook, launches Jupyter, and Jupyter launches the notebook.
Create and Open a Notebook Defined in a GitHub Gist
Taking it a step further, let’s shift the notebook definition to a GitHub gist instead of hardcoding it into the Python script. The gist contains the JSON and also placeholders for <<PowerBIServer>> and <<PowerBIDatabase>>, which the Python script will replace with the actual sys.argv[1] and sys.argv[2] values.
Sample gist (not embedded in the post): https://gist.github.com/deldersveld/fdada06b82621233d6e40fac13cceb55
I used the same [tool].pbitool.json file since only the contents of the powerbinotebook.py script change. Note how the script is much shorter now that it no longer contains the notebook definition. This uses the requests library to get the JSON notebook template from the URL defined in gist_url. Note as well how replace swaps in the current values of the Power BI Server and Database.
Extending the Notebook Definition
Finally, let’s build a slightly more complex notebook using an updated powerbinotebook.py script. As before, no modification is required for [tool].pbitool.json. Only the Python script updates.
Here is the updated notebook Gist for reference (not embedded in post):
https://gist.github.com/deldersveld/70e05dbc97086181641592f03d314800
As with the last example, the Python script downloads the new notebook definition contained in a different GitHub gist. Also as before, it replaces the <<PowerBIServer>> and <<PowerBIDatabase>> placeholders with the actual values passed into sys.argv from Power BI Desktop.
What’s new here is that this script downloads the ssas_api.py module mentioned in the Part 3 post, which is one among many possible ways to connect Python to AMO / TOM and other .NET libraries. The updated sample notebook loads the module and demonstrates basic model connectivity.
NOTE that the script intentionally does not overwrite a notebook once it’s created for a specific port number. To get a new blank template if you are testing this out simply requires closing and re-opening Power BI Desktop. Connecting to TOM in this sample also requires a Power BI Desktop connection with at least one existing table.
Here’s a look at the sample notebook in action. It illustrates two things: a basic connection to TOM, and the evaluation of a DAX query against the model open in Power BI Desktop.
Unlocking More
With a live connection to the Power BI Desktop model from Python, you can unlock a lot of possibilities. As described in Part 1, there are many examples ranging from data visualization to updating the model. I’m curious to hear your thoughts on how you might want to use Python in Power BI.
Let me know in the comments, and don’t forget to subscribe to the DataVeld blog.
This is great!
Could you please consider a post on how to connect via Python to the models in the Service instead? In some usecases people don’t have access to the .pbix
Thanks for commenting. It’s possible now with this setup simply by changing the connection string, but I do not plan to publish on that particular method since it would mean keeping userid and password in plain text in a notebook. There are better ways to handle authentication.
How do I embed report page2 in notebook to export dat from visuals present in page2 .i see report in defualt page 1 and emebed url also not working
Regarding: There are better ways to handle authentication from above post 🙂
Do you know if it’s possible to create a Service Principal user in Active Directory and let that user handle the authentication when we want to update data from example Dynamics 365? (Guess it would be direct to Power BI Online and not Desktop if possible?)
How can i use Jupyter notebook with Power Bi without installing Anaconda?
The JSON files will have to be edited to a path that does not reference Anaconda and instead references your Python path. “C:\Users\luis2\Anaconda3\python.exe” – > “C:\Users\luis2\PYTHONPATH\python.exe.
My jupyter was installed on “D”,when I create the jupyter using the code it cause error.
Creating Power BI Desktop Localhost 65154.ipynb
Opening Jupyter
[I 18:19:53.182 NotebookApp] JupyterLab extension loaded from D:\Anaconda\lib\site-packages\jupyterlab
[I 18:19:53.182 NotebookApp] JupyterLab application directory is D:\Anaconda\share\jupyter\lab
[I 18:19:53.184 NotebookApp] Serving notebooks from local directory: D:/code
[I 18:19:53.184 NotebookApp] The Jupyter Notebook is running at:
[I 18:19:53.184 NotebookApp] http://localhost:8888/?token=a0f548a972d5e11d8b202fa7707a11be0e32731d556dd93a
[I 18:19:53.185 NotebookApp] or http://127.0.0.1:8888/?token=a0f548a972d5e11d8b202fa7707a11be0e32731d556dd93a
[I 18:19:53.185 NotebookApp] Use Control-C to stop this server and shut down all kernels (twice to skip confirmation).
Traceback (most recent call last):
File “D:/Anaconda/Scripts/jupyter-notebook-script.py”, line 10, in
sys.exit(main())
File “D:\Anaconda\lib\site-packages\jupyter_core\application.py”, line 268, in launch_instance
return super(JupyterApp, cls).launch_instance(argv=argv, **kwargs)
File “D:\Anaconda\lib\site-packages\traitlets\config\application.py”, line 664, in launch_instance
app.start()
File “D:\Anaconda\lib\site-packages\notebook\notebookapp.py”, line 1936, in start
self.launch_browser()
File “D:\Anaconda\lib\site-packages\notebook\notebookapp.py”, line 1886, in launch_browser
relpath = os.path.relpath(self.file_to_run, self.notebook_dir)
File “D:\Anaconda\lib\ntpath.py”, line 562, in relpath
path_drive, start_drive))
ValueError: path is on mount ‘C:’, start on mount ‘D:’
Hi David,
I follow the exact same instructions but I get an error in the section of “Create and Open a Notebook (Hardcoded)”.
The error is:
“The system cannot find the path specified.”
I was wondering if you could help me at that step.
Thanks.