Assets

What’s an Asset?

Each Pharaoh components has a directory called asset_scripts, that contain scripts Python scripts or Jupyter Notebooks, that extract information from resources (local result files like HDF5, databases or other sources) and transform this information into plot, tables and any other kind of types.

The generated files are called Assets and may be directly referenced/embedded by a template (like a picture, plot or table) or indirectly referenced by a template to provide context values for template rendering.

The component’s templates then consume these assets while Sphinx renders the template to HTML. Consume means that some of the assets are directly included in the template via the Pharaoh asset directive and others are read by local context scripts to use their content during templating.

Executing Asset Generation

Asset generation is executed before the actual Sphinx build via generate_assets().

PharaohProject.generate_assets(component_filters: Iterable[str] = ('.*',)) list[Path][source]

Generate all assets by executing the asset scripts of a selected or all components.

All asset scripts are executed in separate parallel child processes (number of workers determined by asset_gen.worker_processes setting; setting 0 executes all asset scripts sequentially in the current process).

Setting asset_gen.script_ignore_pattern determines if a script is ignored.

Putting the comment # pharaoh: ignore at the start of a script will also ignore the file.

Parameters:

component_filters – A list of regular expressions that are matched against each component name. If a component name matches any of the regular expressions, the component’s assets are regenerated (containing directory will be cleared)

Besides calling the API function, also the CLI can be used:

  • From within PyCharm or wherever you have the venv activated that has Pharaoh installed, you can use the Pharaoh CLI:

    cd <your Pharaoh project directory>
    pharaoh generate  # for generating assets of all components
    pharaoh generate -f "dummy_.*"  # for generating assets of all components starting with "dummy_"
    pharaoh generate -f dummy_1  -f dummy_2  # only for dummy_1 and dummy_2
    
  • Via a terminal (PS or CMD) from within the project directory, where CMD scripts are generated for your convenience:

    .\pharaoh-generate-assets.cmd  # for generating assets of all components
    .\pharaoh.cmd generate -f dummy_1
    

Debugging Asset Scripts

Asset scripts can be debugged in two different ways:

  • Via Pharaoh Asset Generation

    Generated Pharaoh project ship a debug script debug.py. Just open your generated project in an IDE with an interpreter/venv that has Pharaoh installed and start a debug session on debug.py.

    The script should be modified to your current needs, e.g. if you just want to debug asset generation, comment out the proj.build_report() line. If you like to debug asset scripts of a single component, pass the component name like this proj.generate_assets(component_filters=("dummy_1",)).

    Important

    If the setting asset_gen.worker_processes is set to a non-zero integer or "auto", then asset scripts are executed as child processes.

    Make sure you have the PyCharm debugger option File | Settings | Build, Execution, Deployment | Python Debugger -> Attach to subprocess automatically while debugging enabled.

  • Directly debug your script

    You can execute your asset scripts directly via your IDE of choice. Any functions you import from pharaoh.api or pharaoh.assetlib.api are built in a way to support being executed outside of a Pharaoh asset generation.

    The main difference is, that Pharaoh is not monkey-patching any toolkit (Plotly, Bokeh, Pandas) function, so the functions are behaving according to their official documentation. So statements like `fig.write_image("myplot.png") are storing the plot in the current working directory (this is the same directory your asset script is in), instead of report-project\.asset_build.

Implementing Asset Scripts

This section shows how asset scripts are developed.

For simplicity we will show first how to generate assets without the use of resources. An example on how to access resources can be found in the Resources Reference.

Refresher: A components asset scripts are always located in the component’s subdirectory asset_scripts.

The script names do not matter but certain names may be ignored (see generate_assets()).

Let’s assume we want to make a plot as our first asset in a script called my_plot.py:

import plotly.express as px

from pharaoh.assetlib.api import metadata_context

fig = px.scatter(
    data_frame=px.data.iris(),  # Famous example IRIS data set
    x="sepal_width",
    y="sepal_length",
    color="species",
    symbol="species",
    title="IRIS Example",
)

with metadata_context(label="my_plot"):
    fig.write_html(file="iris_scatter.html")

Now let’s analyse what we got:

  1. import plotly.express as px

    Import a plotting framework of choice. Some frameworks are better supported by Pharaoh than others, we’ll see this later also in the Patched Frameworks section.

  2. from pharaoh.assetlib.api import metadata_context

    Any API function you will need can be imported from the pharaoh.assetlib.api module. The purpose of metadata_context is explained later.

  3. fig = px.scatter(...)

    Create a Plotly figure with some example data. Your own scripts would actually read the resources specified with the component and create plots, tables etc.

  4. with metadata_context(label="my_plot"):

    This context manager determines the metadata the asset will be stored with. This metadata is then used to include assets in a template, in this case this would look like this:

    .. pharaoh-asset::
        :filter: label == "my_plot"
        :template: iframe
        :iframe-width: 500px
        :iframe-height: 500px
    

    This so-called Sphinx directive is a custom Sphinx directive (click here for docs) Pharaoh adds via the plugin system of Sphinx.

    metadata_context is explained in more detail in the next section.

  5. fig.write_html(file="iris_scatter.html")

    This is where Pharaoh is doing it’s magic.

    plotly.graph_objects.Figure.write_html is one of many function which is monkey-patched by Pharaoh. More info about the other patched frameworks you can find here.

    If the asset script is executed via Pharaoh, the patched write_html function will not store the plot to "iris_scatter.html" in the current working directory, but rather in a predefined location for assets inside your Pharaoh project report_project/.asset_build/<component-name> with a unique suffix, for example iris_scatter_9c30799b.html.

    Next to the actual asset, an asset-info file iris_scatter_9c30799b.assetinfo is stored that holds the metadata for the asset. Here is a (shortened) content example:

    {
        "asset": {
            "script_name": "my_plot.py",
            "script_path": "C:\\...\\project\\report-project\\components\\dummy_1\\asset_scripts\\my_plot.py",
            "index": 1,
            "component_name": "dummy_1",
            "user_filepath": "iris_scatter.html",
            "file": "C:\\...\\project\\report-project\\.asset_build\\dummy_1\\iris_scatter_9c30799b.html",
            "name": "iris_scatter_9c30799b.html",
            "stem": "iris_scatter_9c30799b",
            "suffix": ".html",
            "template": "raw_html"
        },
        "label": "my_plot"
    }
    

    This has following advantages:

    • Finding names for assets is completely unnecessary, since they get a unique name assigned and are included in templating through their metadata rather than their file name. You could basically name all your assets asset.<suffix>.

      So for example if you would create a single plot of a signal for 100 operating conditions like Vdd, you would not add the value of Vdd to the file name, but rather put it in the metadata (see here) and name the file always "<signal-name>.png".

      Note

      Though the assets name does not matter, its suffix does. It is used internally to infer the template to render it, so make sure you add it.

    • If the script is directly executed by the user, write_html is not patched and saves the plot to "iris_scatter.html" in the current working directory.

Metadata Stack

The metadata stack is a construct that helps assigning metadata to generated assets.

Use the context manager returned by metadata_context() in your asset scripts to dynamically add/remove/update metadata that is added to your assets.

Let’s explained based on an example:

from pharaoh.assetlib.api import metadata_context, register_asset

# Will have no metadata (except the default ones added by Pharaoh)
register_asset("<some-file>")

# Will have metadata: {"foo": "bar"} only
register_asset("<some-file>", dict(foo="bar"))

with metadata_context(a=1):
    register_asset("<some-file>")  # Will have metadata: {"a": 1} only

    with metadata_context(a=2, b=3):
        register_asset("<some-file>")  # Will have metadata: {"a": 2, "b": 3}

    with metadata_context(c=4):
        register_asset("<some-file>")  # Will have metadata: {"a": 1, "c": 4}

    for i in range(10):
        with metadata_context(e=i):
            register_asset("<some-file>")  # Will have metadata: {"a": 1, "e": i}

Note

If you don’t want to use the context manager, because the metadata should be set globally in the script anyway or you’re asset script is a Jupyter Notebook (no context manager over multiple cells possible), you can also use the activate() and deactivate() method:

metadata_context(...).activate()
register_asset("<some-file>")

Or if you want to deactivate the context again:

mc = metadata_context(...).activate()
register_asset("<some-file>")
mc.deactivate()

The function register_asset() is explained in the next section.

Manually Registering Assets

The function register_asset() may be used to manually register assets of any type.

This is mainly used if you like to create assets that are not generated via patched framework functions, like txt, json, rst files or generated via frameworks not yet supported, like seaborn.

Here some examples:

register_asset("some-path.html", dict(label="my_html_snippet"), template="raw_html")

data = b"""<div><p>This HTML text was generated by an asset script!</p></div>"""
register_asset("raw.html", dict(label="my_html_snippet"), template="raw_html", data=io.BytesIO(data))

data = b""".. important:: This reStructuredText text was generated by an asset script!"""
register_asset("raw.rst", dict(label="my_rst_snippet"), template="raw_rst", data=io.BytesIO(data))

data = b'This text was generated by an asset script and will be included via "literalinclude"'
register_asset("raw.txt", dict(label="my_txt_snippet"), template="raw_txt", data=io.BytesIO(data))

Catching Errors

If uncaught exceptions occur in asset scripts, Pharaoh will catch them and log them to the console as well as fail the generation process and report to the user.

Sometimes this might not be desired, especially for bigger reports where errors in asset scripts might occur occasionally, but nevertheless the report should be generated.

To catch errors and include them in the report, you can use pharaoh.assetlib.api.catch_exceptions() as a context manager:

from pharaoh.assetlib.api import catch_exceptions

with catch_exceptions():
    generate_asset_that_raises(...)

Important

If you want to include the exception message in the report instead of the actual asset which failed generating, make sure catch_exceptions is used within the same metadata_context .

# do this!
with metadata_context(foo="bar"):  # error asset will have the same metadata
    with catch_exceptions():
        generate_asset_that_raises(...)

# NOT this!
with catch_exceptions():  # error asset won't have the same metadata
    with metadata_context(foo="bar"):
        generate_asset_that_raises(...)
# Now the Sphinx directive `.. pharaoh-asset::` will include the error message instead of the asset.

In the report, the rendered error message & traceback will look like this:

This is a ValueError
Traceback (most recent call last):
  File "C:\Users\loibljoh\AppData\Local\Temp\pytest-of-loibljoh\pytest-1342\test_error_report_component_wi0\project\report-project\components\dummy\asset_scripts\assets.py", line 5, in <module>
    raise ValueError("This is a ValueError")
ValueError: This is a ValueError

To explicitly render the error asset in the report template, you can use the following code:

{# Search all error assets in the current component #}
{% for asset in search_error_assets() %}
.. pharaoh-asset:: {{ asset.id }}

{% endfor %}


{# Search all error assets in all components #}
{% for component, assets in search_error_assets_global().items() %}
{{ h3(component) }}

{% for asset in assets %}
.. pharaoh-asset:: {{ asset.id }}

{% endfor %}
{% endfor %}

Patched Frameworks

Pharaoh monkey-patches plot/table export functions of popular tools like Pandas, Plotly etc.

This has the advantage, that users can work with the official plotting APIs of the respective frameworks without having to take care about Manually Registering Assets.

The following APIs are patched by Pharaoh:

Pandas

Examples:

import pandas as pd

df = pd.DataFrame(
    {
        "blabla": ["a", "b", "c"],
        "ints": [1, 2, 3],
        "float": [1.5, 2.5, 3.5],
        "hex": ["0x11", "0x12", "0x13"],
    }
)
df.to_html(buf="table.html")
Plotly

Examples:

import plotly.express as px

fig = px.scatter(
    data_frame=px.data.iris(), x="sepal_width", y="sepal_length",
    color="species", symbol="species", title=r"A title",
)

fig.write_image(file="iris_scatter1.svg")
fig.write_html(file="iris_scatter2.html")
fig.write_image(file="iris_scatter3.png", width=500, height=500)

Bokeh

Examples:

from bokeh.io import save
from bokeh.plotting import figure
from bokeh.sampledata.iris import flowers

colormap = {"setosa": "red", "versicolor": "green", "virginica": "blue"}
colors = [colormap[x] for x in flowers["species"]]

p = figure(title=f"Iris Morphology", width=400, height=400)
p.xaxis.axis_label = "Petal Length"
p.yaxis.axis_label = "Petal Width"
p.scatter(flowers["petal_length"], flowers["petal_width"], color=colors, fill_alpha=0.2, size=10)

save(p, filename="iris_scatter.html")

Matplotlib

Examples:

import matplotlib.pyplot as plt
import numpy as np

fig, ax = plt.subplots()
ax.plot(np.arange(0.0, 2.0, 0.01), 1 + np.sin(2 * np.pi * t))
ax.set(xlabel='time (s)', ylabel='voltage (mV)', title='About as simple as it gets, folks')
ax.grid()

fig.savefig("test.png")
plt.show()
Holoviews

Examples:

import holoviews as hv
data = [(i, chr(97 + j), i * j) for i in range(5) for j in range(5) if i != j]

with metadata_context(ext="plotly"):
    hv.extension("plotly")
    model = hv.HeatMap(data).opts(cmap="RdBu_r", width=400, height=400)
    hv.save(model, "heatmap_holo_plotly.html")
    hv.save(model, "heatmap_holo_plotly.svg")
    hv.save(model, "heatmap_holo_plotly.png")

with metadata_context(ext="bokeh"):
    hv.extension("bokeh")
    model = hv.HeatMap(data).opts(cmap="RdBu_r", width=400, height=400)
    hv.save(model, "heatmap_holo_bokeh.html")
    hv.save(model, "heatmap_holo_bokeh.png")

with metadata_context(ext="matplotlib"):
    hv.extension("matplotlib")
    model = hv.HeatMap(data).opts(cmap="RdBu_r")
    hv.save(model, "heatmap_holo_mpl.svg")
    hv.save(model, "heatmap_holo_mpl.png")

plotly.graph_objects.Figure.write_html is one of many function which is

by Pharaoh. More info about the other patched frameworks you can find here.

If the asset script is executed via Pharaoh, the patched write_html function will not store the plot to "iris_scatter.html" in the current working directory, but rather in a predefined location for assets inside your Pharaoh project report_project/.asset_build/<component-name> with a unique suffix, for example iris_scatter_9c30799b.html.

Force Static Exports

While for debugging and interactive review it’s nice to have all plots rendered as dynamic elements in HTML, other build targets might be used for export purposes like Confluence or LaTeX.

Since those targets don’t support embedding dynamic HTML elements you would have to tweak your asset generation scripts to export different formats for different report output formats, which is suboptimal.

Pharaoh therefore provides a setting asset_gen.force_static which can be used to signal asset scripts to export static assets instead of HTML.

For all patched plotting APIs, if asset_gen.force_static is set to true, Pharaoh takes care and exports static images instead of HTML when plotting APIs like plotly.io.write_html() is used.

Since not all frameworks are patched, you might have to add support by yourself inside asset scripts like this:

import altair as alt  # altair not supported at the moment

from pharaoh.api import get_project
from pharaoh.assetlib.api import register_asset

pharaoh_project = get_project(__file__)
force_static = project.get_setting("asset_gen.force_static")

chart = alt.Chart(...)

if force_static:
    filename = "chart.png"
else:
    filename = "chart.html"

chart.save(filename)
register_asset(filename, dict(plotting_framework="altair"))

Matlab Integration

Pharaoh supports generating assets via Matlab scripts/function through the Matlab API pharaoh.assetlib.api.Matlab.

The asset scripts still have to be Python scripts and any asset the Matlab scripts generate, have to be manually registered using register_asset().

Here an example:

from pharaoh.assetlib.api import Matlab, register_asset, FileResource, get_resource

resource: FileResource = get_resource(alias="<resource_alias>")
some_resource_path: Path = resource.locate()

with Matlab() as matlab:
    plot_path, out, err = matlab.execute_function("generate_plot", [str(some_resource_path)], nargout=1)

register_asset(plot_path, dict(from_matlab=True, foo="bar"))

To use the Matlab API you have to install an additional dependency, depending on your Matlab version:

  • R2020B: pip install matlabengine==9.9.*

  • R2021A: pip install matlabengine==9.10.*

  • R2021B: pip install matlabengine==9.11.*

  • R2022A: pip install matlabengine==9.12.*

  • R2022B: pip install matlabengine==9.13.*

  • R2023A: pip install matlabengine==9.14.*

  • R2023B: pip install matlabengine==9.15.*

Asset Lookup

This section deals with how to access assets in local context scripts and build-time templates.

Asset Finder

The AssetFinder is responsible for discovering and searching assets based on filters using AssetFinder.search_assets().

This function is available in various places:

  • In local context scripts.

    The following script searches all JSON files that have "context_name" set, loads them and exports their content as context for build-time templating:

    import json
    from pharaoh.assetlib.api import get_asset_finder, get_current_component
    
    finder = get_asset_finder()
    component_name = get_current_component()
    
    context = {
        asset.context.context_name: json.loads(asset.read_text())
        for asset in finder.search_assets(
            'asset.suffix == ".json" and "context_name" in asset.context',
            [component_name]
        )
    }
    context["component_name"] = component_name
    return context
    

    In some use cases this is used to make measurement information available during templating to render the information in a table.

  • In templates inside Jinja statements.

    The following snippet searches all image assets of the current component (the component the template is rendered in) and stores the list of assets in a variable:

    {% set image_assets = search_assets("asset.suffix in ['.png', '.svg']") %}
    

    The following snippet searches all image assets of the all components in the project. This can be used to create components that summarize information from other components.

    {% set all_image_assets = search_assets_global("asset.suffix in ['.png', '.svg']") %}
    

    The used functions search_assets and search_assets_global are partial functions where the component argument is preset.

Manual Include

If you like to include some of your assets manually, the template environment provides two functions for your convenience, that return the (relative) path of an asset to the including template:

asset_rel_path_from_project(asset)

Returns the relative path from the Sphinx source directory to the passed asset. This path is needed for Sphinx directives that themselves take care to copy the asset into the build directory, like directives literalinclude, image and figure.

Here’s an example:

{% set matches = search_assets("<some condition>") %}

.. figure:: {{ asset_rel_path_from_project(matches[0]) }}
   :scale: 50 %

   This is the caption of the figure (a simple paragraph).

where {{ asset_rel_path_from_project(matches[0]) }} would render something like this: /.asset_build/dummy/mytxt_f65223c4.txt

asset_rel_path_from_build(asset)

Returns the relative path from the Sphinx build directory to the passed asset. This path is needed for Sphinx directives that themselves do NOT take care to copy the asset into the build directory, like when using the raw_html directive and including a picture or HTMl file using an iframe.

In this case calling asset_rel_path_from_build will automatically copy the asset from the asset build folder to the build directory and return it’s relative path from the including template.

Here’s an example:

{% set matches = search_assets("<some condition>") %}

.. raw:: html

    <iframe src="{{ asset_rel_path_from_build(matches[0]) }}"
        loading="lazy"
    ></iframe><br>

where {{ asset_rel_path_from_build(matches[0]) }} would render something like this: ../../pharaoh_assets/iris_scatter_3e7d7ab7.html

Note

Another way to include assets with a custom template is to use the template option of the Pharaoh asset directive with a path to your custom template.

Grouping Assets By Metadata

Imaging in your asset scripts you are creating a plot for a measured signal for many different operating conditions:

from pharaoh.assetlib.api import metadata_context

for vdd in (8.0, 10.0, 12.0):
    for iout in (1.0, 2.0, 5.0):
        with metadata_context(vdd=vdd, iout=iout, signal_name="idd"):
            fig = ...
            fig.write_image(file="idd_plot.png")

Now in your report you may like to have all those plots added separately each with it’s own title containing the values of either vdd or iout.

Defining the template like this is not really viable, since in the template you usually have no idea about what values were iterated over in the asset script. But let’s assume you know, then this would be the template:

{% for vdd in (8.0, 10.0, 12.0) %}
{{ heading("Plots for Vdd:%.1fV, Iout:%.1fA" % vout, 2) }}

{% for iout in (1.0, 2.0, 5.0) %}
{{ heading("Plots for Iout:%.1fA" % iout, 3) }}

.. pharaoh-asset:: vdd == {{ vdd }} and iout == {{ iout }} and signal_name == 'idd'

{% endfor %}
{% endfor %}

But luckily there is a better option using asset_groupby to group assets first by vdd and then by iout:

{% set idd_plots = search_assets("signal_name == 'idd'") %}
{% for vdd, assets_grby_vdd in agroupby(idd_plots, key="vdd").items() %}
{{ heading("Plots for Vdd:%.1fV" % vout, 2) }}

    {% for iout, assets_grby_iout in agroupby(assets_grby_vdd, key="iout").items() %}
{{ heading("Plots for Iout:%.1fA" % iout, 3) }}

        {% for asset in assets_grby_iout %}
.. pharaoh-asset:: {{ asset.id }}

        {% endfor %}
    {% endfor %}
{% endfor %}

Note

agroupby is an alias for asset_groupby. Both are available as global function during templating.