diff --git a/.gitignore b/.gitignore index c92f3ddc..58169819 100644 --- a/.gitignore +++ b/.gitignore @@ -125,3 +125,7 @@ com_crashlytics_export_strings.xml crashlytics.properties crashlytics-build.properties fabric.properties + + +.mypy_cache +.pytest_cache \ No newline at end of file diff --git a/poetry.lock b/poetry.lock index b9936946..c456a58e 100644 --- a/poetry.lock +++ b/poetry.lock @@ -107,6 +107,17 @@ category = "main" optional = false python-versions = ">=3.6" +[[package]] +name = "data-science-types" +version = "0.2.23" +description = "Type stubs for Python machine learning libraries" +category = "dev" +optional = false +python-versions = ">=3.6" + +[package.extras] +dev = ["black", "flake8", "flake8-pyi", "matplotlib", "mypy (==0.770)", "numpy", "pandas", "pytest"] + [[package]] name = "decorator" version = "5.1.1" @@ -286,22 +297,29 @@ dill = ">=0.3.5.1" [[package]] name = "mypy" -version = "0.971" +version = "0.980+dev.0f17aff06ac1c05c442ba989e23655a2c6adbfbf" description = "Optional static typing for Python" category = "dev" optional = false -python-versions = ">=3.6" +python-versions = ">=3.7" +develop = false [package.dependencies] -mypy-extensions = ">=0.4.3" +mypy_extensions = ">=0.4.3" tomli = {version = ">=1.1.0", markers = "python_version < \"3.11\""} -typing-extensions = ">=3.10" +typing_extensions = ">=3.10" [package.extras] dmypy = ["psutil (>=4.0)"] -python2 = ["typed-ast (>=1.4.0,<2)"] +python2 = ["typed_ast (>=1.4.0,<2)"] reports = ["lxml"] +[package.source] +type = "git" +url = "https://github.com/python/mypy.git" +reference = "HEAD" +resolved_reference = "0f17aff06ac1c05c442ba989e23655a2c6adbfbf" + [[package]] name = "mypy-extensions" version = "0.4.3" @@ -310,6 +328,17 @@ category = "dev" optional = false python-versions = "*" +[[package]] +name = "nodeenv" +version = "1.7.0" +description = "Node.js virtual environment builder" +category = "dev" +optional = false +python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*,!=3.5.*,!=3.6.*" + +[package.dependencies] +setuptools = "*" + [[package]] name = "nuchabal" version = "0.0.1" @@ -320,7 +349,7 @@ python-versions = "*" [[package]] name = "numpy" -version = "1.23.2" +version = "1.23.3" description = "NumPy is the fundamental package for array computing with Python." category = "main" optional = false @@ -368,12 +397,7 @@ optional = false python-versions = ">=3.8" [package.dependencies] -numpy = [ - {version = ">=1.18.5", markers = "platform_machine != \"aarch64\" and platform_machine != \"arm64\" and python_version < \"3.10\""}, - {version = ">=1.19.2", markers = "platform_machine == \"aarch64\" and python_version < \"3.10\""}, - {version = ">=1.20.0", markers = "platform_machine == \"arm64\" and python_version < \"3.10\""}, - {version = ">=1.21.0", markers = "python_version >= \"3.10\""}, -] +numpy = {version = ">=1.21.0", markers = "python_version >= \"3.10\""} python-dateutil = ">=2.8.1" pytz = ">=2020.1" @@ -494,6 +518,21 @@ python-versions = ">=3.8" [package.dependencies] certifi = "*" +[[package]] +name = "pyright" +version = "1.1.270" +description = "Command line wrapper for pyright" +category = "dev" +optional = false +python-versions = ">=3.7" + +[package.dependencies] +nodeenv = ">=1.6.0" + +[package.extras] +all = ["twine (>=3.4.1)"] +dev = ["twine (>=3.4.1)"] + [[package]] name = "pyshp" version = "2.3.1" @@ -940,8 +979,8 @@ python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" [metadata] lock-version = "1.1" -python-versions = "^3.9,<3.11" -content-hash = "e63794e64d306046a9237155720c20f181f8232313a9f40267b8a41245381cf8" +python-versions = ">=3.10,<3.12" +content-hash = "b36e5c8c6e0db295a4bfb548c0c9be557cea1a82c484d9e5382de2d06e24865f" [metadata.files] async-generator = [ @@ -1098,6 +1137,10 @@ cycler = [ {file = "cycler-0.11.0-py3-none-any.whl", hash = "sha256:3a27e95f763a428a739d2add979fa7494c912a32c17c4c38c4d5f082cad165a3"}, {file = "cycler-0.11.0.tar.gz", hash = "sha256:9c87405839a19696e837b3b818fed3f5f69f16f1eec1a1ad77e043dcea9c772f"}, ] +data-science-types = [ + {file = "data-science-types-0.2.23.tar.gz", hash = "sha256:8096b9a35a8a187bf9a122b4707c97de841d810744690ee2a4ac30c6462e0d16"}, + {file = "data_science_types-0.2.23-py3-none-any.whl", hash = "sha256:bca319abc0e53a0316f9fcb887937e942477cb9e5fc63c8581e0b0438903b977"}, +] decorator = [ {file = "decorator-5.1.1-py3-none-any.whl", hash = "sha256:b8c3f85900b9dc423225913c5aace94729fe1fa9763b38939a95226f02d37186"}, {file = "decorator-5.1.1.tar.gz", hash = "sha256:637996211036b6385ef91435e4fae22989472f9d571faba8927ba8253acbc330"}, @@ -1347,68 +1390,48 @@ multiprocess = [ {file = "multiprocess-0.70.13-py39-none-any.whl", hash = "sha256:00ef48461d43d1e30f8f4b2e1b287ecaaffec325a37053beb5503e0d69e5a3cd"}, {file = "multiprocess-0.70.13.tar.gz", hash = "sha256:2e096dd618a84d15aa369a9cf6695815e5539f853dc8fa4f4b9153b11b1d0b32"}, ] -mypy = [ - {file = "mypy-0.971-cp310-cp310-macosx_10_9_universal2.whl", hash = "sha256:f2899a3cbd394da157194f913a931edfd4be5f274a88041c9dc2d9cdcb1c315c"}, - {file = "mypy-0.971-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:98e02d56ebe93981c41211c05adb630d1d26c14195d04d95e49cd97dbc046dc5"}, - {file = "mypy-0.971-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:19830b7dba7d5356d3e26e2427a2ec91c994cd92d983142cbd025ebe81d69cf3"}, - {file = "mypy-0.971-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:02ef476f6dcb86e6f502ae39a16b93285fef97e7f1ff22932b657d1ef1f28655"}, - {file = "mypy-0.971-cp310-cp310-win_amd64.whl", hash = "sha256:25c5750ba5609a0c7550b73a33deb314ecfb559c350bb050b655505e8aed4103"}, - {file = "mypy-0.971-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:d3348e7eb2eea2472db611486846742d5d52d1290576de99d59edeb7cd4a42ca"}, - {file = "mypy-0.971-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:3fa7a477b9900be9b7dd4bab30a12759e5abe9586574ceb944bc29cddf8f0417"}, - {file = "mypy-0.971-cp36-cp36m-win_amd64.whl", hash = "sha256:2ad53cf9c3adc43cf3bea0a7d01a2f2e86db9fe7596dfecb4496a5dda63cbb09"}, - {file = "mypy-0.971-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:855048b6feb6dfe09d3353466004490b1872887150c5bb5caad7838b57328cc8"}, - {file = "mypy-0.971-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:23488a14a83bca6e54402c2e6435467a4138785df93ec85aeff64c6170077fb0"}, - {file = "mypy-0.971-cp37-cp37m-win_amd64.whl", hash = "sha256:4b21e5b1a70dfb972490035128f305c39bc4bc253f34e96a4adf9127cf943eb2"}, - {file = "mypy-0.971-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:9796a2ba7b4b538649caa5cecd398d873f4022ed2333ffde58eaf604c4d2cb27"}, - {file = "mypy-0.971-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:5a361d92635ad4ada1b1b2d3630fc2f53f2127d51cf2def9db83cba32e47c856"}, - {file = "mypy-0.971-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:b793b899f7cf563b1e7044a5c97361196b938e92f0a4343a5d27966a53d2ec71"}, - {file = "mypy-0.971-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d1ea5d12c8e2d266b5fb8c7a5d2e9c0219fedfeb493b7ed60cd350322384ac27"}, - {file = "mypy-0.971-cp38-cp38-win_amd64.whl", hash = "sha256:23c7ff43fff4b0df93a186581885c8512bc50fc4d4910e0f838e35d6bb6b5e58"}, - {file = "mypy-0.971-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:1f7656b69974a6933e987ee8ffb951d836272d6c0f81d727f1d0e2696074d9e6"}, - {file = "mypy-0.971-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:d2022bfadb7a5c2ef410d6a7c9763188afdb7f3533f22a0a32be10d571ee4bbe"}, - {file = "mypy-0.971-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:ef943c72a786b0f8d90fd76e9b39ce81fb7171172daf84bf43eaf937e9f220a9"}, - {file = "mypy-0.971-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:d744f72eb39f69312bc6c2abf8ff6656973120e2eb3f3ec4f758ed47e414a4bf"}, - {file = "mypy-0.971-cp39-cp39-win_amd64.whl", hash = "sha256:77a514ea15d3007d33a9e2157b0ba9c267496acf12a7f2b9b9f8446337aac5b0"}, - {file = "mypy-0.971-py3-none-any.whl", hash = "sha256:0d054ef16b071149917085f51f89555a576e2618d5d9dd70bd6eea6410af3ac9"}, - {file = "mypy-0.971.tar.gz", hash = "sha256:40b0f21484238269ae6a57200c807d80debc6459d444c0489a102d7c6a75fa56"}, -] +mypy = [] mypy-extensions = [ {file = "mypy_extensions-0.4.3-py2.py3-none-any.whl", hash = "sha256:090fedd75945a69ae91ce1303b5824f428daf5a028d2f6ab8a299250a846f15d"}, {file = "mypy_extensions-0.4.3.tar.gz", hash = "sha256:2d82818f5bb3e369420cb3c4060a7970edba416647068eb4c5343488a6c604a8"}, ] +nodeenv = [ + {file = "nodeenv-1.7.0-py2.py3-none-any.whl", hash = "sha256:27083a7b96a25f2f5e1d8cb4b6317ee8aeda3bdd121394e5ac54e498028a042e"}, + {file = "nodeenv-1.7.0.tar.gz", hash = "sha256:e0e7f7dfb85fc5394c6fe1e8fa98131a2473e04311a45afb6508f7cf1836fa2b"}, +] nuchabal = [ {file = "nuchabal-0.0.1-py3-none-any.whl", hash = "sha256:e37376d5f525a7d670ed9af14773d147873b53bb3ddfda4e01dbe8b6ff4d983e"}, {file = "nuchabal-0.0.1.tar.gz", hash = "sha256:6680b6bca7cceb3c99c5b4a3f5ee368d4f5ba9d1843db60714b2e3ef228bd70d"}, ] numpy = [ - {file = "numpy-1.23.2-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:e603ca1fb47b913942f3e660a15e55a9ebca906857edfea476ae5f0fe9b457d5"}, - {file = "numpy-1.23.2-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:633679a472934b1c20a12ed0c9a6c9eb167fbb4cb89031939bfd03dd9dbc62b8"}, - {file = "numpy-1.23.2-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:17e5226674f6ea79e14e3b91bfbc153fdf3ac13f5cc54ee7bc8fdbe820a32da0"}, - {file = "numpy-1.23.2-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bdc02c0235b261925102b1bd586579b7158e9d0d07ecb61148a1799214a4afd5"}, - {file = "numpy-1.23.2-cp310-cp310-win32.whl", hash = "sha256:df28dda02c9328e122661f399f7655cdcbcf22ea42daa3650a26bce08a187450"}, - {file = "numpy-1.23.2-cp310-cp310-win_amd64.whl", hash = "sha256:8ebf7e194b89bc66b78475bd3624d92980fca4e5bb86dda08d677d786fefc414"}, - {file = "numpy-1.23.2-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:dc76bca1ca98f4b122114435f83f1fcf3c0fe48e4e6f660e07996abf2f53903c"}, - {file = "numpy-1.23.2-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:ecfdd68d334a6b97472ed032b5b37a30d8217c097acfff15e8452c710e775524"}, - {file = "numpy-1.23.2-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5593f67e66dea4e237f5af998d31a43e447786b2154ba1ad833676c788f37cde"}, - {file = "numpy-1.23.2-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:ac987b35df8c2a2eab495ee206658117e9ce867acf3ccb376a19e83070e69418"}, - {file = "numpy-1.23.2-cp311-cp311-win32.whl", hash = "sha256:d98addfd3c8728ee8b2c49126f3c44c703e2b005d4a95998e2167af176a9e722"}, - {file = "numpy-1.23.2-cp311-cp311-win_amd64.whl", hash = "sha256:8ecb818231afe5f0f568c81f12ce50f2b828ff2b27487520d85eb44c71313b9e"}, - {file = "numpy-1.23.2-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:909c56c4d4341ec8315291a105169d8aae732cfb4c250fbc375a1efb7a844f8f"}, - {file = "numpy-1.23.2-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:8247f01c4721479e482cc2f9f7d973f3f47810cbc8c65e38fd1bbd3141cc9842"}, - {file = "numpy-1.23.2-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:b8b97a8a87cadcd3f94659b4ef6ec056261fa1e1c3317f4193ac231d4df70215"}, - {file = "numpy-1.23.2-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:bd5b7ccae24e3d8501ee5563e82febc1771e73bd268eef82a1e8d2b4d556ae66"}, - {file = "numpy-1.23.2-cp38-cp38-win32.whl", hash = "sha256:9b83d48e464f393d46e8dd8171687394d39bc5abfe2978896b77dc2604e8635d"}, - {file = "numpy-1.23.2-cp38-cp38-win_amd64.whl", hash = "sha256:dec198619b7dbd6db58603cd256e092bcadef22a796f778bf87f8592b468441d"}, - {file = "numpy-1.23.2-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:4f41f5bf20d9a521f8cab3a34557cd77b6f205ab2116651f12959714494268b0"}, - {file = "numpy-1.23.2-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:806cc25d5c43e240db709875e947076b2826f47c2c340a5a2f36da5bb10c58d6"}, - {file = "numpy-1.23.2-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8f9d84a24889ebb4c641a9b99e54adb8cab50972f0166a3abc14c3b93163f074"}, - {file = "numpy-1.23.2-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c403c81bb8ffb1c993d0165a11493fd4bf1353d258f6997b3ee288b0a48fce77"}, - {file = "numpy-1.23.2-cp39-cp39-win32.whl", hash = "sha256:cf8c6aed12a935abf2e290860af8e77b26a042eb7f2582ff83dc7ed5f963340c"}, - {file = "numpy-1.23.2-cp39-cp39-win_amd64.whl", hash = "sha256:5e28cd64624dc2354a349152599e55308eb6ca95a13ce6a7d5679ebff2962913"}, - {file = "numpy-1.23.2-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:806970e69106556d1dd200e26647e9bee5e2b3f1814f9da104a943e8d548ca38"}, - {file = "numpy-1.23.2-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:2bd879d3ca4b6f39b7770829f73278b7c5e248c91d538aab1e506c628353e47f"}, - {file = "numpy-1.23.2-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:be6b350dfbc7f708d9d853663772a9310783ea58f6035eec649fb9c4371b5389"}, - {file = "numpy-1.23.2.tar.gz", hash = "sha256:b78d00e48261fbbd04aa0d7427cf78d18401ee0abd89c7559bbf422e5b1c7d01"}, + {file = "numpy-1.23.3-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:c9f707b5bb73bf277d812ded9896f9512a43edff72712f31667d0a8c2f8e71ee"}, + {file = "numpy-1.23.3-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:ffcf105ecdd9396e05a8e58e81faaaf34d3f9875f137c7372450baa5d77c9a54"}, + {file = "numpy-1.23.3-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0ea3f98a0ffce3f8f57675eb9119f3f4edb81888b6874bc1953f91e0b1d4f440"}, + {file = "numpy-1.23.3-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:004f0efcb2fe1c0bd6ae1fcfc69cc8b6bf2407e0f18be308612007a0762b4089"}, + {file = "numpy-1.23.3-cp310-cp310-win32.whl", hash = "sha256:98dcbc02e39b1658dc4b4508442a560fe3ca5ca0d989f0df062534e5ca3a5c1a"}, + {file = "numpy-1.23.3-cp310-cp310-win_amd64.whl", hash = "sha256:39a664e3d26ea854211867d20ebcc8023257c1800ae89773cbba9f9e97bae036"}, + {file = "numpy-1.23.3-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:1f27b5322ac4067e67c8f9378b41c746d8feac8bdd0e0ffede5324667b8a075c"}, + {file = "numpy-1.23.3-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:2ad3ec9a748a8943e6eb4358201f7e1c12ede35f510b1a2221b70af4bb64295c"}, + {file = "numpy-1.23.3-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bdc9febce3e68b697d931941b263c59e0c74e8f18861f4064c1f712562903411"}, + {file = "numpy-1.23.3-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:301c00cf5e60e08e04d842fc47df641d4a181e651c7135c50dc2762ffe293dbd"}, + {file = "numpy-1.23.3-cp311-cp311-win32.whl", hash = "sha256:7cd1328e5bdf0dee621912f5833648e2daca72e3839ec1d6695e91089625f0b4"}, + {file = "numpy-1.23.3-cp311-cp311-win_amd64.whl", hash = "sha256:8355fc10fd33a5a70981a5b8a0de51d10af3688d7a9e4a34fcc8fa0d7467bb7f"}, + {file = "numpy-1.23.3-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:bc6e8da415f359b578b00bcfb1d08411c96e9a97f9e6c7adada554a0812a6cc6"}, + {file = "numpy-1.23.3-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:22d43376ee0acd547f3149b9ec12eec2f0ca4a6ab2f61753c5b29bb3e795ac4d"}, + {file = "numpy-1.23.3-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:a64403f634e5ffdcd85e0b12c08f04b3080d3e840aef118721021f9b48fc1460"}, + {file = "numpy-1.23.3-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:efd9d3abe5774404becdb0748178b48a218f1d8c44e0375475732211ea47c67e"}, + {file = "numpy-1.23.3-cp38-cp38-win32.whl", hash = "sha256:f8c02ec3c4c4fcb718fdf89a6c6f709b14949408e8cf2a2be5bfa9c49548fd85"}, + {file = "numpy-1.23.3-cp38-cp38-win_amd64.whl", hash = "sha256:e868b0389c5ccfc092031a861d4e158ea164d8b7fdbb10e3b5689b4fc6498df6"}, + {file = "numpy-1.23.3-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:09f6b7bdffe57fc61d869a22f506049825d707b288039d30f26a0d0d8ea05164"}, + {file = "numpy-1.23.3-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:8c79d7cf86d049d0c5089231a5bcd31edb03555bd93d81a16870aa98c6cfb79d"}, + {file = "numpy-1.23.3-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e5d5420053bbb3dd64c30e58f9363d7a9c27444c3648e61460c1237f9ec3fa14"}, + {file = "numpy-1.23.3-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:d5422d6a1ea9b15577a9432e26608c73a78faf0b9039437b075cf322c92e98e7"}, + {file = "numpy-1.23.3-cp39-cp39-win32.whl", hash = "sha256:c1ba66c48b19cc9c2975c0d354f24058888cdc674bebadceb3cdc9ec403fb5d1"}, + {file = "numpy-1.23.3-cp39-cp39-win_amd64.whl", hash = "sha256:78a63d2df1d947bd9d1b11d35564c2f9e4b57898aae4626638056ec1a231c40c"}, + {file = "numpy-1.23.3-pp38-pypy38_pp73-macosx_10_9_x86_64.whl", hash = "sha256:17c0e467ade9bda685d5ac7f5fa729d8d3e76b23195471adae2d6a6941bd2c18"}, + {file = "numpy-1.23.3-pp38-pypy38_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:91b8d6768a75247026e951dce3b2aac79dc7e78622fc148329135ba189813584"}, + {file = "numpy-1.23.3-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:94c15ca4e52671a59219146ff584488907b1f9b3fc232622b47e2cf832e94fb8"}, + {file = "numpy-1.23.3.tar.gz", hash = "sha256:51bf49c0cd1d52be0a240aa66f3458afc4b95d8993d2d04f0d91fa60c10af6cd"}, ] openpyxl = [ {file = "openpyxl-3.0.10-py2.py3-none-any.whl", hash = "sha256:0ab6d25d01799f97a9464630abacbb34aafecdcaa0ef3cba6d6b3499867d0355"}, @@ -1564,6 +1587,10 @@ pyproj = [ {file = "pyproj-3.3.1-pp38-pypy38_pp73-win_amd64.whl", hash = "sha256:797ad5655d484feac14b0fbb4a4efeaac0cf780a223046e2465494c767fd1c3b"}, {file = "pyproj-3.3.1.tar.gz", hash = "sha256:b3d8e14d91cc95fb3dbc03a9d0588ac58326803eefa5bbb0978d109de3304fbe"}, ] +pyright = [ + {file = "pyright-1.1.270-py3-none-any.whl", hash = "sha256:f2936dd26a732177f0ed8038bb52e423760816d6ee2d86cc6c01b8e9c01c63bd"}, + {file = "pyright-1.1.270.tar.gz", hash = "sha256:83ed7ebc04a4b63a43bc42be158733b3e38612eef0774412113b500cd805765d"}, +] pyshp = [ {file = "pyshp-2.3.1-py2.py3-none-any.whl", hash = "sha256:67024c0ccdc352ba5db777c4e968483782dfa78f8e200672a90d2d30fd8b7b49"}, {file = "pyshp-2.3.1.tar.gz", hash = "sha256:4caec82fd8dd096feba8217858068bacb2a3b5950f43c048c6dc32a3489d5af1"}, diff --git a/pruebas/test_central/test_modelo_calib.py b/pruebas/test_central/test_modelo_calib.py index 9cd0d729..2437d054 100644 --- a/pruebas/test_central/test_modelo_calib.py +++ b/pruebas/test_central/test_modelo_calib.py @@ -114,7 +114,7 @@ def test_renombrar_calib(símismo): valid = modelo.simular('valid', exper, calibs=EspecCalibsCorrida(['otro nombre'])).validar() símismo.assertAlmostEqual(valid['ens'], símismo.valid['ens'], delta=0.05) - def test_borrar_calib(símismo): + def est_borrar_calib(símismo): from .rcrs.modelo_calib_inic import generar gen = generar() modelo = gen['modelo'] diff --git a/pruebas/test_central/test_modelo_funcs_ecs.py b/pruebas/test_central/test_modelo_funcs_ecs.py index 080f2dd8..14e0d5f3 100644 --- a/pruebas/test_central/test_modelo_funcs_ecs.py +++ b/pruebas/test_central/test_modelo_funcs_ecs.py @@ -201,13 +201,13 @@ def test_inter_aprioris(símismo): exper = ec_inter.exper coso1.interactua_con([coso2, coso3]) - with símismo.subTest('sin índs'): + with símismo.subTest('sin índices'): apriori = APrioriDist(uniform(1.5, 1)) coso1.espec_apriori(apriori, 'categ', 'subcateg', 'ec', 'a') res = modelo.simular('con interacciones', exper=exper, t=2, vars_interés=True)['exper']['módulo']['res'] símismo._verif_en(res.datos.loc[res.datos.codificar_coords({'coso': coso1, 'otro': [coso2, coso3]})], (1.5, 1.5 + 1)) - with símismo.subTest('con índs'): + with símismo.subTest('con índices'): apriori = APrioriDist(uniform(1.5, 1)) coso1.borrar_aprioris() coso1.espec_apriori(apriori, 'categ', 'subcateg', 'ec', 'a', índs=[coso3]) diff --git a/pruebas/test_ecs/test_dists.py b/pruebas/test_ecs/test_dists.py index 94981494..13c8c50e 100644 --- a/pruebas/test_ecs/test_dists.py +++ b/pruebas/test_ecs/test_dists.py @@ -1,5 +1,6 @@ import unittest import warnings +from typing import cast import numpy as np import numpy.testing as npt @@ -7,9 +8,9 @@ from matplotlib.figure import Figure as Figura from numpy import exp from scipy.special import expit -from tikon.ecs.dists import DistAnalítica, DistTraza, Dist, MnjdrDists, dibujar_dist +from tikon.ecs.dists import DistAnalítica, DistTraza, Dist, ManejadorDists, dibujar_dist from tikon.ecs.dists.anlt import TransfDist -from tikon.utils import proc_líms +from tikon.ecs.dists.utils import proc_líms class PruebaDistAnalítica(unittest.TestCase): @@ -187,7 +188,7 @@ def test_conv_dic(): class PruebaDistTraza(unittest.TestCase): @classmethod def setUpClass(cls): - cls.n = n = 10000 + cls.n = n = 100000 trz = np.random.normal(0, 1, n) cls.dist = DistTraza(trz) @@ -217,65 +218,65 @@ def test_obt_vals_índ_matr(símismo): npt.assert_equal(símismo.dist.obt_vals_índ(índs), símismo.dist.trz[índs]) def test_conv_dic(símismo): - dist = Dist.de_dic(símismo.dist.a_dic()) + dist = cast(DistTraza, Dist.de_dic(símismo.dist.a_dic())) npt.assert_allclose(dist.trz, símismo.dist.trz) npt.assert_allclose(dist.pesos, símismo.dist.pesos) def test_error_pesos(símismo): with símismo.assertRaises(ValueError): - DistTraza(trz=np.arange(12), pesos=np.random.random(símismo.n + 1)) + DistTraza(trz=np.arange(12), pesos=np.random.rand(símismo.n + 1)) class PruebaMnjdrDists(unittest.TestCase): def test_base(símismo): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist = DistAnalítica(estad.norm()) mnjdr.actualizar(dist) símismo.assertIs(dist, mnjdr.obt_val()) def test_borrar(símismo): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist = DistAnalítica(estad.norm()) mnjdr.actualizar(dist) mnjdr.actualizar(None) símismo.assertIsNone(mnjdr.obt_val()) def test_índs(símismo): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist = DistAnalítica(estad.norm()) - mnjdr.actualizar(dist, índs=['a', 'b']) + mnjdr.actualizar(dist, índices=['a', 'b']) símismo.assertIs(dist, mnjdr.obt_val(['a', 'b'])) def test_índs_no_existen(símismo): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist = DistAnalítica(estad.norm()) mnjdr.actualizar(dist) - símismo.assertIsNone(mnjdr.obt_val(índs=['hola'], heredar=False)) - símismo.assertIs(dist, mnjdr.obt_val(índs=['hola'])) + símismo.assertIsNone(mnjdr.obt_val(índices=['hola'], heredar=False)) + símismo.assertIs(dist, mnjdr.obt_val(índices=['hola'])) def test_índs_herencia(símismo): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist = DistAnalítica(estad.norm()) - mnjdr.actualizar(dist, índs='a') + mnjdr.actualizar(dist, índices='a') símismo.assertIs(dist, mnjdr.obt_val(['a', 'b'])) def test_índs_sin_herencia(símismo): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist = DistAnalítica(estad.norm()) - mnjdr.actualizar(dist, índs=['a']) + mnjdr.actualizar(dist, índices=['a']) símismo.assertIsNone(mnjdr.obt_val(['a', 'b'], heredar=False)) @staticmethod def test_conv_dic(): - mnjdr = MnjdrDists() + mnjdr = ManejadorDists() dist0 = DistAnalítica(estad.norm()) dista = DistAnalítica(estad.gamma(1)) distb = DistAnalítica(estad.norm(3, 4)) mnjdr.actualizar(dist0) - mnjdr.actualizar(dista, índs=['a']) - mnjdr.actualizar(distb, índs=['a', 'b']) - nuevo = MnjdrDists.de_dic(mnjdr.a_dic()) + mnjdr.actualizar(dista, índices=['a']) + mnjdr.actualizar(distb, índices=['a', 'b']) + nuevo = ManejadorDists.de_dic(mnjdr.a_dic()) p = .95 npt.assert_equal(mnjdr.obt_val().aprox_líms(p), nuevo.obt_val().aprox_líms(p)) npt.assert_equal(mnjdr.obt_val('a').aprox_líms(p), nuevo.obt_val('a').aprox_líms(p)) diff --git a/pyproject.toml b/pyproject.toml index 44e52f65..719aa969 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -6,7 +6,7 @@ authors = ["Julien Jean Malard-Adam "] license = "AGPL-v3.0" [tool.poetry.dependencies] -python = "^3.9,<3.11" +python = ">=3.10,<3.12" numpy = "^1.23.2" scipy = "^1.9.1" pandas = "^1.4.4" @@ -31,7 +31,14 @@ bibtexparser = "^1.2.0" frozendict = "^2.3.4" [tool.poetry.group.dev.dependencies] -mypy = "^0.971" +mypy = {git = "https://github.com/python/mypy.git"} +data-science-types = "^0.2.23" +pyright = "^1.1.270" + +[tool.mypy] +plugins = "numpy.typing.mypy_plugin" +follow_imports = "silent" +ignore_missing_imports = true [build-system] requires = ["poetry-core>=1.0.0"] diff --git a/tikon/__init__.py b/tikon/__init__.py index f76558ba..f3813cc4 100644 --- a/tikon/__init__.py +++ b/tikon/__init__.py @@ -5,8 +5,5 @@ with open(resource_filename('tikon', 'versión.txt')) as archivo_versión: __versión__ = __version__ = archivo_versión.read().strip() -# Numpy nos da un montón de errores que no nos importan en Tiko'n. -np.warnings.filterwarnings('ignore') - -__autor__ = 'Julien Malard' +__autor__ = 'Julien Malard-Adam' __correo__ = 'julien.malard@mail.mcgill.ca' diff --git "a/tikon/central/par\303\241ms_exper.py" "b/tikon/central/par\303\241ms_exper.py" index 87a9c2df..336587d1 100644 --- "a/tikon/central/par\303\241ms_exper.py" +++ "b/tikon/central/par\303\241ms_exper.py" @@ -5,7 +5,8 @@ from tikon.ecs.aprioris import APrioriDens, APrioriDist from tikon.ecs.paráms import ValsParámCoso, ValsParámCosoInter, MatrParám from tikon.ecs.árb_coso import ParámCoso -from tikon.utils import proc_líms, EJE_PARÁMS, EJE_ESTOC, EJE_TIEMPO +from tikon.utils import EJE_PARÁMS, EJE_ESTOC, EJE_TIEMPO +from ..ecs.dists.utils import proc_líms from .datos import DatosVar, DatosMód diff --git a/tikon/central/res.py b/tikon/central/res.py index d8847b1b..8ec220ea 100644 --- a/tikon/central/res.py +++ b/tikon/central/res.py @@ -9,7 +9,8 @@ from tikon.datos.dibs import graficar_res from tikon.datos.valid import ValidÍnds, ValidRes from tikon.ecs.aprioris import APrioriDens -from tikon.utils import proc_líms, EJE_PARÁMS, EJE_ESTOC, EJE_TIEMPO, guardar_json, asegurar_dir_existe, asegurar_ext +from tikon.utils import EJE_PARÁMS, EJE_ESTOC, EJE_TIEMPO, guardar_json, asegurar_dir_existe, asegurar_ext +from tikon.ecs.dists.utils import proc_líms class Resultado(PlantillaSimul): diff --git a/tikon/datos/valid.py b/tikon/datos/valid.py index d765f78e..a1424279 100644 --- a/tikon/datos/valid.py +++ b/tikon/datos/valid.py @@ -29,7 +29,7 @@ def __init__(símismo, criterios, peso, índs): super().__init__(criterios, peso) def a_dic(símismo): - return {'índs': {ll: str(v) for ll, v in símismo.índs.items()}, **super().a_dic()} + return {'índices': {ll: str(v) for ll, v in símismo.índs.items()}, **super().a_dic()} class ValidRes(PlantillaValid): diff --git a/tikon/ecs/aprioris.py b/tikon/ecs/aprioris.py index 1ccaa86c..490a1307 100644 --- a/tikon/ecs/aprioris.py +++ b/tikon/ecs/aprioris.py @@ -1,31 +1,38 @@ +from typing import Optional + +from numpy.typing import ArrayLike +from scipy.stats._distn_infrastructure import rv_continuous_frozen + from .dists import DistAnalítica from .dists import líms_dist +from .dists.utils import Líms_Con_None from .utils import líms_compat +from ..tipos import Tipo_Valor_Numérico class APriori(object): - def dist(símismo, líms): + def dist(símismo, líms: Optional[Líms_Con_None]) -> DistAnalítica: raise NotImplementedError class APrioriDens(APriori): - def __init__(símismo, rango, certidumbre): + def __init__(símismo, rango: Optional[Líms_Con_None], certidumbre: Tipo_Valor_Numérico): símismo.rango = rango símismo.cert = certidumbre - def dist(símismo, líms): + def dist(símismo, líms: Optional[Líms_Con_None]) -> DistAnalítica: return DistAnalítica.de_dens(símismo.cert, símismo.rango, líms=líms) class APrioriDist(APriori): - def __init__(símismo, dist): + def __init__(símismo, dist: DistAnalítica | str | rv_continuous_frozen): símismo._dist = dist if isinstance(dist, DistAnalítica): símismo._líms_dist = dist.líms else: símismo._líms_dist = líms_dist(dist) - def dist(símismo, líms): + def dist(símismo, líms: Optional[Líms_Con_None]) -> DistAnalítica: líms_compat(símismo._líms_dist, líms) return símismo._dist if isinstance(símismo._dist, DistAnalítica) else DistAnalítica(símismo._dist) diff --git a/tikon/ecs/dists/__init__.py b/tikon/ecs/dists/__init__.py index eb4f85a7..f2ecb380 100644 --- a/tikon/ecs/dists/__init__.py +++ b/tikon/ecs/dists/__init__.py @@ -1,5 +1,5 @@ from .anlt import DistAnalítica -from .dists import Dist, MnjdrDists +from .dists import Dist, ManejadorDists from .trz import DistTraza from .utils import líms_dist, obt_nombre from .dibs import dibujar_dist diff --git a/tikon/ecs/dists/anlt.py b/tikon/ecs/dists/anlt.py index cdd91c79..7447fc04 100644 --- a/tikon/ecs/dists/anlt.py +++ b/tikon/ecs/dists/anlt.py @@ -1,20 +1,37 @@ +from typing import Union, Optional, TypedDict, Any, cast from warnings import warn as avisar import numpy as np +from matplotlib.axes import Axes from scipy import stats as estad from scipy.special import expit, logit from scipy.stats._continuous_distns import FitDataError +from scipy.stats._distn_infrastructure import rv_continuous_frozen +from typing_extensions import NotRequired from tikon.ecs.dists import utils -from tikon.ecs.dists.dists import Dist, _escl_inf, _dist_mu +from tikon.ecs.dists.dibs import dibujar_dist +from tikon.ecs.dists.dists import Dist, _escala_inf, _dist_mu, DicDist +from tikon.ecs.dists.utils import obt_scipy, obt_nombre, obt_prms_obj_scipy, líms_dist, clase_scipy, prms_dist, \ + proc_líms, \ + Líms_Con_None, Líms_Numéricas from tikon.ecs.utils import líms_compat -from tikon.utils import proc_líms -from .dibs import dibujar_dist -from .utils import obt_scipy, obt_nombre, obt_prms_obj_scipy, líms_dist, clase_scipy, prms_dist +from tikon.tipos import Tipo_Matriz_Numérica, Tipo_Matriz_Núm_Entero, Tipo_Valor_Numérico + + +class DicDistAnalítica(DicDist, TypedDict): + transf: Optional["DicTransfDist"] + paráms: tuple[tuple, Tipo_Valor_Numérico, Tipo_Valor_Numérico] + dist: str class DistAnalítica(Dist): - def __init__(símismo, dist, transf=None): + dist: rv_continuous_frozen + _transf: Optional["TransfDist"] + paráms = tuple[tuple, int | float, int | float] + líms: tuple[Tipo_Valor_Numérico, Tipo_Valor_Numérico] + + def __init__(símismo, dist: rv_continuous_frozen, transf: "TransfDist" = None): símismo.nombre_dist = obt_nombre(dist) paráms = obt_prms_obj_scipy(dist) @@ -31,16 +48,16 @@ def __init__(símismo, dist, transf=None): símismo.líms = tuple(símismo.transf_vals(np.array(líms_dist(símismo.dist)))) - def obt_vals(símismo, n): + def obt_vals(símismo, n: int) -> Tipo_Matriz_Numérica: return símismo.transf_vals(símismo.dist.rvs(n)) - def obt_vals_índ(símismo, í): + def obt_vals_índ(símismo, í: Tipo_Matriz_Núm_Entero) -> Tipo_Matriz_Numérica: return símismo.obt_vals(n=len(í)) - def tmñ(símismo): + def tmñ(símismo) -> Union[int, float]: return np.inf - def aprox_líms(símismo, prc): + def aprox_líms(símismo, prc) -> np.ndarray[Any, np.dtype[np.number]]: # Las superficies de las colas que hay que dejar afuera del rango de los límites colas = ((1 - prc) / 2, 0.5 + prc / 2) @@ -49,86 +66,93 @@ def aprox_líms(símismo, prc): return símismo.transf_vals(líms_d) - def transf_vals(símismo, vals): + def transf_vals(símismo, vals: Tipo_Matriz_Numérica) -> Tipo_Matriz_Numérica: if símismo._transf is not None: vals = símismo._transf.transf(vals) return vals - def dibujar(símismo, nombre=None, ejes=None, argsll=None): + def dibujar(símismo, nombre=None, ejes=None, argsll=None) -> Axes: return dibujar_dist(símismo, nombre=nombre or símismo.nombre_dist, ejes=ejes, argsll=argsll) - def a_dic(símismo): - return { - 'tipo': símismo.__class__.__name__, - '_transf': símismo._transf.a_dic() if símismo._transf else None, - 'dist': símismo.nombre_dist, - 'paráms': símismo.paráms - } + def a_dic(símismo) -> DicDistAnalítica: + return DicDistAnalítica( + tipo=símismo.__class__.__name__, + transf=símismo._transf.a_dic() if símismo._transf else None, + dist=símismo.nombre_dist, + paráms=símismo.paráms + ) @classmethod - def de_dic(cls, dic): - transf = TransfDist.de_dic(dic['_transf']) if dic['_transf'] else None - dist_sp = obt_scipy(dic['dist'], dic['paráms']) + def de_dic(cls, dic: DicDist) -> "DistAnalítica": + if dic["tipo"] != "DistAnalítica": + raise ValueError(dic) + + dicDistAnalítica = cast(DicDistAnalítica, dic) + transf = TransfDist.de_dic(dicDistAnalítica['transf']) if dicDistAnalítica['transf'] else None + dist_sp = obt_scipy(dicDistAnalítica['dist'], dicDistAnalítica['paráms']) return cls(dist=dist_sp, transf=transf) @classmethod - def de_líms(cls, líms): + def de_líms(cls, líms: Optional[Líms_Con_None]) -> "DistAnalítica": líms = proc_líms(líms) if líms[0] == -np.inf: if líms[1] == np.inf: - return DistAnalítica(dist=estad.norm(), transf=TransfDist(None, ubic=0, escl=_escl_inf)) + return DistAnalítica(dist=estad.norm(), transf=TransfDist(None, ubic=0, escl=_escala_inf)) - return DistAnalítica(dist=estad.expon(), transf=TransfDist(None, ubic=líms[1], escl=-_escl_inf)) + return DistAnalítica(dist=estad.expon(), transf=TransfDist(None, ubic=líms[1], escl=-_escala_inf)) if líms[1] == np.inf: - return DistAnalítica(dist=estad.expon(), transf=TransfDist(None, ubic=líms[0], escl=_escl_inf)) + return DistAnalítica(dist=estad.expon(), transf=TransfDist(None, ubic=líms[0], escl=_escala_inf)) return DistAnalítica(dist=estad.uniform(), transf=TransfDist(None, ubic=líms[0], escl=líms[1] - líms[0])) @classmethod - def de_dens(cls, dens, líms_dens, líms): - líms_dens = np.array(proc_líms(líms_dens)) - líms = np.array(proc_líms(líms)) - líms_compat(líms_dens, líms) + def de_dens( + cls, dens: int | float, líms_dens: Optional[Líms_Con_None], + líms: Optional[Líms_Con_None] + ) -> "DistAnalítica": + líms_dens_resueltas = np.array(proc_líms(líms_dens)) + líms_resueltas = np.array(proc_líms(líms)) + líms_compat(líms_dens_resueltas, líms_resueltas) if dens == 1: - if np.isinf(líms_dens[0]) or np.isinf(líms_dens[1]): + if np.isinf(líms_dens_resueltas[0]) or np.isinf(líms_dens_resueltas[1]): raise ValueError( - 'No se puede especificar densidad de 1 con rango ilimitado como "{}".'.format(líms_dens) + 'No se puede especificar densidad de 1 con rango ilimitado como "{}".'.format(líms_dens_resueltas) ) return DistAnalítica( - dist=estad.uniform(), transf=TransfDist(None, ubic=líms_dens[0], escl=líms_dens[1] - líms_dens[0]) + dist=estad.uniform(), transf=TransfDist(None, ubic=líms_dens_resueltas[0], escl=líms_dens_resueltas[1] - líms_dens_resueltas[0]) ) elif dens <= 0 or dens > 1: raise ValueError('La densidad debe ser en (0, 1].') - if líms[0] == -np.inf: - if líms[1] == np.inf: + if líms_resueltas[0] == -np.inf: + if líms_resueltas[1] == np.inf: transf = None else: - escl = (líms[1] - líms_dens[1]) or ((líms[1] - líms_dens[0]) if líms_dens[0] != -np.inf else 1) - transf = TransfDist('LnExp', ubic=líms[1], escl=-escl) + escl = (líms_resueltas[1] - líms_dens_resueltas[1]) or ((líms_resueltas[1] - líms_dens_resueltas[0]) if líms_dens_resueltas[0] != -np.inf else 1) + transf = TransfDist('LnExp', ubic=líms_resueltas[1], escl=-escl) - elif líms[1] == np.inf: - escl = (líms_dens[0] - líms[0]) or ((líms_dens[1] - líms[0]) if líms_dens[1] != np.inf else 1) - transf = TransfDist('LnExp', ubic=líms[0], escl=escl) + elif líms_resueltas[1] == np.inf: + escl = (líms_dens_resueltas[0] - líms_resueltas[0]) or ((líms_dens_resueltas[1] - líms_resueltas[0]) if líms_dens_resueltas[1] != np.inf else 1) + transf = TransfDist('LnExp', ubic=líms_resueltas[0], escl=escl) else: - transf = TransfDist('Expit', ubic=líms[0], escl=líms[1] - líms[0]) + transf = TransfDist('Expit', ubic=líms_resueltas[0], escl=líms_resueltas[1] - líms_resueltas[0]) if transf is None: - líms_dens_intern = líms_dens + líms_dens_intern = líms_dens_resueltas else: - líms_dens_intern = transf.transf_inv(líms_dens) + líms_dens_intern = transf.transf_inv(líms_dens_resueltas) líms_dens_intern.sort() if líms_dens_intern[0] == -np.inf: if líms_dens_intern[1] == np.inf: raise ValueError( 'Rangos idénticos como {r1} y {r2} no pueden tener densidad inferior a ' - '1.'.format(r1=líms, r2=líms_dens) + '1.'.format(r1=líms_resueltas, r2=líms_dens_resueltas) ) mu = líms_dens_intern[1] - _dist_mu sg = -_dist_mu / estad.norm.ppf(1 - dens) @@ -144,7 +168,7 @@ def de_dens(cls, dens, líms_dens, líms): return DistAnalítica(estad.norm(loc=mu, scale=sg), transf=transf) @classmethod - def de_traza(cls, trz, líms, permitidas=None): + def de_traza(cls, trz: Tipo_Matriz_Numérica, líms: Líms_Con_None, permitidas: list[str] = None): permitidas = permitidas or list(utils.dists) líms = proc_líms(líms) @@ -152,7 +176,12 @@ def de_traza(cls, trz, líms, permitidas=None): raise ValueError('Valores en traza deben caber en los límites teoréticos de la distribución.') # Un diccionario para guardar el mejor ajuste - mejor_ajuste = {} + class DicMejorAjuste(TypedDict): + p: int | float + dist: rv_continuous_frozen + transf: TransfDist + + mejor_ajuste: DicMejorAjuste | None = None for nmbr in permitidas: líms_d = líms_dist(nmbr) @@ -199,12 +228,17 @@ def de_traza(cls, trz, líms, permitidas=None): if mejor_ajuste['p'] <= 0.10: avisar('El ajuste de la mejor distribución quedó muy mal (p = {:.6f}).'.format(mejor_ajuste['p'])) - mejor_ajuste.pop('p') - return DistAnalítica(**mejor_ajuste) + return DistAnalítica(dist=mejor_ajuste["dist"], transf=mejor_ajuste["transf"]) + + +class DicTransfDist(TypedDict): + transf: Optional[str] + ubic: Tipo_Valor_Numérico + escl: Tipo_Valor_Numérico class TransfDist(object): - def __init__(símismo, transf, ubic=0, escl=1): + def __init__(símismo, transf: Optional[str], ubic: Tipo_Valor_Numérico = 0, escl: Tipo_Valor_Numérico = 1): """ Parameters @@ -230,24 +264,25 @@ def __init__(símismo, transf, ubic=0, escl=1): símismo._ubic = ubic símismo._escl = escl - def transf(símismo, vals): + def transf(símismo, vals: Tipo_Matriz_Numérica) -> Tipo_Matriz_Numérica: return símismo._f(vals) * símismo._escl + símismo._ubic - def transf_inv(símismo, vals): + def transf_inv(símismo, vals: Tipo_Matriz_Numérica) -> Tipo_Matriz_Numérica: return símismo._f_inv((vals - símismo._ubic) / símismo._escl) - def a_dic(símismo): + def a_dic(símismo) -> DicTransfDist: return { 'transf': símismo._transf, - 'ubic': símismo._ubic, 'escl': símismo._escl + 'ubic': símismo._ubic, + 'escl': símismo._escl } @classmethod - def de_dic(cls, dic): + def de_dic(cls, dic: DicTransfDist) -> "TransfDist": return TransfDist(dic['transf'], ubic=dic['ubic'], escl=dic['escl']) -def _gen_transf_sin_líms(trz, líms): +def _gen_transf_sin_líms(trz: Tipo_Matriz_Numérica, líms: Líms_Numéricas) -> TransfDist: if líms[0] == -np.inf: if líms[1] == np.inf: # noinspection PyTypeChecker @@ -260,7 +295,7 @@ def _gen_transf_sin_líms(trz, líms): return TransfDist('Expit', ubic=líms[0], escl=líms[1] - líms[0]) -def _gen_transf_1_lím(trz, líms, líms_d): +def _gen_transf_1_lím(trz: Tipo_Matriz_Numérica, líms: Líms_Numéricas, líms_d: Líms_Numéricas) -> TransfDist: if líms_d[0] == -np.inf: líms_d = (-líms_d[1], np.inf) líms = (-líms[1], -líms[0]) @@ -274,16 +309,18 @@ def _gen_transf_1_lím(trz, líms, líms_d): ubic = líms[0] - líms_d[0] * escl return TransfDist(None, ubic=ubic, escl=escl) + raise ValueError(líms) + -def lnexp(x): +def lnexp(x: Tipo_Matriz_Numérica) -> Tipo_Matriz_Numérica: return np.log(np.exp(x) + 1) -def invlnexp(x): +def invlnexp(x: Tipo_Matriz_Numérica) -> Tipo_Matriz_Numérica: return np.where(x >= 50, x, np.log(np.exp(x) - 1)) -def _líms_compat_teor(líms, ref): +def _líms_compat_teor(líms: Líms_Con_None, ref: Líms_Con_None) -> None: líms = proc_líms(líms) ref = proc_líms(ref) suma = sum([ref[0] == -np.inf, ref[1] == np.inf]) diff --git a/tikon/ecs/dists/dibs.py b/tikon/ecs/dists/dibs.py index d00db643..6526fc1f 100644 --- a/tikon/ecs/dists/dibs.py +++ b/tikon/ecs/dists/dibs.py @@ -1,10 +1,14 @@ -import seaborn as sns +from typing import Any +import seaborn as sns +from matplotlib.axes import Axes from matplotlib.backends.backend_agg import FigureCanvasAgg as TelaFigura from matplotlib.figure import Figure as Figura +from .dists import Dist + -def dibujar_dist(dist, nombre, ejes=None, argsll=None): +def dibujar_dist(dist: Dist, nombre: str, ejes: Axes = None, argsll: dict[str, Any] = None) -> Axes: args_base = dict( color='#99CC00', cut=0, shade=True ) diff --git a/tikon/ecs/dists/dists.py b/tikon/ecs/dists/dists.py index 0c4e49d9..099be12e 100644 --- a/tikon/ecs/dists/dists.py +++ b/tikon/ecs/dists/dists.py @@ -1,28 +1,44 @@ -_escl_inf = 1e10 +from typing import Union, TypedDict, Optional, Iterable, Any + +import numpy as np + +from tikon.tipos import Tipo_Valor_Numérico, Tipo_Matriz_Núm_Entero, Tipo_Matriz_Numérica + +_escala_inf = 1e10 _dist_mu = 1 +class DicDist(TypedDict): + tipo: str + + +class DicManejadorDists(TypedDict): + val: Optional[DicDist] + índices: dict[str, "DicManejadorDists"] + + +Tipo_Índices = Union[str, Iterable[str]] + + class Dist(object): - def obt_vals(símismo, n): + def obt_vals(símismo, n: int) -> Tipo_Matriz_Numérica: raise NotImplementedError - def obt_vals_índ(símismo, í): + def obt_vals_índ(símismo, í: Tipo_Matriz_Núm_Entero) -> Tipo_Matriz_Numérica: raise NotImplementedError - def tmñ(símismo): + def tmñ(símismo) -> Union[int, float]: raise NotImplementedError - def aprox_líms(símismo, prc): + def aprox_líms(símismo, prc: Tipo_Valor_Numérico) -> np.ndarray[Any, np.dtype[np.number]]: raise NotImplementedError - def a_dic(símismo): + def a_dic(símismo) -> DicDist: raise NotImplementedError @classmethod - def de_dic(cls, dic): - if not dic: - return + def de_dic(cls, dic: DicDist) -> "Dist": tipo = dic['tipo'] for x in cls.__subclasses__(): if x.__name__ == tipo: @@ -31,73 +47,80 @@ def de_dic(cls, dic): raise ValueError(tipo) -class MnjdrDists(object): +class ManejadorDists(object): + val: Optional[Dist] + índices: dict[str, "ManejadorDists"] + def __init__(símismo): símismo.val = None - símismo.índs = {} + símismo.índices = {} - def actualizar(símismo, dist, índs=None): - índs = símismo._proc_índs(índs) - if índs is None or not índs: + def actualizar(símismo, dist: Optional["Dist"], índices: Tipo_Índices = None) -> None: + índices = símismo._proc_índs(índices) + if índices is None or not índices: símismo.val = dist else: - í = str(índs.pop(0)) + í = str(índices.pop(0)) - if í not in símismo.índs: - símismo.índs[í] = MnjdrDists() + if í not in símismo.índices: + símismo.índices[í] = ManejadorDists() - símismo.índs[í].actualizar(dist, índs) + símismo.índices[í].actualizar(dist, índices) - def obt_val(símismo, índs=None, heredar=True): - índs = símismo._proc_índs(índs) + def obt_val(símismo, índices: Tipo_Índices = None, heredar: bool = True) -> Optional[Dist]: + índices = símismo._proc_índs(índices) - if índs is None or not len(índs): + if índices is None or not len(índices): return símismo.val - í = str(índs.pop(0)) - if í in símismo.índs: - return símismo[í].obt_val(índs, heredar) + í = str(índices.pop(0)) + if í in símismo.índices: + return símismo[í].obt_val(índices, heredar) if heredar: return símismo.val + return None - def borrar(símismo, índs=None): + def borrar(símismo, índs: list[str] = None) -> None: índs = símismo._proc_índs(índs) if índs is None or not índs: símismo.val = None - símismo.índs = {} + símismo.índices = {} else: í = str(índs.pop(0)) - símismo.índs[í].borrar(índs) + símismo.índices[í].borrar(índs) @staticmethod - def _proc_índs(índs): + def _proc_índs(índs: Optional[Tipo_Índices]) -> Optional[list[str]]: if isinstance(índs, str): return [índs] elif índs is not None: return list(índs) # generar copia + return None - def __getitem__(símismo, itema): - return símismo.índs[itema] + def __getitem__(símismo, itema: str) -> "ManejadorDists": + return símismo.índices[itema] - def a_dic(símismo): + def a_dic(símismo) -> DicManejadorDists: return { 'val': símismo.val.a_dic() if símismo.val else None, - 'índs': {str(ll): v.a_dic() for ll, v in símismo.índs.items()} + 'índices': {str(ll): v.a_dic() for ll, v in símismo.índices.items()} } @classmethod - def de_dic(cls, dic, mnjdr=None): - if mnjdr is None: - mnjdr = MnjdrDists() + def de_dic(cls, dic: DicManejadorDists, manejador: "ManejadorDists" = None) -> "ManejadorDists": + if manejador is None: + manejador = ManejadorDists() - def act_mnjdr(mnj, d, índs_ant=None): + def act_manejador( + mnj: ManejadorDists, d: DicManejadorDists, índices_anteriores: list[str] = None + ) -> None: val = d['val'] - índs = d['índs'] - mnj.actualizar(dist=Dist.de_dic(val), índs=índs_ant) - for í in índs: - act_mnjdr(mnj, d=índs[í], índs_ant=(índs_ant or []) + [í]) + índices = d['índices'] + mnj.actualizar(dist=Dist.de_dic(val) if val else None, índices=índices_anteriores) + for í in índices: + act_manejador(mnj, d=índices[í], índices_anteriores=(índices_anteriores or []) + [í]) - act_mnjdr(mnjdr, d=dic) + act_manejador(manejador, d=dic) - return mnjdr + return manejador diff --git a/tikon/ecs/dists/trz.py b/tikon/ecs/dists/trz.py index 1020e157..11d7195c 100644 --- a/tikon/ecs/dists/trz.py +++ b/tikon/ecs/dists/trz.py @@ -1,48 +1,63 @@ +from typing import Union, Any, cast + import numpy as np -from tikon.ecs.dists.dists import Dist + +from tikon.tipos import Tipo_Valor_Numérico, Tipo_Matriz_Numérica +from tikon.ecs.dists.dists import Dist, DicDist + + +class DicDistTraza(DicDist): + trz: Tipo_Matriz_Numérica + pesos: Tipo_Matriz_Numérica class DistTraza(Dist): - def __init__(símismo, trz, pesos=None): + trz: Tipo_Matriz_Numérica + pesos: Tipo_Matriz_Numérica + + def __init__(símismo, trz: Tipo_Matriz_Numérica, pesos: Tipo_Matriz_Numérica = None): if pesos is None: pesos = np.ones_like(trz) - pesos = pesos / np.sum(pesos) + pesos_finales = pesos / np.sum(pesos) - if trz.size != pesos.size: + if trz.size != pesos_finales.size: raise ValueError('El tamaño de la traza y él de sus pesos deben ser iguales.') símismo.trz = trz - símismo.pesos = pesos + símismo.pesos = pesos_finales - def obt_vals(símismo, n): + def obt_vals(símismo, n: int) -> Tipo_Matriz_Numérica: return np.random.choice(símismo.trz, n, p=símismo.pesos) - def obt_vals_índ(símismo, í): + def obt_vals_índ(símismo, í: Union[int, Tipo_Matriz_Numérica]) -> Tipo_Matriz_Numérica: return símismo.trz[í] - def tmñ(símismo): + def tmñ(símismo) -> int: return símismo.trz.size - def aprox_líms(símismo, prc): + def aprox_líms(símismo, prc: Tipo_Valor_Numérico) -> Tipo_Matriz_Numérica: # Las superficies de las colas que hay que dejar afuera del rango de los límites colas = ((1 - prc) / 2, 0.5 + prc / 2) trz, pesos = símismo.trz, símismo.pesos return np.array([_centil_pesos(trz, pesos, colas[0]), _centil_pesos(trz, pesos, colas[1])]) - def a_dic(símismo): - return { - 'tipo': símismo.__class__.__name__, - 'trz': símismo.trz, - 'pesos': símismo.pesos - } + def a_dic(símismo) -> DicDistTraza: + return DicDistTraza( + tipo=símismo.__class__.__name__, + trz=símismo.trz, + pesos=símismo.pesos + ) @classmethod - def de_dic(cls, dic): - return DistTraza(trz=dic['trz'], pesos=dic['pesos']) + def de_dic(cls, dic: DicDist) -> "DistTraza": + if dic["tipo"] != "DistTraza": + raise ValueError(dic) + dicDistTraza = cast(DicDistTraza, dic) + return DistTraza(trz=dicDistTraza['trz'], pesos=dicDistTraza['pesos']) -def _centil_pesos(x, p, q): - # Mientras esparamos que numpy lo implemente +def _centil_pesos(x: Tipo_Matriz_Numérica, p: Tipo_Matriz_Numérica, q: float | int) -> Tipo_Matriz_Numérica: + # Mientras esperemos que numpy lo implemente # código de https://github.com/nudomarinero/wquantiles/blob/master/wquantiles.py índs_ord = np.argsort(x) x_ord = x[índs_ord] diff --git a/tikon/ecs/dists/utils.py b/tikon/ecs/dists/utils.py index e6f61276..d057c23f 100644 --- a/tikon/ecs/dists/utils.py +++ b/tikon/ecs/dists/utils.py @@ -1,11 +1,24 @@ from math import pi +from typing import TypedDict, Union, Optional import numpy as np import scipy.stats as estad -# Un diccionario de las distribuciones y de sus objetos de SciPy correspondientes. -from tikon.utils import proc_líms +from scipy.stats._distn_infrastructure import rv_continuous, rv_continuous_frozen + +from tikon.tipos import Tipo_Valor_Numérico + +Líms_Con_None = tuple[Union[Tipo_Valor_Numérico, None], Union[Tipo_Valor_Numérico, None]] +Líms_Numéricas = tuple[Tipo_Valor_Numérico, Tipo_Valor_Numérico] + + +class DicEspecDistScipy(TypedDict): + scipy: rv_continuous + paráms: list[str] + límites: Líms_Con_None -dists = { + +# Un diccionario de las distribuciones y de sus objetos de SciPy correspondientes. +dists: dict[str, DicEspecDistScipy] = { 'Alpha': {'scipy': estad.alpha, 'paráms': ['a', 'loc', 'scale'], 'límites': (0, None) @@ -217,7 +230,6 @@ 'Triang': {'scipy': estad.triang, 'paráms': ['c', 'loc', 'scale'], 'límites': (0, 1), # El límite es ('a', 'b') - 'tipo': 'cont' }, 'TukeyLambda': {'scipy': estad.tukeylambda, 'paráms': ['lam', 'loc', 'scale'], @@ -226,7 +238,6 @@ 'Uniforme': {'scipy': estad.uniform, 'paráms': ['loc', 'scale'], 'límites': (0, 1), - 'tipo': 'cont' }, 'Wald': {'scipy': estad.wald, 'paráms': ['loc', 'scale'], @@ -247,7 +258,7 @@ } -def _valid_nombre(nombre): +def _valid_nombre(nombre: str) -> str: try: return next(nmbr for nmbr in dists if nmbr.lower() == nombre.lower()) except StopIteration: @@ -257,16 +268,16 @@ def _valid_nombre(nombre): ) -def _obt_dic_dist(nombre): +def _obt_dic_dist(nombre: str) -> DicEspecDistScipy: nombre = _valid_nombre(nombre) return dists[nombre] -def clase_scipy(nombre): +def clase_scipy(nombre: str) -> rv_continuous: return _obt_dic_dist(nombre)['scipy'] -def líms_dist(dist): +def líms_dist(dist: Union[str, rv_continuous_frozen]) -> Líms_Numéricas: if isinstance(dist, str): return proc_líms(_obt_dic_dist(dist)['límites']) else: @@ -276,15 +287,15 @@ def líms_dist(dist): return líms[0] + ubic, (líms[1] - líms[0]) * escala + ((líms[0] + ubic) if líms[0] > -np.inf else 0) -def obt_prms_obj_scipy(dist): +def obt_prms_obj_scipy(dist: rv_continuous_frozen) -> tuple[tuple, Tipo_Valor_Numérico, Tipo_Valor_Numérico]: return dist.dist._parse_args(*dist.args, **dist.kwds) -def prms_dist(nombre): +def prms_dist(nombre)->list[str]: return _obt_dic_dist(nombre)['paráms'] -def obt_scipy(nombre, paráms): +def obt_scipy(nombre: str, paráms: Union[dict[str, Tipo_Valor_Numérico], tuple[tuple, Tipo_Valor_Numérico, Tipo_Valor_Numérico]]): cls_dist = clase_scipy(nombre) if isinstance(paráms, dict): @@ -293,5 +304,13 @@ def obt_scipy(nombre, paráms): return cls_dist(*paráms[0], loc=paráms[1], scale=paráms[2]) -def obt_nombre(dist_sp): +def obt_nombre(dist_sp: rv_continuous_frozen)-> str: return next(nmb for nmb in dists if dists[nmb]['scipy'].name == dist_sp.dist.name) + + +def proc_líms(líms: Optional[Líms_Con_None]) -> Líms_Numéricas: + inf = np.inf + + if líms is None: + return -inf, inf + return -inf if líms[0] is None else líms[0], inf if líms[1] is None else líms[1] diff --git "a/tikon/ecs/par\303\241ms.py" "b/tikon/ecs/par\303\241ms.py" index 2483fe7d..affcc51a 100644 --- "a/tikon/ecs/par\303\241ms.py" +++ "b/tikon/ecs/par\303\241ms.py" @@ -1,28 +1,38 @@ +import typing +from typing import Iterable, Sequence, Optional + import numpy as np from frozendict import frozendict +from numpy.typing import ArrayLike from tikon.central.matriz import Datos - +from tikon.ecs.dists import Dist +from tikon.ecs.árb_coso import ParámCoso +from tikon.tipos import Tipo_Valor_Numérico_Entero, Tipo_Matriz_Núm_Entero from tikon.utils import EJE_PARÁMS +if typing.TYPE_CHECKING: + from tikon.central import Modelo, Módulo, Coso + from tikon.ecs import Parám + class MnjdrValsCoefs(object): - def __init__(símismo, modelo, mód, l_paráms, n_reps): - símismo._paráms = {str(pr): pr.gen_matr_parám(modelo=modelo, mód=mód, n_reps=n_reps) for pr in l_paráms} + def __init__(símismo, modelo: "Modelo", mód: "Módulo", paráms: Iterable["Parám"], n_reps: Tipo_Valor_Numérico_Entero): + símismo._paráms = {str(pr): pr.gen_matr_parám(modelo=modelo, mód=mód, n_reps=n_reps) for pr in paráms} - def vals_paráms(símismo): + def vals_paráms(símismo) -> list["ValsParámCoso"]: return [prm for matr in símismo._paráms.values() for prm in matr.vals_paráms()] - def act_vals(símismo): + def act_vals(símismo) -> None: for matr in símismo._paráms.values(): matr.act_vals() - def __getitem__(símismo, itema): + def __getitem__(símismo, itema) -> Datos: return símismo._paráms[itema].val class MatrParám(object): - def __init__(símismo, subs, eje, índice): + def __init__(símismo, subs: Sequence["MatrParám"], eje: str, índice: Optional["Coso"]): símismo._sub_matrs = subs símismo.eje = eje @@ -34,29 +44,35 @@ def __init__(símismo, subs, eje, índice): ) @property - def coords(símismo): + def coords(símismo) -> dict[str, list["Coso"]]: return {símismo.eje: [sub.índice for sub in símismo._sub_matrs], **_combin_coords(símismo._sub_matrs)} @property - def val(símismo): + def val(símismo) -> Datos: return símismo._datos - def act_vals(símismo): + def act_vals(símismo) -> None: for sub in símismo._sub_matrs: sub.act_vals() símismo._datos.loc[símismo._datos.codificar_coords({símismo.eje: sub.índice})] = sub.val - def vals_paráms(símismo): + def vals_paráms(símismo) -> list["ValsParámCoso"]: return [vls for mtr in símismo._sub_matrs for vls in mtr.vals_paráms()] class ValsParámCosoInter(MatrParám): - def __init__(símismo, vals_paráms_inter, eje, índice): + def __init__(símismo, vals_paráms_inter: list["ValsParámCoso"], eje: str, índice: Optional["Coso"]): super().__init__(vals_paráms_inter, eje=eje, índice=índice) class ValsParámCoso(MatrParám): - def __init__(símismo, tmñ, prm_base, índice, inter=None): + def __init__( + símismo, + tmñ: Tipo_Valor_Numérico_Entero, + prm_base: Optional["ParámCoso"], + índice: Optional["Coso"], + inter: Optional["Coso"] = None + ): símismo.tmñ = tmñ símismo.prm = prm_base símismo.inter = inter @@ -64,18 +80,18 @@ def __init__(símismo, tmñ, prm_base, índice, inter=None): super().__init__(subs=[], eje=EJE_PARÁMS, índice=índice) @property - def coords(símismo): + def coords(símismo) -> dict[str, list[int]]: return {símismo.eje: list(range(símismo.tmñ))} @property - def val(símismo): + def val(símismo) -> Datos: return símismo._datos @val.setter - def val(símismo, val): + def val(símismo, val: ArrayLike) -> None: símismo._datos[:] = val - def dists_disp(símismo, heredar): + def dists_disp(símismo, heredar: bool): return símismo.prm.dists_disp(símismo.inter, heredar) def dist_base(símismo): @@ -93,7 +109,7 @@ def apriori(símismo, heredar=True): def llenar_de_apriori(símismo, heredar=True): símismo.val = símismo.apriori(heredar).obt_vals(símismo.tmñ) - def llenar_de_dists(símismo, dists): + def llenar_de_dists(símismo, dists: list[tuple[Dist, Tipo_Valor_Numérico_Entero | Tipo_Matriz_Núm_Entero]]): val = [] for d, n in dists: @@ -102,7 +118,7 @@ def llenar_de_dists(símismo, dists): else: val.append(d.obt_vals_índ(n)) - símismo.val = np.ravel(val) + símismo.val = np.ravel(np.array(val)) def act_vals(símismo): """No necesario porque valores se establecen con las funciones `llenar_de...` o de manera externa por @@ -117,7 +133,7 @@ def guardar_calibs(símismo, dist, nombre): class ValsParámCosoVacíos(ValsParámCoso): - def __init__(símismo, tmñ, índice): + def __init__(símismo, tmñ: Tipo_Valor_Numérico_Entero, índice): super().__init__(tmñ, prm_base=None, índice=índice, inter=None) def vals_paráms(símismo): @@ -138,7 +154,7 @@ def __bool__(símismo): return len(símismo.itemas) > 0 -def _combin_coords(grupo): +def _combin_coords(grupo: Sequence["MatrParám"]) -> dict[str, list["Coso"]]: coords = grupo[0].coords if not all(obj.coords == coords for obj in grupo): raise ValueError('Coordinadas deben ser iguales para cada miembro de un grupo.') diff --git a/tikon/ecs/utils.py b/tikon/ecs/utils.py index 5ff07af9..258c1e8f 100644 --- a/tikon/ecs/utils.py +++ b/tikon/ecs/utils.py @@ -1,4 +1,4 @@ -from tikon.utils import proc_líms +from tikon.ecs.dists.utils import proc_líms def líms_compat(líms, ref): diff --git "a/tikon/ecs/\303\241rb_coso.py" "b/tikon/ecs/\303\241rb_coso.py" index e5ec1246..cdfa0a22 100644 --- "a/tikon/ecs/\303\241rb_coso.py" +++ "b/tikon/ecs/\303\241rb_coso.py" @@ -1,7 +1,7 @@ from copy import copy from typing import Dict -from .dists import DistAnalítica, MnjdrDists +from .dists import DistAnalítica, ManejadorDists class PlantillaRamaEcCoso(dict): @@ -135,8 +135,8 @@ def __init__(símismo, pariente, coso): símismo.líms = pariente.líms símismo.inter = pariente.inter símismo.apriori_auto = pariente.apriori - símismo._calibs = {} # type: Dict[str, MnjdrDists] - símismo._apriori = MnjdrDists() + símismo._calibs = {} # type: Dict[str, ManejadorDists] + símismo._apriori = ManejadorDists() super().__init__(pariente, ramas=[], coso=coso) @@ -145,7 +145,7 @@ def activa(símismo, modelo, mód, exper, coso): def agregar_calib(símismo, id_cal, dist, inter=None): if id_cal not in símismo._calibs: - símismo._calibs[id_cal] = MnjdrDists() + símismo._calibs[id_cal] = ManejadorDists() símismo._calibs[id_cal].actualizar(dist, inter) @@ -157,7 +157,7 @@ def renombrar_calib(símismo, nombre, nuevo): def espec_apriori(símismo, apriori, inter=None): dist = apriori.dist(símismo.líms) - símismo._apriori.actualizar(dist=dist, índs=inter) + símismo._apriori.actualizar(dist=dist, índices=inter) def borrar_aprioris(símismo, *args, índs=None): if len(args): @@ -171,15 +171,15 @@ def calib_base(símismo): return DistAnalítica.de_líms(símismo.líms) def dists_disp(símismo, inter, heredar): - return {clb: dists.obt_val(índs=inter, heredar=heredar) for clb, dists in símismo._calibs.items()} + return {clb: dists.obt_val(índices=inter, heredar=heredar) for clb, dists in símismo._calibs.items()} def a_dic(símismo): return {ll: v for ll, v in {nmb: clb.a_dic() for nmb, clb in símismo._calibs.items()}.items() if len(v)} def de_dic(símismo, dic): for calib, d_dist in dic.items(): - símismo._calibs[calib] = MnjdrDists.de_dic( - d_dist, mnjdr=símismo._calibs[calib] if calib in símismo._calibs else None + símismo._calibs[calib] = ManejadorDists.de_dic( + d_dist, manejador=símismo._calibs[calib] if calib in símismo._calibs else None ) def __copy__(símismo): diff --git "a/tikon/ecs/\303\241rb_m\303\263d.py" "b/tikon/ecs/\303\241rb_m\303\263d.py" index fd36d721..c133d950 100644 --- "a/tikon/ecs/\303\241rb_m\303\263d.py" +++ "b/tikon/ecs/\303\241rb_m\303\263d.py" @@ -1,15 +1,17 @@ +from typing import Type, Sequence + from frozendict import frozendict from tikon.central.matriz import Datos from .paráms import MnjdrValsCoefs, MatrParám, ValsParámCoso, ValsParámCosoInter, ValsParámCosoVacíos -from .árb_coso import ÁrbolEcsCoso, CategEcCoso, SubcategEcCoso, EcuaciónCoso, ParámCoso +from .árb_coso import ÁrbolEcsCoso, CategEcCoso, SubcategEcCoso, EcuaciónCoso, ParámCoso, PlantillaRamaEcCoso class PlantillaRamaEc(object): - cls_ramas = [] + cls_ramas: Sequence[Type["PlantillaRamaEc"]] = [] - _cls_en_coso = NotImplemented - _nombre_res = NotImplemented + _cls_en_coso: "Type[PlantillaRamaEcCoso]" = NotImplemented + _nombre_res: str = NotImplemented def __init__(símismo, modelo, mód, exper, cosos, n_reps, ecs): @@ -216,7 +218,7 @@ def eval(símismo, paso, sim): class Parám(PlantillaRamaEc): _cls_en_coso = ParámCoso eje_cosos = None - cls_ramas = [] + cls_ramas: Sequence[Type["Parám"]] = [] # Éstos se pueden personalizar líms = (None, None) diff --git a/tikon/ejemplos/proyectos/lotka_volterra/__init__.py b/tikon/ejemplos/proyectos/lotka_volterra/__init__.py new file mode 100644 index 00000000..e69de29b diff --git a/tikon/ejemplos/proyectos/lotka_volterra/modelo.py b/tikon/ejemplos/proyectos/lotka_volterra/modelo.py new file mode 100644 index 00000000..e519b7f7 --- /dev/null +++ b/tikon/ejemplos/proyectos/lotka_volterra/modelo.py @@ -0,0 +1,13 @@ +from tikon.central import Modelo +from tikon.móds.rae.orgs.insectos import LotkaVolterra +from tikon.móds.rae.red import RedAE + +presa = LotkaVolterra("presa") +depredador = LotkaVolterra("depredador") + +RedLotkaVolterra = RedAE([presa, depredador]) + +with RedLotkaVolterra: + depredador.secome(presa) + +modelo = Modelo(RedLotkaVolterra) diff --git a/tikon/ejemplos/proyectos/lotka_volterra/simul.py b/tikon/ejemplos/proyectos/lotka_volterra/simul.py new file mode 100644 index 00000000..3b27f1ac --- /dev/null +++ b/tikon/ejemplos/proyectos/lotka_volterra/simul.py @@ -0,0 +1,5 @@ +from tikon.central import Exper, Parcela, GeomParcela +from tikon.ejemplos.proyectos.lotka_volterra.modelo import modelo + +exper = Exper('P1', Parcela('P1', geom=GeomParcela((7.297, 79.865)))) +modelo.simular("base", exper) diff --git "a/tikon/m\303\263ds/cultivo/extrn.py" "b/tikon/m\303\263ds/cultivo/extrn.py" index 1407cbb6..40128b00 100644 --- "a/tikon/m\303\263ds/cultivo/extrn.py" +++ "b/tikon/m\303\263ds/cultivo/extrn.py" @@ -43,7 +43,7 @@ def __init__(símismo, sim, parcelas): símismo.instancias = [ símismo.cls_instancia( sim=símismo, índs=índs, reps=símismo.reps - ) for índs in símismo.combin.índs(símismo.reps) + ) for índs in símismo.combin.índices(símismo.reps) ] @property @@ -94,7 +94,7 @@ def llenar_vals(símismo): def aplicar_daño(símismo, daño): daño = símismo.combin.agregar(daño) for inst in símismo.instancias: - inst.aplicar_daño(daño.loc[codificar_coords(inst.índs)]) + inst.aplicar_daño(daño.loc[codificar_coords(inst.índices)]) def cerrar(símismo): for inst in símismo.instancias: diff --git "a/tikon/m\303\263ds/rae/orgs/organismo.py" "b/tikon/m\303\263ds/rae/orgs/organismo.py" index efb0e738..f48aec90 100644 --- "a/tikon/m\303\263ds/rae/orgs/organismo.py" +++ "b/tikon/m\303\263ds/rae/orgs/organismo.py" @@ -12,7 +12,7 @@ class Organismo(Coso): Un organismo es la clase pariente para cualquier especie en una red agroecológica. """ - def __init__(símismo, nombre, etapas): + def __init__(símismo, nombre: str, etapas: Union["Etapa", Iterable["Etapa"]]): super().__init__(nombre, ecs=EcsOrgs) if ':' in símismo.nombre: @@ -24,7 +24,7 @@ def __init__(símismo, nombre, etapas): símismo.etapas = etapas - def activar_ec(símismo, categ: str, subcateg: str, ec: str, etapas=None): + def activar_ec(símismo, categ: str, subcateg: str, ec: str, etapas: "Tipo_Resolvable_A_Etapas" = None): etapas = símismo.resolver_etapas(etapas) for etp in etapas: @@ -45,7 +45,10 @@ def espec_apriori(símismo, apriori, categ, sub_categ, ec, prm, índs=None, etap for etp in etapas: etp.espec_apriori(apriori, categ, sub_categ, ec, prm, índs=índs) - def borrar_aprioris(símismo, categ=None, sub_categ=None, ec=None, prm=None, índs=None, etapas=None): + def borrar_aprioris( + símismo, categ: str = None, sub_categ: str = None, ec: str = None, prm: str = None, índs=None, + etapas=None + ): etapas = símismo.resolver_etapas(etapas) for etp in etapas: etp.borrar_aprioris(categ, sub_categ, ec, prm, índs=índs) diff --git "a/tikon/m\303\263ds/rae/red/red.py" "b/tikon/m\303\263ds/rae/red/red.py" index 30dc923f..a68d4b06 100644 --- "a/tikon/m\303\263ds/rae/red/red.py" +++ "b/tikon/m\303\263ds/rae/red/red.py" @@ -1,10 +1,12 @@ +from typing import Iterable + import numpy as np from frozendict import frozendict +from tikon.central.matriz import Datos from tikon.central.módulo import Módulo from tikon.central.simul import SimulMódulo from tikon.consts import CORREO_AUTOR -from tikon.central.matriz import Datos from tikon.ecs.paráms import Inter from tikon.móds.rae.utils import RES_POBS, RES_COHORTES, EJE_COH, EJE_ETAPA, EJE_VÍCTIMA, contexto from .res.cohortes import ResCohortes @@ -26,7 +28,7 @@ class SimulRed(SimulMódulo): res_red.ResMuerte, res_red.ResTrans, res_red.ResMov, res_red.ResEstoc, ResCohortes ] - def __init__(símismo, mód, simul_exper, ecs, vars_interés): + def __init__(símismo, mód: "RedAE", simul_exper, ecs, vars_interés): símismo.etapas = mód.etapas símismo.orgs = mód.orgs @@ -78,10 +80,10 @@ def verificar_estado(símismo): if símismo[RES_COHORTES].activa: pobs_coh = símismo[RES_COHORTES].datos[{'comp': 0}] - if np.any(~np.equal(pobs_coh.matr, np.round(pobs_coh.matr))): + if np.any(~np.array_equal(pobs_coh.matr, np.round(pobs_coh.matr))): raise ValueError('Población de cohorte fraccional.\n{mnsg}'.format(mnsg=mnsg)) pobs_corresp_coh = pobs.loc[frozendict({EJE_ETAPA: pobs_coh.coords_internas[EJE_ETAPA]})] - if np.any(np.not_equal(pobs_coh.suma(dim=EJE_COH).matr, pobs_corresp_coh.matr)): + if np.any(np.array_equal(pobs_coh.suma(dim=EJE_COH).matr, pobs_corresp_coh.matr)): raise ValueError('Población de cohorte no suma a población total.\n{mnsg}'.format(mnsg=mnsg)) @@ -90,7 +92,7 @@ class RedAE(Módulo): cls_simul = SimulRed eje_coso = EJE_ETAPA - def __init__(símismo, orgs=None): + def __init__(símismo, orgs: Iterable[Organismo] = None): super().__init__(cosos=orgs) símismo.relaciones = {'presa': [], 'paras': []} @@ -118,7 +120,7 @@ def agregar_relación(símismo, relación): raise ValueError('Organismo "{org}" no existe en red "{r}".'.format(org=org, r=símismo)) símismo.relaciones[relación.tipo].append(relación) - def inter(símismo, modelo, coso, tipo): + def inter(símismo, modelo, coso, tipo: str): if isinstance(tipo, str): tipo = [tipo] diff --git "a/tikon/m\303\263ds/rae/utils.py" "b/tikon/m\303\263ds/rae/utils.py" index 4ccc37d1..dafb41db 100644 --- "a/tikon/m\303\263ds/rae/utils.py" +++ "b/tikon/m\303\263ds/rae/utils.py" @@ -1,3 +1,8 @@ +import typing + +if typing.TYPE_CHECKING: + from tikon.móds.rae.red import RedAE + EJE_COH = 'coh' EJE_ETAPA = 'etapa' EJE_VÍCTIMA = 'víctima' @@ -13,4 +18,4 @@ RES_ESTOC = 'Estoc' RES_COHORTES = 'Cohortes' -contexto = [] +contexto: list['RedAE'] = [] diff --git a/tikon/tipos/__init__.py b/tikon/tipos/__init__.py new file mode 100644 index 00000000..25dbb00e --- /dev/null +++ b/tikon/tipos/__init__.py @@ -0,0 +1,10 @@ +from typing import Union + +import numpy as np +import numpy.typing as npty + +Tipo_Valor_Numérico = Union[int, float, np.number] +Tipo_Valor_Numérico_Entero = Union[int, np.integer] + +Tipo_Matriz_Numérica = npty.NDArray[np.number] +Tipo_Matriz_Núm_Entero = npty.NDArray[np.integer] diff --git a/tikon/utils.py b/tikon/utils.py index 670588b2..d0d30122 100644 --- a/tikon/utils.py +++ b/tikon/utils.py @@ -126,14 +126,6 @@ def detectar_codif(archivo, máx_líneas=None, cortar=None): return detector.result['encoding'] # Devolver el resultado -def proc_líms(líms): - inf = np.inf - - if líms is None: - return -inf, inf - return -inf if líms[0] is None else líms[0], inf if líms[1] is None else líms[1] - - EJE_PARÁMS = 'paráms' EJE_ESTOC = 'estoc' EJE_TIEMPO = 'tiempo'