diff --git a/.flake8 b/.flake8 index 79a16af..84bd549 100644 --- a/.flake8 +++ b/.flake8 @@ -1,2 +1,5 @@ [flake8] -max-line-length = 120 \ No newline at end of file +max-line-length = 120 +ignore = E741 +exclude = .git,__pycache__ +max-complexity = 10 diff --git a/poetry.lock b/poetry.lock index dd26d8e..c61a133 100644 --- a/poetry.lock +++ b/poetry.lock @@ -20,52 +20,6 @@ docs = ["furo", "sphinx", "zope.interface", "sphinx-notfound-page"] tests = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins", "zope.interface"] tests_no_zope = ["coverage[toml] (>=5.0.2)", "hypothesis", "pympler", "pytest (>=4.3.0)", "six", "mypy", "pytest-mypy-plugins"] -[[package]] -name = "cairocffi" -version = "1.2.0" -description = "cffi-based cairo bindings for Python" -category = "main" -optional = false -python-versions = ">=3.6" - -[package.dependencies] -cffi = ">=1.1.0" - -[package.extras] -doc = ["sphinx", "sphinx-rtd-theme"] -test = ["pytest-runner", "pytest-cov", "pytest-flake8", "pytest-isort"] -xcb = ["xcffib (>=0.3.2)"] - -[[package]] -name = "cairosvg" -version = "2.5.2" -description = "A Simple SVG Converter based on Cairo" -category = "main" -optional = false -python-versions = ">=3.5" - -[package.dependencies] -cairocffi = "*" -cssselect2 = "*" -defusedxml = "*" -pillow = "*" -tinycss2 = "*" - -[package.extras] -doc = ["sphinx", "sphinx-rtd-theme"] -test = ["pytest-runner", "pytest-cov", "pytest-flake8", "pytest-isort"] - -[[package]] -name = "cffi" -version = "1.15.0" -description = "Foreign Function Interface for Python calling C code." -category = "main" -optional = false -python-versions = "*" - -[package.dependencies] -pycparser = "*" - [[package]] name = "click" version = "7.1.2" @@ -93,30 +47,6 @@ python-versions = "*" [package.dependencies] colorama = {version = "*", markers = "platform_system == \"Windows\""} -[[package]] -name = "cssselect2" -version = "0.4.1" -description = "cssselect2" -category = "main" -optional = false -python-versions = ">=3.6" - -[package.dependencies] -tinycss2 = "*" -webencodings = "*" - -[package.extras] -doc = ["sphinx", "sphinx-rtd-theme"] -test = ["pytest", "pytest-cov", "pytest-flake8", "pytest-isort", "coverage"] - -[[package]] -name = "defusedxml" -version = "0.7.1" -description = "XML bomb protection for Python stdlib modules" -category = "main" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" - [[package]] name = "flask" version = "1.1.4" @@ -147,24 +77,6 @@ python-versions = ">=2.7,!=3.0.*,!=3.1.*,!=3.2.*,!=3.3.*,!=3.4.*" [package.extras] docs = ["sphinx"] -[[package]] -name = "html5lib" -version = "1.1" -description = "HTML parser based on the WHATWG HTML specification" -category = "main" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*, !=3.4.*" - -[package.dependencies] -six = ">=1.9" -webencodings = "*" - -[package.extras] -all = ["genshi", "chardet (>=2.2)", "lxml"] -chardet = ["chardet (>=2.2)"] -genshi = ["genshi"] -lxml = ["lxml"] - [[package]] name = "importlib-metadata" version = "4.8.3" @@ -251,14 +163,6 @@ python-versions = ">=3.6" [package.dependencies] pyparsing = ">=2.0.2,<3.0.5 || >3.0.5" -[[package]] -name = "pillow" -version = "8.4.0" -description = "Python Imaging Library (Fork)" -category = "main" -optional = false -python-versions = ">=3.6" - [[package]] name = "pluggy" version = "1.0.0" @@ -301,14 +205,6 @@ category = "main" optional = false python-versions = "*" -[[package]] -name = "pycparser" -version = "2.21" -description = "C parser in Python" -category = "main" -optional = false -python-versions = ">=2.7, !=3.0.*, !=3.1.*, !=3.2.*, !=3.3.*" - [[package]] name = "pyparsing" version = "3.0.6" @@ -320,17 +216,6 @@ python-versions = ">=3.6" [package.extras] diagrams = ["jinja2", "railroad-diagrams"] -[[package]] -name = "pyphen" -version = "0.11.0" -description = "Pure Python module to hyphenate text" -category = "main" -optional = false -python-versions = ">=3.6" - -[package.extras] -test = ["pytest", "pytest-cov", "pytest-flake8", "pytest-isort", "coverage"] - [[package]] name = "pytest" version = "6.2.5" @@ -425,21 +310,6 @@ postgresql_psycopg2cffi = ["psycopg2cffi"] pymysql = ["pymysql (<1)", "pymysql"] sqlcipher = ["sqlcipher3-binary"] -[[package]] -name = "tinycss2" -version = "1.1.1" -description = "A tiny CSS parser" -category = "main" -optional = false -python-versions = ">=3.6" - -[package.dependencies] -webencodings = ">=0.4" - -[package.extras] -doc = ["sphinx", "sphinx-rtd-theme"] -test = ["pytest", "pytest-cov", "pytest-flake8", "pytest-isort", "coverage"] - [[package]] name = "toml" version = "0.10.2" @@ -464,36 +334,6 @@ category = "main" optional = false python-versions = "*" -[[package]] -name = "weasyprint" -version = "52.5" -description = "The Awesome Document Factory" -category = "main" -optional = false -python-versions = ">=3.6" - -[package.dependencies] -cairocffi = ">=0.9.0" -CairoSVG = ">=2.4.0" -cffi = ">=0.6" -cssselect2 = ">=0.1" -html5lib = ">=0.999999999" -Pillow = ">=4.0.0" -Pyphen = ">=0.9.1" -tinycss2 = ">=1.0.0" - -[package.extras] -doc = ["sphinx", "sphinx-rtd-theme"] -test = ["pytest-runner", "pytest-cov", "pytest-flake8", "pytest-isort"] - -[[package]] -name = "webencodings" -version = "0.5.1" -description = "Character encoding aliases for legacy web content" -category = "main" -optional = false -python-versions = "*" - [[package]] name = "werkzeug" version = "1.0.1" @@ -529,7 +369,7 @@ testing = ["pytest (>=4.6)", "pytest-checkdocs (>=2.4)", "pytest-flake8", "pytes [metadata] lock-version = "1.1" python-versions = "^3.6.1" -content-hash = "05801cd5e8b55510bd46861fd13f119c0269d3b343e22aa86c261eca50455d09" +content-hash = "d42ac98f37e9df3afbff04bb958a53bedd346fdfe0ea81b0bd4464eeb96f6837" [metadata.files] atomicwrites = [ @@ -540,65 +380,6 @@ attrs = [ {file = "attrs-21.2.0-py2.py3-none-any.whl", hash = "sha256:149e90d6d8ac20db7a955ad60cf0e6881a3f20d37096140088356da6c716b0b1"}, {file = "attrs-21.2.0.tar.gz", hash = "sha256:ef6aaac3ca6cd92904cdd0d83f629a15f18053ec84e6432106f7a4d04ae4f5fb"}, ] -cairocffi = [ - {file = "cairocffi-1.2.0.tar.gz", hash = "sha256:9a979b500c64c8179fec286f337e8fe644eca2f2cd05860ce0b62d25f22ea140"}, -] -cairosvg = [ - {file = "CairoSVG-2.5.2-py3-none-any.whl", hash = "sha256:98c276b7e4f0caf01e5c7176765c104ffa1aa1461d63b2053b04ab663cf7052b"}, - {file = "CairoSVG-2.5.2.tar.gz", hash = "sha256:b0b9929cf5dba005178d746a8036fcf0025550f498ca54db61873322384783bc"}, -] -cffi = [ - {file = "cffi-1.15.0-cp27-cp27m-macosx_10_9_x86_64.whl", hash = "sha256:c2502a1a03b6312837279c8c1bd3ebedf6c12c4228ddbad40912d671ccc8a962"}, - {file = "cffi-1.15.0-cp27-cp27m-manylinux1_i686.whl", hash = "sha256:23cfe892bd5dd8941608f93348c0737e369e51c100d03718f108bf1add7bd6d0"}, - {file = "cffi-1.15.0-cp27-cp27m-manylinux1_x86_64.whl", hash = "sha256:41d45de54cd277a7878919867c0f08b0cf817605e4eb94093e7516505d3c8d14"}, - {file = "cffi-1.15.0-cp27-cp27m-win32.whl", hash = "sha256:4a306fa632e8f0928956a41fa8e1d6243c71e7eb59ffbd165fc0b41e316b2474"}, - {file = "cffi-1.15.0-cp27-cp27m-win_amd64.whl", hash = "sha256:e7022a66d9b55e93e1a845d8c9eba2a1bebd4966cd8bfc25d9cd07d515b33fa6"}, - {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_i686.whl", hash = "sha256:14cd121ea63ecdae71efa69c15c5543a4b5fbcd0bbe2aad864baca0063cecf27"}, - {file = "cffi-1.15.0-cp27-cp27mu-manylinux1_x86_64.whl", hash = "sha256:d4d692a89c5cf08a8557fdeb329b82e7bf609aadfaed6c0d79f5a449a3c7c023"}, - {file = "cffi-1.15.0-cp310-cp310-macosx_10_9_x86_64.whl", hash = "sha256:0104fb5ae2391d46a4cb082abdd5c69ea4eab79d8d44eaaf79f1b1fd806ee4c2"}, - {file = "cffi-1.15.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:91ec59c33514b7c7559a6acda53bbfe1b283949c34fe7440bcf917f96ac0723e"}, - {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:f5c7150ad32ba43a07c4479f40241756145a1f03b43480e058cfd862bf5041c7"}, - {file = "cffi-1.15.0-cp310-cp310-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:00c878c90cb53ccfaae6b8bc18ad05d2036553e6d9d1d9dbcf323bbe83854ca3"}, - {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:abb9a20a72ac4e0fdb50dae135ba5e77880518e742077ced47eb1499e29a443c"}, - {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:a5263e363c27b653a90078143adb3d076c1a748ec9ecc78ea2fb916f9b861962"}, - {file = "cffi-1.15.0-cp310-cp310-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:f54a64f8b0c8ff0b64d18aa76675262e1700f3995182267998c31ae974fbc382"}, - {file = "cffi-1.15.0-cp310-cp310-win32.whl", hash = "sha256:c21c9e3896c23007803a875460fb786118f0cdd4434359577ea25eb556e34c55"}, - {file = "cffi-1.15.0-cp310-cp310-win_amd64.whl", hash = "sha256:5e069f72d497312b24fcc02073d70cb989045d1c91cbd53979366077959933e0"}, - {file = "cffi-1.15.0-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:64d4ec9f448dfe041705426000cc13e34e6e5bb13736e9fd62e34a0b0c41566e"}, - {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2756c88cbb94231c7a147402476be2c4df2f6078099a6f4a480d239a8817ae39"}, - {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3b96a311ac60a3f6be21d2572e46ce67f09abcf4d09344c49274eb9e0bf345fc"}, - {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:75e4024375654472cc27e91cbe9eaa08567f7fbdf822638be2814ce059f58032"}, - {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.whl", hash = "sha256:59888172256cac5629e60e72e86598027aca6bf01fa2465bdb676d37636573e8"}, - {file = "cffi-1.15.0-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.whl", hash = "sha256:27c219baf94952ae9d50ec19651a687b826792055353d07648a5695413e0c605"}, - {file = "cffi-1.15.0-cp36-cp36m-win32.whl", hash = "sha256:4958391dbd6249d7ad855b9ca88fae690783a6be9e86df65865058ed81fc860e"}, - {file = "cffi-1.15.0-cp36-cp36m-win_amd64.whl", hash = "sha256:f6f824dc3bce0edab5f427efcfb1d63ee75b6fcb7282900ccaf925be84efb0fc"}, - {file = "cffi-1.15.0-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:06c48159c1abed75c2e721b1715c379fa3200c7784271b3c46df01383b593636"}, - {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:c2051981a968d7de9dd2d7b87bcb9c939c74a34626a6e2f8181455dd49ed69e4"}, - {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:fd8a250edc26254fe5b33be00402e6d287f562b6a5b2152dec302fa15bb3e997"}, - {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:91d77d2a782be4274da750752bb1650a97bfd8f291022b379bb8e01c66b4e96b"}, - {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:45db3a33139e9c8f7c09234b5784a5e33d31fd6907800b316decad50af323ff2"}, - {file = "cffi-1.15.0-cp37-cp37m-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:263cc3d821c4ab2213cbe8cd8b355a7f72a8324577dc865ef98487c1aeee2bc7"}, - {file = "cffi-1.15.0-cp37-cp37m-win32.whl", hash = "sha256:17771976e82e9f94976180f76468546834d22a7cc404b17c22df2a2c81db0c66"}, - {file = "cffi-1.15.0-cp37-cp37m-win_amd64.whl", hash = "sha256:3415c89f9204ee60cd09b235810be700e993e343a408693e80ce7f6a40108029"}, - {file = "cffi-1.15.0-cp38-cp38-macosx_10_9_x86_64.whl", hash = "sha256:4238e6dab5d6a8ba812de994bbb0a79bddbdf80994e4ce802b6f6f3142fcc880"}, - {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0808014eb713677ec1292301ea4c81ad277b6cdf2fdd90fd540af98c0b101d20"}, - {file = "cffi-1.15.0-cp38-cp38-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:57e9ac9ccc3101fac9d6014fba037473e4358ef4e89f8e181f8951a2c0162024"}, - {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:8b6c2ea03845c9f501ed1313e78de148cd3f6cad741a75d43a29b43da27f2e1e"}, - {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:10dffb601ccfb65262a27233ac273d552ddc4d8ae1bf93b21c94b8511bffe728"}, - {file = "cffi-1.15.0-cp38-cp38-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:786902fb9ba7433aae840e0ed609f45c7bcd4e225ebb9c753aa39725bb3e6ad6"}, - {file = "cffi-1.15.0-cp38-cp38-win32.whl", hash = "sha256:da5db4e883f1ce37f55c667e5c0de439df76ac4cb55964655906306918e7363c"}, - {file = "cffi-1.15.0-cp38-cp38-win_amd64.whl", hash = "sha256:181dee03b1170ff1969489acf1c26533710231c58f95534e3edac87fff06c443"}, - {file = "cffi-1.15.0-cp39-cp39-macosx_10_9_x86_64.whl", hash = "sha256:45e8636704eacc432a206ac7345a5d3d2c62d95a507ec70d62f23cd91770482a"}, - {file = "cffi-1.15.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:31fb708d9d7c3f49a60f04cf5b119aeefe5644daba1cd2a0fe389b674fd1de37"}, - {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:6dc2737a3674b3e344847c8686cf29e500584ccad76204efea14f451d4cc669a"}, - {file = "cffi-1.15.0-cp39-cp39-manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:74fdfdbfdc48d3f47148976f49fab3251e550a8720bebc99bf1483f5bfb5db3e"}, - {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:ffaa5c925128e29efbde7301d8ecaf35c8c60ffbcd6a1ffd3a552177c8e5e796"}, - {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:3f7d084648d77af029acb79a0ff49a0ad7e9d09057a9bf46596dac9514dc07df"}, - {file = "cffi-1.15.0-cp39-cp39-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:ef1f279350da2c586a69d32fc8733092fd32cc8ac95139a00377841f59a3f8d8"}, - {file = "cffi-1.15.0-cp39-cp39-win32.whl", hash = "sha256:2a23af14f408d53d5e6cd4e3d9a24ff9e05906ad574822a10563efcef137979a"}, - {file = "cffi-1.15.0-cp39-cp39-win_amd64.whl", hash = "sha256:3773c4d81e6e818df2efbc7dd77325ca0dcb688116050fb2b3011218eda36139"}, - {file = "cffi-1.15.0.tar.gz", hash = "sha256:920f0d66a896c2d99f0adbb391f990a84091179542c205fa53ce5787aff87954"}, -] click = [ {file = "click-7.1.2-py2.py3-none-any.whl", hash = "sha256:dacca89f4bfadd5de3d7489b7c8a566eee0d3676333fbb50030263894c38c0dc"}, {file = "click-7.1.2.tar.gz", hash = "sha256:d2b5255c7c6349bc1bd1e59e08cd12acbbd63ce649f2588755783aa94dfb6b1a"}, @@ -611,14 +392,6 @@ colorful = [ {file = "colorful-0.5.4-py2.py3-none-any.whl", hash = "sha256:8d264b52a39aae4c0ba3e2a46afbaec81b0559a99be0d2cfe2aba4cf94531348"}, {file = "colorful-0.5.4.tar.gz", hash = "sha256:86848ad4e2eda60cd2519d8698945d22f6f6551e23e95f3f14dfbb60997807ea"}, ] -cssselect2 = [ - {file = "cssselect2-0.4.1-py3-none-any.whl", hash = "sha256:2f4a9f20965367bae459e3bb42561f7927e0cfe5b7ea1692757cf67ef5d7dace"}, - {file = "cssselect2-0.4.1.tar.gz", hash = "sha256:93fbb9af860e95dd40bf18c3b2b6ed99189a07c0f29ba76f9c5be71344664ec8"}, -] -defusedxml = [ - {file = "defusedxml-0.7.1-py2.py3-none-any.whl", hash = "sha256:a352e7e428770286cc899e2542b6cdaedb2b4953ff269a210103ec58f6198a61"}, - {file = "defusedxml-0.7.1.tar.gz", hash = "sha256:1bb3032db185915b62d7c6209c5a8792be6a32ab2fedacc84e01b52c51aa3e69"}, -] flask = [ {file = "Flask-1.1.4-py2.py3-none-any.whl", hash = "sha256:c34f04500f2cbbea882b1acb02002ad6fe6b7ffa64a6164577995657f50aed22"}, {file = "Flask-1.1.4.tar.gz", hash = "sha256:0fbeb6180d383a9186d0d6ed954e0042ad9f18e0e8de088b2b419d526927d196"}, @@ -680,10 +453,6 @@ greenlet = [ {file = "greenlet-1.1.2-cp39-cp39-win_amd64.whl", hash = "sha256:013d61294b6cd8fe3242932c1c5e36e5d1db2c8afb58606c5a67efce62c1f5fd"}, {file = "greenlet-1.1.2.tar.gz", hash = "sha256:e30f5ea4ae2346e62cedde8794a56858a67b878dd79f7df76a0767e356b1744a"}, ] -html5lib = [ - {file = "html5lib-1.1-py2.py3-none-any.whl", hash = "sha256:0d78f8fde1c230e99fe37986a60526d7049ed4bf8a9fadbad5f00e22e58e041d"}, - {file = "html5lib-1.1.tar.gz", hash = "sha256:b2e5b40261e20f354d198eae92afc10d750afb487ed5e50f9c4eaf07c184146f"}, -] importlib-metadata = [ {file = "importlib_metadata-4.8.3-py3-none-any.whl", hash = "sha256:65a9576a5b2d58ca44d133c42a241905cc45e34d2c06fd5ba2bafa221e5d7b5e"}, {file = "importlib_metadata-4.8.3.tar.gz", hash = "sha256:766abffff765960fcc18003801f7044eb6755ffae4521c8e8ce8e83b9c9b0668"}, @@ -710,6 +479,9 @@ markupsafe = [ {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:2d7d807855b419fc2ed3e631034685db6079889a1f01d5d9dac950f764da3dad"}, {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:add36cb2dbb8b736611303cd3bfcee00afd96471b09cda130da3581cbdc56a6d"}, {file = "MarkupSafe-2.0.1-cp310-cp310-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:168cd0a3642de83558a5153c8bd34f175a9a6e7f6dc6384b9655d2697312a646"}, + {file = "MarkupSafe-2.0.1-cp310-cp310-musllinux_1_1_aarch64.whl", hash = "sha256:4dc8f9fb58f7364b63fd9f85013b780ef83c11857ae79f2feda41e270468dd9b"}, + {file = "MarkupSafe-2.0.1-cp310-cp310-musllinux_1_1_i686.whl", hash = "sha256:20dca64a3ef2d6e4d5d615a3fd418ad3bde77a47ec8a23d984a12b5b4c74491a"}, + {file = "MarkupSafe-2.0.1-cp310-cp310-musllinux_1_1_x86_64.whl", hash = "sha256:cdfba22ea2f0029c9261a4bd07e830a8da012291fbe44dc794e488b6c9bb353a"}, {file = "MarkupSafe-2.0.1-cp310-cp310-win32.whl", hash = "sha256:99df47edb6bda1249d3e80fdabb1dab8c08ef3975f69aed437cb69d0a5de1e28"}, {file = "MarkupSafe-2.0.1-cp310-cp310-win_amd64.whl", hash = "sha256:e0f138900af21926a02425cf736db95be9f4af72ba1bb21453432a07f6082134"}, {file = "MarkupSafe-2.0.1-cp36-cp36m-macosx_10_9_x86_64.whl", hash = "sha256:f9081981fe268bd86831e5c75f7de206ef275defcb82bc70740ae6dc507aee51"}, @@ -721,6 +493,9 @@ markupsafe = [ {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:bf5d821ffabf0ef3533c39c518f3357b171a1651c1ff6827325e4489b0e46c3c"}, {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:0d4b31cc67ab36e3392bbf3862cfbadac3db12bdd8b02a2731f509ed5b829724"}, {file = "MarkupSafe-2.0.1-cp36-cp36m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:baa1a4e8f868845af802979fcdbf0bb11f94f1cb7ced4c4b8a351bb60d108145"}, + {file = "MarkupSafe-2.0.1-cp36-cp36m-musllinux_1_1_aarch64.whl", hash = "sha256:deb993cacb280823246a026e3b2d81c493c53de6acfd5e6bfe31ab3402bb37dd"}, + {file = "MarkupSafe-2.0.1-cp36-cp36m-musllinux_1_1_i686.whl", hash = "sha256:63f3268ba69ace99cab4e3e3b5840b03340efed0948ab8f78d2fd87ee5442a4f"}, + {file = "MarkupSafe-2.0.1-cp36-cp36m-musllinux_1_1_x86_64.whl", hash = "sha256:8d206346619592c6200148b01a2142798c989edcb9c896f9ac9722a99d4e77e6"}, {file = "MarkupSafe-2.0.1-cp36-cp36m-win32.whl", hash = "sha256:6c4ca60fa24e85fe25b912b01e62cb969d69a23a5d5867682dd3e80b5b02581d"}, {file = "MarkupSafe-2.0.1-cp36-cp36m-win_amd64.whl", hash = "sha256:b2f4bf27480f5e5e8ce285a8c8fd176c0b03e93dcc6646477d4630e83440c6a9"}, {file = "MarkupSafe-2.0.1-cp37-cp37m-macosx_10_9_x86_64.whl", hash = "sha256:0717a7390a68be14b8c793ba258e075c6f4ca819f15edfc2a3a027c823718567"}, @@ -732,6 +507,9 @@ markupsafe = [ {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:e9936f0b261d4df76ad22f8fee3ae83b60d7c3e871292cd42f40b81b70afae85"}, {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:2a7d351cbd8cfeb19ca00de495e224dea7e7d919659c2841bbb7f420ad03e2d6"}, {file = "MarkupSafe-2.0.1-cp37-cp37m-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:60bf42e36abfaf9aff1f50f52644b336d4f0a3fd6d8a60ca0d054ac9f713a864"}, + {file = "MarkupSafe-2.0.1-cp37-cp37m-musllinux_1_1_aarch64.whl", hash = "sha256:d6c7ebd4e944c85e2c3421e612a7057a2f48d478d79e61800d81468a8d842207"}, + {file = "MarkupSafe-2.0.1-cp37-cp37m-musllinux_1_1_i686.whl", hash = "sha256:f0567c4dc99f264f49fe27da5f735f414c4e7e7dd850cfd8e69f0862d7c74ea9"}, + {file = "MarkupSafe-2.0.1-cp37-cp37m-musllinux_1_1_x86_64.whl", hash = "sha256:89c687013cb1cd489a0f0ac24febe8c7a666e6e221b783e53ac50ebf68e45d86"}, {file = "MarkupSafe-2.0.1-cp37-cp37m-win32.whl", hash = "sha256:a30e67a65b53ea0a5e62fe23682cfe22712e01f453b95233b25502f7c61cb415"}, {file = "MarkupSafe-2.0.1-cp37-cp37m-win_amd64.whl", hash = "sha256:611d1ad9a4288cf3e3c16014564df047fe08410e628f89805e475368bd304914"}, {file = "MarkupSafe-2.0.1-cp38-cp38-macosx_10_9_universal2.whl", hash = "sha256:5bb28c636d87e840583ee3adeb78172efc47c8b26127267f54a9c0ec251d41a9"}, @@ -744,6 +522,9 @@ markupsafe = [ {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6fcf051089389abe060c9cd7caa212c707e58153afa2c649f00346ce6d260f1b"}, {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:5855f8438a7d1d458206a2466bf82b0f104a3724bf96a1c781ab731e4201731a"}, {file = "MarkupSafe-2.0.1-cp38-cp38-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:3dd007d54ee88b46be476e293f48c85048603f5f516008bee124ddd891398ed6"}, + {file = "MarkupSafe-2.0.1-cp38-cp38-musllinux_1_1_aarch64.whl", hash = "sha256:aca6377c0cb8a8253e493c6b451565ac77e98c2951c45f913e0b52facdcff83f"}, + {file = "MarkupSafe-2.0.1-cp38-cp38-musllinux_1_1_i686.whl", hash = "sha256:04635854b943835a6ea959e948d19dcd311762c5c0c6e1f0e16ee57022669194"}, + {file = "MarkupSafe-2.0.1-cp38-cp38-musllinux_1_1_x86_64.whl", hash = "sha256:6300b8454aa6930a24b9618fbb54b5a68135092bc666f7b06901f897fa5c2fee"}, {file = "MarkupSafe-2.0.1-cp38-cp38-win32.whl", hash = "sha256:023cb26ec21ece8dc3907c0e8320058b2e0cb3c55cf9564da612bc325bed5e64"}, {file = "MarkupSafe-2.0.1-cp38-cp38-win_amd64.whl", hash = "sha256:984d76483eb32f1bcb536dc27e4ad56bba4baa70be32fa87152832cdd9db0833"}, {file = "MarkupSafe-2.0.1-cp39-cp39-macosx_10_9_universal2.whl", hash = "sha256:2ef54abee730b502252bcdf31b10dacb0a416229b72c18b19e24a4509f273d26"}, @@ -756,6 +537,9 @@ markupsafe = [ {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c47adbc92fc1bb2b3274c4b3a43ae0e4573d9fbff4f54cd484555edbf030baf1"}, {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_5_i686.manylinux1_i686.manylinux_2_12_i686.manylinux2010_i686.whl", hash = "sha256:37205cac2a79194e3750b0af2a5720d95f786a55ce7df90c3af697bfa100eaac"}, {file = "MarkupSafe-2.0.1-cp39-cp39-manylinux_2_5_x86_64.manylinux1_x86_64.manylinux_2_12_x86_64.manylinux2010_x86_64.whl", hash = "sha256:1f2ade76b9903f39aa442b4aadd2177decb66525062db244b35d71d0ee8599b6"}, + {file = "MarkupSafe-2.0.1-cp39-cp39-musllinux_1_1_aarch64.whl", hash = "sha256:4296f2b1ce8c86a6aea78613c34bb1a672ea0e3de9c6ba08a960efe0b0a09047"}, + {file = "MarkupSafe-2.0.1-cp39-cp39-musllinux_1_1_i686.whl", hash = "sha256:9f02365d4e99430a12647f09b6cc8bab61a6564363f313126f775eb4f6ef798e"}, + {file = "MarkupSafe-2.0.1-cp39-cp39-musllinux_1_1_x86_64.whl", hash = "sha256:5b6d930f030f8ed98e3e6c98ffa0652bdb82601e7a016ec2ab5d7ff23baa78d1"}, {file = "MarkupSafe-2.0.1-cp39-cp39-win32.whl", hash = "sha256:10f82115e21dc0dfec9ab5c0223652f7197feb168c940f3ef61563fc2d6beb74"}, {file = "MarkupSafe-2.0.1-cp39-cp39-win_amd64.whl", hash = "sha256:693ce3f9e70a6cf7d2fb9e6c9d8b204b6b39897a2c4a1aa65728d5ac97dcc1d8"}, {file = "MarkupSafe-2.0.1.tar.gz", hash = "sha256:594c67807fb16238b30c44bdf74f36c02cdf22d1c8cda91ef8a0ed8dabf5620a"}, @@ -768,49 +552,6 @@ packaging = [ {file = "packaging-21.3-py3-none-any.whl", hash = "sha256:ef103e05f519cdc783ae24ea4e2e0f508a9c99b2d4969652eed6a2e1ea5bd522"}, {file = "packaging-21.3.tar.gz", hash = "sha256:dd47c42927d89ab911e606518907cc2d3a1f38bbd026385970643f9c5b8ecfeb"}, ] -pillow = [ - {file = "Pillow-8.4.0-cp310-cp310-macosx_10_10_universal2.whl", hash = "sha256:81f8d5c81e483a9442d72d182e1fb6dcb9723f289a57e8030811bac9ea3fef8d"}, - {file = "Pillow-8.4.0-cp310-cp310-macosx_11_0_arm64.whl", hash = "sha256:3f97cfb1e5a392d75dd8b9fd274d205404729923840ca94ca45a0af57e13dbe6"}, - {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:eb9fc393f3c61f9054e1ed26e6fe912c7321af2f41ff49d3f83d05bacf22cc78"}, - {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:d82cdb63100ef5eedb8391732375e6d05993b765f72cb34311fab92103314649"}, - {file = "Pillow-8.4.0-cp310-cp310-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:62cc1afda735a8d109007164714e73771b499768b9bb5afcbbee9d0ff374b43f"}, - {file = "Pillow-8.4.0-cp310-cp310-win32.whl", hash = "sha256:e3dacecfbeec9a33e932f00c6cd7996e62f53ad46fbe677577394aaa90ee419a"}, - {file = "Pillow-8.4.0-cp310-cp310-win_amd64.whl", hash = "sha256:620582db2a85b2df5f8a82ddeb52116560d7e5e6b055095f04ad828d1b0baa39"}, - {file = "Pillow-8.4.0-cp36-cp36m-macosx_10_10_x86_64.whl", hash = "sha256:1bc723b434fbc4ab50bb68e11e93ce5fb69866ad621e3c2c9bdb0cd70e345f55"}, - {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:72cbcfd54df6caf85cc35264c77ede902452d6df41166010262374155947460c"}, - {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:70ad9e5c6cb9b8487280a02c0ad8a51581dcbbe8484ce058477692a27c151c0a"}, - {file = "Pillow-8.4.0-cp36-cp36m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:25a49dc2e2f74e65efaa32b153527fc5ac98508d502fa46e74fa4fd678ed6645"}, - {file = "Pillow-8.4.0-cp36-cp36m-win32.whl", hash = "sha256:93ce9e955cc95959df98505e4608ad98281fff037350d8c2671c9aa86bcf10a9"}, - {file = "Pillow-8.4.0-cp36-cp36m-win_amd64.whl", hash = "sha256:2e4440b8f00f504ee4b53fe30f4e381aae30b0568193be305256b1462216feff"}, - {file = "Pillow-8.4.0-cp37-cp37m-macosx_10_10_x86_64.whl", hash = "sha256:8c803ac3c28bbc53763e6825746f05cc407b20e4a69d0122e526a582e3b5e153"}, - {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:c8a17b5d948f4ceeceb66384727dde11b240736fddeda54ca740b9b8b1556b29"}, - {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:1394a6ad5abc838c5cd8a92c5a07535648cdf6d09e8e2d6df916dfa9ea86ead8"}, - {file = "Pillow-8.4.0-cp37-cp37m-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:792e5c12376594bfcb986ebf3855aa4b7c225754e9a9521298e460e92fb4a488"}, - {file = "Pillow-8.4.0-cp37-cp37m-win32.whl", hash = "sha256:d99ec152570e4196772e7a8e4ba5320d2d27bf22fdf11743dd882936ed64305b"}, - {file = "Pillow-8.4.0-cp37-cp37m-win_amd64.whl", hash = "sha256:7b7017b61bbcdd7f6363aeceb881e23c46583739cb69a3ab39cb384f6ec82e5b"}, - {file = "Pillow-8.4.0-cp38-cp38-macosx_10_10_x86_64.whl", hash = "sha256:d89363f02658e253dbd171f7c3716a5d340a24ee82d38aab9183f7fdf0cdca49"}, - {file = "Pillow-8.4.0-cp38-cp38-macosx_11_0_arm64.whl", hash = "sha256:0a0956fdc5defc34462bb1c765ee88d933239f9a94bc37d132004775241a7585"}, - {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:5b7bb9de00197fb4261825c15551adf7605cf14a80badf1761d61e59da347779"}, - {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:72b9e656e340447f827885b8d7a15fc8c4e68d410dc2297ef6787eec0f0ea409"}, - {file = "Pillow-8.4.0-cp38-cp38-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:a5a4532a12314149d8b4e4ad8ff09dde7427731fcfa5917ff16d0291f13609df"}, - {file = "Pillow-8.4.0-cp38-cp38-win32.whl", hash = "sha256:82aafa8d5eb68c8463b6e9baeb4f19043bb31fefc03eb7b216b51e6a9981ae09"}, - {file = "Pillow-8.4.0-cp38-cp38-win_amd64.whl", hash = "sha256:066f3999cb3b070a95c3652712cffa1a748cd02d60ad7b4e485c3748a04d9d76"}, - {file = "Pillow-8.4.0-cp39-cp39-macosx_10_10_x86_64.whl", hash = "sha256:5503c86916d27c2e101b7f71c2ae2cddba01a2cf55b8395b0255fd33fa4d1f1a"}, - {file = "Pillow-8.4.0-cp39-cp39-macosx_11_0_arm64.whl", hash = "sha256:4acc0985ddf39d1bc969a9220b51d94ed51695d455c228d8ac29fcdb25810e6e"}, - {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:0b052a619a8bfcf26bd8b3f48f45283f9e977890263e4571f2393ed8898d331b"}, - {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:493cb4e415f44cd601fcec11c99836f707bb714ab03f5ed46ac25713baf0ff20"}, - {file = "Pillow-8.4.0-cp39-cp39-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b8831cb7332eda5dc89b21a7bce7ef6ad305548820595033a4b03cf3091235ed"}, - {file = "Pillow-8.4.0-cp39-cp39-win32.whl", hash = "sha256:5e9ac5f66616b87d4da618a20ab0a38324dbe88d8a39b55be8964eb520021e02"}, - {file = "Pillow-8.4.0-cp39-cp39-win_amd64.whl", hash = "sha256:3eb1ce5f65908556c2d8685a8f0a6e989d887ec4057326f6c22b24e8a172c66b"}, - {file = "Pillow-8.4.0-pp36-pypy36_pp73-macosx_10_10_x86_64.whl", hash = "sha256:ddc4d832a0f0b4c52fff973a0d44b6c99839a9d016fe4e6a1cb8f3eea96479c2"}, - {file = "Pillow-8.4.0-pp36-pypy36_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:9a3e5ddc44c14042f0844b8cf7d2cd455f6cc80fd7f5eefbe657292cf601d9ad"}, - {file = "Pillow-8.4.0-pp36-pypy36_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:c70e94281588ef053ae8998039610dbd71bc509e4acbc77ab59d7d2937b10698"}, - {file = "Pillow-8.4.0-pp37-pypy37_pp73-macosx_10_10_x86_64.whl", hash = "sha256:3862b7256046fcd950618ed22d1d60b842e3a40a48236a5498746f21189afbbc"}, - {file = "Pillow-8.4.0-pp37-pypy37_pp73-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:a4901622493f88b1a29bd30ec1a2f683782e57c3c16a2dbc7f2595ba01f639df"}, - {file = "Pillow-8.4.0-pp37-pypy37_pp73-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:84c471a734240653a0ec91dec0996696eea227eafe72a33bd06c92697728046b"}, - {file = "Pillow-8.4.0-pp37-pypy37_pp73-win_amd64.whl", hash = "sha256:244cf3b97802c34c41905d22810846802a3329ddcb93ccc432870243211c79fc"}, - {file = "Pillow-8.4.0.tar.gz", hash = "sha256:b8e2f83c56e141920c39464b852de3719dfbfb6e3c99a2d8da0edf4fb33176ed"}, -] pluggy = [ {file = "pluggy-1.0.0-py2.py3-none-any.whl", hash = "sha256:74134bbf457f031a36d68416e1509f34bd5ccc019f0bcc952c7b909d06b37bd3"}, {file = "pluggy-1.0.0.tar.gz", hash = "sha256:4224373bacce55f955a878bf9cfa763c1e360858e330072059e10bad68531159"}, @@ -826,18 +567,10 @@ py = [ pycountry = [ {file = "pycountry-20.7.3.tar.gz", hash = "sha256:81084a53d3454344c0292deebc20fcd0a1488c136d4900312cbd465cf552cb42"}, ] -pycparser = [ - {file = "pycparser-2.21-py2.py3-none-any.whl", hash = "sha256:8ee45429555515e1f6b185e78100aea234072576aa43ab53aefcae078162fca9"}, - {file = "pycparser-2.21.tar.gz", hash = "sha256:e644fdec12f7872f86c58ff790da456218b10f863970249516d60a5eaca77206"}, -] pyparsing = [ {file = "pyparsing-3.0.6-py3-none-any.whl", hash = "sha256:04ff808a5b90911829c55c4e26f75fa5ca8a2f5f36aa3a51f68e27033341d3e4"}, {file = "pyparsing-3.0.6.tar.gz", hash = "sha256:d9bdec0013ef1eb5a84ab39a3b3868911598afa494f5faa038647101504e2b81"}, ] -pyphen = [ - {file = "pyphen-0.11.0-py3-none-any.whl", hash = "sha256:e3c1b1d05deaa31acdd78e2e24005402358feb7a4b407c4af7e5cc2e41c4d608"}, - {file = "pyphen-0.11.0.tar.gz", hash = "sha256:e2c3ed82c3a04317df5102addafe89652b0876bc6c6265f5dd4c3efaf02315e8"}, -] pytest = [ {file = "pytest-6.2.5-py3-none-any.whl", hash = "sha256:7310f8d27bc79ced999e760ca304d69f6ba6c6649c0b60fb0e04a4a77cacc134"}, {file = "pytest-6.2.5.tar.gz", hash = "sha256:131b36680866a76e6781d13f101efb86cf674ebb9762eb70d3082b6f29889e89"}, @@ -922,10 +655,6 @@ sqlalchemy = [ {file = "SQLAlchemy-1.4.28-cp39-cp39-win_amd64.whl", hash = "sha256:853de08e881dae0305647dd61b4429758f11d1bf02a9faf02793cad44bb2e0d5"}, {file = "SQLAlchemy-1.4.28.tar.gz", hash = "sha256:7fdb7b775fb0739d3e71461509f978beb788935bc0aa9e47df14837cb33e5226"}, ] -tinycss2 = [ - {file = "tinycss2-1.1.1-py3-none-any.whl", hash = "sha256:fe794ceaadfe3cf3e686b22155d0da5780dd0e273471a51846d0a02bc204fec8"}, - {file = "tinycss2-1.1.1.tar.gz", hash = "sha256:b2e44dd8883c360c35dd0d1b5aad0b610e5156c2cb3b33434634e539ead9d8bf"}, -] toml = [ {file = "toml-0.10.2-py2.py3-none-any.whl", hash = "sha256:806143ae5bfb6a3c6e736a764057db0e6a0e05e338b5630894a5f779cabb4f9b"}, {file = "toml-0.10.2.tar.gz", hash = "sha256:b3bda1d108d5dd99f4a20d24d9c348e91c4db7ab1b749200bded2f839ccbe68f"}, @@ -938,14 +667,6 @@ wcwidth = [ {file = "wcwidth-0.2.5-py2.py3-none-any.whl", hash = "sha256:beb4802a9cebb9144e99086eff703a642a13d6a0052920003a230f3294bbe784"}, {file = "wcwidth-0.2.5.tar.gz", hash = "sha256:c4d647b99872929fdb7bdcaa4fbe7f01413ed3d98077df798530e5b04f116c83"}, ] -weasyprint = [ - {file = "WeasyPrint-52.5-py3-none-any.whl", hash = "sha256:3433d657049a65d7d63f545fd71f5efa8aae7f05d24e49e01a757973fd3799f1"}, - {file = "WeasyPrint-52.5.tar.gz", hash = "sha256:b37ea02d75ca04babd7becad7341426be332ae560d8f02d664bfa1e9afb18481"}, -] -webencodings = [ - {file = "webencodings-0.5.1-py2.py3-none-any.whl", hash = "sha256:a0af1213f3c2226497a97e2b3aa01a7e4bee4f403f95be16fc9acd2947514a78"}, - {file = "webencodings-0.5.1.tar.gz", hash = "sha256:b36a1c245f2d304965eb4e0a82848379241dc04b865afcc4aab16748587e1923"}, -] werkzeug = [ {file = "Werkzeug-1.0.1-py2.py3-none-any.whl", hash = "sha256:2de2a5db0baeae7b2d2664949077c2ac63fbd16d98da0ff71837f7d1dea3fd43"}, {file = "Werkzeug-1.0.1.tar.gz", hash = "sha256:6c80b1e5ad3665290ea39320b91e1be1e0d5f60652b964a3070216de83d2e47c"}, diff --git a/pyproject.toml b/pyproject.toml index a7aa0f0..ed28daa 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -17,7 +17,6 @@ schwifty = "^2020.9.0" prompt-toolkit = "^3.0.8" pytest = "^6.1.2" python-dateutil = "^2.8.1" -WeasyPrint = "^52.4" [tool.poetry.dev-dependencies] @@ -28,3 +27,14 @@ build-backend = "poetry.core.masonry.api" [tool.poetry.scripts] schmeckels = "schmeckels.cli:main" + +[tool.isort] +profile = "black" +multi_line_output = 3 +line_length = 120 +default_section = "THIRDPARTY" +known_first_party = [] +known_third_party = [] + +[tool.black] +line-length = 120 diff --git a/schmeckels/autosort.py b/schmeckels/autosort.py index 33b76f4..bbc9439 100644 --- a/schmeckels/autosort.py +++ b/schmeckels/autosort.py @@ -1,22 +1,27 @@ #! /usr/bin/env python3 -from sqlalchemy import create_engine -from sqlalchemy.orm import sessionmaker -from sqlalchemy.ext.declarative import declarative_base -from schmeckels.models import Tag, Transaction -from schmeckels.helper import get_session, create_tag, get_rules import re import sys + import click import colorful as cf +from sqlalchemy import create_engine +from sqlalchemy.ext.declarative import declarative_base +from sqlalchemy.orm import sessionmaker + +from schmeckels.helper import create_tag, get_rules, get_session +from schmeckels.models import Tag, Transaction @click.command(name="autosort") -@click.option("--profile", "-p") @click.option("--dry-run", default=False, is_flag=True) @click.option("--verbose", "-v", default=False, is_flag=True) -def command(profile, dry_run, verbose): - session = get_session(profile) - rules = get_rules(profile) +def command(dry_run, verbose): + session = get_session() + rules = get_rules() + + if not rules: + print(cf.red("Empty rules file.")) + sys.exit(0) unsorted = session.query(Transaction).filter(Transaction.tags == None).all() print(cf.yellow("Found {} unsorted transcations".format(len(unsorted)))) @@ -33,7 +38,7 @@ def command(profile, dry_run, verbose): for tag_label in taglist: tag = session.query(Tag).filter(Tag.name == tag_label).first() if not tag and not dry_run: - tag = create_tag(tag_label, profile, session) + tag = create_tag(tag_label, session) transaction.tags.append(tag) new.append(transaction) diff --git a/schmeckels/banks.py b/schmeckels/banks.py index c5fd4bd..703dfeb 100644 --- a/schmeckels/banks.py +++ b/schmeckels/banks.py @@ -1,9 +1,11 @@ #! /usr/bin/env python3 -import mt940 import csv -from schmeckels import models from datetime import datetime +import mt940 + +from schmeckels import models + SUPPORTED_BANKS = ( ("dkb", "DKB"), ("sparkasse-mt940", "Sparkasse MT940"), diff --git a/schmeckels/categories.py b/schmeckels/categories.py index d60c07b..3436fda 100644 --- a/schmeckels/categories.py +++ b/schmeckels/categories.py @@ -1,12 +1,13 @@ #! /usr/bin/env python3 -from sqlalchemy import create_engine -from sqlalchemy.orm import sessionmaker -from sqlalchemy.ext.declarative import declarative_base -from schmeckels.models import * import sys from prompt_toolkit.completion import FuzzyWordCompleter from prompt_toolkit.shortcuts import prompt +from sqlalchemy import create_engine +from sqlalchemy.ext.declarative import declarative_base +from sqlalchemy.orm import sessionmaker + +from schmeckels.models import * engine = create_engine("sqlite:///app.db") Base = declarative_base() diff --git a/schmeckels/cli.py b/schmeckels/cli.py index e1e35bc..0df4ce7 100644 --- a/schmeckels/cli.py +++ b/schmeckels/cli.py @@ -1,17 +1,15 @@ #! /usr/bin/env python3 -import click import os import sys from pathlib import Path - -from schmeckels import importer, serve, autosort, validate, info, sort, models, report - -from schmeckels.helper import build_database_filename, create_dirs, build_rules_filename +import click from sqlalchemy import create_engine +from sqlalchemy.ext.declarative import declarative_base from sqlalchemy.orm import sessionmaker -from sqlalchemy.ext.declarative import declarative_base +from schmeckels import autosort, importer, info, models, serve, sort, validate +from schmeckels.helper import build_database_filename, build_rules_filename, get_data_dir __version__ = "0.0.1" @@ -22,15 +20,20 @@ def cli(): @cli.command(name="init") -@click.argument("profile_name", required=True) -def init(profile_name): - create_dirs() +def init(): + data_dir = get_data_dir() - db_path = build_database_filename(profile_name) - rule_path = build_rules_filename(profile_name) + # make sure folder exists + try: + os.mkdir(data_dir) + except FileExistsError: + pass - if os.path.exists(db_path) and os.path.exists(rule_path): - print(f"Profile '{profile_name}' already exists") + db_path = build_database_filename(data_dir) + rule_path = build_rules_filename(data_dir) + + if os.path.exists(db_path) or os.path.exists(rule_path): + print(f"Init already ran") sys.exit(1) else: # database @@ -39,7 +42,7 @@ def init(profile_name): # rules Path(rule_path).touch() - print(f"Sucessfully create the profile '{profile_name}'") + print(f"Sucessfull init at {data_dir}'") def main(): @@ -49,5 +52,5 @@ def main(): cli.add_command(autosort.command) cli.add_command(validate.command) cli.add_command(info.command) - cli.add_command(report.command) + #cli.add_command(report.command) cli() diff --git a/schmeckels/helper.py b/schmeckels/helper.py index a58b660..ded6895 100644 --- a/schmeckels/helper.py +++ b/schmeckels/helper.py @@ -1,13 +1,15 @@ #! /usr/bin/env python3 +import locale import os -import sys -import yaml import re -from xdg import XDG_CONFIG_HOME, XDG_DATA_HOME +import sys + +import yaml from sqlalchemy import create_engine from sqlalchemy.orm import sessionmaker +from xdg import XDG_CONFIG_HOME, XDG_DATA_HOME + import schmeckels.models -import locale from schmeckels import banks, models try: @@ -16,13 +18,9 @@ except ImportError: from yaml import Loader -CONFIG_DIR = os.path.join(XDG_CONFIG_HOME, "schmeckels") -DATA_DIR = os.path.join(XDG_DATA_HOME, "schmeckels") - - def format_amount(d): if type(d) is dict: - for k,v in d.items(): + for k, v in d.items(): amount = v / 100 d[k] = f"{amount:,.2f}" return d @@ -34,69 +32,50 @@ def format_amount(d): raise Exception() - -def create_dirs(): - for directory in [CONFIG_DIR, DATA_DIR]: - try: - os.mkdir(directory) - except FileExistsError: - pass +def get_data_dir(): + fallback_dir = os.path.join(XDG_DATA_HOME, "schmeckels") + return os.getenv("SCHMECKELS_DIR", fallback_dir) -def build_database_filename(profile_name): - return f"{DATA_DIR}/{profile_name}.db" +def build_database_filename(base_path): + return f"{base_path}/schmeckels.db" -def build_rules_filename(profile_name): - return f"{CONFIG_DIR}/{profile_name}.yaml" +def build_rules_filename(base_path): + return f"{base_path}/rules.yaml" -def check_single_profile(): - files = os.listdir(DATA_DIR) - if len(files) == 1: - return files[0].split(".")[0] - else: - print("--profile is required when you have more than one profile.") - sys.exit(1) - - -def list_profiles(): - files = os.listdir(DATA_DIR) - return [x.split(".")[0] for x in files] - - -def get_session(profile_name): - if not profile_name: - profile_name = check_single_profile() - filename = build_database_filename(profile_name) +def get_session(): + data_dir = get_data_dir() + filename = build_database_filename(data_dir) if os.path.exists(filename) and os.path.isfile(filename): engine = create_engine(f"sqlite:///{filename}") Session = sessionmaker(bind=engine) return Session() else: - print(f"No database for profile '{profile_name}'. Did you run 'init'?") + print(f"No database found in {data_dir}! Did you run 'init'?") sys.exit(1) -def get_rules(profile_name): - if not profile_name: - profile_name = check_single_profile() - filename = build_rules_filename(profile_name) +def get_rules(): + data_dir = get_data_dir() + filename = build_rules_filename(data_dir) if os.path.exists(filename) and os.path.isfile(filename): with open(filename) as fh: data = yaml.load(fh, Loader=Loader) - for rule in data: - if rule.get("name"): - rule["name_regex"] = re.compile(rule["name"]) - if rule.get("description"): - rule["desc_regex"] = re.compile(rule["description"]) + if data: + for rule in data: + if rule.get("name"): + rule["name_regex"] = re.compile(rule["name"]) + if rule.get("description"): + rule["desc_regex"] = re.compile(rule["description"]) return data else: - print(f"No rules for profile '{profile_name}'. Did you run 'init'?") + print(f"No rules.json found in {data_dir}! Did you run 'init'?") sys.exit(1) -def create_tag(name, profile, session): +def create_tag(name, session): t = session.query(models.Tag).filter(models.Tag.name == name).first() if not t: t = models.Tag(name=name) diff --git a/schmeckels/importer.py b/schmeckels/importer.py index 6a5dd7d..0b69359 100644 --- a/schmeckels/importer.py +++ b/schmeckels/importer.py @@ -1,21 +1,22 @@ #! /usr/bin/env python3 +import sys + import click +import colorful as cf from sqlalchemy import create_engine, desc from sqlalchemy.orm import sessionmaker -from schmeckels.models import Transaction -from schmeckels.helper import get_session, import_transactions -import sys + from schmeckels import banks -import colorful as cf +from schmeckels.helper import get_session, import_transactions +from schmeckels.models import Transaction @click.command(name="import") @click.option("--filetype", "-t", type=click.Choice([b[0] for b in banks.SUPPORTED_BANKS], case_sensitive=False)) -@click.option("--profile", "-p") @click.option("--dry-run", default=False, is_flag=True) @click.argument("filename", type=click.Path(exists=True)) -def command(filetype, profile, dry_run, filename): - session = get_session(profile) +def command(filetype, dry_run, filename): + session = get_session() new = import_transactions(session, filetype, filename) if dry_run: diff --git a/schmeckels/info.py b/schmeckels/info.py index e84f53d..04dbea3 100644 --- a/schmeckels/info.py +++ b/schmeckels/info.py @@ -1,35 +1,33 @@ #! /usr/bin/env python3 -from sqlalchemy import create_engine -from schmeckels.models import Tag, Transaction -from schmeckels.helper import get_session, list_profiles, build_database_filename, build_rules_filename import sys + import click +from sqlalchemy import create_engine + +from schmeckels.helper import build_database_filename, build_rules_filename, get_data_dir, get_session, get_rules +from schmeckels.models import Tag, Transaction @click.command(name="info") def command(): - first = True - for profile in list_profiles(): - session = get_session(profile) - db_path = build_database_filename(profile) - rules_path = build_rules_filename(profile) + session = get_session() + data_dir = get_data_dir() + db_path = build_database_filename(data_dir) + rules_path = build_rules_filename(data_dir) + rules = get_rules() - print(f"Profile: {profile}") - print(f" DB: {db_path}") - print(f" Rules: {rules_path}") + print(f" DB: {db_path}") + print(f" Rules: {rules_path}") - unsorted = session.query(Transaction).filter(Transaction.tags == None).count() - transactions = session.query(Transaction).count() - tags = session.query(Tag).count() - try: - unsorted_percent = round(unsorted / (transactions / 100), 1) - except ZeroDivisionError: - unsorted_percent = 0 + unsorted = session.query(Transaction).filter(Transaction.tags == None).count() + transactions = session.query(Transaction).count() + tags = session.query(Tag).count() + try: + unsorted_percent = round(unsorted / (transactions / 100), 1) + except ZeroDivisionError: + unsorted_percent = 0 - print(f"Transactions: {transactions}") - print(f" Unsorted: {unsorted} ({unsorted_percent}%)") - print(f" Tags: {tags}") - if first: - first = False - else: - print("----------") + print(f"Transactions: {transactions}") + print(f" Unsorted: {unsorted} ({unsorted_percent}%)") + print(f" Tags: {tags}") + print(f" Rules: {len(rules)}") diff --git a/schmeckels/models.py b/schmeckels/models.py index 9701722..6e37727 100644 --- a/schmeckels/models.py +++ b/schmeckels/models.py @@ -1,10 +1,12 @@ #! /usr/bin/env python3 -from sqlalchemy.ext.declarative import declarative_base -from sqlalchemy import Column, Integer, String, DateTime, ForeignKey, Table, Boolean -from sqlalchemy.orm import relationship, backref -from schmeckels import helper from hashlib import md5 +from sqlalchemy import Boolean, Column, DateTime, ForeignKey, Integer, String, Table +from sqlalchemy.ext.declarative import declarative_base +from sqlalchemy.orm import backref, relationship + +from schmeckels import helper + Base = declarative_base() association_table = Table( diff --git a/schmeckels/report.py b/schmeckels/report.py index f4eae4c..95c6cda 100644 --- a/schmeckels/report.py +++ b/schmeckels/report.py @@ -1,21 +1,20 @@ #! /usr/bin/env python3 -from schmeckels.models import Tag, Transaction -from schmeckels.helper import get_session, format_amount import sys -import click -from jinja2 import Environment, FileSystemLoader -from sqlalchemy import func, and_ -from weasyprint import HTML -from sqlalchemy import func -from weasyprint import HTML from datetime import date from pprint import pprint +import click +from jinja2 import Environment, FileSystemLoader +from sqlalchemy import and_, func +from weasyprint import HTML + +from schmeckels.helper import format_amount, get_data_dir, get_session +from schmeckels.models import Tag, Transaction + @click.command(name="report") -@click.option("--profile", "-p") @click.option("--year", "-y") -def command(profile, year): +def command(year): if not year: year = date.today().year() @@ -23,7 +22,8 @@ def command(profile, year): end_date = date(day=1, month=12, year=int(year)) # Get a list of all tags - session = get_session(profile) + data_dir = get_data_dir() + session = get_session() tags = session.query(Tag).filter_by(reporting=True).all() data = {"Einnahmen": {}, "Ausgaben": {}} @@ -42,7 +42,6 @@ def command(profile, year): else: data["Ausgaben"][name] = format_amount(tag_sum) - pprint(data) context = { "year": year, "tags": data, @@ -51,7 +50,7 @@ def command(profile, year): } context["diff"] = context["sum_in"] + context["sum_out"] - filename = "output/report_{}_{}.pdf".format(profile, year) + filename = f"{data_dir}/reports/{year}.pdf" env = Environment(loader=FileSystemLoader("schmeckels/templates/")) template = env.get_template("report.tpl") html_out = template.render(context) diff --git a/schmeckels/serve.py b/schmeckels/serve.py index 4bcd212..8b43e77 100644 --- a/schmeckels/serve.py +++ b/schmeckels/serve.py @@ -1,15 +1,16 @@ #! /usr/bin/env python3 -from flask import Flask, render_template, request, redirect, flash -from schmeckels.helper import get_session, format_amount -from datetime import datetime, date -from dateutil.relativedelta import relativedelta -from sqlalchemy import func, and_ -import click -from math import floor import calendar -from schmeckels.models import Transaction, Tag -from schmeckels.helper import import_transactions +from datetime import date, datetime +from math import floor + +import click +from dateutil.relativedelta import relativedelta +from flask import Flask, flash, redirect, render_template, request +from sqlalchemy import and_, func + from schmeckels import banks +from schmeckels.helper import format_amount, get_session, import_transactions +from schmeckels.models import Tag, Transaction session = None app = Flask(__name__) @@ -157,9 +158,8 @@ def upload(): @click.command(name="serve") -@click.option("--profile", "-p") -def command(profile): +def command(): global session - session = get_session(profile) + session = get_session() app.secret_key = "WeDon'tHaveSessionsSoThisMustNotBeVerySecure!" app.run(host="0.0.0.0", port=8080) diff --git a/schmeckels/sort.py b/schmeckels/sort.py index 4fe9f5c..9fe7115 100644 --- a/schmeckels/sort.py +++ b/schmeckels/sort.py @@ -1,17 +1,17 @@ #! /usr/bin/env python3 -from schmeckels.models import Transaction, Tag -from schmeckels.helper import get_session, create_tag import sys -import click +import click from prompt_toolkit.completion import FuzzyWordCompleter from prompt_toolkit.shortcuts import prompt +from schmeckels.helper import create_tag, get_session +from schmeckels.models import Tag, Transaction + @click.command(name="sort") -@click.option("--profile", "-p") -def command(profile): - session = get_session(profile) +def command(): + session = get_session() tags = session.query(Tag).all() tag_names = FuzzyWordCompleter([x.name for x in tags]) tag_lookup = {tag.name: tag for tag in tags} @@ -36,7 +36,7 @@ def command(profile): tag = tag_lookup.get(select, None) if not tag: print(f"Creating new category '{select}'") - tag = create_tag(select, profile, session) + tag = create_tag(select, session) tags = session.query(Tag).all() tag_names = FuzzyWordCompleter([x.name for x in tags]) diff --git a/schmeckels/validate.py b/schmeckels/validate.py index 1bf5a6d..230d585 100644 --- a/schmeckels/validate.py +++ b/schmeckels/validate.py @@ -1,20 +1,22 @@ #! /usr/bin/env python3 +import os +import re +import sys + +import click +import colorful as cf +import yaml +from schwifty import IBAN from sqlalchemy import create_engine from sqlalchemy.orm import sessionmaker -from sqlalchemy.ext.declarative import declarative_base -from schmeckels.helper import list_profiles, build_rules_filename, build_database_filename -from schwifty import IBAN -import sys -import click -import os -import yaml -import re -import colorful as cf + +from schmeckels.helper import build_database_filename, build_rules_filename, get_data_dir try: - from yaml import CLoader as Loader, CDumper as Dumper + from yaml import CDumper as Dumper + from yaml import CLoader as Loader except ImportError: - from yaml import Loader, Dumper + from yaml import Dumper, Loader ALLOWED_KEYS = ["name", "tags", "iban", "description"] TAGS_REGEXP = re.compile(r"[\w]+(?:,\s{0,1}[\w]+)*") @@ -28,71 +30,70 @@ def verbose_print(text, verbose): @click.command(name="validate") @click.option("--verbose", "-v", default=False, is_flag=True) def command(verbose): - for profile in list_profiles(): - print(f"Checking {profile}:") - db_path = build_database_filename(profile) - rules_path = build_rules_filename(profile) + data_dir = get_data_dir() + db_path = build_database_filename(data_dir) + rules_path = build_rules_filename(data_dir) - # Rules - if os.path.exists(rules_path) and os.path.isfile(rules_path): - with open(rules_path) as fh: - data = yaml.load(fh, Loader=Loader) + # Rules + if os.path.exists(rules_path) and os.path.isfile(rules_path): + with open(rules_path) as fh: + data = yaml.load(fh, Loader=Loader) - if not data: - print(cf.green(" Ruleset is empty")) + if not data: + print(cf.green(" Ruleset is empty")) + else: + errors = 0 + for rule in data: + # validate yaml keys: + for key in rule.keys(): + if key not in ALLOWED_KEYS: + verbose_print(cf.red(f"Rule uses the invalid key '{key}': {rule}"), verbose) + errors += 1 + + if not rule.get("tags"): + verbose_print(cf.red(f"Rule has no tags: {rule}"), verbose) + errors += 1 + + if not TAGS_REGEXP.fullmatch(rule.get("tags")): + verbose_print(cf.red(f"Rule has an invalid list of tags: {rule}"), verbose) + errors += 1 + + # validate name regex + if rule.get("name"): + try: + re.compile(rule["name"]) + except: + verbose_print(cf.red(f" Invalid name regex: '{rule.get('name')}'"), verbose) + errors += 1 + + # validate description regex + if rule.get("description"): + try: + re.compile(rule["description"]) + except: + verbose_print(cf.red(f" Invalid description regex: '{rule.get('description')}'"), verbose) + errors += 1 + + # validate IBAN + if rule.get("iban"): + try: + IBAN(rule.get("iban")) + except: + verbose_print(cf.red(f" Invalid IBAN: '{rule.get('iban')}'"), verbose) + errors += 1 + + if errors == 0: + print(cf.green(f" All rules are valid")) else: - errors = 0 - for rule in data: - # validate yaml keys: - for key in rule.keys(): - if key not in ALLOWED_KEYS: - verbose_print(cf.red(f"Rule uses the invalid key '{key}': {rule}"), verbose) - errors += 1 + print(cf.red(f" Found {errors} invalid rules")) - if not rule.get("tags"): - verbose_print(cf.red(f"Rule has no tags: {rule}"), verbose) - errors += 1 + else: + print(cf.red(f" The rule file doesn't exists")) + sys.exit(1) - if not TAGS_REGEXP.fullmatch(rule.get("tags")): - verbose_print(cf.red(f"Rule has an invalid list of tags: {rule}"), verbose) - errors += 1 - - # validate name regex - if rule.get("name"): - try: - re.compile(rule["name"]) - except: - verbose_print(cf.red(f" Invalid name regex: '{rule.get('name')}'"), verbose) - errors += 1 - - # validate description regex - if rule.get("description"): - try: - re.compile(rule["description"]) - except: - verbose_print(cf.red(f" Invalid description regex: '{rule.get('description')}'"), verbose) - errors += 1 - - # validate IBAN - if rule.get("iban"): - try: - IBAN(rule.get("iban")) - except: - verbose_print(cf.red(f" Invalid IBAN: '{rule.get('iban')}'"), verbose) - errors += 1 - - if errors == 0: - print(cf.green(f" All rules are valid")) - else: - print(cf.red(f" Found {errors} invalid rules")) - - else: - print(cf.red(f" The rule file doesn't exists")) - sys.exit(1) - - # Database - if os.path.exists(db_path) and os.path.isfile(db_path): - print(cf.green(f" Database exists")) - else: - print(cf.red(f" The database file doesn't exists")) - sys.exit(1) + # Database + if os.path.exists(db_path) and os.path.isfile(db_path): + print(cf.green(f" Database exists")) + else: + print(cf.red(f" The database file doesn't exists")) + sys.exit(1) diff --git a/shell.nix b/shell.nix index 658bf50..b049fbd 100644 --- a/shell.nix +++ b/shell.nix @@ -1,11 +1,16 @@ { pkgs ? import {} }: pkgs.mkShell { - buildInputs = [ - pkgs.python3 - pkgs.python3Packages.pip - pkgs.python3Packages.setuptools - pkgs.poetry + buildInputs = with pkgs; [ + black + python3 + python3Packages.flake8 + python3Packages.isort + python3Packages.pip + python3Packages.poetry + python3Packages.setuptools + # python3Packages.pygobject3 + # python3Packages.cairocffi ]; }