mirror of
https://github.com/bec-project/bec_widgets.git
synced 2026-05-09 08:12:15 +02:00
Compare commits
17 Commits
| Author | SHA1 | Date | |
|---|---|---|---|
| 603edede9c | |||
| 30ef25533a | |||
| 73b44cffb2 | |||
| a614d662d6 | |||
| 3f1aa80756 | |||
| 409c9e5bfa | |||
| 19b5c8f724 | |||
| 5056ef8946 | |||
| 551d38d901 | |||
| 999b7a2321 | |||
| 5dc373bd8e | |||
| 91afc775d5 | |||
| 55694ff2b9 | |||
| 5b68a51aaa | |||
| f13fa75e25 | |||
| 0cf84cd1d8 | |||
| 3e77f54034 |
@@ -0,0 +1,169 @@
|
|||||||
|
##########################
|
||||||
|
### AI-generated file. ###
|
||||||
|
##########################
|
||||||
|
|
||||||
|
"""Aggregate and merge benchmark JSON files.
|
||||||
|
|
||||||
|
The workflow runs the same benchmark suite on multiple independent runners.
|
||||||
|
This script reads every JSON file produced by those attempts, normalizes the
|
||||||
|
contained benchmark values, and writes a compact mapping JSON where each value is
|
||||||
|
the median across attempts. It can also merge independent hyperfine JSON files
|
||||||
|
from one runner into a single hyperfine-style JSON file.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import json
|
||||||
|
import statistics
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
from compare_benchmarks import Benchmark, extract_benchmarks
|
||||||
|
|
||||||
|
|
||||||
|
def collect_benchmarks(paths: list[Path]) -> dict[str, list[Benchmark]]:
|
||||||
|
"""Collect benchmarks from multiple JSON files.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
paths (list[Path]): Paths to hyperfine, pytest-benchmark, or compact
|
||||||
|
mapping JSON files.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, list[Benchmark]]: Benchmarks grouped by benchmark name.
|
||||||
|
"""
|
||||||
|
|
||||||
|
collected: dict[str, list[Benchmark]] = {}
|
||||||
|
for path in paths:
|
||||||
|
for name, benchmark in extract_benchmarks(path).items():
|
||||||
|
collected.setdefault(name, []).append(benchmark)
|
||||||
|
return collected
|
||||||
|
|
||||||
|
|
||||||
|
def aggregate(collected: dict[str, list[Benchmark]]) -> dict[str, dict[str, object]]:
|
||||||
|
"""Aggregate grouped benchmarks using the median value.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
collected (dict[str, list[Benchmark]]): Benchmarks grouped by benchmark
|
||||||
|
name.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, dict[str, object]]: Compact mapping JSON data. Each benchmark
|
||||||
|
contains ``value``, ``unit``, ``metric``, ``attempts``, and
|
||||||
|
``attempt_values``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
aggregated: dict[str, dict[str, object]] = {}
|
||||||
|
for name, benchmarks in sorted(collected.items()):
|
||||||
|
values = [benchmark.value for benchmark in benchmarks]
|
||||||
|
unit = next((benchmark.unit for benchmark in benchmarks if benchmark.unit), "")
|
||||||
|
metric = next((benchmark.metric for benchmark in benchmarks if benchmark.metric), "value")
|
||||||
|
aggregated[name] = {
|
||||||
|
"value": statistics.median(values),
|
||||||
|
"unit": unit,
|
||||||
|
"metric": f"median-of-attempt-{metric}",
|
||||||
|
"attempts": len(values),
|
||||||
|
"attempt_values": values,
|
||||||
|
}
|
||||||
|
return aggregated
|
||||||
|
|
||||||
|
|
||||||
|
def merge_hyperfine_results(paths: list[Path]) -> dict[str, Any]:
|
||||||
|
"""Merge hyperfine result files.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
paths (list[Path]): Hyperfine JSON files to merge.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, Any]: Hyperfine-style JSON object containing all result rows.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: If any file has no hyperfine ``results`` list.
|
||||||
|
"""
|
||||||
|
|
||||||
|
merged: dict[str, Any] = {"results": []}
|
||||||
|
for path in paths:
|
||||||
|
data = json.loads(path.read_text(encoding="utf-8"))
|
||||||
|
results = data.get("results", []) if isinstance(data, dict) else None
|
||||||
|
if not isinstance(results, list):
|
||||||
|
raise ValueError(f"{path} has no hyperfine results list")
|
||||||
|
merged["results"].extend(results)
|
||||||
|
return merged
|
||||||
|
|
||||||
|
|
||||||
|
def main_from_paths(input_dir: Path, output: Path) -> int:
|
||||||
|
"""Aggregate all JSON files in a directory and write the result.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
input_dir (Path): Directory containing benchmark JSON files.
|
||||||
|
output (Path): Path where the aggregate JSON should be written.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
int: Always ``0`` on success.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: If no JSON files are found in ``input_dir``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
paths = sorted(input_dir.rglob("*.json"))
|
||||||
|
if not paths:
|
||||||
|
raise ValueError(f"No benchmark JSON files found in {input_dir}")
|
||||||
|
|
||||||
|
output.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
output.write_text(
|
||||||
|
json.dumps(aggregate(collect_benchmarks(paths)), indent=2, sort_keys=True) + "\n",
|
||||||
|
encoding="utf-8",
|
||||||
|
)
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
def merge_from_paths(input_dir: Path, output: Path) -> int:
|
||||||
|
"""Merge all hyperfine JSON files in a directory and write the result.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
input_dir (Path): Directory containing hyperfine JSON files.
|
||||||
|
output (Path): Path where the merged JSON should be written.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
int: Always ``0`` on success.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: If no JSON files are found in ``input_dir``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
paths = sorted(input_dir.glob("*.json"))
|
||||||
|
if not paths:
|
||||||
|
raise ValueError(f"No hyperfine JSON files found in {input_dir}")
|
||||||
|
|
||||||
|
output.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
output.write_text(
|
||||||
|
json.dumps(merge_hyperfine_results(paths), indent=2, sort_keys=True) + "\n",
|
||||||
|
encoding="utf-8",
|
||||||
|
)
|
||||||
|
return 0
|
||||||
|
|
||||||
|
|
||||||
|
def main() -> int:
|
||||||
|
"""Run the benchmark aggregation command line interface.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
int: Always ``0`` on success.
|
||||||
|
"""
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument(
|
||||||
|
"--mode",
|
||||||
|
choices=("aggregate", "merge-hyperfine"),
|
||||||
|
default="aggregate",
|
||||||
|
help="Operation to perform.",
|
||||||
|
)
|
||||||
|
parser.add_argument("--input-dir", required=True, type=Path)
|
||||||
|
parser.add_argument("--output", required=True, type=Path)
|
||||||
|
args = parser.parse_args()
|
||||||
|
if args.mode == "merge-hyperfine":
|
||||||
|
return merge_from_paths(input_dir=args.input_dir, output=args.output)
|
||||||
|
return main_from_paths(input_dir=args.input_dir, output=args.output)
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
raise SystemExit(main())
|
||||||
@@ -0,0 +1,454 @@
|
|||||||
|
##########################
|
||||||
|
### AI-generated file. ###
|
||||||
|
##########################
|
||||||
|
|
||||||
|
"""Compare benchmark JSON files and write a GitHub Actions summary.
|
||||||
|
|
||||||
|
The script supports JSON emitted by hyperfine, JSON emitted by pytest-benchmark,
|
||||||
|
and a compact mapping format generated by ``aggregate_benchmarks.py``. Timing
|
||||||
|
formats prefer median values and fall back to mean values when median values are
|
||||||
|
not present.
|
||||||
|
"""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import json
|
||||||
|
import math
|
||||||
|
from dataclasses import dataclass
|
||||||
|
from pathlib import Path
|
||||||
|
from typing import Any
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class Benchmark:
|
||||||
|
"""Normalized benchmark result.
|
||||||
|
|
||||||
|
Attributes:
|
||||||
|
name (str): Stable benchmark name used to match baseline and current results.
|
||||||
|
value (float): Numeric benchmark value used for comparison.
|
||||||
|
unit (str): Display unit for the value, for example ``"s"``.
|
||||||
|
metric (str): Source metric name, for example ``"median"`` or ``"mean"``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
name: str
|
||||||
|
value: float
|
||||||
|
unit: str
|
||||||
|
metric: str = "value"
|
||||||
|
|
||||||
|
|
||||||
|
@dataclass(frozen=True)
|
||||||
|
class Comparison:
|
||||||
|
"""Comparison between one baseline benchmark and one current benchmark.
|
||||||
|
|
||||||
|
Attributes:
|
||||||
|
name (str): Benchmark name.
|
||||||
|
baseline (float): Baseline benchmark value.
|
||||||
|
current (float): Current benchmark value.
|
||||||
|
delta_percent (float): Percent change from baseline to current.
|
||||||
|
unit (str): Display unit for both values.
|
||||||
|
metric (str): Current result metric used for comparison.
|
||||||
|
regressed (bool): Whether the change exceeds the configured threshold in
|
||||||
|
the worse direction.
|
||||||
|
improved (bool): Whether the change exceeds the configured threshold in
|
||||||
|
the better direction.
|
||||||
|
"""
|
||||||
|
|
||||||
|
name: str
|
||||||
|
baseline: float
|
||||||
|
current: float
|
||||||
|
delta_percent: float
|
||||||
|
unit: str
|
||||||
|
metric: str
|
||||||
|
regressed: bool
|
||||||
|
improved: bool
|
||||||
|
|
||||||
|
|
||||||
|
def _read_json(path: Path) -> Any:
|
||||||
|
"""Read JSON data from a file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
path (Path): Path to the JSON file.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
Any: Parsed JSON value.
|
||||||
|
"""
|
||||||
|
|
||||||
|
with path.open("r", encoding="utf-8") as stream:
|
||||||
|
return json.load(stream)
|
||||||
|
|
||||||
|
|
||||||
|
def _as_float(value: Any) -> float | None:
|
||||||
|
"""Convert a value to a finite float.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
value (Any): Value to convert.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
float | None: Converted finite float, or ``None`` if conversion fails.
|
||||||
|
"""
|
||||||
|
|
||||||
|
try:
|
||||||
|
result = float(value)
|
||||||
|
except (TypeError, ValueError):
|
||||||
|
return None
|
||||||
|
if math.isfinite(result):
|
||||||
|
return result
|
||||||
|
return None
|
||||||
|
|
||||||
|
|
||||||
|
def _extract_hyperfine(data: dict[str, Any]) -> dict[str, Benchmark]:
|
||||||
|
"""Extract normalized benchmarks from hyperfine JSON.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
data (dict[str, Any]): Parsed hyperfine JSON object.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, Benchmark]: Benchmarks keyed by command name.
|
||||||
|
"""
|
||||||
|
|
||||||
|
benchmarks: dict[str, Benchmark] = {}
|
||||||
|
for result in data.get("results", []):
|
||||||
|
if not isinstance(result, dict):
|
||||||
|
continue
|
||||||
|
name = str(result.get("command") or result.get("name") or "").strip()
|
||||||
|
metric = "median"
|
||||||
|
value = _as_float(result.get(metric))
|
||||||
|
if value is None:
|
||||||
|
metric = "mean"
|
||||||
|
value = _as_float(result.get(metric))
|
||||||
|
if name and value is not None:
|
||||||
|
benchmarks[name] = Benchmark(name=name, value=value, unit="s", metric=metric)
|
||||||
|
return benchmarks
|
||||||
|
|
||||||
|
|
||||||
|
def _extract_pytest_benchmark(data: dict[str, Any]) -> dict[str, Benchmark]:
|
||||||
|
"""Extract normalized benchmarks from pytest-benchmark JSON.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
data (dict[str, Any]): Parsed pytest-benchmark JSON object.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, Benchmark]: Benchmarks keyed by full benchmark name.
|
||||||
|
"""
|
||||||
|
|
||||||
|
benchmarks: dict[str, Benchmark] = {}
|
||||||
|
for benchmark in data.get("benchmarks", []):
|
||||||
|
if not isinstance(benchmark, dict):
|
||||||
|
continue
|
||||||
|
|
||||||
|
name = str(benchmark.get("fullname") or benchmark.get("name") or "").strip()
|
||||||
|
stats = benchmark.get("stats", {})
|
||||||
|
value = None
|
||||||
|
metric = "median"
|
||||||
|
if isinstance(stats, dict):
|
||||||
|
value = _as_float(stats.get(metric))
|
||||||
|
if value is None:
|
||||||
|
metric = "mean"
|
||||||
|
value = _as_float(stats.get(metric))
|
||||||
|
if name and value is not None:
|
||||||
|
benchmarks[name] = Benchmark(name=name, value=value, unit="s", metric=metric)
|
||||||
|
return benchmarks
|
||||||
|
|
||||||
|
|
||||||
|
def _extract_simple_mapping(data: dict[str, Any]) -> dict[str, Benchmark]:
|
||||||
|
"""Extract normalized benchmarks from a compact mapping JSON object.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
data (dict[str, Any]): Parsed mapping where each benchmark is either a
|
||||||
|
raw number or an object containing ``value``, ``unit``, and ``metric``.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, Benchmark]: Benchmarks keyed by mapping key.
|
||||||
|
"""
|
||||||
|
|
||||||
|
benchmarks: dict[str, Benchmark] = {}
|
||||||
|
|
||||||
|
for name, raw_value in data.items():
|
||||||
|
if name in {"version", "context", "commit", "timestamp"}:
|
||||||
|
continue
|
||||||
|
|
||||||
|
value = _as_float(raw_value)
|
||||||
|
unit = ""
|
||||||
|
metric = "value"
|
||||||
|
if value is None and isinstance(raw_value, dict):
|
||||||
|
value = _as_float(raw_value.get("value"))
|
||||||
|
unit = str(raw_value.get("unit") or "")
|
||||||
|
metric = str(raw_value.get("metric") or "value")
|
||||||
|
|
||||||
|
if value is not None:
|
||||||
|
benchmarks[str(name)] = Benchmark(name=str(name), value=value, unit=unit, metric=metric)
|
||||||
|
|
||||||
|
return benchmarks
|
||||||
|
|
||||||
|
|
||||||
|
def extract_benchmarks(path: Path) -> dict[str, Benchmark]:
|
||||||
|
"""Extract normalized benchmarks from a supported JSON file.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
path (Path): Path to a hyperfine, pytest-benchmark, or compact mapping
|
||||||
|
JSON file.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
dict[str, Benchmark]: Normalized benchmarks keyed by name.
|
||||||
|
|
||||||
|
Raises:
|
||||||
|
ValueError: If the JSON root is not an object or no supported benchmark
|
||||||
|
entries can be extracted.
|
||||||
|
"""
|
||||||
|
|
||||||
|
data = _read_json(path)
|
||||||
|
if not isinstance(data, dict):
|
||||||
|
raise ValueError(f"{path} must contain a JSON object")
|
||||||
|
|
||||||
|
extractors = (_extract_hyperfine, _extract_pytest_benchmark, _extract_simple_mapping)
|
||||||
|
for extractor in extractors:
|
||||||
|
benchmarks = extractor(data)
|
||||||
|
if benchmarks:
|
||||||
|
return benchmarks
|
||||||
|
|
||||||
|
raise ValueError(f"No supported benchmark entries found in {path}")
|
||||||
|
|
||||||
|
|
||||||
|
def compare_benchmarks(
|
||||||
|
baseline: dict[str, Benchmark],
|
||||||
|
current: dict[str, Benchmark],
|
||||||
|
threshold_percent: float,
|
||||||
|
higher_is_better: bool,
|
||||||
|
) -> tuple[list[Comparison], list[str], list[str]]:
|
||||||
|
"""Compare baseline benchmarks with current benchmarks.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
baseline (dict[str, Benchmark]): Baseline benchmarks keyed by name.
|
||||||
|
current (dict[str, Benchmark]): Current benchmarks keyed by name.
|
||||||
|
threshold_percent (float): Regression threshold in percent.
|
||||||
|
higher_is_better (bool): If ``True``, lower current values are treated as
|
||||||
|
regressions. If ``False``, higher current values are treated as
|
||||||
|
regressions.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
tuple[list[Comparison], list[str], list[str]]: Comparisons for common
|
||||||
|
benchmark names, names missing from current results, and names newly
|
||||||
|
present in current results.
|
||||||
|
"""
|
||||||
|
|
||||||
|
comparisons: list[Comparison] = []
|
||||||
|
missing_in_current: list[str] = []
|
||||||
|
new_in_current: list[str] = []
|
||||||
|
|
||||||
|
for name, baseline_benchmark in sorted(baseline.items()):
|
||||||
|
current_benchmark = current.get(name)
|
||||||
|
if current_benchmark is None:
|
||||||
|
missing_in_current.append(name)
|
||||||
|
continue
|
||||||
|
|
||||||
|
if baseline_benchmark.value == 0:
|
||||||
|
delta_percent = 0.0
|
||||||
|
else:
|
||||||
|
delta_percent = (
|
||||||
|
(current_benchmark.value - baseline_benchmark.value)
|
||||||
|
/ abs(baseline_benchmark.value)
|
||||||
|
* 100
|
||||||
|
)
|
||||||
|
|
||||||
|
if higher_is_better:
|
||||||
|
regressed = delta_percent <= -threshold_percent
|
||||||
|
improved = delta_percent >= threshold_percent
|
||||||
|
else:
|
||||||
|
regressed = delta_percent >= threshold_percent
|
||||||
|
improved = delta_percent <= -threshold_percent
|
||||||
|
|
||||||
|
comparisons.append(
|
||||||
|
Comparison(
|
||||||
|
name=name,
|
||||||
|
baseline=baseline_benchmark.value,
|
||||||
|
current=current_benchmark.value,
|
||||||
|
delta_percent=delta_percent,
|
||||||
|
unit=current_benchmark.unit or baseline_benchmark.unit,
|
||||||
|
metric=current_benchmark.metric,
|
||||||
|
regressed=regressed,
|
||||||
|
improved=improved,
|
||||||
|
)
|
||||||
|
)
|
||||||
|
|
||||||
|
for name in sorted(set(current) - set(baseline)):
|
||||||
|
new_in_current.append(name)
|
||||||
|
|
||||||
|
return comparisons, missing_in_current, new_in_current
|
||||||
|
|
||||||
|
|
||||||
|
def _format_value(value: float, unit: str) -> str:
|
||||||
|
"""Format a benchmark value for Markdown output.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
value (float): Numeric benchmark value.
|
||||||
|
unit (str): Display unit.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: Formatted value with optional unit suffix.
|
||||||
|
"""
|
||||||
|
|
||||||
|
suffix = f" {unit}" if unit else ""
|
||||||
|
return f"{value:.6g}{suffix}"
|
||||||
|
|
||||||
|
|
||||||
|
def _format_status(comparison: Comparison) -> str:
|
||||||
|
"""Format a comparison status for Markdown output."""
|
||||||
|
|
||||||
|
if comparison.regressed:
|
||||||
|
return ":red_circle: regressed"
|
||||||
|
if comparison.improved:
|
||||||
|
return ":green_circle: improved"
|
||||||
|
return "ok"
|
||||||
|
|
||||||
|
|
||||||
|
def write_summary(
|
||||||
|
path: Path,
|
||||||
|
comparisons: list[Comparison],
|
||||||
|
missing_in_current: list[str],
|
||||||
|
new_in_current: list[str],
|
||||||
|
threshold_percent: float,
|
||||||
|
higher_is_better: bool,
|
||||||
|
) -> None:
|
||||||
|
"""Write a Markdown benchmark comparison summary.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
path (Path): Path where the summary should be written.
|
||||||
|
comparisons (list[Comparison]): Comparison rows for matching benchmarks.
|
||||||
|
missing_in_current (list[str]): Baseline benchmark names missing from the
|
||||||
|
current result.
|
||||||
|
new_in_current (list[str]): Current benchmark names not present in the
|
||||||
|
baseline result.
|
||||||
|
threshold_percent (float): Regression threshold in percent.
|
||||||
|
higher_is_better (bool): Whether higher benchmark values are considered
|
||||||
|
better.
|
||||||
|
"""
|
||||||
|
|
||||||
|
regressions = [comparison for comparison in comparisons if comparison.regressed]
|
||||||
|
improvements = [comparison for comparison in comparisons if comparison.improved]
|
||||||
|
direction = "higher is better" if higher_is_better else "lower is better"
|
||||||
|
sorted_comparisons = sorted(comparisons, key=lambda comparison: comparison.name)
|
||||||
|
|
||||||
|
lines = [
|
||||||
|
"<!-- bw-benchmark-comment -->",
|
||||||
|
"## Benchmark comparison",
|
||||||
|
"",
|
||||||
|
f"Threshold: {threshold_percent:g}% ({direction}).",
|
||||||
|
f"Result: {len(regressions)} regression(s), {len(improvements)} improvement(s) beyond threshold.",
|
||||||
|
]
|
||||||
|
lines.append("")
|
||||||
|
|
||||||
|
if regressions:
|
||||||
|
lines.extend(
|
||||||
|
[
|
||||||
|
f"{len(regressions)} benchmark(s) regressed beyond the configured threshold.",
|
||||||
|
"",
|
||||||
|
"| Benchmark | Baseline | Current | Change |",
|
||||||
|
"| --- | ---: | ---: | ---: |",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
for comparison in regressions:
|
||||||
|
lines.append(
|
||||||
|
"| "
|
||||||
|
f"{comparison.name} | "
|
||||||
|
f"{_format_value(comparison.baseline, comparison.unit)} | "
|
||||||
|
f"{_format_value(comparison.current, comparison.unit)} | "
|
||||||
|
f"{comparison.delta_percent:+.2f}% |"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
lines.append("No benchmark regression exceeded the configured threshold.")
|
||||||
|
|
||||||
|
lines.append("")
|
||||||
|
|
||||||
|
if improvements:
|
||||||
|
lines.extend(
|
||||||
|
[
|
||||||
|
f"{len(improvements)} benchmark(s) improved beyond the configured threshold.",
|
||||||
|
"",
|
||||||
|
"| Benchmark | Baseline | Current | Change |",
|
||||||
|
"| --- | ---: | ---: | ---: |",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
for comparison in improvements:
|
||||||
|
lines.append(
|
||||||
|
"| "
|
||||||
|
f"{comparison.name} | "
|
||||||
|
f"{_format_value(comparison.baseline, comparison.unit)} | "
|
||||||
|
f"{_format_value(comparison.current, comparison.unit)} | "
|
||||||
|
f"{comparison.delta_percent:+.2f}% |"
|
||||||
|
)
|
||||||
|
else:
|
||||||
|
lines.append("No benchmark improvement exceeded the configured threshold.")
|
||||||
|
|
||||||
|
if sorted_comparisons:
|
||||||
|
lines.extend(
|
||||||
|
[
|
||||||
|
"",
|
||||||
|
"<details>",
|
||||||
|
"<summary>All benchmark results</summary>",
|
||||||
|
"",
|
||||||
|
"| Benchmark | Baseline | Current | Change | Status |",
|
||||||
|
"| --- | ---: | ---: | ---: | --- |",
|
||||||
|
]
|
||||||
|
)
|
||||||
|
for comparison in sorted_comparisons:
|
||||||
|
lines.append(
|
||||||
|
"| "
|
||||||
|
f"{comparison.name} | "
|
||||||
|
f"{_format_value(comparison.baseline, comparison.unit)} | "
|
||||||
|
f"{_format_value(comparison.current, comparison.unit)} | "
|
||||||
|
f"{comparison.delta_percent:+.2f}% | "
|
||||||
|
f"{_format_status(comparison)} |"
|
||||||
|
)
|
||||||
|
lines.extend(["", "</details>"])
|
||||||
|
|
||||||
|
if missing_in_current:
|
||||||
|
lines.extend(["", "Missing benchmarks in the current run:"])
|
||||||
|
lines.extend(f"- `{name}`" for name in missing_in_current)
|
||||||
|
|
||||||
|
if new_in_current:
|
||||||
|
lines.extend(["", "New benchmarks in the current run:"])
|
||||||
|
lines.extend(f"- `{name}`" for name in new_in_current)
|
||||||
|
|
||||||
|
path.parent.mkdir(parents=True, exist_ok=True)
|
||||||
|
path.write_text("\n".join(lines) + "\n", encoding="utf-8")
|
||||||
|
|
||||||
|
|
||||||
|
def main() -> int:
|
||||||
|
"""Run the benchmark comparison command line interface.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
int: ``1`` when a regression exceeds the threshold, otherwise ``0``.
|
||||||
|
"""
|
||||||
|
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument("--baseline", required=True, type=Path)
|
||||||
|
parser.add_argument("--current", required=True, type=Path)
|
||||||
|
parser.add_argument("--summary", required=True, type=Path)
|
||||||
|
parser.add_argument("--threshold-percent", required=True, type=float)
|
||||||
|
parser.add_argument("--higher-is-better", action="store_true")
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
baseline = extract_benchmarks(args.baseline)
|
||||||
|
current = extract_benchmarks(args.current)
|
||||||
|
comparisons, missing_in_current, new_in_current = compare_benchmarks(
|
||||||
|
baseline=baseline,
|
||||||
|
current=current,
|
||||||
|
threshold_percent=args.threshold_percent,
|
||||||
|
higher_is_better=args.higher_is_better,
|
||||||
|
)
|
||||||
|
|
||||||
|
write_summary(
|
||||||
|
path=args.summary,
|
||||||
|
comparisons=comparisons,
|
||||||
|
missing_in_current=missing_in_current,
|
||||||
|
new_in_current=new_in_current,
|
||||||
|
threshold_percent=args.threshold_percent,
|
||||||
|
higher_is_better=args.higher_is_better,
|
||||||
|
)
|
||||||
|
|
||||||
|
return 1 if any(comparison.regressed for comparison in comparisons) else 0
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
raise SystemExit(main())
|
||||||
@@ -0,0 +1,74 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
|
||||||
|
##########################
|
||||||
|
### AI-generated file. ###
|
||||||
|
##########################
|
||||||
|
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
mkdir -p benchmark-results
|
||||||
|
benchmark_json="${BENCHMARK_JSON:-benchmark-results/current.json}"
|
||||||
|
benchmark_root="$(dirname "$benchmark_json")"
|
||||||
|
hyperfine_benchmark_dir="${BENCHMARK_HYPERFINE_DIR:-tests/benchmarks/hyperfine}"
|
||||||
|
pytest_benchmark_dirs="${BENCHMARK_PYTEST_DIRS:-${BENCHMARK_PYTEST_DIR:-}}"
|
||||||
|
benchmark_work_dir="$benchmark_root/raw-results"
|
||||||
|
hyperfine_json_dir="$benchmark_work_dir/hyperfine"
|
||||||
|
pytest_json="$benchmark_work_dir/pytest.json"
|
||||||
|
|
||||||
|
shopt -s nullglob
|
||||||
|
benchmark_scripts=()
|
||||||
|
benchmark_scripts=("$hyperfine_benchmark_dir"/benchmark_*.sh)
|
||||||
|
shopt -u nullglob
|
||||||
|
|
||||||
|
pytest_dirs=()
|
||||||
|
for pytest_benchmark_dir in $pytest_benchmark_dirs; do
|
||||||
|
if [ -d "$pytest_benchmark_dir" ]; then
|
||||||
|
pytest_dirs+=("$pytest_benchmark_dir")
|
||||||
|
else
|
||||||
|
echo "Pytest benchmark directory not found: $pytest_benchmark_dir" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
done
|
||||||
|
|
||||||
|
if [ "${#benchmark_scripts[@]}" -eq 0 ] && [ "${#pytest_dirs[@]}" -eq 0 ]; then
|
||||||
|
echo "No benchmark scripts or pytest benchmarks found" >&2
|
||||||
|
exit 1
|
||||||
|
fi
|
||||||
|
|
||||||
|
echo "Benchmark Python: $(command -v python)"
|
||||||
|
python -c 'import sys; print(sys.version)'
|
||||||
|
|
||||||
|
rm -rf "$benchmark_work_dir"
|
||||||
|
mkdir -p "$hyperfine_json_dir"
|
||||||
|
|
||||||
|
if [ "${#benchmark_scripts[@]}" -gt 0 ]; then
|
||||||
|
for benchmark_script in "${benchmark_scripts[@]}"; do
|
||||||
|
title="$(sed -n 's/^# BENCHMARK_TITLE:[[:space:]]*//p' "$benchmark_script" | head -n 1)"
|
||||||
|
if [ -z "$title" ]; then
|
||||||
|
title="$(basename "$benchmark_script" .sh)"
|
||||||
|
fi
|
||||||
|
benchmark_name="$(basename "$benchmark_script" .sh)"
|
||||||
|
benchmark_result_json="$hyperfine_json_dir/$benchmark_name.json"
|
||||||
|
echo "Preflight benchmark script: $benchmark_script"
|
||||||
|
bash "$benchmark_script"
|
||||||
|
|
||||||
|
hyperfine \
|
||||||
|
--show-output \
|
||||||
|
--warmup 1 \
|
||||||
|
--runs 5 \
|
||||||
|
--command-name "$title" \
|
||||||
|
--export-json "$benchmark_result_json" \
|
||||||
|
"bash $(printf "%q" "$benchmark_script")"
|
||||||
|
done
|
||||||
|
fi
|
||||||
|
|
||||||
|
if [ "${#pytest_dirs[@]}" -gt 0 ]; then
|
||||||
|
pytest \
|
||||||
|
-q "${pytest_dirs[@]}" \
|
||||||
|
--benchmark-only \
|
||||||
|
--benchmark-json "$pytest_json"
|
||||||
|
fi
|
||||||
|
|
||||||
|
python .github/scripts/aggregate_benchmarks.py \
|
||||||
|
--input-dir "$benchmark_work_dir" \
|
||||||
|
--output "$benchmark_json"
|
||||||
@@ -0,0 +1,125 @@
|
|||||||
|
##########################
|
||||||
|
### AI-generated file. ###
|
||||||
|
##########################
|
||||||
|
|
||||||
|
"""Run a command with BEC e2e services available."""
|
||||||
|
|
||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import argparse
|
||||||
|
import os
|
||||||
|
import shutil
|
||||||
|
import subprocess
|
||||||
|
import tempfile
|
||||||
|
import time
|
||||||
|
from pathlib import Path
|
||||||
|
|
||||||
|
import bec_lib
|
||||||
|
from bec_ipython_client import BECIPythonClient
|
||||||
|
from bec_lib.redis_connector import RedisConnector
|
||||||
|
from bec_lib.service_config import ServiceConfig, ServiceConfigModel
|
||||||
|
from redis import Redis
|
||||||
|
|
||||||
|
|
||||||
|
def _wait_for_redis(host: str, port: int) -> None:
|
||||||
|
client = Redis(host=host, port=port)
|
||||||
|
deadline = time.monotonic() + 10
|
||||||
|
while time.monotonic() < deadline:
|
||||||
|
try:
|
||||||
|
if client.ping():
|
||||||
|
return
|
||||||
|
except Exception:
|
||||||
|
time.sleep(0.1)
|
||||||
|
raise RuntimeError(f"Redis did not start on {host}:{port}")
|
||||||
|
|
||||||
|
|
||||||
|
def _start_redis(files_path: Path, host: str, port: int) -> subprocess.Popen:
|
||||||
|
redis_server = shutil.which("redis-server")
|
||||||
|
if redis_server is None:
|
||||||
|
raise RuntimeError("redis-server executable not found")
|
||||||
|
|
||||||
|
return subprocess.Popen(
|
||||||
|
[
|
||||||
|
redis_server,
|
||||||
|
"--bind",
|
||||||
|
host,
|
||||||
|
"--port",
|
||||||
|
str(port),
|
||||||
|
"--save",
|
||||||
|
"",
|
||||||
|
"--appendonly",
|
||||||
|
"no",
|
||||||
|
"--dir",
|
||||||
|
str(files_path),
|
||||||
|
]
|
||||||
|
)
|
||||||
|
|
||||||
|
|
||||||
|
def _write_configs(files_path: Path, host: str, port: int) -> Path:
|
||||||
|
test_config = files_path / "test_config.yaml"
|
||||||
|
services_config = files_path / "services_config.yaml"
|
||||||
|
|
||||||
|
bec_lib_path = Path(bec_lib.__file__).resolve().parent
|
||||||
|
shutil.copyfile(bec_lib_path / "tests" / "test_config.yaml", test_config)
|
||||||
|
|
||||||
|
service_config = ServiceConfigModel(
|
||||||
|
redis={"host": host, "port": port}, file_writer={"base_path": str(files_path)}
|
||||||
|
)
|
||||||
|
services_config.write_text(service_config.model_dump_json(indent=4), encoding="utf-8")
|
||||||
|
return services_config
|
||||||
|
|
||||||
|
|
||||||
|
def _load_demo_config(services_config: Path) -> None:
|
||||||
|
bec = BECIPythonClient(ServiceConfig(services_config), RedisConnector, forced=True)
|
||||||
|
bec.start()
|
||||||
|
try:
|
||||||
|
bec.config.load_demo_config()
|
||||||
|
finally:
|
||||||
|
bec.shutdown()
|
||||||
|
bec._client._reset_singleton()
|
||||||
|
|
||||||
|
|
||||||
|
def main() -> int:
|
||||||
|
parser = argparse.ArgumentParser()
|
||||||
|
parser.add_argument("command", nargs=argparse.REMAINDER)
|
||||||
|
args = parser.parse_args()
|
||||||
|
|
||||||
|
if args.command[:1] == ["--"]:
|
||||||
|
args.command = args.command[1:]
|
||||||
|
if not args.command:
|
||||||
|
raise ValueError("No command provided")
|
||||||
|
|
||||||
|
host = "127.0.0.1"
|
||||||
|
port = 6379
|
||||||
|
|
||||||
|
with tempfile.TemporaryDirectory(prefix="bec-benchmark-") as tmp:
|
||||||
|
files_path = Path(tmp)
|
||||||
|
services_config = _write_configs(files_path, host, port)
|
||||||
|
redis_process = _start_redis(files_path, host, port)
|
||||||
|
processes = None
|
||||||
|
service_handler = None
|
||||||
|
try:
|
||||||
|
_wait_for_redis(host, port)
|
||||||
|
|
||||||
|
from bec_server.bec_server_utils.service_handler import ServiceHandler
|
||||||
|
|
||||||
|
service_handler = ServiceHandler(
|
||||||
|
bec_path=files_path, config_path=services_config, interface="subprocess"
|
||||||
|
)
|
||||||
|
processes = service_handler.start()
|
||||||
|
_load_demo_config(services_config)
|
||||||
|
|
||||||
|
env = os.environ.copy()
|
||||||
|
return subprocess.run(args.command, env=env, check=False).returncode
|
||||||
|
finally:
|
||||||
|
if service_handler is not None and processes is not None:
|
||||||
|
service_handler.stop(processes)
|
||||||
|
redis_process.terminate()
|
||||||
|
try:
|
||||||
|
redis_process.wait(timeout=10)
|
||||||
|
except subprocess.TimeoutExpired:
|
||||||
|
redis_process.kill()
|
||||||
|
|
||||||
|
|
||||||
|
if __name__ == "__main__":
|
||||||
|
raise SystemExit(main())
|
||||||
@@ -0,0 +1,242 @@
|
|||||||
|
name: BW Benchmarks
|
||||||
|
|
||||||
|
on: [ workflow_call ]
|
||||||
|
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
|
||||||
|
env:
|
||||||
|
BENCHMARK_JSON: benchmark-results/current.json
|
||||||
|
BENCHMARK_BASELINE_JSON: gh-pages-benchmark-data/benchmarks/latest.json
|
||||||
|
BENCHMARK_SUMMARY: benchmark-results/summary.md
|
||||||
|
BENCHMARK_COMMAND: "bash .github/scripts/run_benchmarks.sh"
|
||||||
|
BENCHMARK_THRESHOLD_PERCENT: 20
|
||||||
|
BENCHMARK_HIGHER_IS_BETTER: false
|
||||||
|
|
||||||
|
jobs:
|
||||||
|
benchmark_attempt:
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
continue-on-error: true
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
defaults:
|
||||||
|
run:
|
||||||
|
shell: bash -el {0}
|
||||||
|
strategy:
|
||||||
|
fail-fast: false
|
||||||
|
matrix:
|
||||||
|
attempt: [ 1, 2, 3 ]
|
||||||
|
|
||||||
|
env:
|
||||||
|
BENCHMARK_JSON: benchmark-results/current-${{ matrix.attempt }}.json
|
||||||
|
BEC_CORE_BRANCH: main
|
||||||
|
OPHYD_DEVICES_BRANCH: main
|
||||||
|
PLUGIN_REPO_BRANCH: main
|
||||||
|
BENCHMARK_PYTEST_DIRS: tests/unit_tests/benchmarks
|
||||||
|
QTWEBENGINE_DISABLE_SANDBOX: 1
|
||||||
|
QT_QPA_PLATFORM: "offscreen"
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout BEC Widgets
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
repository: bec-project/bec_widgets
|
||||||
|
ref: ${{ github.event.pull_request.head.sha || github.sha }}
|
||||||
|
|
||||||
|
- name: Set up Conda
|
||||||
|
uses: conda-incubator/setup-miniconda@v3
|
||||||
|
with:
|
||||||
|
auto-update-conda: true
|
||||||
|
auto-activate-base: true
|
||||||
|
python-version: "3.11"
|
||||||
|
|
||||||
|
- name: Install system dependencies
|
||||||
|
run: |
|
||||||
|
sudo apt-get update
|
||||||
|
sudo apt-get install -y libgl1 libegl1 x11-utils libxkbcommon-x11-0 libdbus-1-3 xvfb
|
||||||
|
sudo apt-get -y install libnss3 libxdamage1 libasound2t64 libatomic1 libxcursor1
|
||||||
|
sudo apt-get -y install ttyd hyperfine redis-server
|
||||||
|
|
||||||
|
- name: Install full e2e environment
|
||||||
|
run: |
|
||||||
|
echo -e "\033[35;1m Using branch $BEC_CORE_BRANCH of BEC CORE \033[0;m";
|
||||||
|
git clone --branch "$BEC_CORE_BRANCH" https://github.com/bec-project/bec.git
|
||||||
|
echo -e "\033[35;1m Using branch $OPHYD_DEVICES_BRANCH of OPHYD_DEVICES \033[0;m";
|
||||||
|
git clone --branch "$OPHYD_DEVICES_BRANCH" https://github.com/bec-project/ophyd_devices.git
|
||||||
|
export OHPYD_DEVICES_PATH=$PWD/ophyd_devices
|
||||||
|
echo -e "\033[35;1m Using branch $PLUGIN_REPO_BRANCH of bec_testing_plugin \033[0;m";
|
||||||
|
git clone --branch "$PLUGIN_REPO_BRANCH" https://github.com/bec-project/bec_testing_plugin.git
|
||||||
|
cd ./bec
|
||||||
|
conda create -q -n test-environment python=3.11
|
||||||
|
conda activate test-environment
|
||||||
|
source ./bin/install_bec_dev.sh -t
|
||||||
|
cd ../
|
||||||
|
python -m pip install -e ./ophyd_devices -e .[dev,pyside6] -e ./bec_testing_plugin pytest-benchmark
|
||||||
|
|
||||||
|
mkdir -p "$(dirname "$BENCHMARK_JSON")"
|
||||||
|
python .github/scripts/run_with_bec_servers.py -- bash -lc "$BENCHMARK_COMMAND"
|
||||||
|
test -s "$BENCHMARK_JSON"
|
||||||
|
|
||||||
|
- name: Upload benchmark artifact
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: bw-benchmark-json-${{ matrix.attempt }}
|
||||||
|
path: ${{ env.BENCHMARK_JSON }}
|
||||||
|
|
||||||
|
benchmark:
|
||||||
|
needs: [ benchmark_attempt ]
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: read
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout BEC Widgets
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
repository: bec-project/bec_widgets
|
||||||
|
ref: ${{ github.event.pull_request.head.sha || github.sha }}
|
||||||
|
|
||||||
|
- name: Download benchmark attempts
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
pattern: bw-benchmark-json-*
|
||||||
|
path: benchmark-results/attempts
|
||||||
|
merge-multiple: true
|
||||||
|
|
||||||
|
- name: Aggregate benchmark attempts
|
||||||
|
run: |
|
||||||
|
python .github/scripts/aggregate_benchmarks.py \
|
||||||
|
--input-dir benchmark-results/attempts \
|
||||||
|
--output "$BENCHMARK_JSON"
|
||||||
|
|
||||||
|
- name: Upload aggregate benchmark artifact
|
||||||
|
uses: actions/upload-artifact@v4
|
||||||
|
with:
|
||||||
|
name: bw-benchmark-json
|
||||||
|
path: ${{ env.BENCHMARK_JSON }}
|
||||||
|
|
||||||
|
- name: Fetch gh-pages benchmark data
|
||||||
|
run: |
|
||||||
|
if git ls-remote --exit-code --heads origin gh-pages; then
|
||||||
|
git clone --depth=1 --branch gh-pages "$GITHUB_SERVER_URL/$GITHUB_REPOSITORY.git" gh-pages-benchmark-data
|
||||||
|
else
|
||||||
|
mkdir -p gh-pages-benchmark-data
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Compare with latest gh-pages benchmark
|
||||||
|
id: compare
|
||||||
|
continue-on-error: true
|
||||||
|
run: |
|
||||||
|
if [ ! -s "$BENCHMARK_BASELINE_JSON" ]; then
|
||||||
|
mkdir -p "$(dirname "$BENCHMARK_SUMMARY")"
|
||||||
|
{
|
||||||
|
echo "<!-- bw-benchmark-comment -->"
|
||||||
|
echo "## Benchmark comparison"
|
||||||
|
echo
|
||||||
|
echo "No benchmark baseline was found on gh-pages."
|
||||||
|
} > "$BENCHMARK_SUMMARY"
|
||||||
|
exit 0
|
||||||
|
fi
|
||||||
|
|
||||||
|
args=(
|
||||||
|
--baseline "$BENCHMARK_BASELINE_JSON"
|
||||||
|
--current "$BENCHMARK_JSON"
|
||||||
|
--summary "$BENCHMARK_SUMMARY"
|
||||||
|
--threshold-percent "$BENCHMARK_THRESHOLD_PERCENT"
|
||||||
|
)
|
||||||
|
|
||||||
|
if [ "$BENCHMARK_HIGHER_IS_BETTER" = "true" ]; then
|
||||||
|
args+=(--higher-is-better)
|
||||||
|
fi
|
||||||
|
|
||||||
|
set +e
|
||||||
|
python .github/scripts/compare_benchmarks.py "${args[@]}"
|
||||||
|
status=$?
|
||||||
|
set -e
|
||||||
|
|
||||||
|
if [ ! -s "$BENCHMARK_SUMMARY" ]; then
|
||||||
|
mkdir -p "$(dirname "$BENCHMARK_SUMMARY")"
|
||||||
|
{
|
||||||
|
echo "<!-- bw-benchmark-comment -->"
|
||||||
|
echo "## Benchmark comparison"
|
||||||
|
echo
|
||||||
|
echo "Benchmark comparison failed before writing a summary."
|
||||||
|
} > "$BENCHMARK_SUMMARY"
|
||||||
|
fi
|
||||||
|
|
||||||
|
exit "$status"
|
||||||
|
|
||||||
|
- name: Find existing benchmark PR comment
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
id: fc
|
||||||
|
uses: peter-evans/find-comment@v3
|
||||||
|
with:
|
||||||
|
issue-number: ${{ github.event.pull_request.number }}
|
||||||
|
comment-author: github-actions[bot]
|
||||||
|
body-includes: "<!-- bw-benchmark-comment -->"
|
||||||
|
|
||||||
|
- name: Create or update benchmark PR comment
|
||||||
|
if: github.event_name == 'pull_request'
|
||||||
|
uses: peter-evans/create-or-update-comment@v5
|
||||||
|
with:
|
||||||
|
issue-number: ${{ github.event.pull_request.number }}
|
||||||
|
comment-id: ${{ steps.fc.outputs.comment-id }}
|
||||||
|
body-path: ${{ env.BENCHMARK_SUMMARY }}
|
||||||
|
edit-mode: replace
|
||||||
|
|
||||||
|
- name: Fail on benchmark regression
|
||||||
|
if: github.event_name == 'pull_request' && steps.compare.outcome == 'failure'
|
||||||
|
run: exit 1
|
||||||
|
|
||||||
|
publish:
|
||||||
|
needs: [ benchmark ]
|
||||||
|
if: github.event_name == 'push' && github.ref == 'refs/heads/main'
|
||||||
|
runs-on: ubuntu-latest
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
|
||||||
|
steps:
|
||||||
|
- name: Checkout BEC Widgets
|
||||||
|
uses: actions/checkout@v4
|
||||||
|
with:
|
||||||
|
repository: bec-project/bec_widgets
|
||||||
|
ref: ${{ github.sha }}
|
||||||
|
|
||||||
|
- name: Download aggregate benchmark artifact
|
||||||
|
uses: actions/download-artifact@v4
|
||||||
|
with:
|
||||||
|
name: bw-benchmark-json
|
||||||
|
path: benchmark-results
|
||||||
|
|
||||||
|
- name: Verify aggregate benchmark artifact
|
||||||
|
run: test -s "$BENCHMARK_JSON"
|
||||||
|
|
||||||
|
- name: Prepare gh-pages for publishing
|
||||||
|
run: |
|
||||||
|
# Clean up any existing worktree/directory
|
||||||
|
if [ -d gh-pages-benchmark-data ]; then
|
||||||
|
git worktree remove gh-pages-benchmark-data --force || rm -rf gh-pages-benchmark-data
|
||||||
|
fi
|
||||||
|
|
||||||
|
if git ls-remote --exit-code --heads origin gh-pages; then
|
||||||
|
git fetch --depth=1 origin gh-pages
|
||||||
|
git worktree add gh-pages-benchmark-data FETCH_HEAD
|
||||||
|
else
|
||||||
|
git worktree add --detach gh-pages-benchmark-data
|
||||||
|
git -C gh-pages-benchmark-data checkout --orphan gh-pages
|
||||||
|
git -C gh-pages-benchmark-data rm -rf .
|
||||||
|
fi
|
||||||
|
|
||||||
|
- name: Publish benchmark data to gh-pages
|
||||||
|
working-directory: gh-pages-benchmark-data
|
||||||
|
run: |
|
||||||
|
mkdir -p benchmarks/history
|
||||||
|
cp "../$BENCHMARK_JSON" benchmarks/latest.json
|
||||||
|
cp "../$BENCHMARK_JSON" "benchmarks/history/${GITHUB_SHA}.json"
|
||||||
|
git config user.name "github-actions[bot]"
|
||||||
|
git config user.email "41898282+github-actions[bot]@users.noreply.github.com"
|
||||||
|
git add benchmarks/latest.json "benchmarks/history/${GITHUB_SHA}.json"
|
||||||
|
git commit -m "Update BW benchmark data for ${GITHUB_SHA}" || exit 0
|
||||||
|
git push origin HEAD:gh-pages
|
||||||
@@ -5,15 +5,15 @@ on:
|
|||||||
workflow_dispatch:
|
workflow_dispatch:
|
||||||
inputs:
|
inputs:
|
||||||
BEC_WIDGETS_BRANCH:
|
BEC_WIDGETS_BRANCH:
|
||||||
description: 'Branch of BEC Widgets to install'
|
description: "Branch of BEC Widgets to install"
|
||||||
required: false
|
required: false
|
||||||
type: string
|
type: string
|
||||||
BEC_CORE_BRANCH:
|
BEC_CORE_BRANCH:
|
||||||
description: 'Branch of BEC Core to install'
|
description: "Branch of BEC Core to install"
|
||||||
required: false
|
required: false
|
||||||
type: string
|
type: string
|
||||||
OPHYD_DEVICES_BRANCH:
|
OPHYD_DEVICES_BRANCH:
|
||||||
description: 'Branch of Ophyd Devices to install'
|
description: "Branch of Ophyd Devices to install"
|
||||||
required: false
|
required: false
|
||||||
type: string
|
type: string
|
||||||
|
|
||||||
@@ -23,6 +23,7 @@ concurrency:
|
|||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
pull-requests: write
|
pull-requests: write
|
||||||
|
contents: read
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
check_pr_status:
|
check_pr_status:
|
||||||
@@ -33,6 +34,15 @@ jobs:
|
|||||||
if: needs.check_pr_status.outputs.branch-pr == ''
|
if: needs.check_pr_status.outputs.branch-pr == ''
|
||||||
uses: ./.github/workflows/formatter.yml
|
uses: ./.github/workflows/formatter.yml
|
||||||
|
|
||||||
|
benchmark:
|
||||||
|
needs: [check_pr_status]
|
||||||
|
if: needs.check_pr_status.outputs.branch-pr == ''
|
||||||
|
permissions:
|
||||||
|
contents: write
|
||||||
|
issues: write
|
||||||
|
pull-requests: write
|
||||||
|
uses: ./.github/workflows/benchmark.yml
|
||||||
|
|
||||||
unit-test:
|
unit-test:
|
||||||
needs: [check_pr_status, formatter]
|
needs: [check_pr_status, formatter]
|
||||||
if: needs.check_pr_status.outputs.branch-pr == ''
|
if: needs.check_pr_status.outputs.branch-pr == ''
|
||||||
|
|||||||
@@ -3,23 +3,23 @@ on:
|
|||||||
workflow_call:
|
workflow_call:
|
||||||
inputs:
|
inputs:
|
||||||
pr_number:
|
pr_number:
|
||||||
description: 'Pull request number'
|
description: "Pull request number"
|
||||||
required: false
|
required: false
|
||||||
type: number
|
type: number
|
||||||
BEC_CORE_BRANCH:
|
BEC_CORE_BRANCH:
|
||||||
description: 'Branch of BEC Core to install'
|
description: "Branch of BEC Core to install"
|
||||||
required: false
|
required: false
|
||||||
default: 'main'
|
default: "main"
|
||||||
type: string
|
type: string
|
||||||
OPHYD_DEVICES_BRANCH:
|
OPHYD_DEVICES_BRANCH:
|
||||||
description: 'Branch of Ophyd Devices to install'
|
description: "Branch of Ophyd Devices to install"
|
||||||
required: false
|
required: false
|
||||||
default: 'main'
|
default: "main"
|
||||||
type: string
|
type: string
|
||||||
BEC_WIDGETS_BRANCH:
|
BEC_WIDGETS_BRANCH:
|
||||||
description: 'Branch of BEC Widgets to install'
|
description: "Branch of BEC Widgets to install"
|
||||||
required: false
|
required: false
|
||||||
default: 'main'
|
default: "main"
|
||||||
type: string
|
type: string
|
||||||
|
|
||||||
jobs:
|
jobs:
|
||||||
@@ -30,15 +30,14 @@ jobs:
|
|||||||
python-version: ["3.11", "3.12", "3.13"]
|
python-version: ["3.11", "3.12", "3.13"]
|
||||||
|
|
||||||
env:
|
env:
|
||||||
BEC_WIDGETS_BRANCH: main # Set the branch you want for bec_widgets
|
BEC_WIDGETS_BRANCH: main # Set the branch you want for bec_widgets
|
||||||
BEC_CORE_BRANCH: main # Set the branch you want for bec
|
BEC_CORE_BRANCH: main # Set the branch you want for bec
|
||||||
OPHYD_DEVICES_BRANCH: main # Set the branch you want for ophyd_devices
|
OPHYD_DEVICES_BRANCH: main # Set the branch you want for ophyd_devices
|
||||||
PROJECT_PATH: ${{ github.repository }}
|
PROJECT_PATH: ${{ github.repository }}
|
||||||
QTWEBENGINE_DISABLE_SANDBOX: 1
|
QTWEBENGINE_DISABLE_SANDBOX: 1
|
||||||
QT_QPA_PLATFORM: "offscreen"
|
QT_QPA_PLATFORM: "offscreen"
|
||||||
|
|
||||||
steps:
|
steps:
|
||||||
|
|
||||||
- name: Checkout BEC Widgets
|
- name: Checkout BEC Widgets
|
||||||
uses: actions/checkout@v4
|
uses: actions/checkout@v4
|
||||||
with:
|
with:
|
||||||
@@ -56,4 +55,4 @@ jobs:
|
|||||||
- name: Run Pytest
|
- name: Run Pytest
|
||||||
run: |
|
run: |
|
||||||
pip install pytest pytest-random-order
|
pip install pytest pytest-random-order
|
||||||
pytest -v --junitxml=report.xml --random-order ./tests/unit_tests
|
pytest -v --junitxml=report.xml --random-order --ignore=tests/unit_tests/benchmarks ./tests/unit_tests
|
||||||
|
|||||||
@@ -3,30 +3,28 @@ on:
|
|||||||
workflow_call:
|
workflow_call:
|
||||||
inputs:
|
inputs:
|
||||||
pr_number:
|
pr_number:
|
||||||
description: 'Pull request number'
|
description: "Pull request number"
|
||||||
required: false
|
required: false
|
||||||
type: number
|
type: number
|
||||||
BEC_CORE_BRANCH:
|
BEC_CORE_BRANCH:
|
||||||
description: 'Branch of BEC Core to install'
|
description: "Branch of BEC Core to install"
|
||||||
required: false
|
required: false
|
||||||
default: 'main'
|
default: "main"
|
||||||
type: string
|
type: string
|
||||||
OPHYD_DEVICES_BRANCH:
|
OPHYD_DEVICES_BRANCH:
|
||||||
description: 'Branch of Ophyd Devices to install'
|
description: "Branch of Ophyd Devices to install"
|
||||||
required: false
|
required: false
|
||||||
default: 'main'
|
default: "main"
|
||||||
type: string
|
type: string
|
||||||
BEC_WIDGETS_BRANCH:
|
BEC_WIDGETS_BRANCH:
|
||||||
description: 'Branch of BEC Widgets to install'
|
description: "Branch of BEC Widgets to install"
|
||||||
required: false
|
required: false
|
||||||
default: 'main'
|
default: "main"
|
||||||
type: string
|
type: string
|
||||||
secrets:
|
secrets:
|
||||||
CODECOV_TOKEN:
|
CODECOV_TOKEN:
|
||||||
required: true
|
required: true
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
permissions:
|
permissions:
|
||||||
pull-requests: write
|
pull-requests: write
|
||||||
|
|
||||||
@@ -55,7 +53,7 @@ jobs:
|
|||||||
|
|
||||||
- name: Run Pytest with Coverage
|
- name: Run Pytest with Coverage
|
||||||
id: coverage
|
id: coverage
|
||||||
run: pytest --random-order --cov=bec_widgets --cov-config=pyproject.toml --cov-branch --cov-report=xml --no-cov-on-fail tests/unit_tests/
|
run: pytest --random-order --cov=bec_widgets --cov-config=pyproject.toml --cov-branch --cov-report=xml --no-cov-on-fail --ignore=tests/unit_tests/benchmarks tests/unit_tests/
|
||||||
|
|
||||||
- name: Upload test artifacts
|
- name: Upload test artifacts
|
||||||
uses: actions/upload-artifact@v4
|
uses: actions/upload-artifact@v4
|
||||||
|
|||||||
@@ -1,6 +1,79 @@
|
|||||||
# CHANGELOG
|
# CHANGELOG
|
||||||
|
|
||||||
|
|
||||||
|
## v3.7.2 (2026-04-29)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
- **dock-area**: Avoid switching profile when saving new profile
|
||||||
|
([`73b44cf`](https://github.com/bec-project/bec_widgets/commit/73b44cffb219347cacb609f3b93068eda6701b42))
|
||||||
|
|
||||||
|
- **workspace-actions**: Use try/finally and restore previous blocked state in refresh_profiles
|
||||||
|
([`30ef255`](https://github.com/bec-project/bec_widgets/commit/30ef25533af9df5a9bd9e69808dc49fdf22f4318))
|
||||||
|
|
||||||
|
Agent-Logs-Url:
|
||||||
|
https://github.com/bec-project/bec_widgets/sessions/004cb4bc-5015-485e-a803-1e63876b7024
|
||||||
|
|
||||||
|
Co-authored-by: wyzula-jan <133381102+wyzula-jan@users.noreply.github.com>
|
||||||
|
|
||||||
|
### Build System
|
||||||
|
|
||||||
|
- Add pytest-benchmark dependency
|
||||||
|
([`551d38d`](https://github.com/bec-project/bec_widgets/commit/551d38d90111361e64bfcf10a1409be71dd298bc))
|
||||||
|
|
||||||
|
### Chores
|
||||||
|
|
||||||
|
- Update header comments in script files to indicate AI generation
|
||||||
|
([`3f1aa80`](https://github.com/bec-project/bec_widgets/commit/3f1aa80756368454c3631cb6cf9d29db28af177a))
|
||||||
|
|
||||||
|
### Continuous Integration
|
||||||
|
|
||||||
|
- Add benchmark workflow
|
||||||
|
([`999b7a2`](https://github.com/bec-project/bec_widgets/commit/999b7a2321f2f222c04b056a2db4280f66de9c48))
|
||||||
|
|
||||||
|
- Fix benchmark upload
|
||||||
|
([`19b5c8f`](https://github.com/bec-project/bec_widgets/commit/19b5c8f724dbdb7421957c290f9213bf072392df))
|
||||||
|
|
||||||
|
- Increase threshold to 20 percent
|
||||||
|
([`409c9e5`](https://github.com/bec-project/bec_widgets/commit/409c9e5bfacdfc003f1bc8c9944f01798bd3818e))
|
||||||
|
|
||||||
|
### Testing
|
||||||
|
|
||||||
|
- Fix assertions after updating ophyd devices templates
|
||||||
|
([`a614d66`](https://github.com/bec-project/bec_widgets/commit/a614d662d6da716a49afb4ed3a3903f108210386))
|
||||||
|
|
||||||
|
Co-authored-by: Copilot <copilot@github.com>
|
||||||
|
|
||||||
|
- Remove references to "scan_motors" in tests
|
||||||
|
([`5056ef8`](https://github.com/bec-project/bec_widgets/commit/5056ef8946d03a20e802709d3fe81c84c195fe41))
|
||||||
|
|
||||||
|
|
||||||
|
## v3.7.1 (2026-04-21)
|
||||||
|
|
||||||
|
### Bug Fixes
|
||||||
|
|
||||||
|
- **heatmap**: Fix access to status from metadata
|
||||||
|
([`55694ff`](https://github.com/bec-project/bec_widgets/commit/55694ff2b96581e03c63c8b8e068e2db79bcf780))
|
||||||
|
|
||||||
|
### Testing
|
||||||
|
|
||||||
|
- Fix exit status and status access in tests
|
||||||
|
([`91afc77`](https://github.com/bec-project/bec_widgets/commit/91afc775d59b4ba31bed3585847a67c301acf9b0))
|
||||||
|
|
||||||
|
|
||||||
|
## v3.7.0 (2026-04-21)
|
||||||
|
|
||||||
|
### Features
|
||||||
|
|
||||||
|
- Move companion app to applications
|
||||||
|
([`0cf84cd`](https://github.com/bec-project/bec_widgets/commit/0cf84cd1d839ac4a39ffb5fb9ba57d432e04348a))
|
||||||
|
|
||||||
|
### Refactoring
|
||||||
|
|
||||||
|
- Cleanup of imports
|
||||||
|
([`3e77f54`](https://github.com/bec-project/bec_widgets/commit/3e77f540345f56b9f184a332fcdd50d4d4c8c621))
|
||||||
|
|
||||||
|
|
||||||
## v3.6.0 (2026-04-21)
|
## v3.6.0 (2026-04-21)
|
||||||
|
|
||||||
### Bug Fixes
|
### Bug Fixes
|
||||||
|
|||||||
+12
-18
@@ -1,19 +1,13 @@
|
|||||||
import os
|
|
||||||
import sys
|
|
||||||
|
|
||||||
import bec_widgets.widgets.containers.qt_ads as QtAds
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
|
||||||
|
|
||||||
if sys.platform.startswith("linux"):
|
|
||||||
qt_platform = os.environ.get("QT_QPA_PLATFORM", "")
|
|
||||||
if qt_platform != "offscreen":
|
|
||||||
os.environ["QT_QPA_PLATFORM"] = "xcb"
|
|
||||||
|
|
||||||
# Default QtAds configuration
|
|
||||||
QtAds.CDockManager.setConfigFlag(QtAds.CDockManager.eConfigFlag.FocusHighlighting, True)
|
|
||||||
QtAds.CDockManager.setConfigFlag(
|
|
||||||
QtAds.CDockManager.eConfigFlag.RetainTabSizeWhenCloseButtonHidden, True
|
|
||||||
)
|
|
||||||
|
|
||||||
__all__ = ["BECWidget", "SafeSlot", "SafeProperty"]
|
__all__ = ["BECWidget", "SafeSlot", "SafeProperty"]
|
||||||
|
|
||||||
|
|
||||||
|
def __getattr__(name):
|
||||||
|
if name == "BECWidget":
|
||||||
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
|
|
||||||
|
return BECWidget
|
||||||
|
if name in {"SafeSlot", "SafeProperty"}:
|
||||||
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
|
|
||||||
|
return {"SafeSlot": SafeSlot, "SafeProperty": SafeProperty}[name]
|
||||||
|
raise AttributeError(f"module {__name__!r} has no attribute {name!r}")
|
||||||
|
|||||||
@@ -0,0 +1,15 @@
|
|||||||
|
import os
|
||||||
|
import sys
|
||||||
|
|
||||||
|
import bec_widgets.widgets.containers.qt_ads as QtAds
|
||||||
|
|
||||||
|
if sys.platform.startswith("linux"):
|
||||||
|
qt_platform = os.environ.get("QT_QPA_PLATFORM", "")
|
||||||
|
if qt_platform != "offscreen":
|
||||||
|
os.environ["QT_QPA_PLATFORM"] = "xcb"
|
||||||
|
|
||||||
|
# Default QtAds configuration
|
||||||
|
QtAds.CDockManager.setConfigFlag(QtAds.CDockManager.eConfigFlag.FocusHighlighting, True)
|
||||||
|
QtAds.CDockManager.setConfigFlag(
|
||||||
|
QtAds.CDockManager.eConfigFlag.RetainTabSizeWhenCloseButtonHidden, True
|
||||||
|
)
|
||||||
|
|||||||
@@ -19,8 +19,8 @@ from qtpy.QtWidgets import QApplication
|
|||||||
|
|
||||||
import bec_widgets
|
import bec_widgets
|
||||||
from bec_widgets.applications.launch_window import LaunchWindow
|
from bec_widgets.applications.launch_window import LaunchWindow
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.utils.bec_dispatcher import BECDispatcher
|
from bec_widgets.utils.bec_dispatcher import BECDispatcher
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
|
|
||||||
logger = bec_logger.logger
|
logger = bec_logger.logger
|
||||||
|
|
||||||
@@ -20,13 +20,13 @@ from qtpy.QtWidgets import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
import bec_widgets
|
import bec_widgets
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.utils.bec_plugin_helper import get_all_plugin_widgets
|
from bec_widgets.utils.bec_plugin_helper import get_all_plugin_widgets
|
||||||
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.name_utils import pascal_to_snake
|
from bec_widgets.utils.name_utils import pascal_to_snake
|
||||||
from bec_widgets.utils.plugin_utils import get_plugin_auto_updates
|
from bec_widgets.utils.plugin_utils import get_plugin_auto_updates
|
||||||
from bec_widgets.utils.round_frame import RoundedFrame
|
from bec_widgets.utils.round_frame import RoundedFrame
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
from bec_widgets.utils.screen_utils import apply_window_geometry, centered_geometry_for_app
|
from bec_widgets.utils.screen_utils import apply_window_geometry, centered_geometry_for_app
|
||||||
from bec_widgets.utils.toolbars.toolbar import ModularToolBar
|
from bec_widgets.utils.toolbars.toolbar import ModularToolBar
|
||||||
from bec_widgets.utils.ui_loader import UILoader
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
|
|||||||
@@ -0,0 +1 @@
|
|||||||
|
from bec_widgets.cli.rpc import rpc_base
|
||||||
|
|||||||
@@ -25,8 +25,8 @@ from qtpy.QtWidgets import (
|
|||||||
QWidget,
|
QWidget,
|
||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_widget_handler import widget_handler
|
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import apply_theme
|
||||||
|
from bec_widgets.utils.rpc_widget_handler import widget_handler
|
||||||
from bec_widgets.utils.widget_io import WidgetHierarchy as wh
|
from bec_widgets.utils.widget_io import WidgetHierarchy as wh
|
||||||
from bec_widgets.widgets.editors.jupyter_console.jupyter_console import BECJupyterConsole
|
from bec_widgets.widgets.editors.jupyter_console.jupyter_console import BECJupyterConsole
|
||||||
|
|
||||||
|
|||||||
@@ -1,13 +1 @@
|
|||||||
from qtpy.QtWebEngineWidgets import QWebEngineView
|
|
||||||
|
|
||||||
from .bec_connector import BECConnector, ConnectionConfig
|
|
||||||
from .bec_dispatcher import BECDispatcher
|
|
||||||
from .bec_table import BECTable
|
|
||||||
from .colors import Colors
|
|
||||||
from .container_utils import WidgetContainerUtils
|
|
||||||
from .crosshair import Crosshair
|
|
||||||
from .entry_validator import EntryValidator
|
|
||||||
from .layout_manager import GridLayoutManager
|
|
||||||
from .rpc_decorator import register_rpc_methods, rpc_public
|
|
||||||
from .ui_loader import UILoader
|
|
||||||
from .validator_delegate import DoubleValidationDelegate
|
|
||||||
|
|||||||
@@ -15,9 +15,9 @@ from pydantic import BaseModel, Field, field_validator
|
|||||||
from qtpy.QtCore import Property, QObject, QRunnable, QThreadPool, Signal
|
from qtpy.QtCore import Property, QObject, QRunnable, QThreadPool, Signal
|
||||||
from qtpy.QtWidgets import QApplication
|
from qtpy.QtWidgets import QApplication
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.utils.error_popups import ErrorPopupUtility, SafeSlot
|
from bec_widgets.utils.error_popups import ErrorPopupUtility, SafeSlot
|
||||||
from bec_widgets.utils.name_utils import sanitize_namespace
|
from bec_widgets.utils.name_utils import sanitize_namespace
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
from bec_widgets.utils.widget_io import WidgetHierarchy
|
from bec_widgets.utils.widget_io import WidgetHierarchy
|
||||||
from bec_widgets.utils.yaml_dialog import load_yaml, load_yaml_gui, save_yaml, save_yaml_gui
|
from bec_widgets.utils.yaml_dialog import load_yaml, load_yaml_gui, save_yaml, save_yaml_gui
|
||||||
|
|
||||||
|
|||||||
@@ -10,11 +10,11 @@ from qtpy.QtGui import QFont, QPixmap
|
|||||||
from qtpy.QtWidgets import QApplication, QFileDialog, QLabel, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QApplication, QFileDialog, QLabel, QVBoxLayout, QWidget
|
||||||
|
|
||||||
import bec_widgets.widgets.containers.qt_ads as QtAds
|
import bec_widgets.widgets.containers.qt_ads as QtAds
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.utils.bec_connector import BECConnector, ConnectionConfig
|
from bec_widgets.utils.bec_connector import BECConnector, ConnectionConfig
|
||||||
from bec_widgets.utils.busy_loader import install_busy_loader
|
from bec_widgets.utils.busy_loader import install_busy_loader
|
||||||
from bec_widgets.utils.error_popups import SafeConnect, SafeSlot
|
from bec_widgets.utils.error_popups import SafeConnect, SafeSlot
|
||||||
from bec_widgets.utils.rpc_decorator import rpc_timeout
|
from bec_widgets.utils.rpc_decorator import rpc_timeout
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
from bec_widgets.utils.widget_io import WidgetHierarchy
|
from bec_widgets.utils.widget_io import WidgetHierarchy
|
||||||
from bec_widgets.widgets.utility.spinner.spinner import SpinnerWidget
|
from bec_widgets.widgets.utility.spinner.spinner import SpinnerWidget
|
||||||
|
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ from typing import TYPE_CHECKING, Iterable
|
|||||||
from bec_lib.plugin_helper import _get_available_plugins
|
from bec_lib.plugin_helper import _get_available_plugins
|
||||||
from qtpy.QtWidgets import QWidget
|
from qtpy.QtWidgets import QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
|
|
||||||
if TYPE_CHECKING: # pragma: no cover
|
if TYPE_CHECKING: # pragma: no cover
|
||||||
|
|||||||
@@ -14,11 +14,11 @@ from qtpy.QtCore import Qt, QTimer
|
|||||||
from qtpy.QtWidgets import QWidget
|
from qtpy.QtWidgets import QWidget
|
||||||
from redis.exceptions import RedisError
|
from redis.exceptions import RedisError
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.utils import BECDispatcher
|
|
||||||
from bec_widgets.utils.bec_connector import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
|
from bec_widgets.utils.bec_dispatcher import BECDispatcher
|
||||||
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
||||||
from bec_widgets.utils.error_popups import ErrorPopupUtility
|
from bec_widgets.utils.error_popups import ErrorPopupUtility
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
from bec_widgets.utils.screen_utils import apply_window_geometry
|
from bec_widgets.utils.screen_utils import apply_window_geometry
|
||||||
from bec_widgets.widgets.containers.dock_area.dock_area import BECDockArea
|
from bec_widgets.widgets.containers.dock_area.dock_area import BECDockArea
|
||||||
from bec_widgets.widgets.containers.main_window.main_window import BECMainWindow, BECMainWindowNoRPC
|
from bec_widgets.widgets.containers.main_window.main_window import BECMainWindow, BECMainWindowNoRPC
|
||||||
|
|||||||
@@ -26,7 +26,7 @@ from qtpy.QtWidgets import (
|
|||||||
from bec_widgets.widgets.utility.toggle.toggle import ToggleSwitch
|
from bec_widgets.widgets.utility.toggle.toggle import ToggleSwitch
|
||||||
|
|
||||||
if TYPE_CHECKING: # pragma: no cover
|
if TYPE_CHECKING: # pragma: no cover
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
|
|
||||||
logger = bec_logger.logger
|
logger = bec_logger.logger
|
||||||
|
|
||||||
@@ -418,7 +418,7 @@ class WidgetHierarchy:
|
|||||||
only_bec_widgets(bool, optional): Whether to print only widgets that are instances of BECWidget.
|
only_bec_widgets(bool, optional): Whether to print only widgets that are instances of BECWidget.
|
||||||
show_parent(bool, optional): Whether to display which BECWidget is the parent of each discovered BECWidget.
|
show_parent(bool, optional): Whether to display which BECWidget is the parent of each discovered BECWidget.
|
||||||
"""
|
"""
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
||||||
|
|
||||||
for node in WidgetHierarchy.iter_widget_tree(
|
for node in WidgetHierarchy.iter_widget_tree(
|
||||||
@@ -468,7 +468,7 @@ class WidgetHierarchy:
|
|||||||
|
|
||||||
from qtpy.QtWidgets import QApplication
|
from qtpy.QtWidgets import QApplication
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.widgets.plots.plot_base import PlotBase
|
from bec_widgets.widgets.plots.plot_base import PlotBase
|
||||||
|
|
||||||
# 1) Gather ALL QWidget-based BECConnector objects
|
# 1) Gather ALL QWidget-based BECConnector objects
|
||||||
@@ -534,7 +534,7 @@ class WidgetHierarchy:
|
|||||||
Returns:
|
Returns:
|
||||||
The nearest ancestor that is a BECConnector, or None if not found.
|
The nearest ancestor that is a BECConnector, or None if not found.
|
||||||
"""
|
"""
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
|
|
||||||
# Guard against deleted/invalid Qt wrappers
|
# Guard against deleted/invalid Qt wrappers
|
||||||
if not shb.isValid(widget):
|
if not shb.isValid(widget):
|
||||||
@@ -636,7 +636,7 @@ class WidgetHierarchy:
|
|||||||
Return all BECConnector instances whose closest BECConnector ancestor is the given widget,
|
Return all BECConnector instances whose closest BECConnector ancestor is the given widget,
|
||||||
including the widget itself if it is a BECConnector.
|
including the widget itself if it is a BECConnector.
|
||||||
"""
|
"""
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
|
|
||||||
connectors: list[BECConnector] = []
|
connectors: list[BECConnector] = []
|
||||||
if isinstance(widget, BECConnector):
|
if isinstance(widget, BECConnector):
|
||||||
@@ -664,7 +664,7 @@ class WidgetHierarchy:
|
|||||||
return None
|
return None
|
||||||
|
|
||||||
try:
|
try:
|
||||||
from bec_widgets.utils import BECConnector # local import to avoid cycles
|
from bec_widgets.utils.bec_connector import BECConnector # local import to avoid cycles
|
||||||
|
|
||||||
is_bec_target = False
|
is_bec_target = False
|
||||||
if isinstance(ancestor_class, str):
|
if isinstance(ancestor_class, str):
|
||||||
|
|||||||
@@ -13,9 +13,9 @@ from shiboken6 import isValid
|
|||||||
|
|
||||||
import bec_widgets.widgets.containers.qt_ads as QtAds
|
import bec_widgets.widgets.containers.qt_ads as QtAds
|
||||||
from bec_widgets import BECWidget, SafeSlot
|
from bec_widgets import BECWidget, SafeSlot
|
||||||
from bec_widgets.cli.rpc.rpc_widget_handler import widget_handler
|
|
||||||
from bec_widgets.utils.bec_connector import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.utils.property_editor import PropertyEditor
|
from bec_widgets.utils.property_editor import PropertyEditor
|
||||||
|
from bec_widgets.utils.rpc_widget_handler import widget_handler
|
||||||
from bec_widgets.utils.toolbars.actions import MaterialIconAction
|
from bec_widgets.utils.toolbars.actions import MaterialIconAction
|
||||||
from bec_widgets.widgets.containers.qt_ads import (
|
from bec_widgets.widgets.containers.qt_ads import (
|
||||||
CDockAreaWidget,
|
CDockAreaWidget,
|
||||||
|
|||||||
@@ -20,10 +20,10 @@ from qtpy.QtWidgets import (
|
|||||||
import bec_widgets.widgets.containers.qt_ads as QtAds
|
import bec_widgets.widgets.containers.qt_ads as QtAds
|
||||||
from bec_widgets import BECWidget, SafeProperty, SafeSlot
|
from bec_widgets import BECWidget, SafeProperty, SafeSlot
|
||||||
from bec_widgets.applications.views.view import ViewTourSteps
|
from bec_widgets.applications.views.view import ViewTourSteps
|
||||||
from bec_widgets.cli.rpc.rpc_widget_handler import widget_handler
|
from bec_widgets.utils.bec_dispatcher import BECDispatcher
|
||||||
from bec_widgets.utils import BECDispatcher
|
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import apply_theme
|
||||||
from bec_widgets.utils.rpc_decorator import rpc_timeout
|
from bec_widgets.utils.rpc_decorator import rpc_timeout
|
||||||
|
from bec_widgets.utils.rpc_widget_handler import widget_handler
|
||||||
from bec_widgets.utils.toolbars.actions import (
|
from bec_widgets.utils.toolbars.actions import (
|
||||||
ExpandableMenuAction,
|
ExpandableMenuAction,
|
||||||
MaterialIconAction,
|
MaterialIconAction,
|
||||||
@@ -235,11 +235,8 @@ class BECDockArea(DockAreaWidget):
|
|||||||
def _load_initial_profile(self, name: str) -> None:
|
def _load_initial_profile(self, name: str) -> None:
|
||||||
"""Load the initial profile."""
|
"""Load the initial profile."""
|
||||||
self.load_profile(name)
|
self.load_profile(name)
|
||||||
combo = self.toolbar.components.get_action("workspace_combo").widget
|
|
||||||
combo.blockSignals(True)
|
|
||||||
if not self._empty_profile_active:
|
if not self._empty_profile_active:
|
||||||
combo.setCurrentText(name)
|
self._set_workspace_combo_text_silent(name)
|
||||||
combo.blockSignals(False)
|
|
||||||
|
|
||||||
def _start_empty_workspace(self) -> None:
|
def _start_empty_workspace(self) -> None:
|
||||||
"""
|
"""
|
||||||
@@ -669,6 +666,14 @@ class BECDockArea(DockAreaWidget):
|
|||||||
combo = self.toolbar.components.get_action("workspace_combo").widget
|
combo = self.toolbar.components.get_action("workspace_combo").widget
|
||||||
combo.refresh_profiles(active_profile=name)
|
combo.refresh_profiles(active_profile=name)
|
||||||
|
|
||||||
|
def _set_workspace_combo_text_silent(self, text: str) -> None:
|
||||||
|
combo = self.toolbar.components.get_action("workspace_combo").widget
|
||||||
|
was_blocked = combo.blockSignals(True)
|
||||||
|
try:
|
||||||
|
combo.setCurrentText(text)
|
||||||
|
finally:
|
||||||
|
combo.blockSignals(was_blocked)
|
||||||
|
|
||||||
def _enter_empty_profile_state(self) -> None:
|
def _enter_empty_profile_state(self) -> None:
|
||||||
"""
|
"""
|
||||||
Switch to the transient empty workspace state.
|
Switch to the transient empty workspace state.
|
||||||
@@ -796,7 +801,6 @@ class BECDockArea(DockAreaWidget):
|
|||||||
self._pending_autosave_skip = (current_profile, name)
|
self._pending_autosave_skip = (current_profile, name)
|
||||||
else:
|
else:
|
||||||
self._pending_autosave_skip = None
|
self._pending_autosave_skip = None
|
||||||
workspace_combo.setCurrentText(name)
|
|
||||||
self._finalize_profile_change(name, namespace)
|
self._finalize_profile_change(name, namespace)
|
||||||
|
|
||||||
@SafeSlot()
|
@SafeSlot()
|
||||||
|
|||||||
@@ -24,19 +24,9 @@ class ProfileComboBox(QComboBox):
|
|||||||
def set_quick_profile_provider(self, provider: Callable[[], list[str]]) -> None:
|
def set_quick_profile_provider(self, provider: Callable[[], list[str]]) -> None:
|
||||||
self._quick_provider = provider
|
self._quick_provider = provider
|
||||||
|
|
||||||
def refresh_profiles(
|
def _refresh_profiles(
|
||||||
self, active_profile: str | None = None, show_empty_profile: bool = False
|
self, current_text: str, active_profile: str | None = None, show_empty_profile: bool = False
|
||||||
) -> None:
|
) -> None:
|
||||||
"""
|
|
||||||
Refresh the profile list and ensure the active profile is visible.
|
|
||||||
|
|
||||||
Args:
|
|
||||||
active_profile(str | None): The currently active profile name.
|
|
||||||
show_empty_profile(bool): If True, show an explicit empty unsaved workspace entry.
|
|
||||||
"""
|
|
||||||
|
|
||||||
current_text = active_profile or self.currentText()
|
|
||||||
self.blockSignals(True)
|
|
||||||
self.clear()
|
self.clear()
|
||||||
|
|
||||||
quick_profiles = self._quick_provider()
|
quick_profiles = self._quick_provider()
|
||||||
@@ -103,7 +93,6 @@ class ProfileComboBox(QComboBox):
|
|||||||
if index >= 0:
|
if index >= 0:
|
||||||
self.setCurrentIndex(index)
|
self.setCurrentIndex(index)
|
||||||
|
|
||||||
self.blockSignals(False)
|
|
||||||
if active_profile and self.currentText() != active_profile:
|
if active_profile and self.currentText() != active_profile:
|
||||||
idx = self.findText(active_profile)
|
idx = self.findText(active_profile)
|
||||||
if idx >= 0:
|
if idx >= 0:
|
||||||
@@ -115,6 +104,24 @@ class ProfileComboBox(QComboBox):
|
|||||||
else:
|
else:
|
||||||
self.setToolTip("")
|
self.setToolTip("")
|
||||||
|
|
||||||
|
def refresh_profiles(
|
||||||
|
self, active_profile: str | None = None, show_empty_profile: bool = False
|
||||||
|
) -> None:
|
||||||
|
"""
|
||||||
|
Refresh the profile list and ensure the active profile is visible.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
active_profile(str | None): The currently active profile name.
|
||||||
|
show_empty_profile(bool): If True, show an explicit empty unsaved workspace entry.
|
||||||
|
"""
|
||||||
|
|
||||||
|
current_text = active_profile or self.currentText()
|
||||||
|
was_blocked = self.blockSignals(True)
|
||||||
|
try:
|
||||||
|
self._refresh_profiles(current_text, active_profile, show_empty_profile)
|
||||||
|
finally:
|
||||||
|
self.blockSignals(was_blocked)
|
||||||
|
|
||||||
|
|
||||||
def workspace_bundle(components: ToolbarComponents, enable_tools: bool = True) -> ToolbarBundle:
|
def workspace_bundle(components: ToolbarComponents, enable_tools: bool = True) -> ToolbarBundle:
|
||||||
"""
|
"""
|
||||||
@@ -122,6 +129,7 @@ def workspace_bundle(components: ToolbarComponents, enable_tools: bool = True) -
|
|||||||
|
|
||||||
Args:
|
Args:
|
||||||
components (ToolbarComponents): The components to be added to the bundle.
|
components (ToolbarComponents): The components to be added to the bundle.
|
||||||
|
enable_tools(bool): If True, show the workspace management tools; otherwise, only show the profile combo.
|
||||||
|
|
||||||
Returns:
|
Returns:
|
||||||
ToolbarBundle: The workspace toolbar bundle.
|
ToolbarBundle: The workspace toolbar bundle.
|
||||||
|
|||||||
@@ -22,7 +22,7 @@ from qtpy.QtWidgets import (
|
|||||||
)
|
)
|
||||||
from typeguard import typechecked
|
from typeguard import typechecked
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_widget_handler import widget_handler
|
from bec_widgets.utils.rpc_widget_handler import widget_handler
|
||||||
|
|
||||||
|
|
||||||
class LayoutManagerWidget(QWidget):
|
class LayoutManagerWidget(QWidget):
|
||||||
|
|||||||
+1
-1
@@ -28,7 +28,7 @@ from qtpy.QtCore import QObject, QTimer
|
|||||||
from qtpy.QtWidgets import QApplication, QFrame, QMainWindow, QScrollArea, QWidget
|
from qtpy.QtWidgets import QApplication, QFrame, QMainWindow, QScrollArea, QWidget
|
||||||
|
|
||||||
from bec_widgets import SafeProperty, SafeSlot
|
from bec_widgets import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import apply_theme
|
||||||
from bec_widgets.utils.widget_io import WidgetIO
|
from bec_widgets.utils.widget_io import WidgetIO
|
||||||
|
|
||||||
|
|||||||
@@ -18,10 +18,10 @@ from qtpy.QtWidgets import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
import bec_widgets
|
import bec_widgets
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import apply_theme
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.widgets.containers.main_window.addons.hover_widget import HoverWidget
|
from bec_widgets.widgets.containers.main_window.addons.hover_widget import HoverWidget
|
||||||
from bec_widgets.widgets.containers.main_window.addons.notification_center.notification_banner import (
|
from bec_widgets.widgets.containers.main_window.addons.notification_center.notification_banner import (
|
||||||
BECNotificationBroker,
|
BECNotificationBroker,
|
||||||
|
|||||||
+1
-1
@@ -11,9 +11,9 @@ from qtpy.QtCore import Qt, Signal
|
|||||||
from qtpy.QtGui import QDoubleValidator
|
from qtpy.QtGui import QDoubleValidator
|
||||||
from qtpy.QtWidgets import QDoubleSpinBox
|
from qtpy.QtWidgets import QDoubleSpinBox
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.colors import apply_theme, get_accent_colors
|
from bec_widgets.utils.colors import apply_theme, get_accent_colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.widgets.control.device_control.positioner_box.positioner_box_base import (
|
from bec_widgets.widgets.control.device_control.positioner_box.positioner_box_base import (
|
||||||
DeviceUpdateUIComponents,
|
DeviceUpdateUIComponents,
|
||||||
PositionerBoxBase,
|
PositionerBoxBase,
|
||||||
|
|||||||
+1
-1
@@ -12,9 +12,9 @@ from qtpy.QtCore import Signal
|
|||||||
from qtpy.QtGui import QDoubleValidator
|
from qtpy.QtGui import QDoubleValidator
|
||||||
from qtpy.QtWidgets import QDoubleSpinBox
|
from qtpy.QtWidgets import QDoubleSpinBox
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import apply_theme
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.widgets.control.device_control.positioner_box.positioner_box_base import (
|
from bec_widgets.widgets.control.device_control.positioner_box.positioner_box_base import (
|
||||||
DeviceUpdateUIComponents,
|
DeviceUpdateUIComponents,
|
||||||
PositionerBoxBase,
|
PositionerBoxBase,
|
||||||
|
|||||||
@@ -7,7 +7,7 @@ from bec_lib.device import Signal as BECSignal
|
|||||||
from bec_lib.logger import bec_logger
|
from bec_lib.logger import bec_logger
|
||||||
from pydantic import field_validator
|
from pydantic import field_validator
|
||||||
|
|
||||||
from bec_widgets.utils import ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.filter_io import FilterIO
|
from bec_widgets.utils.filter_io import FilterIO
|
||||||
|
|||||||
@@ -3,7 +3,7 @@ from bec_lib.device import Signal
|
|||||||
from bec_lib.logger import bec_logger
|
from bec_lib.logger import bec_logger
|
||||||
from qtpy.QtCore import Property
|
from qtpy.QtCore import Property
|
||||||
|
|
||||||
from bec_widgets.utils import ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.filter_io import FilterIO
|
from bec_widgets.utils.filter_io import FilterIO
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ from qtpy.QtWidgets import (
|
|||||||
QWidget,
|
QWidget,
|
||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets.utils import ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.colors import apply_theme, get_accent_colors
|
from bec_widgets.utils.colors import apply_theme, get_accent_colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
|
|||||||
@@ -5,10 +5,10 @@ from bec_lib.logger import bec_logger
|
|||||||
from qtpy.QtCore import Signal
|
from qtpy.QtCore import Signal
|
||||||
from qtpy.QtWidgets import QPushButton, QSizePolicy, QTreeWidgetItem, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QPushButton, QSizePolicy, QTreeWidgetItem, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.colors import get_accent_colors
|
from bec_widgets.utils.colors import get_accent_colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
|
|
||||||
logger = bec_logger.logger
|
logger = bec_logger.logger
|
||||||
|
|
||||||
|
|||||||
@@ -19,8 +19,8 @@ from scipy.interpolate import (
|
|||||||
from scipy.spatial import cKDTree
|
from scipy.spatial import cKDTree
|
||||||
from toolz import partition
|
from toolz import partition
|
||||||
|
|
||||||
from bec_widgets.utils import Colors
|
|
||||||
from bec_widgets.utils.bec_connector import ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingsDialog
|
from bec_widgets.utils.settings_dialog import SettingsDialog
|
||||||
from bec_widgets.utils.toolbars.actions import MaterialIconAction
|
from bec_widgets.utils.toolbars.actions import MaterialIconAction
|
||||||
@@ -611,7 +611,7 @@ class Heatmap(ImageBase):
|
|||||||
scan_msg = self.scan_item.status_message
|
scan_msg = self.scan_item.status_message
|
||||||
elif hasattr(self.scan_item, "metadata"):
|
elif hasattr(self.scan_item, "metadata"):
|
||||||
metadata = self.scan_item.metadata["bec"]
|
metadata = self.scan_item.metadata["bec"]
|
||||||
status = metadata["exit_status"]
|
status = metadata["status"]
|
||||||
scan_id = metadata["scan_id"]
|
scan_id = metadata["scan_id"]
|
||||||
scan_name = metadata["scan_name"]
|
scan_name = metadata["scan_name"]
|
||||||
scan_type = metadata["scan_type"]
|
scan_type = metadata["scan_type"]
|
||||||
|
|||||||
@@ -4,9 +4,9 @@ import os
|
|||||||
|
|
||||||
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout
|
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingWidget
|
from bec_widgets.utils.settings_dialog import SettingWidget
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
|
|
||||||
|
|
||||||
class HeatmapSettings(SettingWidget):
|
class HeatmapSettings(SettingWidget):
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ from pydantic import BaseModel, Field, field_validator
|
|||||||
from qtpy.QtCore import QTimer
|
from qtpy.QtCore import QTimer
|
||||||
from qtpy.QtWidgets import QWidget
|
from qtpy.QtWidgets import QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.colors import Colors, apply_theme
|
from bec_widgets.utils.colors import Colors, apply_theme
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.widgets.plots.image.image_base import ImageBase
|
from bec_widgets.widgets.plots.image.image_base import ImageBase
|
||||||
|
|||||||
@@ -9,7 +9,7 @@ from pydantic import BaseModel, ConfigDict, Field, ValidationError
|
|||||||
from qtpy.QtCore import QPointF, Signal, SignalInstance
|
from qtpy.QtCore import QPointF, Signal, SignalInstance
|
||||||
from qtpy.QtWidgets import QDialog, QVBoxLayout
|
from qtpy.QtWidgets import QDialog, QVBoxLayout
|
||||||
|
|
||||||
from bec_widgets.utils import Colors
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.side_panel import SidePanel
|
from bec_widgets.utils.side_panel import SidePanel
|
||||||
|
|||||||
@@ -9,7 +9,8 @@ from pydantic import Field, ValidationError, field_validator
|
|||||||
from qtpy.QtCore import Signal
|
from qtpy.QtCore import Signal
|
||||||
from qtpy.QtGui import QTransform
|
from qtpy.QtGui import QTransform
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector, Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import BECConnector, ConnectionConfig
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.widgets.plots.image.image_processor import (
|
from bec_widgets.widgets.plots.image.image_processor import (
|
||||||
ImageProcessor,
|
ImageProcessor,
|
||||||
ImageStats,
|
ImageStats,
|
||||||
|
|||||||
@@ -20,7 +20,8 @@ from qtpy.QtWidgets import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets import BECWidget
|
from bec_widgets import BECWidget
|
||||||
from bec_widgets.utils import BECDispatcher, ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
|
from bec_widgets.utils.bec_dispatcher import BECDispatcher
|
||||||
from bec_widgets.utils.toolbars.actions import WidgetAction
|
from bec_widgets.utils.toolbars.actions import WidgetAction
|
||||||
from bec_widgets.utils.toolbars.bundles import ToolbarBundle
|
from bec_widgets.utils.toolbars.bundles import ToolbarBundle
|
||||||
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction, ModularToolBar
|
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction, ModularToolBar
|
||||||
|
|||||||
@@ -10,8 +10,8 @@ from qtpy.QtCore import Signal
|
|||||||
from qtpy.QtGui import QColor
|
from qtpy.QtGui import QColor
|
||||||
from qtpy.QtWidgets import QHBoxLayout, QMainWindow, QWidget
|
from qtpy.QtWidgets import QHBoxLayout, QMainWindow, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import Colors, apply_theme
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingsDialog
|
from bec_widgets.utils.settings_dialog import SettingsDialog
|
||||||
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction
|
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction
|
||||||
|
|||||||
@@ -2,9 +2,9 @@ import os
|
|||||||
|
|
||||||
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingWidget
|
from bec_widgets.utils.settings_dialog import SettingWidget
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.utils.widget_io import WidgetIO
|
from bec_widgets.utils.widget_io import WidgetIO
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -10,7 +10,8 @@ from pydantic import Field, ValidationError, field_validator
|
|||||||
from qtpy.QtCore import Signal
|
from qtpy.QtCore import Signal
|
||||||
from qtpy.QtWidgets import QWidget
|
from qtpy.QtWidgets import QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.side_panel import SidePanel
|
from bec_widgets.utils.side_panel import SidePanel
|
||||||
from bec_widgets.widgets.control.device_input.device_combobox.device_combobox import DeviceComboBox
|
from bec_widgets.widgets.control.device_input.device_combobox.device_combobox import DeviceComboBox
|
||||||
|
|||||||
@@ -2,9 +2,9 @@ import os
|
|||||||
|
|
||||||
from qtpy.QtWidgets import QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingWidget
|
from bec_widgets.utils.settings_dialog import SettingWidget
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.utils.widget_io import WidgetIO
|
from bec_widgets.utils.widget_io import WidgetIO
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -8,8 +8,10 @@ from bec_lib import bec_logger
|
|||||||
from qtpy.QtCore import QPoint, QPointF, Qt, Signal
|
from qtpy.QtCore import QPoint, QPointF, Qt, Signal
|
||||||
from qtpy.QtWidgets import QHBoxLayout, QLabel, QMainWindow, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QHBoxLayout, QLabel, QMainWindow, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import ConnectionConfig, Crosshair, EntryValidator
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
|
from bec_widgets.utils.crosshair import Crosshair
|
||||||
|
from bec_widgets.utils.entry_validator import EntryValidator
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.fps_counter import FPSCounter
|
from bec_widgets.utils.fps_counter import FPSCounter
|
||||||
from bec_widgets.utils.plot_indicator_items import BECArrowItem, BECTickItem
|
from bec_widgets.utils.plot_indicator_items import BECArrowItem, BECTickItem
|
||||||
|
|||||||
@@ -10,7 +10,7 @@ from qtpy import QtCore
|
|||||||
from qtpy.QtCore import QObject, Signal
|
from qtpy.QtCore import QObject, Signal
|
||||||
|
|
||||||
from bec_widgets import SafeProperty
|
from bec_widgets import SafeProperty
|
||||||
from bec_widgets.utils import BECConnector, ConnectionConfig
|
from bec_widgets.utils.bec_connector import BECConnector, ConnectionConfig
|
||||||
from bec_widgets.utils.colors import Colors
|
from bec_widgets.utils.colors import Colors
|
||||||
|
|
||||||
if TYPE_CHECKING:
|
if TYPE_CHECKING:
|
||||||
|
|||||||
@@ -8,7 +8,8 @@ from bec_lib import bec_logger
|
|||||||
from pydantic import BaseModel, Field, ValidationError, field_validator
|
from pydantic import BaseModel, Field, ValidationError, field_validator
|
||||||
from qtpy import QtCore
|
from qtpy import QtCore
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector, Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import BECConnector, ConnectionConfig
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
|
|
||||||
if TYPE_CHECKING: # pragma: no cover
|
if TYPE_CHECKING: # pragma: no cover
|
||||||
from bec_widgets.widgets.plots.scatter_waveform.scatter_waveform import ScatterWaveform
|
from bec_widgets.widgets.plots.scatter_waveform.scatter_waveform import ScatterWaveform
|
||||||
|
|||||||
@@ -7,7 +7,8 @@ from pydantic import Field, ValidationError, field_validator
|
|||||||
from qtpy.QtCore import QTimer, Signal
|
from qtpy.QtCore import QTimer, Signal
|
||||||
from qtpy.QtWidgets import QHBoxLayout, QMainWindow, QWidget
|
from qtpy.QtWidgets import QHBoxLayout, QMainWindow, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
from bec_widgets.utils.error_popups import SafeProperty, SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingsDialog
|
from bec_widgets.utils.settings_dialog import SettingsDialog
|
||||||
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction
|
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction
|
||||||
|
|||||||
@@ -2,9 +2,9 @@ import os
|
|||||||
|
|
||||||
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout
|
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingWidget
|
from bec_widgets.utils.settings_dialog import SettingWidget
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
|
|
||||||
|
|
||||||
class ScatterCurveSettings(SettingWidget):
|
class ScatterCurveSettings(SettingWidget):
|
||||||
|
|||||||
@@ -2,9 +2,9 @@ import os
|
|||||||
|
|
||||||
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QFrame, QScrollArea, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingWidget
|
from bec_widgets.utils.settings_dialog import SettingWidget
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.utils.widget_io import WidgetIO
|
from bec_widgets.utils.widget_io import WidgetIO
|
||||||
|
|
||||||
|
|
||||||
|
|||||||
@@ -8,7 +8,8 @@ from bec_lib import bec_logger
|
|||||||
from pydantic import BaseModel, Field, field_validator
|
from pydantic import BaseModel, Field, field_validator
|
||||||
from qtpy import QtCore
|
from qtpy import QtCore
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector, Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import BECConnector, ConnectionConfig
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
|
|
||||||
if TYPE_CHECKING: # pragma: no cover
|
if TYPE_CHECKING: # pragma: no cover
|
||||||
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
||||||
|
|||||||
@@ -50,9 +50,10 @@ from qtpy.QtWidgets import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets import SafeSlot
|
from bec_widgets import SafeSlot
|
||||||
from bec_widgets.utils import ConnectionConfig, EntryValidator
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.colors import Colors
|
from bec_widgets.utils.colors import Colors
|
||||||
|
from bec_widgets.utils.entry_validator import EntryValidator
|
||||||
from bec_widgets.utils.toolbars.actions import WidgetAction
|
from bec_widgets.utils.toolbars.actions import WidgetAction
|
||||||
from bec_widgets.utils.toolbars.bundles import ToolbarBundle
|
from bec_widgets.utils.toolbars.bundles import ToolbarBundle
|
||||||
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction, ModularToolBar
|
from bec_widgets.utils.toolbars.toolbar import MaterialIconAction, ModularToolBar
|
||||||
|
|||||||
@@ -25,7 +25,7 @@ from qtpy.QtWidgets import (
|
|||||||
QWidget,
|
QWidget,
|
||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets.utils import ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_signal_proxy import BECSignalProxy
|
from bec_widgets.utils.bec_signal_proxy import BECSignalProxy
|
||||||
from bec_widgets.utils.colors import Colors, apply_theme
|
from bec_widgets.utils.colors import Colors, apply_theme
|
||||||
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
from bec_widgets.utils.container_utils import WidgetContainerUtils
|
||||||
|
|||||||
@@ -6,8 +6,8 @@ from bec_lib.logger import bec_logger
|
|||||||
from qtpy.QtCore import QPointF, QSize, Qt
|
from qtpy.QtCore import QPointF, QSize, Qt
|
||||||
from qtpy.QtWidgets import QHBoxLayout, QLabel, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QHBoxLayout, QLabel, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import Colors
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.utils.error_popups import SafeProperty
|
from bec_widgets.utils.error_popups import SafeProperty
|
||||||
from bec_widgets.utils.settings_dialog import SettingsDialog
|
from bec_widgets.utils.settings_dialog import SettingsDialog
|
||||||
from bec_widgets.utils.toolbars.actions import MaterialIconAction
|
from bec_widgets.utils.toolbars.actions import MaterialIconAction
|
||||||
|
|||||||
@@ -19,9 +19,9 @@ from qtpy.QtWidgets import (
|
|||||||
QWidget,
|
QWidget,
|
||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets.utils import UILoader
|
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.settings_dialog import SettingWidget
|
from bec_widgets.utils.settings_dialog import SettingWidget
|
||||||
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.widgets.progress.ring_progress_bar.ring import Ring
|
from bec_widgets.widgets.progress.ring_progress_bar.ring import Ring
|
||||||
from bec_widgets.widgets.utility.visual.colormap_widget.colormap_widget import BECColorMapWidget
|
from bec_widgets.widgets.utility.visual.colormap_widget.colormap_widget import BECColorMapWidget
|
||||||
|
|
||||||
|
|||||||
@@ -12,10 +12,10 @@ from pyqtgraph import SignalProxy
|
|||||||
from qtpy.QtCore import QThreadPool, Signal
|
from qtpy.QtCore import QThreadPool, Signal
|
||||||
from qtpy.QtWidgets import QFileDialog, QListWidget, QToolButton, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QFileDialog, QListWidget, QToolButton, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.error_popups import SafeSlot
|
from bec_widgets.utils.error_popups import SafeSlot
|
||||||
from bec_widgets.utils.list_of_expandable_frames import ListOfExpandableFrames
|
from bec_widgets.utils.list_of_expandable_frames import ListOfExpandableFrames
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
from bec_widgets.utils.ui_loader import UILoader
|
from bec_widgets.utils.ui_loader import UILoader
|
||||||
from bec_widgets.widgets.services.device_browser.device_item import DeviceItem
|
from bec_widgets.widgets.services.device_browser.device_item import DeviceItem
|
||||||
from bec_widgets.widgets.services.device_browser.device_item.device_config_dialog import (
|
from bec_widgets.widgets.services.device_browser.device_item.device_config_dialog import (
|
||||||
|
|||||||
@@ -3,8 +3,8 @@ from qtpy import QtCore, QtGui
|
|||||||
from qtpy.QtCore import Property, Signal, Slot
|
from qtpy.QtCore import Property, Signal, Slot
|
||||||
from qtpy.QtWidgets import QSizePolicy, QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QSizePolicy, QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import Colors
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
|
from bec_widgets.utils.colors import Colors
|
||||||
|
|
||||||
|
|
||||||
class RoundedColorMapButton(ColorMapButton):
|
class RoundedColorMapButton(ColorMapButton):
|
||||||
|
|||||||
@@ -19,7 +19,7 @@ from qtpy.QtWidgets import (
|
|||||||
QWidget,
|
QWidget,
|
||||||
)
|
)
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.utils.widget_highlighter import WidgetHighlighter
|
from bec_widgets.utils.widget_highlighter import WidgetHighlighter
|
||||||
from bec_widgets.utils.widget_io import WidgetHierarchy
|
from bec_widgets.utils.widget_io import WidgetHierarchy
|
||||||
|
|
||||||
|
|||||||
+10
-11
@@ -1,6 +1,6 @@
|
|||||||
[project]
|
[project]
|
||||||
name = "bec_widgets"
|
name = "bec_widgets"
|
||||||
version = "3.6.0"
|
version = "3.7.2"
|
||||||
description = "BEC Widgets"
|
description = "BEC Widgets"
|
||||||
requires-python = ">=3.11"
|
requires-python = ">=3.11"
|
||||||
classifiers = [
|
classifiers = [
|
||||||
@@ -12,19 +12,19 @@ dependencies = [
|
|||||||
"PyJWT~=2.9",
|
"PyJWT~=2.9",
|
||||||
"PySide6==6.9.0",
|
"PySide6==6.9.0",
|
||||||
"PySide6-QtAds==4.4.0",
|
"PySide6-QtAds==4.4.0",
|
||||||
"bec_ipython_client~=3.107,>=3.107.2", # needed for jupyter console
|
"bec_ipython_client~=3.107,>=3.107.2", # needed for jupyter console
|
||||||
"bec_lib~=3.107,>=3.107.2",
|
"bec_lib~=3.107,>=3.107.2",
|
||||||
"bec_qthemes~=1.0, >=1.3.4",
|
"bec_qthemes~=1.0, >=1.3.4",
|
||||||
"black>=26,<27", # needed for bw-generate-cli
|
"black>=26,<27", # needed for bw-generate-cli
|
||||||
"copier~=9.7",
|
"copier~=9.7",
|
||||||
"darkdetect~=0.8",
|
"darkdetect~=0.8",
|
||||||
"isort>=5.13, <9.0", # needed for bw-generate-cli
|
"isort>=5.13, <9.0", # needed for bw-generate-cli
|
||||||
"markdown~=3.9",
|
"markdown~=3.9",
|
||||||
"ophyd_devices~=1.29, >=1.29.1",
|
"ophyd_devices~=1.29, >=1.29.1",
|
||||||
"pydantic~=2.0",
|
"pydantic~=2.0",
|
||||||
"pylsp-bec~=1.2",
|
"pylsp-bec~=1.2",
|
||||||
"pyqtgraph==0.13.7",
|
"pyqtgraph==0.13.7",
|
||||||
"qtconsole~=5.5, >=5.5.1", # needed for jupyter console
|
"qtconsole~=5.5, >=5.5.1", # needed for jupyter console
|
||||||
"qtmonaco~=0.8, >=0.8.1",
|
"qtmonaco~=0.8, >=0.8.1",
|
||||||
"qtpy~=2.4",
|
"qtpy~=2.4",
|
||||||
"thefuzz~=0.22",
|
"thefuzz~=0.22",
|
||||||
@@ -38,8 +38,8 @@ Homepage = "https://gitlab.psi.ch/bec/bec_widgets"
|
|||||||
[project.scripts]
|
[project.scripts]
|
||||||
bec-app = "bec_widgets.applications.main_app:main"
|
bec-app = "bec_widgets.applications.main_app:main"
|
||||||
bec-designer = "bec_widgets.utils.bec_designer:main"
|
bec-designer = "bec_widgets.utils.bec_designer:main"
|
||||||
bec-gui-server = "bec_widgets.cli.server:main"
|
bec-gui-server = "bec_widgets.applications.companion_app:main"
|
||||||
bw-generate-cli = "bec_widgets.cli.generate_cli:main"
|
bw-generate-cli = "bec_widgets.utils.generate_cli:main"
|
||||||
|
|
||||||
[project.optional-dependencies]
|
[project.optional-dependencies]
|
||||||
dev = [
|
dev = [
|
||||||
@@ -52,12 +52,11 @@ dev = [
|
|||||||
"pytest-xvfb~=3.0",
|
"pytest-xvfb~=3.0",
|
||||||
"pytest~=8.0",
|
"pytest~=8.0",
|
||||||
"pytest-cov~=6.1.1",
|
"pytest-cov~=6.1.1",
|
||||||
|
"pytest-benchmark~=5.2",
|
||||||
"watchdog~=6.0",
|
"watchdog~=6.0",
|
||||||
"pre_commit~=4.2",
|
"pre_commit~=4.2",
|
||||||
]
|
]
|
||||||
qtermwidget = [
|
qtermwidget = ["pyside6_qtermwidget"]
|
||||||
"pyside6_qtermwidget",
|
|
||||||
]
|
|
||||||
|
|
||||||
[build-system]
|
[build-system]
|
||||||
requires = ["hatchling"]
|
requires = ["hatchling"]
|
||||||
@@ -68,7 +67,7 @@ line-length = 100
|
|||||||
skip-magic-trailing-comma = true
|
skip-magic-trailing-comma = true
|
||||||
|
|
||||||
[tool.coverage.report]
|
[tool.coverage.report]
|
||||||
skip_empty = true # exclude empty *files*, e.g. __init__.py, from the report
|
skip_empty = true # exclude empty *files*, e.g. __init__.py, from the report
|
||||||
exclude_lines = [
|
exclude_lines = [
|
||||||
"pragma: no cover",
|
"pragma: no cover",
|
||||||
"if TYPE_CHECKING:",
|
"if TYPE_CHECKING:",
|
||||||
|
|||||||
@@ -0,0 +1,5 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
# BENCHMARK_TITLE: Import bec_widgets
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
python -c 'import bec_widgets; print(bec_widgets.__file__)'
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
# BENCHMARK_TITLE: BEC IPython client with companion app
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
bec --post-startup-file tests/benchmarks/hyperfine/utils/exit_bec_startup.py
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
#!/usr/bin/env bash
|
||||||
|
# BENCHMARK_TITLE: BEC IPython client without companion app
|
||||||
|
set -euo pipefail
|
||||||
|
|
||||||
|
bec --nogui --post-startup-file tests/benchmarks/hyperfine/utils/exit_bec_startup.py
|
||||||
@@ -0,0 +1,5 @@
|
|||||||
|
import time
|
||||||
|
|
||||||
|
_ip = get_ipython()
|
||||||
|
_ip.confirm_exit = False
|
||||||
|
_ip.ask_exit()
|
||||||
@@ -59,5 +59,4 @@ def test_run_line_scan_with_parameters_e2e(scan_control, bec_client_lib, qtbot):
|
|||||||
last_scan = queue.scan_storage.storage[-1]
|
last_scan = queue.scan_storage.storage[-1]
|
||||||
assert last_scan.status_message.info["scan_name"] == scan_name
|
assert last_scan.status_message.info["scan_name"] == scan_name
|
||||||
assert last_scan.status_message.info["exp_time"] == kwargs["exp_time"]
|
assert last_scan.status_message.info["exp_time"] == kwargs["exp_time"]
|
||||||
assert last_scan.status_message.info["scan_motors"] == [args["device"]]
|
|
||||||
assert last_scan.status_message.info["num_points"] == kwargs["steps"]
|
assert last_scan.status_message.info["num_points"] == kwargs["steps"]
|
||||||
|
|||||||
@@ -84,7 +84,6 @@ def test_scan_metadata_for_custom_scan(
|
|||||||
last_scan = queue.scan_storage.storage[-1]
|
last_scan = queue.scan_storage.storage[-1]
|
||||||
assert last_scan.status_message.info["scan_name"] == scan_name
|
assert last_scan.status_message.info["scan_name"] == scan_name
|
||||||
assert last_scan.status_message.info["exp_time"] == kwargs["exp_time"]
|
assert last_scan.status_message.info["exp_time"] == kwargs["exp_time"]
|
||||||
assert last_scan.status_message.info["scan_motors"] == [args["device"]]
|
|
||||||
assert last_scan.status_message.info["num_points"] == kwargs["steps"]
|
assert last_scan.status_message.info["num_points"] == kwargs["steps"]
|
||||||
|
|
||||||
if valid:
|
if valid:
|
||||||
|
|||||||
@@ -0,0 +1,27 @@
|
|||||||
|
from __future__ import annotations
|
||||||
|
|
||||||
|
import pytest
|
||||||
|
|
||||||
|
from bec_widgets.widgets.containers.dock_area.dock_area import BECDockArea
|
||||||
|
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
||||||
|
from tests.unit_tests.client_mocks import mocked_client
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.fixture
|
||||||
|
def dock_area(qtbot, mocked_client):
|
||||||
|
widget = BECDockArea(client=mocked_client)
|
||||||
|
qtbot.addWidget(widget)
|
||||||
|
qtbot.waitExposed(widget)
|
||||||
|
yield widget
|
||||||
|
|
||||||
|
|
||||||
|
def test_add_waveform_to_dock_area(benchmark, dock_area, qtbot, mocked_client):
|
||||||
|
"""Benchmark adding a Waveform widget to an existing dock area."""
|
||||||
|
|
||||||
|
def add_waveform():
|
||||||
|
dock_area.new("Waveform")
|
||||||
|
return dock_area
|
||||||
|
|
||||||
|
dock = benchmark(add_waveform)
|
||||||
|
|
||||||
|
assert dock is not None
|
||||||
@@ -23,11 +23,11 @@ from pytestqt.exceptions import TimeoutError as QtBotTimeoutError
|
|||||||
from qtpy.QtCore import QEvent, QEventLoop
|
from qtpy.QtCore import QEvent, QEventLoop
|
||||||
from qtpy.QtWidgets import QApplication, QMessageBox
|
from qtpy.QtWidgets import QApplication, QMessageBox
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
|
||||||
from bec_widgets.tests.utils import DEVICES, DMMock
|
from bec_widgets.tests.utils import DEVICES, DMMock
|
||||||
from bec_widgets.utils import bec_dispatcher as bec_dispatcher_module
|
from bec_widgets.utils import bec_dispatcher as bec_dispatcher_module
|
||||||
from bec_widgets.utils import error_popups
|
from bec_widgets.utils import error_popups
|
||||||
from bec_widgets.utils.bec_dispatcher import QtRedisConnector
|
from bec_widgets.utils.bec_dispatcher import QtRedisConnector
|
||||||
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
|
|
||||||
# Patch to set default RAISE_ERROR_DEFAULT to True for tests
|
# Patch to set default RAISE_ERROR_DEFAULT to True for tests
|
||||||
# This means that by default, error popups will raise exceptions during tests
|
# This means that by default, error popups will raise exceptions during tests
|
||||||
@@ -227,7 +227,7 @@ def create_history_file(file_path, data: dict, metadata: dict) -> messages.ScanH
|
|||||||
msg = messages.ScanHistoryMessage(
|
msg = messages.ScanHistoryMessage(
|
||||||
scan_id=metadata["scan_id"],
|
scan_id=metadata["scan_id"],
|
||||||
scan_name=metadata["scan_name"],
|
scan_name=metadata["scan_name"],
|
||||||
exit_status=metadata["exit_status"],
|
exit_status=metadata["status"],
|
||||||
file_path=file_path,
|
file_path=file_path,
|
||||||
scan_number=metadata["scan_number"],
|
scan_number=metadata["scan_number"],
|
||||||
dataset_number=metadata["dataset_number"],
|
dataset_number=metadata["dataset_number"],
|
||||||
@@ -274,7 +274,7 @@ def grid_scan_history_msg(tmpdir):
|
|||||||
"scan_id": "test_scan",
|
"scan_id": "test_scan",
|
||||||
"scan_name": "grid_scan",
|
"scan_name": "grid_scan",
|
||||||
"scan_type": "step",
|
"scan_type": "step",
|
||||||
"exit_status": "closed",
|
"status": "closed",
|
||||||
"scan_number": 1,
|
"scan_number": 1,
|
||||||
"dataset_number": 1,
|
"dataset_number": 1,
|
||||||
"request_inputs": {
|
"request_inputs": {
|
||||||
@@ -354,7 +354,7 @@ def scan_history_factory(tmpdir):
|
|||||||
"scan_id": scan_id,
|
"scan_id": scan_id,
|
||||||
"scan_name": scan_name,
|
"scan_name": scan_name,
|
||||||
"scan_type": scan_type,
|
"scan_type": scan_type,
|
||||||
"exit_status": "closed",
|
"status": "closed",
|
||||||
"scan_number": scan_number,
|
"scan_number": scan_number,
|
||||||
"dataset_number": dataset_number,
|
"dataset_number": dataset_number,
|
||||||
"request_inputs": {
|
"request_inputs": {
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import pytest
|
|||||||
from qtpy.QtCore import QObject
|
from qtpy.QtCore import QObject
|
||||||
from qtpy.QtWidgets import QApplication, QWidget
|
from qtpy.QtWidgets import QApplication, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.utils.error_popups import SafeProperty
|
from bec_widgets.utils.error_popups import SafeProperty
|
||||||
from bec_widgets.utils.error_popups import SafeSlot as Slot
|
from bec_widgets.utils.error_popups import SafeSlot as Slot
|
||||||
|
|
||||||
|
|||||||
@@ -71,7 +71,6 @@ def bec_queue_msg_full():
|
|||||||
},
|
},
|
||||||
"report_instructions": [{"scan_progress": 20}],
|
"report_instructions": [{"scan_progress": 20}],
|
||||||
"scan_id": "2d704cc3-c172-404c-866d-608ce09fce40",
|
"scan_id": "2d704cc3-c172-404c-866d-608ce09fce40",
|
||||||
"scan_motors": ["samx"],
|
|
||||||
"scan_number": 1289,
|
"scan_number": 1289,
|
||||||
}
|
}
|
||||||
],
|
],
|
||||||
|
|||||||
@@ -4,9 +4,9 @@ from pydantic import ValidationError
|
|||||||
from qtpy.QtGui import QColor
|
from qtpy.QtGui import QColor
|
||||||
from qtpy.QtWidgets import QVBoxLayout, QWidget
|
from qtpy.QtWidgets import QVBoxLayout, QWidget
|
||||||
|
|
||||||
from bec_widgets.utils import Colors, ConnectionConfig
|
from bec_widgets.utils.bec_connector import ConnectionConfig
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.colors import apply_theme
|
from bec_widgets.utils.colors import Colors, apply_theme
|
||||||
from bec_widgets.widgets.plots.waveform.curve import CurveConfig
|
from bec_widgets.widgets.plots.waveform.curve import CurveConfig
|
||||||
from tests.unit_tests.client_mocks import mocked_client
|
from tests.unit_tests.client_mocks import mocked_client
|
||||||
from tests.unit_tests.conftest import create_widget
|
from tests.unit_tests.conftest import create_widget
|
||||||
|
|||||||
@@ -4,7 +4,7 @@ import pytest
|
|||||||
from qtpy.QtCore import QPointF, Qt
|
from qtpy.QtCore import QPointF, Qt
|
||||||
from qtpy.QtGui import QTransform
|
from qtpy.QtGui import QTransform
|
||||||
|
|
||||||
from bec_widgets.utils import Crosshair
|
from bec_widgets.utils.crosshair import Crosshair
|
||||||
from bec_widgets.widgets.plots.image.image_item import ImageItem
|
from bec_widgets.widgets.plots.image.image_item import ImageItem
|
||||||
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
from bec_widgets.widgets.plots.waveform.waveform import Waveform
|
||||||
from tests.unit_tests.client_mocks import mocked_client
|
from tests.unit_tests.client_mocks import mocked_client
|
||||||
|
|||||||
@@ -146,10 +146,7 @@ class TestDeviceManagerViewDialogs:
|
|||||||
group_combo: QtWidgets.QComboBox = dialog._control_widgets["group_combo"]
|
group_combo: QtWidgets.QComboBox = dialog._control_widgets["group_combo"]
|
||||||
assert group_combo.count() == len(OPHYD_DEVICE_TEMPLATES)
|
assert group_combo.count() == len(OPHYD_DEVICE_TEMPLATES)
|
||||||
|
|
||||||
# Test select a group from available templates
|
|
||||||
variant_combo = dialog._control_widgets["variant_combo"]
|
variant_combo = dialog._control_widgets["variant_combo"]
|
||||||
assert variant_combo.isEnabled() is False
|
|
||||||
|
|
||||||
with qtbot.waitSignal(group_combo.currentTextChanged):
|
with qtbot.waitSignal(group_combo.currentTextChanged):
|
||||||
epics_signal_index = group_combo.findText("EpicsSignal")
|
epics_signal_index = group_combo.findText("EpicsSignal")
|
||||||
group_combo.setCurrentIndex(epics_signal_index) # Select "EpicsSignal" group
|
group_combo.setCurrentIndex(epics_signal_index) # Select "EpicsSignal" group
|
||||||
@@ -235,7 +232,7 @@ class TestDeviceManagerViewDialogs:
|
|||||||
sample_config = {
|
sample_config = {
|
||||||
"name": "TestDevice",
|
"name": "TestDevice",
|
||||||
"enabled": True,
|
"enabled": True,
|
||||||
"deviceClass": "ophyd.EpicsSignal",
|
"deviceClass": "ophyd_devices.EpicsSignal",
|
||||||
"readoutPriority": "baseline",
|
"readoutPriority": "baseline",
|
||||||
"deviceConfig": {"read_pv": "X25DA-ES1-MOT:GET"},
|
"deviceConfig": {"read_pv": "X25DA-ES1-MOT:GET"},
|
||||||
}
|
}
|
||||||
@@ -248,7 +245,7 @@ class TestDeviceManagerViewDialogs:
|
|||||||
assert variant_combo.currentText() == "EpicsSignal"
|
assert variant_combo.currentText() == "EpicsSignal"
|
||||||
config = dialog._device_config_template.get_config_fields()
|
config = dialog._device_config_template.get_config_fields()
|
||||||
assert config["name"] == "TestDevice"
|
assert config["name"] == "TestDevice"
|
||||||
assert config["deviceClass"] == "ophyd.EpicsSignal"
|
assert config["deviceClass"] == "ophyd_devices.EpicsSignal"
|
||||||
assert config["deviceConfig"]["read_pv"] == "X25DA-ES1-MOT:GET"
|
assert config["deviceConfig"]["read_pv"] == "X25DA-ES1-MOT:GET"
|
||||||
|
|
||||||
# Test now to add the device config with different validation results
|
# Test now to add the device config with different validation results
|
||||||
|
|||||||
@@ -1863,9 +1863,14 @@ class TestWorkspaceProfileOperations:
|
|||||||
with patch(
|
with patch(
|
||||||
"bec_widgets.widgets.containers.dock_area.dock_area.SaveProfileDialog", StubDialog
|
"bec_widgets.widgets.containers.dock_area.dock_area.SaveProfileDialog", StubDialog
|
||||||
):
|
):
|
||||||
advanced_dock_area.save_profile(show_dialog=True)
|
widgets_before_save = list(advanced_dock_area.widget_list())
|
||||||
|
with patch.object(advanced_dock_area, "load_profile") as mock_load_profile:
|
||||||
|
advanced_dock_area.save_profile(show_dialog=True)
|
||||||
|
qtbot.wait(100)
|
||||||
|
mock_load_profile.assert_not_called()
|
||||||
|
|
||||||
qtbot.wait(500)
|
qtbot.wait(500)
|
||||||
|
assert list(advanced_dock_area.widget_list()) == widgets_before_save
|
||||||
source_manifest = read_manifest(helper.open_user(source_profile))
|
source_manifest = read_manifest(helper.open_user(source_profile))
|
||||||
new_manifest = read_manifest(helper.open_user(new_profile))
|
new_manifest = read_manifest(helper.open_user(new_profile))
|
||||||
|
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import black
|
|||||||
import isort
|
import isort
|
||||||
import pytest
|
import pytest
|
||||||
|
|
||||||
from bec_widgets.cli.generate_cli import ClientGenerator
|
from bec_widgets.utils.generate_cli import ClientGenerator
|
||||||
from bec_widgets.utils.plugin_utils import BECClassContainer, BECClassInfo
|
from bec_widgets.utils.plugin_utils import BECClassContainer, BECClassInfo
|
||||||
|
|
||||||
# pylint: disable=missing-function-docstring
|
# pylint: disable=missing-function-docstring
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ from qtpy.QtCore import QEvent, QPoint, QPointF, Qt
|
|||||||
from qtpy.QtGui import QColor, QMouseEvent
|
from qtpy.QtGui import QColor, QMouseEvent
|
||||||
from qtpy.QtWidgets import QApplication
|
from qtpy.QtWidgets import QApplication
|
||||||
|
|
||||||
from bec_widgets.utils import Colors
|
from bec_widgets.utils.colors import Colors
|
||||||
from bec_widgets.widgets.progress.ring_progress_bar.ring_progress_bar import (
|
from bec_widgets.widgets.progress.ring_progress_bar.ring_progress_bar import (
|
||||||
RingProgressBar,
|
RingProgressBar,
|
||||||
RingProgressContainerWidget,
|
RingProgressContainerWidget,
|
||||||
|
|||||||
@@ -1,4 +1,4 @@
|
|||||||
from bec_widgets.cli.rpc.rpc_register import RPCRegister
|
from bec_widgets.utils.rpc_register import RPCRegister
|
||||||
|
|
||||||
|
|
||||||
class FakeObject:
|
class FakeObject:
|
||||||
|
|||||||
@@ -5,7 +5,7 @@ import pytest
|
|||||||
from bec_lib.service_config import ServiceConfig
|
from bec_lib.service_config import ServiceConfig
|
||||||
from qtpy.QtWidgets import QWidget
|
from qtpy.QtWidgets import QWidget
|
||||||
|
|
||||||
from bec_widgets.cli.server import GUIServer
|
from bec_widgets.applications.companion_app import GUIServer
|
||||||
from bec_widgets.utils.bec_connector import BECConnector
|
from bec_widgets.utils.bec_connector import BECConnector
|
||||||
from bec_widgets.utils.rpc_server import RegistryNotReadyError, RPCServer, SingleshotRPCRepeat
|
from bec_widgets.utils.rpc_server import RegistryNotReadyError, RPCServer, SingleshotRPCRepeat
|
||||||
|
|
||||||
|
|||||||
@@ -1,8 +1,8 @@
|
|||||||
from unittest.mock import patch
|
from unittest.mock import patch
|
||||||
|
|
||||||
from bec_widgets.cli.rpc.rpc_widget_handler import RPCWidgetHandler
|
|
||||||
from bec_widgets.utils.bec_widget import BECWidget
|
from bec_widgets.utils.bec_widget import BECWidget
|
||||||
from bec_widgets.utils.plugin_utils import BECClassContainer, BECClassInfo
|
from bec_widgets.utils.plugin_utils import BECClassContainer, BECClassInfo
|
||||||
|
from bec_widgets.utils.rpc_widget_handler import RPCWidgetHandler
|
||||||
|
|
||||||
|
|
||||||
def test_rpc_widget_handler():
|
def test_rpc_widget_handler():
|
||||||
@@ -16,7 +16,7 @@ class _TestPluginWidget(BECWidget): ...
|
|||||||
|
|
||||||
|
|
||||||
@patch(
|
@patch(
|
||||||
"bec_widgets.cli.rpc.rpc_widget_handler.get_all_plugin_widgets",
|
"bec_widgets.utils.rpc_widget_handler.get_all_plugin_widgets",
|
||||||
return_value=BECClassContainer(
|
return_value=BECClassContainer(
|
||||||
[
|
[
|
||||||
BECClassInfo(name="DeviceComboBox", obj=_TestPluginWidget, module="", file=""),
|
BECClassInfo(name="DeviceComboBox", obj=_TestPluginWidget, module="", file=""),
|
||||||
|
|||||||
@@ -501,6 +501,7 @@ def test_changing_scans_remember_parameters(scan_control, mocked_client):
|
|||||||
assert grid_kwargs["burst_at_each_point"] == kwargs["burst_at_each_point"]
|
assert grid_kwargs["burst_at_each_point"] == kwargs["burst_at_each_point"]
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.skip(reason="Unreliable - GH issue #1134")
|
||||||
def test_get_scan_parameters_from_redis(scan_control, mocked_client):
|
def test_get_scan_parameters_from_redis(scan_control, mocked_client):
|
||||||
scan_name = "line_scan"
|
scan_name = "line_scan"
|
||||||
scan_control.comboBox_scan_selection.setCurrentText(scan_name)
|
scan_control.comboBox_scan_selection.setCurrentText(scan_name)
|
||||||
@@ -585,6 +586,7 @@ def test_scan_metadata_is_passed_to_scan_function(scan_control: ScanControl):
|
|||||||
scans.grid_scan.assert_called_once_with(metadata=TEST_MD)
|
scans.grid_scan.assert_called_once_with(metadata=TEST_MD)
|
||||||
|
|
||||||
|
|
||||||
|
@pytest.mark.skip(reason="Unreliable - GH issue #1134")
|
||||||
def test_restore_parameters_with_fewer_arg_bundles(scan_control, qtbot):
|
def test_restore_parameters_with_fewer_arg_bundles(scan_control, qtbot):
|
||||||
"""
|
"""
|
||||||
Ensure that when more argument bundles are present than exist in the
|
Ensure that when more argument bundles are present than exist in the
|
||||||
|
|||||||
Reference in New Issue
Block a user