import io
import pathlib
from typing import Optional, Union
import plotly.graph_objects as go
from jinja2 import Environment, FileSystemLoader
from qibocal.auto.history import History
from qibocal.auto.operation import QubitId, QubitPairId
from qibocal.auto.output import Output
from qibocal.auto.task import Completed
from qibocal.config import log
from qibocal.web.report import (
SCRIPT,
STYLES,
TEMPLATES,
Report,
report_css_styles,
report_script,
)
ReportOutcome = tuple[str, list[go.Figure]]
"""Report produced by protocol."""
[docs]def plotter(
node: Completed, target: Union[QubitId, QubitPairId, list[QubitId]]
) -> tuple[str, str]:
"""Run plotly pipeline for generating html.
Performs conversions of plotly figures in html rendered code for
completed node on specific target.
"""
figures, fitting_report = generate_figures_and_report(node, target)
buffer = io.StringIO()
html_list = []
for figure in figures:
figure.write_html(buffer, include_plotlyjs=False, full_html=False)
buffer.seek(0)
html_list.append(buffer.read())
buffer.close()
all_html = "".join(html_list)
return all_html, fitting_report
[docs]def report(path: pathlib.Path, history: Optional[History] = None):
"""Report generation.
Generates the report for protocol dumped in `path`.
Executor can be passed to generate report on the fly.
"""
if (path / "index.html").exists(): # pragma: no cover
log.warning(f"Regenerating {path}/index.html")
# load meta
output = Output.load(path)
if history is None:
history = output.history
env = Environment(loader=FileSystemLoader(TEMPLATES))
template = env.get_template("template.html")
html = template.render(
is_static=True,
css_styles=report_css_styles(STYLES),
js_script=report_script(SCRIPT),
path=path,
title=path.name,
report=Report(
path=path,
history=history,
meta=output.meta.dump(),
plotter=plotter,
),
)
(path / "index.html").write_text(html)