wip
This commit is contained in:
parent
3511095be8
commit
0612cb530f
16 changed files with 255 additions and 525 deletions
3
.flake8
3
.flake8
|
@ -1,2 +1,5 @@
|
|||
[flake8]
|
||||
max-line-length = 120
|
||||
ignore = E741
|
||||
exclude = .git,__pycache__
|
||||
max-complexity = 10
|
||||
|
|
311
poetry.lock
generated
311
poetry.lock
generated
|
@ -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"},
|
||||
|
|
|
@ -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
|
||||
|
|
|
@ -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)
|
||||
|
||||
|
|
|
@ -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"),
|
||||
|
|
|
@ -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()
|
||||
|
|
|
@ -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()
|
||||
|
|
|
@ -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)
|
||||
|
|
|
@ -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:
|
||||
|
|
|
@ -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)}")
|
||||
|
|
|
@ -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(
|
||||
|
|
|
@ -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)
|
||||
|
|
|
@ -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)
|
||||
|
|
|
@ -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])
|
||||
|
|
|
@ -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)
|
||||
|
|
15
shell.nix
15
shell.nix
|
@ -1,11 +1,16 @@
|
|||
{ pkgs ? import <nixpkgs> {} }:
|
||||
|
||||
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
|
||||
];
|
||||
|
||||
}
|
||||
|
|
Loading…
Add table
Reference in a new issue