import sys
from pathlib import Path
from typing import Union, Tuple, List, Optional, Any
if sys.version_info.minor >= 11:
from typing import Self
else:
Self = Any
import pandas as pd
from iode.util import table2str, JUSTIFY, split_list
from iode.iode_database.abstract_database import IodeDatabase, PositionalIndexer
from iode.iode_cython import Lists as CythonLists
[docs]
class Lists(IodeDatabase):
r"""
IODE Lists database.
Attributes
----------
filename: str
description: str
Parameters
----------
filepath: str, optional
file containing the IODE lists to load.
Returns
-------
Lists
Examples
--------
>>> from iode import lists, SAMPLE_DATA_DIR
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> len(lists)
17
>>> lists # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Workspace: Lists
nb lists: 17
filename: ...\tests\data\fun.lst
description: Modèle fun
<BLANKLINE>
name lists
COPY $COPY0; $COPY1
COPY0 ACAF; ACAG; AOUC; AQC; BENEF; BQY; BVY; CGU; COEFON; COTRES; DPU; DPUF; DPUG; DPUGO; DPUH; DPUU;
DTF; DTH; DTH1; DTH1C; EXC; EXCC; FLF; FLG; GAP; GOSF; GOSG; GOSH; GOSH_; IDF; IDG; IDH; IFU; IHU;
IT; ITCEE; ITCR; ITD; ITEP; ITF; ITF5; ITFC; ITFGI; ITFGO; ITFQ; ITGR; ITI5R; ITIFR; ITIGR; ITM;
ITMQ; ITMQR; ITNQ; ITON; ITONQ; ITPL; ITPR; ITPS; ITT; IUG; KN5; KNF; KNFF; KNFFY; KNFY; KNI; KNIY;
NFY; NFYH; OCUF; OCUG; OCUH; PAF_; PAG; PAH; PBBP; PBNP; PC; PC_; PDPUG; PFI; PFI_; PFND; PG; PI5;
PIF; PIG; PM; PMAB; PME; PMS; PMT; PQOG; PW3; PWBG; PX; PXAB; PXB; PXE; PXS; PXT; QAF; QAFF; QAFF_;
QAF_; QAG; QAH; QAI_; QAT; QAT_; QBBP; QBBPPOT_; QBBP_B; QBBP_P; QBNP; QC; QC_; QFND; QG; QGO; QI;
QI5; QIF; QIG; QM; QMAB; QME; QMS; QMT; QQMAB_; QS; QX; QXAB; QXB; QXE; QXS; QXT; Q_F; Q_I; RIDG;
RIDGG; RIPBE; RLBE; RLBERHP; RSBE; SBF; SBF3L; SBG; SBH; SF; SG; SH; SSF; SSF3; SSF3L; SSF3P;
SSFDOM; SSFF; SSFFIC; SSFG; SSH; SSH3GP; SSH3O; SSH3P; SSH3W; SSH3WA; SSH3WW; SSH3ZA; SSH3ZW; SSHFF;
SUBCEE; TWG; TWGP; ULCP; UY; VAF; VAFF; VAFF_; VAG; VAH; VAI; VAI_; VAMARE; VAT; VAT_; VBBP; VBBP_B;
VBBP_P; VBNP; VBNP_B; VBNP_I; VBNP_P; VC; VC_; VI; VI5; VIF; VM
COPY1 VMAB; VME; VMK; VMN; VMS; VMT; VS; VS_; VX; VXAB; VXB; VXE; VXK; VXN; VXS; VXT; WBF; WBF_; WBG;
WBGO; WBGP; WBU; WBU_; WCF; WCF_; WDOM; WG; WIND; WIND_; WLCP; WNF; WNF_; YDH; YDH_; YDTG; YIDG; YK;
YN; YSEFP; YSEFT1; YSEFT2; YSFIC; YSSF; YSSG; ZF; ZJ; ZKF; ZZF_; DPUHO; EFMY; EFXY; EX; EXCCR; HF;
ITCEEM; ITDR; ITEPR; ITF5R; ITFGR; ITFIGR; ITMR; ITNQHP; KLFHP; KNFFO; NATY; NDOMY; NGY; NIY; PCO;
PI5O; PIFO; PIGO; PMABO; PMDVD; PMINTP; PMSO; PMTO; POIL; PRDAL; PROF; PROIHP_; PWMAB; PWMS; PWXAB;
PWXS; PXABO; PXSO; PXTO; QAGHP; QAHHP; QAI; QCO; QI5O; QMABO; QMEO; QMSO; QMTO; QOUG; QSO; QS_;
QWXSS; QWXS; QXABO; QXEO; QXSO; QXTO; RIPBEO; SSF3PR; SSH3RO; SSH3RW; SSHRZW; SSRDOM; SUB; SUBR;
TFPFHP_; TIME; TVACEE; TWGR; ULAG; ULCPO; ULIL; ULIO; VAF_; VAIO; VATPC; VKF; VKFF; VSO; VS_O; VXKO;
VXNO; ZKFO; BRUGP; VK5; SSFFX
ENDO $endo0; $endo1
ENDO0 ACAF; ACAG; AOUC; AQC; BENEF; BQY; BVY; CGU; COEFON; COTRES; DEBT; DPU; DPUF; DPUG; DPUGO; DPUH;
DPUU; DTF; DTH; DTH1; DTH1C; EXC; EXCC; FLF; FLG; FLGR; GAP; GOSF; GOSG; GOSH; GOSH_; IDF; IDG; IDH;
IFU; IHU; IT; ITCEE; ITCR; ITD; ITEP; ITF; ITF5; ITFC; ITFGI; ITFGO; ITFQ; ITGR; ITI5R; ITIFR;
ITIGR; ITM; ITMQ; ITMQR; ITNQ; ITON; ITONQ; ITPL; ITPR; ITPS; ITT; IUG; KL; KLHP; KN5; KNF; KNFF;
KNFFY; KNFY; KNI; KNIY; NFY; NFYH; OCUF; OCUG; OCUH; PAF_; PAG; PAH; PBBP; PBNP; PC; PC_; PDPUG;
PFI; PFI_; PFND; PG; PI5; PIF; PIG; PM; PMAB; PME; PMS; PMT; PQOG; PROD; PW3; PWBG; PX; PXAB; PXB;
PXE; PXS; PXT; QAF; QAFF; QAFF_; QAF_; QAG; QAH; QAI_; QAT; QAT_; QBBP; QBBPPOT_; QBBP_B; QBBP_P;
QBNP; QC; QC_; QFND; QG; QGO; QI; QI5; QIF; QIG; QL; QM; QMAB; QME; QMS; QMT; QQMAB_; QS; QX; QXAB;
QXB; QXE; QXS; QXT; Q_F; Q_I; RDEBT; RENT; RIDG; RIDGG; RIPBE; RLBE; RLBER; RLBERHP; RSBE; SBF;
SBF3L; SBG; SBGX; SBH; SF; SG; SH; SSF; SSF3; SSF3L; SSF3P; SSFDOM; SSFF; SSFFIC; SSFG; SSH; SSH3GP;
SSH3O; SSH3P; SSH3W; SSH3WA; SSH3WW; SSH3ZA; SSH3ZW; SSHFF; SUBCEE; TWG; TWGP; ULCP; UY; VAF; VAFF;
VAFF_; VAG; VAH; VAI; VAI_; VAMARE; VAT; VAT_; VBBP; VBBP_B; VBBP_P; VBNP; VBNP_B
... ...
TOTAL1 VBNP_I; VBNP_P; VC; VC_; VI; VI5; VIF; VM; VMAB; VME; VMK; VMN; VMS; VMT; VS; VS_; VX; VXAB; VXB;
VXE; VXK; VXN; VXS; VXT; WBF; WBF_; WBG; WBGO; WBGP; WBU; WBU_; WCF; WCF_; WCRH; WDOM; WG; WIND;
WIND_; WLCP; WNF; WNF_; YDH; YDH_; YDTG; YIDG; YK; YN; YSEFP; YSEFT1; YSEFT2; YSFIC; YSSF; YSSG; ZF;
ZJ; ZKF; ZZF_; DPUHO; DTFX; DTHX; EFMY; EFXY; EX; EXCCR; HF; ITCEEM; ITCRX; ITDR; ITEPR; ITF5R;
ITFGR; ITFIGR; ITMR; ITNQHP; IUGR; KLFHP; KNFFO; NATY; NDOMY; NGY; NIY; OCP; PCO; PI5O; PIFO; PIGO;
PMABO; PMDVD; PMINTP; PMSO; PMTO; POIL; PRDAL; PROF; PROIHP; PWMAB; PWMS; PWXAB; PWXS; PXABO; PXSO;
PXTO; QAGHP; QAHHP; QAI; QCO; QI5O; QMABO; QMEO; QMSO; QMTO; QOUG; QSO; QS_; QWXAB; QWXS; QXABO;
QXEO; QXSO; QXTO; Q_; RIPBEO; SBGXR; SSF3PR; SSH3RO; SSH3RW; SSHKB; SSHRZW; SSRDOM; SUB; SUBR_;
TFPFHP; TFPFHP_; TIME; TVACEE; TWGR; ULAG; ULCPO; ULIL; ULIO; VAF_; VAIO; VATPC; VKF; VKF5; VKFF;
VSO; VS_O; VXKO; VXNO; WBFRXX; WBGR; YSEFPR; YSFICR; ZKFO
XENVI XEX; XPWMAB; XPWMS; XPWXAB; XPWXS; XQWXAB; XQWXS; XPOIL; XNATY; XTFP
XSCENARIO XQWXSS; XPWXAB; XPWMAB; XRLBER; XRSBER; XNATY; XTFP; SBGXR; DTFX; DTHX
_SCAL acaf1; acaf2; acaf3; dlnpaf; dpuh_1; dpuh_2; dtf0; dtf1; ifu_1; ifu_2; kl1; kl10; kl11; kl12; kl13;
kl14; kl2; knf2; knf3; knff0; knff1; nfyh1; nfyh10; nfyh11; nfyh2; nfyh3; nfyh4; pc0; pc1; pc2; pc4;
pc5; pc6; pif_1; pif_2; pif_3; pmab0; pmab1; pmab4; pmab6; pms0; pms1; pms3; pxab1; pxs1; pxs2;
pxs3; pxs4; qc0_; qc1_; qc2_; qc3_; qc4_; qc5_; qi5_1; qi5_2; qi5_3; qi5_4; qi5_5; qmab_1; qmab_10;
qmab_2; qmab_4; qmab_5; qme_1; qme_2; qme_3; qms1; qms2; qms3; qms4; qmt_1; qmt_2; qmt_3; qxab0;
qxab1; qxab2; qxab3; qxs0; qxs1; qxs2; qxs3; qxs4; qxs5; qxt0; qxt1; qxt2; qxt3; sbf3l_0; sbf3l_1;
vah_1; vah_2; vah_3; vai1_; vai2_; vai3_; vai4_; vmn_1; vmn_2; vmn_3; vs0_; vs1_; vs2_; vs3_; vxn_1;
vxn_2; vxn_3; zkf1; zkf2; zkf3
_SEARCH ZKF
<BLANKLINE>
"""
[docs]
def __init__(self, filepath: str=None):
raise TypeError("This class cannot be instantiated directly.")
@classmethod
def get_instance(cls) -> Self:
instance = cls.__new__(cls)
instance._cython_instance = CythonLists()
return instance
def _load(self, filepath: str):
self._cython_instance._load(filepath)
def _subset(self, pattern: str, copy: bool) -> Self:
instance = Lists.get_instance()
instance._cython_instance = self._cython_instance.initialize_subset(instance._cython_instance, pattern, copy)
return instance
@property
def i(self) -> PositionalIndexer:
r"""
Allow to select the ith list in the database.
Examples
--------
>>> from iode import lists, SAMPLE_DATA_DIR
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> # get the first list
>>> lists.i[0]
['$COPY0', '$COPY1']
>>> # get the last list
>>> lists.i[-1]
['ZKF']
>>> # update first list
>>> lists.i[0] = ['ACAF', 'ACAG', 'AOUC']
>>> lists.i[0]
['ACAF', 'ACAG', 'AOUC']
>>> # update last list
>>> lists.i[-1] = ['ACAF', 'ACAG', 'AOUC']
>>> lists.i[-1]
['ACAF', 'ACAG', 'AOUC']
"""
return PositionalIndexer(self)
def _get_object(self, key: Union[str, int]) -> List[str]:
name = self._single_object_key_to_name(key)
if not name in self:
raise KeyError(f"Name '{name}' not found in the {type(self).__name__} workspace")
iode_list = self._cython_instance._get_object(name)
return split_list(iode_list)
def _set_object(self, key: Union[str, int], value: Union[str, List[str]]):
name = self._single_object_key_to_name(key)
if isinstance(value, str):
value = value.strip()
value = split_list(value)
else:
value = [item.strip() for item in value]
value = ';'.join(value)
self._cython_instance._set_object(name, value)
[docs]
def __getitem__(self, key: Union[str, List[str]]) -> Union[str, Self]:
r"""
Return the (subset of) IODE list(s) referenced by `key`.
The `key` can represent a single object name (e.g. "ACAF") or a list of object names ("ACAF;ACAG;AOUC")
or a pattern (e.g. "A*") or a list of sub-patterns (e.g. "A*;*_").
If the `key` represents a list of object names or of sub-patterns, each name or sub-pattern is separated
by a `separator` character which is either a whitespace ` `, or a comma `,`, or a semi-colon `;`, or a
tabulation `\t`, or a newline `\n`.
A (sub-)`pattern` is a list of characters representing a group of object names.
It includes some special characters which have a special meaning:
- `*` : any character sequence, even empty
- `?` : any character (one and only one)
- `@` : any alphanumerical char [A-Za-z0-9]
- `&` : any non alphanumerical char
- `|` : any alphanumeric character or none at the beginning and end of a string
- `!` : any non-alphanumeric character or none at the beginning and end of a string
- `\` : escape the next character
Note that the `key` can contain references to IODE lists which are prefixed with the symbol `$`.
Parameters
----------
key: str or list(str)
(the list of) name(s) of the IODE list(s) to get.
The list of lists to get can be specified by a pattern or by a list of sub-patterns (e.g. "A*;*_").
Returns
-------
Single IODE list or a subset of the database.
Examples
--------
>>> from iode import SAMPLE_DATA_DIR
>>> from iode import lists
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> # a) get one list
>>> lists["ENVI"]
['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']
>>> # b) get a subset of the Lists database using a pattern
>>> lists_subset = lists["E*"]
>>> lists_subset.names
['ENDO', 'ENDO0', 'ENDO1', 'ENVI']
>>> # c) get a subset of the Lists database using a list of names
>>> lists_subset = lists[["COPY", "ENDO", "ENVI", "TOTAL"]]
>>> lists_subset.names
['COPY', 'ENDO', 'ENVI', 'TOTAL']
"""
return super().__getitem__(key)
[docs]
def __setitem__(self, key: Union[str, List[str]], value: Union[str, List[str]]):
r"""
Update/add a (subset of) IODE list(s) referenced by `key` from/to the Lists database.
The `key` can represent a single object name (e.g. "ACAF") or a list of object names ("ACAF;ACAG;AOUC")
or a pattern (e.g. "A*") or a list of sub-patterns (e.g. "A*;*_").
If the `key` represents a list of object names or of sub-patterns, each name or sub-pattern is separated
by a `separator` character which is either a whitespace ` `, or a comma `,`, or a semi-colon `;`, or a
tabulation `\t`, or a newline `\n`.
A (sub-)`pattern` is a list of characters representing a group of object names.
It includes some special characters which have a special meaning:
- `*` : any character sequence, even empty
- `?` : any character (one and only one)
- `@` : any alphanumerical char [A-Za-z0-9]
- `&` : any non alphanumerical char
- `|` : any alphanumeric character or none at the beginning and end of a string
- `!` : any non-alphanumeric character or none at the beginning and end of a string
- `\` : escape the next character
Note that the `key` can contain references to IODE lists which are prefixed with the symbol `$`.
Parameters
----------
key: str or list(str)
(the list of) name(s) of the IODE list(s) to update/add.
The list of lists to update/add can be specified by a pattern or by a list of sub-patterns
(e.g. "A*;*_").
value: str or dict(str, str) or pd.Series or Lists
(new) IODE list(s) value(s).
Examples
--------
>>> import pandas as pd
>>> from iode import SAMPLE_DATA_DIR
>>> from iode import lists, variables
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> variables.load(f"{SAMPLE_DATA_DIR}/fun.var") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.var
394 objects loaded
>>> # a) add one list
>>> # --- by passing a string
>>> lists["A_VAR"] = "ACAF;ACAG;AOUC;AOUC_;AQC"
>>> lists["A_VAR"]
['ACAF', 'ACAG', 'AOUC', 'AOUC_', 'AQC']
>>> # --- by passing a Python list
>>> b_vars = variables.get_names("B*")
>>> b_vars
['BENEF', 'BQY', 'BRUGP', 'BVY']
>>> lists["B_VAR"] = b_vars
>>> lists["B_VAR"]
['BENEF', 'BQY', 'BRUGP', 'BVY']
>>> # b) update one list
>>> # --- by passing a string
>>> lists["A_VAR"] = "ACAF;ACAG;AOUC;AQC"
>>> lists["A_VAR"]
['ACAF', 'ACAG', 'AOUC', 'AQC']
>>> # --- by passing a Python list
>>> b_y_vars = variables.get_names("B*Y")
>>> b_y_vars
['BQY', 'BVY']
>>> lists["B_VAR"] = b_y_vars
>>> lists["B_VAR"]
['BQY', 'BVY']
>>> # c) add/update multiple lists at once
>>> # 1) using a dict of values
>>> values = {"ENVI": "PWMAB; PWMS; PWXAB; PWXS; QWXAB; QWXS; POIL; NATY",
... "IDT": "FLGR; KL; PROD; QL; RDEBT; RENT; RLBER; SBGX; WCRH; IUGR; SBGXR; WBGR; YSFICR",
... "MAINEQ": "NFYH; KNFF; PC; PXAB; PMAB; QXAB; QMAB"}
>>> lists["ENVI, IDT, MAINEQ"] = values
>>> lists["ENVI, IDT, MAINEQ"] # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Workspace: Lists
nb lists: 3
filename: ...fun.lst
description: Modèle fun
<BLANKLINE>
name lists
ENVI PWMAB; PWMS; PWXAB; PWXS; QWXAB; QWXS; POIL; NATY
IDT FLGR; KL; PROD; QL; RDEBT; RENT; RLBER; SBGX; WCRH; IUGR; SBGXR; WBGR; YSFICR
MAINEQ NFYH; KNFF; PC; PXAB; PMAB; QXAB; QMAB
>>> # 2) using a pandas series
>>> data = ["PWMS; PWXAB; PWXS; QWXAB; QWXS; POIL",
... "KL; PROD; QL; RDEBT; RENT; RLBER; SBGX; WCRH; IUGR; SBGXR; WBGR",
... "KNFF; PC; PXAB; PMAB; QXAB"]
>>> series = pd.Series(data, index=["ENVI", "IDT", "MAINEQ"])
>>> lists["ENVI, IDT, MAINEQ"] = series
>>> lists["ENVI, IDT, MAINEQ"] # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Workspace: Lists
nb lists: 3
filename: ...fun.lst
description: Modèle fun
<BLANKLINE>
name lists
ENVI PWMS; PWXAB; PWXS; QWXAB; QWXS; POIL
IDT KL; PROD; QL; RDEBT; RENT; RLBER; SBGX; WCRH; IUGR; SBGXR; WBGR
MAINEQ KNFF; PC; PXAB; PMAB; QXAB
>>> # 3) using another Lists database (subset)
>>> lists_subset = lists["ENVI, IDT, MAINEQ"].copy()
>>> lists_subset["ENVI"] = "PWXAB; PWXS; QWXAB; QWXS"
>>> lists_subset["IDT"] = "PROD; QL; RDEBT; RENT; RLBER; SBGX; WCRH; IUGR; SBGXR"
>>> lists_subset["MAINEQ"] = "PC; PXAB; PMAB"
>>> lists["ENVI, IDT, MAINEQ"] = lists_subset
>>> lists["ENVI, IDT, MAINEQ"] # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Workspace: Lists
nb lists: 3
filename: ...fun.lst
description: Modèle fun
<BLANKLINE>
name lists
ENVI PWXAB; PWXS; QWXAB; QWXS
IDT PROD; QL; RDEBT; RENT; RLBER; SBGX; WCRH; IUGR; SBGXR
MAINEQ PC; PXAB; PMAB
>>> # d) working on a subset
>>> # 1) get subset
>>> lists_subset = lists["E*"]
>>> lists_subset.names
['ENDO', 'ENDO0', 'ENDO1', 'ENVI']
>>> # 2) add a list to the subset
>>> lists_subset["E_VAR"] = variables.get_names("E*")
>>> lists_subset["E_VAR"]
['EFMY', 'EFXY', 'EX', 'EXC', 'EXCC', 'EXCCR']
>>> # --> new list also appears in the global workspace
>>> "E_VAR" in lists
True
>>> lists["E_VAR"]
['EFMY', 'EFXY', 'EX', 'EXC', 'EXCC', 'EXCCR']
>>> # 3) update a list in the subset
>>> lists_subset["E_VAR"] = "EX;EXC;EXCC;EXCCR"
>>> lists_subset["E_VAR"]
['EX', 'EXC', 'EXCC', 'EXCCR']
>>> # --> list is also updated in the global workspace
>>> lists["E_VAR"]
['EX', 'EXC', 'EXCC', 'EXCCR']
"""
super().__setitem__(key, value)
[docs]
def __delitem__(self, key):
r"""
Remove the (subset of) IODE list(s) referenced by `key` from the Lists database.
Parameters
----------
key: str or list(str)
(list of) name(s) of the IODE list(s) to be removed.
The list of names can be given as a string pattern (e.g. "A*;*_").
Examples
--------
>>> from iode import SAMPLE_DATA_DIR
>>> from iode import lists
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> # a) delete one IODE list
>>> lists.get_names("C*")
['COPY', 'COPY0', 'COPY1']
>>> del lists["COPY"]
>>> lists.get_names("C*")
['COPY0', 'COPY1']
>>> # b) delete several lists at once using a pattern
>>> del lists["C*"]
>>> lists.get_names("C*")
[]
>>> # c) delete several lists at once using a list of names
>>> lists.get_names("T*")
['TOTAL', 'TOTAL0', 'TOTAL1']
>>> del lists[["TOTAL", "TOTAL0"]]
>>> lists.get_names("T*")
['TOTAL1']
>>> # delete one IODE list from a subset of the global database
>>> lists_subset = lists["E*"]
>>> lists_subset.names
['ENDO', 'ENDO0', 'ENDO1', 'ENVI']
>>> del lists_subset["ENVI"]
>>> lists_subset.names
['ENDO', 'ENDO0', 'ENDO1']
>>> # NOTE: the IODE list has also been deleted from the global database
>>> "ENVI" in lists
False
>>> lists.get_names("E*")
['ENDO', 'ENDO0', 'ENDO1']
"""
super().__delitem__(key)
[docs]
def copy_from(self, input_files: Union[str, List[str]], names: Union[str, List[str]]='*'):
r"""
Copy (a subset of) lists from the input file(s) 'input_files' into the current database.
Parameters
----------
input_file: str or list(str)
file(s) from which the copied lists are read.
names: str or list(str)
list of lists to copy from the input file(s).
Defaults to load all lists from the input file(s).
"""
input_files, names = self._copy_from(input_files, names)
self._cython_instance.copy_from(input_files, names)
[docs]
def from_series(self, s: pd.Series):
r"""
Copy the pandas Series `s` into the IODE Lists database.
The IODE list names to copy are deduced from the index of the Series.
Parameters
----------
s: Series
pandas Series containing the IODE lists to copy into the IODE Lists database.
Notes
-----
The index of the passed Series is sorted in alphabetical order before
copying to IODE Lists database.
See Also
--------
Lists.to_series
Examples
--------
>>> from iode import lists
>>> import pandas as pd
>>> lists.clear()
>>> len(lists)
0
>>> # create the pandas Series
>>> names = ["SEMICOLON_LST", "COMMA_LST", "WHITESPACE_LST", "TAB_LST", "MIX_LST"]
>>> semicolon_lst = "A;B;C;D;E;F"
>>> comma_lst = "A,B,C,D,E,F"
>>> whitespace_lst = "A B C D E F"
>>> tab_lst = "A B C D E F"
>>> mix_lst = " A;B,C D E;F "
>>> data = [semicolon_lst, comma_lst, whitespace_lst, tab_lst, mix_lst]
>>> s = pd.Series(data=data, index=names, dtype=str, name="Lists")
>>> # display the pandas series
>>> s # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
SEMICOLON_LST A;B;C;D;E;F
COMMA_LST A,B,C,D,E,F
WHITESPACE_LST A B C D E F
TAB_LST A B C D E F
MIX_LST A;B,C D E;F
Name: Lists, dtype: object
>>> # load into the IODE Lists database
>>> lists.from_series(s)
>>> len(lists)
5
>>> lists.names
['COMMA_LST', 'MIX_LST', 'SEMICOLON_LST', 'TAB_LST', 'WHITESPACE_LST']
>>> lists["SEMICOLON_LST"]
['A', 'B', 'C', 'D', 'E', 'F']
>>> # note: when added or updated, an IODE list is normalized.
>>> # - leading and trailing whitespaces are stripped
>>> # - items of the IODE list are separated by a semicolon
>>> lists["COMMA_LST"]
['A', 'B', 'C', 'D', 'E', 'F']
>>> lists["MIX_LST"]
['A', 'B', 'C', 'D', 'E', 'F']
"""
if not (self.is_global_workspace or self.is_detached):
# check that all names in the pandas object are present in the current subset
self._check_same_names(self.names, s.index.tolist())
for index, value in s.items():
self._set_object(index, value)
[docs]
def to_series(self) -> pd.Series:
r"""
Create a pandas Series from the current Lists database.
The index of the returned Series is build from the Lists names.
See Also
--------
Lists.from_series
Examples
--------
>>> from iode import SAMPLE_DATA_DIR, lists
>>> import pandas as pd
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> len(lists)
17
>>> # Export the IODE Lists database as a pandas Series
>>> s = lists.to_series()
>>> len(s)
17
>>> s.index.to_list() # doctest: +ELLIPSIS
['COPY', 'COPY0', 'COPY1', ..., 'XSCENARIO', '_SCAL', '_SEARCH']
>>> lists["ENVI"] # doctest: +NORMALIZE_WHITESPACE
['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']
>>> s["ENVI"] # doctest: +NORMALIZE_WHITESPACE
"['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']"
>>> lists["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']
>>> s["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
"['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']"
>>> # Export a subset of the IODE Lists database as a pandas Series
>>> s = lists["E*;MA*"].to_series()
>>> len(s)
5
>>> s.index.to_list()
['ENDO', 'ENDO0', 'ENDO1', 'ENVI', 'MAINEQ']
>>> lists["ENVI"] # doctest: +NORMALIZE_WHITESPACE
['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']
>>> s["ENVI"] # doctest: +NORMALIZE_WHITESPACE
"['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']"
>>> lists["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']
>>> s["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
"['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']"
"""
names = self.names
data = [self._get_object(name) for name in names]
return pd.Series(data=data, index=names, dtype=str, name=self.__class__.__name__)
@property
def series(self) -> pd.Series:
r"""
Create a pandas Series from the current Lists database.
The index of the returned Series is build from the Lists names.
See Also
--------
Lists.to_series
Examples
--------
>>> from iode import SAMPLE_DATA_DIR, lists
>>> import pandas as pd
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> len(lists)
17
>>> # Export the IODE Lists database as a pandas Series
>>> s = lists.series
>>> len(s)
17
>>> s.index.to_list() # doctest: +ELLIPSIS
['COPY', 'COPY0', 'COPY1', ..., 'XSCENARIO', '_SCAL', '_SEARCH']
>>> lists["ENVI"] # doctest: +NORMALIZE_WHITESPACE
['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']
>>> s["ENVI"] # doctest: +NORMALIZE_WHITESPACE
"['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']"
>>> lists["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']
>>> s["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
"['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']"
>>> # Export a subset of the IODE Lists database as a pandas Series
>>> s = lists["E*;MA*"].series
>>> len(s)
5
>>> s.index.to_list()
['ENDO', 'ENDO0', 'ENDO1', 'ENVI', 'MAINEQ']
>>> lists["ENVI"] # doctest: +NORMALIZE_WHITESPACE
['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']
>>> s["ENVI"] # doctest: +NORMALIZE_WHITESPACE
"['EX', 'PWMAB', 'PWMS', 'PWXAB', 'PWXS', 'QWXAB', 'QWXS', 'POIL', 'NATY', 'TFPFHP_']"
>>> lists["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']
>>> s["MAINEQ"] # doctest: +NORMALIZE_WHITESPACE
"['W', 'NFYH', 'KNFF', 'PC', 'PXAB', 'PMAB', 'QXAB', 'QMAB', 'QC_']"
"""
return self.to_series()
def _str_table(self, names: List[str]) -> str:
columns = {"name": names, "lists": ['; '.join(self._get_object(name)) for name in names]}
return table2str(columns, max_lines=10, justify_funcs={"name": JUSTIFY.LEFT, "lists": JUSTIFY.LEFT})
[docs]
def print_to_file(self, filepath: Union[str, Path], names: Union[str, List[str]]=None, format: str=None):
r"""
Print the list IODE lists defined by `names` to the file `filepath` using the format `format`.
Argument `format` must be in the list:
- 'H' (HTML file)
- 'M' (MIF file)
- 'R' (RTF file)
- 'C' (CSV file)
If argument `format` is null (default), the *A2M* format will be used to print the output.
If the filename does not contain an extension, it is automatically added based on
the value of `format`.
If `names` is a string, it is considered as a *pattern* and the function will print
all IODE lists matching the pattern. The following characters in *pattern* have a
special meaning:
- `*` : any character sequence, even empty
- `?` : any character (one and only one)
- `@` : any alphanumerical char [A-Za-z0-9]
- `&` : any non alphanumerical char
- `|` : any alphanumeric character or none at the beginning and end of a string
- `!` : any non-alphanumeric character or none at the beginning and end of a string
- `\` : escape the next character
If `names` is None, print all IODE lists of the (subset of the) current database.
Parameters
----------
filepath: str or Path
path to the file to print.
If the filename does not contain an extension, it is automatically
added based on the value of the format argument.
names: str or list of str, optional
pattern or list of names of the IODE lists to print.
If None, print all IODE lists of the (subset of the) current database.
Defaults to None.
format: str, optional
format of the output file. Possible values are: 'H' (HTML file),
'M' (MIF file), 'R' (RTF file) or 'C' (CSV file).
Defaults to None meaning that the IODE lists will be dumped in the *A2M* format.
Examples
--------
>>> from iode import lists, SAMPLE_DATA_DIR
>>> output_dir = getfixture('tmp_path')
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> lists.print_to_file(output_dir / "lists.csv", ["ENVI", "IDT", "MAINEQ"]) # doctest: +ELLIPSIS
Printing IODE objects definition to file '...lists.csv'...
Printing ENVI ...
Printing IDT ...
Printing MAINEQ ...
Print done
>>> with open(output_dir / "lists.csv") as f: # doctest: +NORMALIZE_WHITESPACE
... print(f.read())
...
" - ENVI : EX;PWMAB;PWMS;PWXAB;PWXS;QWXAB;QWXS;POIL;NATY;TFPFHP_"
" - IDT : OCP;FLGR;KL;PROD;QL;RDEBT;RENT;RLBER;SBGX;WCRH;IUGR;SBGXR; WBGR;YSFICR;YSEFPR"
" - MAINEQ : W;NFYH;KNFF;PC;PXAB;PMAB;QXAB;QMAB;QC_"
"""
super().print_to_file(filepath, names, format)
def __hash__(self) -> int:
r"""
Return a hash value for the current Lists database.
Examples
--------
>>> from iode import SAMPLE_DATA_DIR, lists
>>> lists.load(f"{SAMPLE_DATA_DIR}/fun.lst") # doctest: +ELLIPSIS, +NORMALIZE_WHITESPACE
Loading .../fun.lst
17 objects loaded
>>> len(lists)
17
>>> original_hash = hash(lists)
>>> # rename 1 IODE list
>>> lists.rename("ENVI", "ENVI_")
>>> original_hash == hash(lists)
False
>>> lists.rename("ENVI_", "ENVI") # revert the change
>>> original_hash == hash(lists)
True
>>> # modify one IODE list
>>> original_list = lists["ENVI"]
>>> lists["ENVI"] = original_list[1:-1]
>>> original_hash == hash(lists)
False
>>> lists["ENVI"] = original_list # revert the change
>>> original_hash == hash(lists)
True
>>> # delete a IODE list
>>> original_list = lists["ENVI"]
>>> del lists["ENVI"]
>>> original_hash == hash(lists)
False
>>> lists["ENVI"] = original_list
>>> original_hash == hash(lists)
True
>>> # add a IODE list
>>> lists["NEW"] = "$COPY0"
>>> original_hash == hash(lists)
False
>>> del lists["NEW"]
>>> original_hash == hash(lists)
True
"""
return super().__hash__()
lists: Lists = Lists.get_instance()