Skip to content

Commit

Permalink
Support manipulation of field values
Browse files Browse the repository at this point in the history
  • Loading branch information
gerlero committed Mar 19, 2024
1 parent 4277594 commit b958772
Show file tree
Hide file tree
Showing 2 changed files with 43 additions and 5 deletions.
44 changes: 39 additions & 5 deletions foamlib/_dictionaries.py
Original file line number Diff line number Diff line change
Expand Up @@ -41,7 +41,7 @@ def _cmd(self, args: Sequence[str], *, key: Optional[str] = None) -> str:
) from None

@staticmethod
def _parse(value: str) -> "FoamDictionary.Value":
def _parse(value: str) -> Value:
if value == "yes":
return True
elif value == "no":
Expand Down Expand Up @@ -82,25 +82,45 @@ def _parse(value: str) -> "FoamDictionary.Value":
return value

@staticmethod
def _str(value: Union[Value, "FoamDictionary"]) -> str:
def _str(value: Union[Value, "FoamDictionary"], assume_field: bool = False) -> str:
if isinstance(value, FoamDictionary):
return value._cmd(["-value"])
elif isinstance(value, Mapping):
out = "{ "
for k, v in value.items():
out += f"{k} {FoamDictionary._str(v)}"
assume_field = k == "internalField" or k == "value"
out += f"{k} {FoamDictionary._str(v, assume_field=assume_field)}"
if not isinstance(v, Mapping):
out += "; "
out += "} "
return out
elif isinstance(value, Sequence) and not isinstance(value, str):
out = "( "
out = ""
if assume_field:
if len(value) < 10:
out += "uniform "
else:
out += "nonuniform List<"
if not isinstance(value[0], Sequence):
out += "scalar"
elif len(value[0]) == 3:
out += "vector"
elif len(value[0]) == 6:
out += "symmTensor"
elif len(value[0]) == 9:
out += "tensor"
else:
raise ValueError(f"Unsupported sequence length for field: {len(value[0])}")
out += "> "
out += "( "
for v in value:
out += f"{FoamDictionary._str(v)} "
out += ") "
return out
elif isinstance(value, bool):
return "yes" if value else "no"
elif assume_field and isinstance(value, (int, float)):
return f"uniform {value}"
else:
return str(value)

Expand All @@ -114,7 +134,13 @@ def __getitem__(self, key: str) -> Union[Value, "FoamDictionary"]:
return FoamDictionary._parse(value)

def __setitem__(self, key: str, value: Any) -> None:
self._cmd(["-set", self._str(value)], key=key)
assume_field = False
if key == "internalField":
assume_field = True
elif key == "value" and len(self._keywords) == 2 and self._keywords[0] == "boundaryField":
assume_field = True

self._cmd(["-set", self._str(value, assume_field=assume_field)], key=key)

def __delitem__(self, key: str) -> None:
if key not in self:
Expand Down Expand Up @@ -153,6 +179,10 @@ def internal_field(self) -> FoamDictionary.Value:
if isinstance(ret, FoamDictionary):
raise TypeError("internalField is a dictionary")
return ret

@internal_field.setter
def internal_field(self, value: FoamDictionary.Value) -> None:
self["internalField"] = value

@property
def boundary_field(self) -> FoamDictionary:
Expand All @@ -163,6 +193,10 @@ def boundary_field(self) -> FoamDictionary:
if not isinstance(ret, FoamDictionary):
raise TypeError("boundaryField is not a dictionary")
return ret

@boundary_field.setter
def boundary_field(self, value: Mapping) -> None:
self["boundaryField"] = value

def __fspath__(self) -> str:
return str(self.path)
Expand Down
4 changes: 4 additions & 0 deletions tests/test_pitz.py
Original file line number Diff line number Diff line change
Expand Up @@ -25,6 +25,10 @@ def test_run(pitz: FoamCase) -> None:
assert isinstance(internal, Sequence)
assert len(internal) == 12225

internal[42][1] = 42
pitz[-1]["U"].internal_field = internal
assert pitz[-1]["U"].internal_field[42][1] == 42


def test_double_clean(pitz: FoamCase) -> None:
pitz.clean()
Expand Down

0 comments on commit b958772

Please sign in to comment.