From 2a007dcfdf92c9d18c7c5afd6d4cbdc491ee69ff Mon Sep 17 00:00:00 2001 From: mcflugen Date: Mon, 29 Jan 2024 23:10:44 -0700 Subject: [PATCH] add type annotations and test with mypy --- .pre-commit-config.yaml | 2 +- heat/bmi_heat.py | 120 +++++++++++++++++++++++----------------- heat/heat.py | 40 +++++++++----- pyproject.toml | 8 +++ 4 files changed, 104 insertions(+), 66 deletions(-) diff --git a/.pre-commit-config.yaml b/.pre-commit-config.yaml index ab8dce0..9771c19 100644 --- a/.pre-commit-config.yaml +++ b/.pre-commit-config.yaml @@ -61,4 +61,4 @@ repos: - id: mypy language_version: python3.12 additional_dependencies: [types-all] - files: src/.*\.py$ + files: heat/.*\.py$ diff --git a/heat/bmi_heat.py b/heat/bmi_heat.py index 76daf23..51ddf95 100644 --- a/heat/bmi_heat.py +++ b/heat/bmi_heat.py @@ -1,8 +1,11 @@ #! /usr/bin/env python """Basic Model Interface implementation for the 2D heat model.""" +from typing import Any + import numpy as np from bmipy import Bmi +from numpy.typing import NDArray from .heat import Heat @@ -15,20 +18,21 @@ class BmiHeat(Bmi): _input_var_names = ("plate_surface__temperature",) _output_var_names = ("plate_surface__temperature",) - def __init__(self): + def __init__(self) -> None: """Create a BmiHeat model that is ready for initialization.""" - self._model = None - self._values = {} - self._var_units = {} - self._var_loc = {} - self._grids = {} - self._grid_type = {} + # self._model: Heat | None = None + self._model: Heat + self._values: dict[str, NDArray[Any]] = {} + self._var_units: dict[str, str] = {} + self._var_loc: dict[str, str] = {} + self._grids: dict[int, list[str]] = {} + self._grid_type: dict[int, str] = {} self._start_time = 0.0 - self._end_time = np.finfo("d").max + self._end_time = float(np.finfo("d").max) self._time_units = "s" - def initialize(self, filename=None): + def initialize(self, filename: str | None = None) -> None: """Initialize the Heat model. Parameters @@ -40,7 +44,7 @@ def initialize(self, filename=None): self._model = Heat() elif isinstance(filename, str): with open(filename) as file_obj: - self._model = Heat.from_file_like(file_obj.read()) + self._model = Heat.from_file_like(file_obj) else: self._model = Heat.from_file_like(filename) @@ -50,11 +54,11 @@ def initialize(self, filename=None): self._grids = {0: ["plate_surface__temperature"]} self._grid_type = {0: "uniform_rectilinear"} - def update(self): + def update(self) -> None: """Advance model by one time step.""" self._model.advance_in_time() - def update_frac(self, time_frac): + def update_frac(self, time_frac: float) -> None: """Update model by a fraction of a time step. Parameters @@ -67,7 +71,7 @@ def update_frac(self, time_frac): self.update() self._model.time_step = time_step - def update_until(self, then): + def update_until(self, then: float) -> None: """Update model until a particular time. Parameters @@ -81,11 +85,12 @@ def update_until(self, then): self.update() self.update_frac(n_steps - int(n_steps)) - def finalize(self): + def finalize(self) -> None: """Finalize model.""" - self._model = None + del self._model + # self._model = None - def get_var_type(self, var_name): + def get_var_type(self, var_name: str) -> str: """Data type of variable. Parameters @@ -100,7 +105,7 @@ def get_var_type(self, var_name): """ return str(self.get_value_ptr(var_name).dtype) - def get_var_units(self, var_name): + def get_var_units(self, var_name: str) -> str: """Get units of variable. Parameters @@ -115,7 +120,7 @@ def get_var_units(self, var_name): """ return self._var_units[var_name] - def get_var_nbytes(self, var_name): + def get_var_nbytes(self, var_name: str) -> int: """Get units of variable. Parameters @@ -130,13 +135,13 @@ def get_var_nbytes(self, var_name): """ return self.get_value_ptr(var_name).nbytes - def get_var_itemsize(self, name): + def get_var_itemsize(self, name: str) -> int: return np.dtype(self.get_var_type(name)).itemsize - def get_var_location(self, name): + def get_var_location(self, name: str) -> str: return self._var_loc[name] - def get_var_grid(self, var_name): + def get_var_grid(self, var_name: str) -> int | None: """Grid id for a variable. Parameters @@ -152,8 +157,9 @@ def get_var_grid(self, var_name): for grid_id, var_name_list in self._grids.items(): if var_name in var_name_list: return grid_id + return None - def get_grid_rank(self, grid_id): + def get_grid_rank(self, grid_id: int) -> int: """Rank of grid. Parameters @@ -168,7 +174,7 @@ def get_grid_rank(self, grid_id): """ return len(self._model.shape) - def get_grid_size(self, grid_id): + def get_grid_size(self, grid_id: int) -> int: """Size of grid. Parameters @@ -183,7 +189,7 @@ def get_grid_size(self, grid_id): """ return int(np.prod(self._model.shape)) - def get_value_ptr(self, var_name): + def get_value_ptr(self, var_name: str) -> NDArray[Any]: """Reference to values. Parameters @@ -198,7 +204,7 @@ def get_value_ptr(self, var_name): """ return self._values[var_name] - def get_value(self, var_name, dest): + def get_value(self, var_name: str, dest: NDArray[Any]) -> NDArray[Any]: """Copy of values. Parameters @@ -216,7 +222,9 @@ def get_value(self, var_name, dest): dest[:] = self.get_value_ptr(var_name).flatten() return dest - def get_value_at_indices(self, var_name, dest, indices): + def get_value_at_indices( + self, var_name: str, dest: NDArray[Any], indices: NDArray[np.int_] + ) -> NDArray[Any]: """Get values at particular indices. Parameters @@ -236,7 +244,7 @@ def get_value_at_indices(self, var_name, dest, indices): dest[:] = self.get_value_ptr(var_name).take(indices) return dest - def set_value(self, var_name, src): + def set_value(self, var_name: str, src: NDArray[Any]) -> None: """Set model values. Parameters @@ -249,7 +257,9 @@ def set_value(self, var_name, src): val = self.get_value_ptr(var_name) val[:] = src.reshape(val.shape) - def set_value_at_indices(self, name, inds, src): + def set_value_at_indices( + self, name: str, inds: NDArray[np.int_], src: NDArray[Any] + ) -> None: """Set model values at particular indices. Parameters @@ -264,76 +274,80 @@ def set_value_at_indices(self, name, inds, src): val = self.get_value_ptr(name) val.flat[inds] = src - def get_component_name(self): + def get_component_name(self) -> str: """Name of the component.""" return self._name - def get_input_item_count(self): + def get_input_item_count(self) -> int: """Get names of input variables.""" return len(self._input_var_names) - def get_output_item_count(self): + def get_output_item_count(self) -> int: """Get names of output variables.""" return len(self._output_var_names) - def get_input_var_names(self): + def get_input_var_names(self) -> tuple[str, ...]: """Get names of input variables.""" return self._input_var_names - def get_output_var_names(self): + def get_output_var_names(self) -> tuple[str, ...]: """Get names of output variables.""" return self._output_var_names - def get_grid_shape(self, grid_id, shape): + def get_grid_shape(self, grid_id: int, shape: NDArray[np.int_]) -> NDArray[np.int_]: """Number of rows and columns of uniform rectilinear grid.""" var_name = self._grids[grid_id][0] shape[:] = self.get_value_ptr(var_name).shape return shape - def get_grid_spacing(self, grid_id, spacing): + def get_grid_spacing( + self, grid_id: int, spacing: NDArray[np.float_] + ) -> NDArray[np.float_]: """Spacing of rows and columns of uniform rectilinear grid.""" spacing[:] = self._model.spacing return spacing - def get_grid_origin(self, grid_id, origin): + def get_grid_origin( + self, grid_id: int, origin: NDArray[np.float_] + ) -> NDArray[np.float_]: """Origin of uniform rectilinear grid.""" origin[:] = self._model.origin return origin - def get_grid_type(self, grid_id): + def get_grid_type(self, grid_id: int) -> str: """Type of grid.""" return self._grid_type[grid_id] - def get_start_time(self): + def get_start_time(self) -> float: """Start time of model.""" return self._start_time - def get_end_time(self): + def get_end_time(self) -> float: """End time of model.""" return self._end_time - def get_current_time(self): + def get_current_time(self) -> float: return self._model.time - def get_time_step(self): + def get_time_step(self) -> float: return self._model.time_step - def get_time_units(self): + def get_time_units(self) -> str: return self._time_units - def get_grid_edge_count(self, grid): + def get_grid_edge_count(self, grid: int) -> int: raise NotImplementedError("get_grid_edge_count") - def get_grid_edge_nodes(self, grid, edge_nodes): + def get_grid_edge_nodes(self, grid: int, edge_nodes: NDArray[np.int_]) -> None: raise NotImplementedError("get_grid_edge_nodes") - def get_grid_face_count(self, grid): + def get_grid_face_count(self, grid: int) -> None: raise NotImplementedError("get_grid_face_count") - def get_grid_face_nodes(self, grid, face_nodes): + def get_grid_face_nodes(self, grid: int, face_nodes: NDArray[np.int_]) -> None: raise NotImplementedError("get_grid_face_nodes") - def get_grid_node_count(self, grid): + def get_grid_node_count(self, grid: int) -> int: """Number of grid nodes. Parameters @@ -348,17 +362,19 @@ def get_grid_node_count(self, grid): """ return self.get_grid_size(grid) - def get_grid_nodes_per_face(self, grid, nodes_per_face): + def get_grid_nodes_per_face( + self, grid: int, nodes_per_face: NDArray[np.int_] + ) -> None: raise NotImplementedError("get_grid_nodes_per_face") - def get_grid_face_edges(self, grid, face_edges): + def get_grid_face_edges(self, grid: int, face_edges: NDArray[np.int_]) -> None: raise NotImplementedError("get_grid_face_edges") - def get_grid_x(self, grid, x): + def get_grid_x(self, grid: int, x: NDArray[np.float_]) -> None: raise NotImplementedError("get_grid_x") - def get_grid_y(self, grid, y): + def get_grid_y(self, grid: int, y: NDArray[np.float_]) -> None: raise NotImplementedError("get_grid_y") - def get_grid_z(self, grid, z): + def get_grid_z(self, grid: int, z: NDArray[np.float_]) -> None: raise NotImplementedError("get_grid_z") diff --git a/heat/heat.py b/heat/heat.py index baa20d8..fa31bc1 100644 --- a/heat/heat.py +++ b/heat/heat.py @@ -1,11 +1,21 @@ """The 2D heat model.""" +from __future__ import annotations + +from io import TextIOBase import numpy as np import yaml +from numpy.typing import NDArray from scipy import ndimage -def solve_2d(temp, spacing, out=None, alpha=1.0, time_step=1.0): +def solve_2d( + temp: NDArray[np.float_], + spacing: tuple[float, ...], + out: NDArray[np.float_] | None = None, + alpha: float = 1.0, + time_step: float = 1.0, +) -> NDArray[np.float_]: """Solve the 2D Heat Equation on a uniform mesh. Parameters @@ -82,8 +92,12 @@ class Heat: """ def __init__( - self, shape=(10, 20), spacing=(1.0, 1.0), origin=(0.0, 0.0), alpha=1.0 - ): + self, + shape: tuple[int, int] = (10, 20), + spacing: tuple[float, float] = (1.0, 1.0), + origin: tuple[float, float] = (0.0, 0.0), + alpha: float = 1.0, + ) -> None: """Create a new heat model. Parameters @@ -108,17 +122,17 @@ def __init__( self._next_temperature = np.empty_like(self._temperature) @property - def time(self): + def time(self) -> float: """Current model time.""" return self._time @property - def temperature(self): + def temperature(self) -> NDArray[np.float_]: """Temperature of the plate.""" return self._temperature @temperature.setter - def temperature(self, new_temp): + def temperature(self, new_temp: float) -> None: """Set the temperature of the plate. Parameters @@ -129,32 +143,32 @@ def temperature(self, new_temp): self._temperature[:] = new_temp @property - def time_step(self): + def time_step(self) -> float: """Model time step.""" return self._time_step @time_step.setter - def time_step(self, time_step): + def time_step(self, time_step: float) -> None: """Set model time step.""" self._time_step = time_step @property - def shape(self): + def shape(self) -> tuple[int, int]: """Shape of the model grid.""" return self._shape @property - def spacing(self): + def spacing(self) -> tuple[float, float]: """Spacing between nodes of the model grid.""" return self._spacing @property - def origin(self): + def origin(self) -> tuple[float, float]: """Origin coordinates of the model grid.""" return self._origin @classmethod - def from_file_like(cls, file_like): + def from_file_like(cls: type[Heat], file_like: TextIOBase) -> Heat: """Create a Heat object from a file-like object. Parameters @@ -170,7 +184,7 @@ def from_file_like(cls, file_like): config = yaml.safe_load(file_like) return cls(**config) - def advance_in_time(self): + def advance_in_time(self) -> None: """Calculate new temperatures for the next time step.""" solve_2d( self._temperature, diff --git a/pyproject.toml b/pyproject.toml index 551a1af..8328aed 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -85,6 +85,14 @@ relative_files = true profile = "black" force_single_line = "true" +[tool.mypy] +check_untyped_defs = true +disallow_any_generics = true +disallow_incomplete_defs = true +disallow_untyped_defs = true +warn_redundant_casts = true +warn_unused_ignores = true + [tool.pytest.ini_options] minversion = "6.0" testpaths = [