Skip to content

Commit

Permalink
Simplified FMU observation setup
Browse files Browse the repository at this point in the history
  • Loading branch information
anders-kiaer committed Dec 5, 2024
1 parent e2f1947 commit aa37710
Show file tree
Hide file tree
Showing 8 changed files with 1,323 additions and 88 deletions.
9 changes: 4 additions & 5 deletions pyproject.toml
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,6 @@ dependencies = [
"ert",
"ert>=10.2.0b13",
"fmu-tools",
"grid3d_maps",
"matplotlib",
"numpy<2",
"opm>=2023.04",
Expand All @@ -55,6 +54,8 @@ dependencies = [
"shapely",
"xlrd",
"xtgeo",
"fmu-config",
"odfpy"
]

[project.optional-dependencies]
Expand All @@ -75,7 +76,7 @@ docs = [
"autoapi",
"sphinx",
"sphinx-argparse",
"sphinx-autodoc-typehints<2.4",
"sphinx-autodoc-typehints",
"sphinx-copybutton",
"sphinx-togglebutton",
"sphinx_rtd_theme",
Expand Down Expand Up @@ -117,16 +118,14 @@ sw_model_utilities = "subscript.sw_model_utilities.sw_model_utilities:main"
sunsch = "subscript.sunsch.sunsch:main"
vfp2csv = "subscript.vfp2csv.vfp2csv:main"
welltest_dpds = "subscript.welltest_dpds.welltest_dpds:main"
field_statistics = "subscript.field_statistics.field_statistics:main"

genertobs_unstable = "subscript.genertobs_unstable.main:main"

[project.entry-points.ert]
subscript_jobs = "subscript.hook_implementations.jobs"
CsvMerge = "subscript.csv_merge.csv_merge"
CsvStack = "subscript.csv_stack.csv_stack"
FmuObs = "subscript.fmuobs.fmuobs"
Params2Csv = "subscript.params2csv.params2csv"
FieldStatistics = "subscript.field_statistics.field_statistics"

[tool.setuptools]
script-files = [
Expand Down
File renamed without changes.
325 changes: 325 additions & 0 deletions src/subscript/genertobs_unstable/_datatypes.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,325 @@
"""Pydantic models for genertobs"""

import logging
import warnings
from enum import Enum
from pathlib import Path
from typing import Dict, List, Union

from pydantic import (
BaseModel,
ConfigDict,
Field,
RootModel,
computed_field,
field_validator,
model_validator,
)


def is_number(tocheck):
"""Check that variable can be converted to number
Args:
tocheck (something): what shall be checked
Returns:
bool: check passed or not
"""
try:
float(tocheck)
return True
except TypeError:
return False


def is_percent_range(string):
"""Check if string ending with % is between 0 and 100
Args:
string (str): the string to check
Returns:
bool: True if number is in the range
"""
logger = logging.getLogger(__file__ + ".is_percent_range")
logger.debug("Input is %s", string)
number = float(string.replace("%", ""))
if 0 < number < 100:
return True

if number > 50:
warnings.warn(
"It seems weird to have an error of more than 50%"
f" ({number}, is this correct?)"
)
return False


def is_string_convertible_2_percent(error):
"""Check string
Args:
error (str): string to check
Raises:
ValueError: if string does not end with a percent sign
TypeError: if string cannot be converted to a number
ValueError: if the number ends with %, but is not between 0 and 100
"""
logger = logging.getLogger(__file__ + ".is_string_convertible_2_percent")
logger.debug("Checking this string %s", error)
try:
if not is_number(error[:-1]):
raise TypeError(f"This: {error} is not convertible to a number")
except TypeError:
pass

if not error.endswith("%"):
raise ValueError(
f"When default_error ({error}) given as string it must end with a % sign"
)

if not is_percent_range(error):
raise ValueError(f"The number {error} is not in the valid range 0-100")


def check_error_limits(error, err_min, err_max):
"""Check error limits
Args:
error (Union[str,int,float]): the error to check against
err_min (Union[int,float,None]): the lower limit
err_max (Union[int,float,None]): the higher limit
Raises:
ValueError: if err_min is not None when error is not in percent
ValueError: if err_max is not None when error is not in percent
ValueError: if err_min >= max
"""
logger = logging.getLogger(__file__ + ".check_error_limits")
logger.debug("Checking with error: %s, and limits %s-%s", error, err_min, err_max)
if isinstance(error, (float, int)):
if err_min is not None:
raise ValueError(
"default_error specified as an absolute number,"
f" doesn't make sense to set a lower limit ({err_min})"
)
if err_max is not None:
raise ValueError(
"default_error specified as an absolute number,"
f" doesn't make sense to set a higher limit ({err_max})"
)
else:
if err_min is not None and err_max is not None and err_min >= err_max:
raise ValueError(
f"When using limits, the minimum must be lower than the maximum\n"
f"{err_min}-{err_max}"
)


class ObservationType(Enum):
"""The valid datatypes in a config file
Args:
Enum (Enum): Enumerator
"""

SUMMARY = "summary"
RFT = "rft"


class RftType(Enum):
"""Valid Rft types
Args:
Enum (Enum): Enumerator
"""

PRESSURE = "pressure"
SWAT = "saturation_water"
SOIL = "saturation_oil"
SGAS = "saturation_gas"


class ElementMetaData(BaseModel):
"""Pattern for Metadata element for observations
Args:
BaseModel (BaseModel): pydantic BaseModel
"""

subtype: RftType = Field(
default=RftType.PRESSURE,
description=f"Type of rft observation, can be any of {RftType.__members__}",
)

@property
@computed_field
def columns(self) -> Dict[RftType, Dict[str, str]]:
"""Define columns to expect
Returns:
Dict[RftType, Dict[str, str]]: the expected column with units
"""
if self.subtype == RftType.PRESSURE:
units = {"unit": "bar"}
else:
units = {"unit": "fraction"}
return {self.subtype: units}


class PluginArguments(BaseModel):
"""Plugin arguments for config element"""

model_config = ConfigDict(extra="forbid")
zonemap: str = Field(
default="",
description="path to file with mapping between zone names and grid layers",
)
trajectories: str = Field(
default="", description="path to folder with trajectory files"
)


class ConfigElement(BaseModel):
"""Element in a config file"""

name: str = Field(min_length=5, description="Name of observation")
type: ObservationType = Field(description="Type of observation")
observation: str = Field(
description="path to file containing observations, this can be any csv"
" like file,\n i.e textfile or spreadsheet",
)
active: bool = Field(
default=True,
description="If the observation element shall be used in\n "
"generation of ert observations",
)
default_error: Union[str, float, int] = Field(
default=None,
description="Optional argument. Error to be used\n. Used only when"
"no error column present or where error column is empty."
" Can be supplied as any number, and even with a percentage sign",
)

min_error: Union[int, float] = Field(
default=None,
description="Optional argument. Minimum error, only allowed "
"when default_error is in percent",
)
max_error: Union[int, float] = Field(
default=None,
description="Optional argument. Maximum error, only allowed "
"when default_error is in percent",
)

alias_file: str = Field(default=None, description="Name of file with aliases")

@field_validator("observation")
@classmethod
def validate_path_exists(cls, observation_path: str):
"""Check that observation file exists
Args:
observation_path (str): the path to check
Raises:
OSError: if observation_path does not exist
"""
if not Path(observation_path).exists():
raise OSError(f"Input observation file {observation_path}, does not exist")
return observation_path

@field_validator("default_error")
@classmethod
def validate_default_error(cls, error: Union[str, int, float]):
"""Check that if error is string, and if so then check that it is in %
Args:
observation_path (str): the path to check
Raises:
OSError: if observation_path does not exist
"""
try:
is_string_convertible_2_percent(error)
except AttributeError:
if error < 0: # type: ignore
raise ValueError(f"default error cannot be negative {error}") # pylint: disable=raise-missing-from
return error

@model_validator(mode="after")
def check_when_default_is_number(self):
"""Check
Returns:
Self: self
"""
check_error_limits(self.default_error, self.min_error, self.max_error)
return self


class SummaryConfigElement(ConfigElement):
"""ConfigElement for Summary data
Args:
ConfigElement (ConfigElement): base observation config element
"""

model_config = ConfigDict(extra="forbid")


class RftConfigElement(ConfigElement):
"""Config element for Rft
contains some additional fields
Args:
ConfigElement (pydantic model): base observation config element
"""

plugin_arguments: PluginArguments # = Field(default=None)
metadata: ElementMetaData = Field(
default=ElementMetaData(),
description="Metadata describing the type",
)

# @field_validator("type")
# @classmethod
# def validate_of_rft_type(cls, observation_type: ObservationType):
# """validate that type is rft

# Args:
# observation_type (ObservationType): the type of observation

# Raises:
# TypeError: if type is not RFT

# Returns:
# ObservationType: type of observations
# """
# if observation_type != ObservationType.RFT:
# raise TypeError(f"This is not rft type, but {observation_type}")
# return observation_type


class ObservationsConfig(RootModel):
"""Root model for config file
Args:
RootModel (Rootmodel): pydantic root model
"""

root: List[Union[SummaryConfigElement, RftConfigElement]] = Field(
description="What type of observation",
)

def __iter__(self):
return iter(self.root)

def __getitem__(self, item):
return self.root[item]

def __len__(self):
return len(self.root)
Loading

0 comments on commit aa37710

Please sign in to comment.