Source code for openstef_beam.analysis.models.target_metadata
# SPDX-FileCopyrightText: 2025 Contributors to the OpenSTEF project <openstef@lfenergy.org>
#
# SPDX-License-Identifier: MPL-2.0
"""Target metadata models for analysis pipeline.
This module defines the metadata structure that carries target information
through the analysis pipeline, including grouping and filtering context.
"""
from typing import Self
from pydantic import Field, model_validator
from openstef_beam.evaluation.models import Filtering
from openstef_core.base_model import BaseModel
type TargetName = str
type GroupName = str
type RunName = str
[docs]
class TargetMetadata(BaseModel):
"""Metadata for a forecasting target in the analysis pipeline.
Contains essential information about a target including its grouping context
and lead time filtering criteria. Lead time filtering determines which
predictions are included based on how far ahead they were made (e.g.,
1-hour ahead vs 24-hour ahead forecasts).
Raises:
ValueError: When limit constraints are not met (either 'limit' alone or both
'upper_limit' and 'lower_limit' must be specified).
"""
name: TargetName = Field(description="Name of the target")
group_name: GroupName = Field(description="Name of the group this target belongs to, used for grouping in reports")
filtering: Filtering | None = Field(
description="Lead time filtering criteria - either AvailableAt (data availability time) "
"or LeadTime (forecast horizon)"
)
limit: float | None = Field(default=None, description="Capacity limit of the target in appropriate units")
upper_limit: float | None = Field(
default=None, description="Upper capacity limit of the target in appropriate units"
)
lower_limit: float | None = Field(
default=None, description="Lower capacity limit of the target in appropriate units"
)
run_name: RunName = Field(description="Name of the run associated with this target")
[docs]
@model_validator(mode="after")
def validate_limits(self) -> "TargetMetadata":
"""Validate that either limit or both upper_limit and lower_limit are provided.
Returns:
The validated TargetMetadata instance.
Raises:
ValueError: If neither limit nor (upper_limit and lower_limit) are provided,
or if both limit and (upper_limit or lower_limit) are provided.
"""
has_limit = self.limit is not None
has_upper = self.upper_limit is not None
has_lower = self.lower_limit is not None
if has_limit and (has_upper or has_lower):
raise ValueError(
"Cannot specify both 'limit' and 'upper_limit'/'lower_limit'. "
"Use either 'limit' alone or both 'upper_limit' and 'lower_limit'."
)
if not has_limit and not (has_upper and has_lower):
raise ValueError("Must specify either 'limit' or both 'upper_limit' and 'lower_limit'.")
return self
[docs]
def with_filtering(self, filtering: Filtering) -> Self:
"""Returns a copy of the target metadata with different lead time filtering applied.
Args:
filtering: New lead time filtering criteria to apply
Returns:
New TargetMetadata instance with updated lead time filtering
"""
return type(self)(
name=self.name,
group_name=self.group_name,
filtering=filtering,
limit=self.limit,
upper_limit=self.upper_limit,
lower_limit=self.lower_limit,
run_name=self.run_name,
)
__all__ = ["GroupName", "RunName", "TargetMetadata", "TargetName"]