diff --git a/.github/workflows/docs.yml b/.github/workflows/docs.yml index ff7c66ef8..5cf16b51e 100644 --- a/.github/workflows/docs.yml +++ b/.github/workflows/docs.yml @@ -17,7 +17,7 @@ on: env: FRONTEND_BRANCH: master - GH_TOKEN: ${{ secrets.API_TOKEN_EXT }} + GITHUB_TOKEN: ${{ secrets.DOCS_GITHUB_TOKEN }} jobs: docs: diff --git a/CHANGELOG.md b/CHANGELOG.md index b295118a0..451da2348 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -4,6 +4,23 @@ All notable changes to this project will be documented in this file. The format is based on [Keep a Changelog], and this project adheres to [Semantic Versioning]. +## [7.0.2] - 2023-10-10 + +### Added + +- database: Added `dipdup_wipe` and `dipdup_approve` SQL functions to the schema. + +### Fixed + +- cli: Fixed `schema wipe` command for SQLite databases. +- tezos.tzkt: Fixed regression in `get_transactions` method pagination. + +## [6.5.13] - 2023-10-10 + +### Fixed + +- tzkt: Fixed regression in `get_transactions` method pagination. + ## [7.0.1] - 2023-09-30 ### Added diff --git a/docs/5.advanced/1.reindexing.md b/docs/5.advanced/1.reindexing.md index c7bdd75b6..741964f5f 100644 --- a/docs/5.advanced/1.reindexing.md +++ b/docs/5.advanced/1.reindexing.md @@ -7,13 +7,13 @@ description: "In some cases, DipDup can't proceed with indexing without a full w In some cases, DipDup can't proceed with indexing without a full wipe. Several reasons trigger reindexing: -| reason | description | -| ----------------- | ------------------------------------------------------------------------------------------------------------------------- | -| `manual` | Reindexing triggered manually from callback with `ctx.reindex`. | -| `migration` | Applied migration requires reindexing. Check release notes before switching between major DipDup versions to be prepared. | -| `rollback` | Reorg message received from TzKT can not be processed. | -| `config_modified` | One of the index configs has been modified. | -| `schema_modified` | Database schema has been modified. Try to avoid manual schema modifications in favor of [sql](../5.advanced/6.sql.md). | +| reason | description | +| ----------------- | ------------------------------------------------------------------------------------------------------------------------------ | +| `manual` | Reindexing triggered manually from callback with `ctx.reindex`. | +| `migration` | Applied migration requires reindexing. Check release notes before switching between major DipDup versions to be prepared. | +| `rollback` | Reorg message received from datasource and can not be processed. | +| `config_modified` | One of the index configs has been modified. | +| `schema_modified` | Database schema has been modified. Try to avoid manual schema modifications in favor of [SQL scripts](../5.advanced/3.sql.md). | It is possible to configure desirable action on reindexing triggered by a specific reason. diff --git a/docs/5.advanced/3.internal-tables.md b/docs/5.advanced/3.sql.md similarity index 54% rename from docs/5.advanced/3.internal-tables.md rename to docs/5.advanced/3.sql.md index 2b938bec6..1a9b1170c 100644 --- a/docs/5.advanced/3.internal-tables.md +++ b/docs/5.advanced/3.sql.md @@ -1,11 +1,13 @@ --- -title: "Internal tables" -description: "This page describes the internal tables used by DipDup. They are created automatically and are not intended to be modified by the user. However, they can be useful for external monitoring and debugging." +title: "Advanced SQL" +description: "Put your *.sql scripts to dipdup_indexer/sql. You can run these scripts from any callback with ctx.execute_sql('name'). If name is a directory, each script it contains will be executed." --- -# Internal tables +# Advanced SQL -This page describes the internal tables used by DipDup. They are created automatically and are not intended to be modified by the user. However, they can be useful for external monitoring and debugging. +## Internal tables + +Several tables haing `dipdup_` prefix are created by DipDup automatically and are not intended to be modified by the user. However, they can be useful for external monitoring and debugging. | table | description | |:-------------------------- |:----------------------------------------------------------------------------------------------------------------------------------------- | @@ -15,8 +17,8 @@ This page describes the internal tables used by DipDup. They are created automat | `dipdup_contract` | Info about contracts used by all indexes, including ones added in runtime. | | `dipdup_model_update` | Service table to store model diffs for database rollback. Configured by `advanced.rollback_depth` | | `dipdup_meta` | Arbitrary key-value storage for DipDup internal use. Survives reindexing. You can use it too, but don't touch keys with `dipdup_` prefix. | -| `dipdup_contract_metadata` | See Metadata interface page | -| `dipdup_token_metadata` | See Metadata interface page | +| `dipdup_contract_metadata` | See [Metadata interface](/docs/advanced/metadata-interface) | +| `dipdup_token_metadata` | See [Metadata interface](/docs/advanced/metadata-interface) | See [`dipdup.models` module](https://github.com/dipdup-io/dipdup/blob/next/src/dipdup/models/__init__.py) for exact table definitions. @@ -32,3 +34,28 @@ SELECT name, status FROM dipdup_index; -- Get last reindex time SELECT created_at FROM dipdup_schema WHERE name = 'public'; ``` + +## Scripts + +Put your `*.sql` scripts to `{{ project.package }}/sql`. You can run these scripts from any callback with `ctx.execute_sql('name')`. If `name` is a directory, each script it contains will be executed. + +Scripts are executed without being wrapped with SQL transactions. It's generally a good idea to avoid touching table data in scripts. + +By default, an empty `sql/` directory is generated for every hook in config during init. Remove `ctx.execute_sql` call from hook callback to avoid executing them. + +```python +# Execute all scripts in sql/my_hook directory +await ctx.execute_sql('my_hook') + +# Execute a single script +await ctx.execute_sql('my_hook/my_script.sql') +``` + +## Managing schema + +When using PostgreSQL as database engine you can use `dipdup_approve` and `dipdup_wipe` functions to manage schema state from SQL console if needed: + +```sql +SELECT dipdup_approve('public'); +SELECT dipdup_wipe('public'); +``` diff --git a/docs/5.advanced/6.sql.md b/docs/5.advanced/6.sql.md deleted file mode 100644 index 8580e4b65..000000000 --- a/docs/5.advanced/6.sql.md +++ /dev/null @@ -1,22 +0,0 @@ ---- -title: "SQL scripts" -description: "Put your *.sql scripts to dipdup_indexer/sql. You can run these scripts from any callback with ctx.execute_sql('name'). If name is a directory, each script it contains will be executed." ---- - -# SQL scripts - -Put your `*.sql` scripts to `{{ project.package }}/sql`. You can run these scripts from any callback with `ctx.execute_sql('name')`. If `name` is a directory, each script it contains will be executed. - -Scripts are executed without being wrapped with SQL transactions. It's generally a good idea to avoid touching table data in scripts. - -By default, an empty `sql/` directory is generated for every hook in config during init. Remove `ctx.execute_sql` call from hook callback to avoid executing them. - -## Usage - -```python -# Execute all scripts in sql/my_hook directory -await ctx.execute_sql('my_hook') - -# Execute a single script -await ctx.execute_sql('my_hook/my_script.sql') -``` diff --git a/pdm.lock b/pdm.lock index 8cd771875..1fb497bb3 100644 --- a/pdm.lock +++ b/pdm.lock @@ -251,7 +251,7 @@ files = [ [[package]] name = "black" -version = "23.9.1" +version = "23.10.0" requires_python = ">=3.8" summary = "The uncompromising code formatter." dependencies = [ @@ -262,13 +262,12 @@ dependencies = [ "platformdirs>=2", ] files = [ - {file = "black-23.9.1-cp311-cp311-macosx_10_16_arm64.whl", hash = "sha256:8431445bf62d2a914b541da7ab3e2b4f3bc052d2ccbf157ebad18ea126efb91f"}, - {file = "black-23.9.1-cp311-cp311-macosx_10_16_universal2.whl", hash = "sha256:8fc1ddcf83f996247505db6b715294eba56ea9372e107fd54963c7553f2b6dfe"}, - {file = "black-23.9.1-cp311-cp311-macosx_10_16_x86_64.whl", hash = "sha256:7d30ec46de88091e4316b17ae58bbbfc12b2de05e069030f6b747dfc649ad186"}, - {file = "black-23.9.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:031e8c69f3d3b09e1aa471a926a1eeb0b9071f80b17689a655f7885ac9325a6f"}, - {file = "black-23.9.1-cp311-cp311-win_amd64.whl", hash = "sha256:538efb451cd50f43aba394e9ec7ad55a37598faae3348d723b59ea8e91616300"}, - {file = "black-23.9.1-py3-none-any.whl", hash = "sha256:6ccd59584cc834b6d127628713e4b6b968e5f79572da66284532525a042549f9"}, - {file = "black-23.9.1.tar.gz", hash = "sha256:24b6b3ff5c6d9ea08a8888f6977eae858e1f340d7260cf56d70a49823236b62d"}, + {file = "black-23.10.0-cp311-cp311-macosx_10_16_arm64.whl", hash = "sha256:30b78ac9b54cf87bcb9910ee3d499d2bc893afd52495066c49d9ee6b21eee06e"}, + {file = "black-23.10.0-cp311-cp311-macosx_10_16_x86_64.whl", hash = "sha256:0e232f24a337fed7a82c1185ae46c56c4a6167fb0fe37411b43e876892c76699"}, + {file = "black-23.10.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:31946ec6f9c54ed7ba431c38bc81d758970dd734b96b8e8c2b17a367d7908171"}, + {file = "black-23.10.0-cp311-cp311-win_amd64.whl", hash = "sha256:c870bee76ad5f7a5ea7bd01dc646028d05568d33b0b09b7ecfc8ec0da3f3f39c"}, + {file = "black-23.10.0-py3-none-any.whl", hash = "sha256:e223b731a0e025f8ef427dd79d8cd69c167da807f5710add30cdf131f13dd62e"}, + {file = "black-23.10.0.tar.gz", hash = "sha256:31b9f87b277a68d0e99d2905edae08807c007973eaa609da5f0c62def6b7c0bd"}, ] [[package]] @@ -895,7 +894,7 @@ files = [ [[package]] name = "mypy" -version = "1.5.1" +version = "1.6.0" requires_python = ">=3.8" summary = "Optional static typing for Python" dependencies = [ @@ -903,13 +902,13 @@ dependencies = [ "typing-extensions>=4.1.0", ] files = [ - {file = "mypy-1.5.1-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:6ac9c21bfe7bc9f7f1b6fae441746e6a106e48fc9de530dea29e8cd37a2c0cc4"}, - {file = "mypy-1.5.1-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:51cb1323064b1099e177098cb939eab2da42fea5d818d40113957ec954fc85f4"}, - {file = "mypy-1.5.1-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:596fae69f2bfcb7305808c75c00f81fe2829b6236eadda536f00610ac5ec2243"}, - {file = "mypy-1.5.1-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:32cb59609b0534f0bd67faebb6e022fe534bdb0e2ecab4290d683d248be1b275"}, - {file = "mypy-1.5.1-cp311-cp311-win_amd64.whl", hash = "sha256:159aa9acb16086b79bbb0016145034a1a05360626046a929f84579ce1666b315"}, - {file = "mypy-1.5.1-py3-none-any.whl", hash = "sha256:f757063a83970d67c444f6e01d9550a7402322af3557ce7630d3c957386fa8f5"}, - {file = "mypy-1.5.1.tar.gz", hash = "sha256:b031b9601f1060bf1281feab89697324726ba0c0bae9d7cd7ab4b690940f0b92"}, + {file = "mypy-1.6.0-cp311-cp311-macosx_10_9_x86_64.whl", hash = "sha256:3c8835a07b8442da900db47ccfda76c92c69c3a575872a5b764332c4bacb5a0a"}, + {file = "mypy-1.6.0-cp311-cp311-macosx_11_0_arm64.whl", hash = "sha256:24f3de8b9e7021cd794ad9dfbf2e9fe3f069ff5e28cb57af6f873ffec1cb0425"}, + {file = "mypy-1.6.0-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:856bad61ebc7d21dbc019b719e98303dc6256cec6dcc9ebb0b214b81d6901bd8"}, + {file = "mypy-1.6.0-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:89513ddfda06b5c8ebd64f026d20a61ef264e89125dc82633f3c34eeb50e7d60"}, + {file = "mypy-1.6.0-cp311-cp311-win_amd64.whl", hash = "sha256:9f8464ed410ada641c29f5de3e6716cbdd4f460b31cf755b2af52f2d5ea79ead"}, + {file = "mypy-1.6.0-py3-none-any.whl", hash = "sha256:9e1589ca150a51d9d00bb839bfeca2f7a04f32cd62fad87a847bc0818e15d7dc"}, + {file = "mypy-1.6.0.tar.gz", hash = "sha256:4f3d27537abde1be6d5f2c96c29a454da333a2a271ae7d5bc7110e6d4b7beb3f"}, ] [[package]] @@ -973,21 +972,21 @@ files = [ [[package]] name = "orjson" -version = "3.9.7" -requires_python = ">=3.7" +version = "3.9.9" +requires_python = ">=3.8" summary = "Fast, correct Python JSON library supporting dataclasses, datetimes, and numpy" files = [ - {file = "orjson-3.9.7-cp311-cp311-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:1f8b47650f90e298b78ecf4df003f66f54acdba6a0f763cc4df1eab048fe3738"}, - {file = "orjson-3.9.7-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:f738fee63eb263530efd4d2e9c76316c1f47b3bbf38c1bf45ae9625feed0395e"}, - {file = "orjson-3.9.7-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:38e34c3a21ed41a7dbd5349e24c3725be5416641fdeedf8f56fcbab6d981c900"}, - {file = "orjson-3.9.7-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:21a3344163be3b2c7e22cef14fa5abe957a892b2ea0525ee86ad8186921b6cf0"}, - {file = "orjson-3.9.7-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:23be6b22aab83f440b62a6f5975bcabeecb672bc627face6a83bc7aeb495dc7e"}, - {file = "orjson-3.9.7-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:e5205ec0dfab1887dd383597012199f5175035e782cdb013c542187d280ca443"}, - {file = "orjson-3.9.7-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:8769806ea0b45d7bf75cad253fba9ac6700b7050ebb19337ff6b4e9060f963fa"}, - {file = "orjson-3.9.7-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:f9e01239abea2f52a429fe9d95c96df95f078f0172489d691b4a848ace54a476"}, - {file = "orjson-3.9.7-cp311-none-win32.whl", hash = "sha256:8bdb6c911dae5fbf110fe4f5cba578437526334df381b3554b6ab7f626e5eeca"}, - {file = "orjson-3.9.7-cp311-none-win_amd64.whl", hash = "sha256:9d62c583b5110e6a5cf5169ab616aa4ec71f2c0c30f833306f9e378cf51b6c86"}, - {file = "orjson-3.9.7.tar.gz", hash = "sha256:85e39198f78e2f7e054d296395f6c96f5e02892337746ef5b6a1bf3ed5910142"}, + {file = "orjson-3.9.9-cp311-cp311-macosx_10_15_x86_64.macosx_11_0_arm64.macosx_10_15_universal2.whl", hash = "sha256:31d676bc236f6e919d100fb85d0a99812cff1ebffaa58106eaaec9399693e227"}, + {file = "orjson-3.9.9-cp311-cp311-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:678ffb5c0a6b1518b149cc328c610615d70d9297e351e12c01d0beed5d65360f"}, + {file = "orjson-3.9.9-cp311-cp311-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:a71b0cc21f2c324747bc77c35161e0438e3b5e72db6d3b515310457aba743f7f"}, + {file = "orjson-3.9.9-cp311-cp311-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ae72621f216d1d990468291b1ec153e1b46e0ed188a86d54e0941f3dabd09ee8"}, + {file = "orjson-3.9.9-cp311-cp311-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:512e5a41af008e76451f5a344941d61f48dddcf7d7ddd3073deb555de64596a6"}, + {file = "orjson-3.9.9-cp311-cp311-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:0f89dc338a12f4357f5bf1b098d3dea6072fb0b643fd35fec556f4941b31ae27"}, + {file = "orjson-3.9.9-cp311-cp311-musllinux_1_1_aarch64.whl", hash = "sha256:957a45fb201c61b78bcf655a16afbe8a36c2c27f18a998bd6b5d8a35e358d4ad"}, + {file = "orjson-3.9.9-cp311-cp311-musllinux_1_1_x86_64.whl", hash = "sha256:d1c01cf4b8e00c7e98a0a7cf606a30a26c32adf2560be2d7d5d6766d6f474b31"}, + {file = "orjson-3.9.9-cp311-none-win32.whl", hash = "sha256:397a185e5dd7f8ebe88a063fe13e34d61d394ebb8c70a443cee7661b9c89bda7"}, + {file = "orjson-3.9.9-cp311-none-win_amd64.whl", hash = "sha256:24301f2d99d670ded4fb5e2f87643bc7428a54ba49176e38deb2887e42fe82fb"}, + {file = "orjson-3.9.9.tar.gz", hash = "sha256:02e693843c2959befdd82d1ebae8b05ed12d1cb821605d5f9fe9f98ca5c9fd2b"}, ] [[package]] @@ -1521,40 +1520,40 @@ files = [ [[package]] name = "ruff" -version = "0.0.292" +version = "0.1.0" requires_python = ">=3.7" summary = "An extremely fast Python linter, written in Rust." files = [ - {file = "ruff-0.0.292-py3-none-macosx_10_7_x86_64.whl", hash = "sha256:02f29db018c9d474270c704e6c6b13b18ed0ecac82761e4fcf0faa3728430c96"}, - {file = "ruff-0.0.292-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl", hash = "sha256:69654e564342f507edfa09ee6897883ca76e331d4bbc3676d8a8403838e9fade"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:6c3c91859a9b845c33778f11902e7b26440d64b9d5110edd4e4fa1726c41e0a4"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:f4476f1243af2d8c29da5f235c13dca52177117935e1f9393f9d90f9833f69e4"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:be8eb50eaf8648070b8e58ece8e69c9322d34afe367eec4210fdee9a555e4ca7"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:9889bac18a0c07018aac75ef6c1e6511d8411724d67cb879103b01758e110a81"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:6bdfabd4334684a4418b99b3118793f2c13bb67bf1540a769d7816410402a205"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:aa7c77c53bfcd75dbcd4d1f42d6cabf2485d2e1ee0678da850f08e1ab13081a8"}, - {file = "ruff-0.0.292-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:8e087b24d0d849c5c81516ec740bf4fd48bf363cfb104545464e0fca749b6af9"}, - {file = "ruff-0.0.292-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:f160b5ec26be32362d0774964e218f3fcf0a7da299f7e220ef45ae9e3e67101a"}, - {file = "ruff-0.0.292-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:ac153eee6dd4444501c4bb92bff866491d4bfb01ce26dd2fff7ca472c8df9ad0"}, - {file = "ruff-0.0.292-py3-none-musllinux_1_2_i686.whl", hash = "sha256:87616771e72820800b8faea82edd858324b29bb99a920d6aa3d3949dd3f88fb0"}, - {file = "ruff-0.0.292-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:b76deb3bdbea2ef97db286cf953488745dd6424c122d275f05836c53f62d4016"}, - {file = "ruff-0.0.292-py3-none-win32.whl", hash = "sha256:e854b05408f7a8033a027e4b1c7f9889563dd2aca545d13d06711e5c39c3d003"}, - {file = "ruff-0.0.292-py3-none-win_amd64.whl", hash = "sha256:f27282bedfd04d4c3492e5c3398360c9d86a295be00eccc63914438b4ac8a83c"}, - {file = "ruff-0.0.292-py3-none-win_arm64.whl", hash = "sha256:7f67a69c8f12fbc8daf6ae6d36705037bde315abf8b82b6e1f4c9e74eb750f68"}, - {file = "ruff-0.0.292.tar.gz", hash = "sha256:1093449e37dd1e9b813798f6ad70932b57cf614e5c2b5c51005bf67d55db33ac"}, + {file = "ruff-0.1.0-py3-none-macosx_10_7_x86_64.whl", hash = "sha256:87114e254dee35e069e1b922d85d4b21a5b61aec759849f393e1dbb308a00439"}, + {file = "ruff-0.1.0-py3-none-macosx_10_9_x86_64.macosx_11_0_arm64.macosx_10_9_universal2.whl", hash = "sha256:764f36d2982cc4a703e69fb73a280b7c539fd74b50c9ee531a4e3fe88152f521"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_aarch64.manylinux2014_aarch64.whl", hash = "sha256:65f4b7fb539e5cf0f71e9bd74f8ddab74cabdd673c6fb7f17a4dcfd29f126255"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_armv7l.manylinux2014_armv7l.whl", hash = "sha256:299fff467a0f163baa282266b310589b21400de0a42d8f68553422fa6bf7ee01"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_i686.manylinux2014_i686.whl", hash = "sha256:0d412678bf205787263bb702c984012a4f97e460944c072fd7cfa2bd084857c4"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_ppc64.manylinux2014_ppc64.whl", hash = "sha256:a5391b49b1669b540924640587d8d24128e45be17d1a916b1801d6645e831581"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_ppc64le.manylinux2014_ppc64le.whl", hash = "sha256:ee8cd57f454cdd77bbcf1e11ff4e0046fb6547cac1922cc6e3583ce4b9c326d1"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_s390x.manylinux2014_s390x.whl", hash = "sha256:fa7aeed7bc23861a2b38319b636737bf11cfa55d2109620b49cf995663d3e888"}, + {file = "ruff-0.1.0-py3-none-manylinux_2_17_x86_64.manylinux2014_x86_64.whl", hash = "sha256:b04cd4298b43b16824d9a37800e4c145ba75c29c43ce0d74cad1d66d7ae0a4c5"}, + {file = "ruff-0.1.0-py3-none-musllinux_1_2_aarch64.whl", hash = "sha256:7186ccf54707801d91e6314a016d1c7895e21d2e4cd614500d55870ed983aa9f"}, + {file = "ruff-0.1.0-py3-none-musllinux_1_2_armv7l.whl", hash = "sha256:d88adfd93849bc62449518228581d132e2023e30ebd2da097f73059900d8dce3"}, + {file = "ruff-0.1.0-py3-none-musllinux_1_2_i686.whl", hash = "sha256:ad2ccdb3bad5a61013c76a9c1240fdfadf2c7103a2aeebd7bcbbed61f363138f"}, + {file = "ruff-0.1.0-py3-none-musllinux_1_2_x86_64.whl", hash = "sha256:b77f6cfa72c6eb19b5cac967cc49762ae14d036db033f7d97a72912770fd8e1c"}, + {file = "ruff-0.1.0-py3-none-win32.whl", hash = "sha256:480bd704e8af1afe3fd444cc52e3c900b936e6ca0baf4fb0281124330b6ceba2"}, + {file = "ruff-0.1.0-py3-none-win_amd64.whl", hash = "sha256:a76ba81860f7ee1f2d5651983f87beb835def94425022dc5f0803108f1b8bfa2"}, + {file = "ruff-0.1.0-py3-none-win_arm64.whl", hash = "sha256:45abdbdab22509a2c6052ecf7050b3f5c7d6b7898dc07e82869401b531d46da4"}, + {file = "ruff-0.1.0.tar.gz", hash = "sha256:ad6b13824714b19c5f8225871cf532afb994470eecb74631cd3500fe817e6b3f"}, ] [[package]] name = "sentry-sdk" -version = "1.31.0" +version = "1.32.0" summary = "Python client for Sentry (https://sentry.io)" dependencies = [ "certifi", "urllib3>=1.26.11; python_version >= \"3.6\"", ] files = [ - {file = "sentry-sdk-1.31.0.tar.gz", hash = "sha256:6de2e88304873484207fed836388e422aeff000609b104c802749fd89d56ba5b"}, - {file = "sentry_sdk-1.31.0-py2.py3-none-any.whl", hash = "sha256:64a7141005fb775b9db298a30de93e3b83e0ddd1232dc6f36eb38aebc1553291"}, + {file = "sentry-sdk-1.32.0.tar.gz", hash = "sha256:935e8fbd7787a3702457393b74b13d89a5afb67185bc0af85c00cb27cbd42e7c"}, + {file = "sentry_sdk-1.32.0-py2.py3-none-any.whl", hash = "sha256:eeb0b3550536f3bbc05bb1c7e0feb3a78d74acb43b607159a606ed2ec0a33a4d"}, ] [[package]] @@ -1883,7 +1882,7 @@ files = [ [[package]] name = "web3" -version = "6.10.0" +version = "6.11.0" requires_python = ">=3.7.2" summary = "web3.py" dependencies = [ @@ -1904,8 +1903,8 @@ dependencies = [ "websockets>=10.0.0", ] files = [ - {file = "web3-6.10.0-py3-none-any.whl", hash = "sha256:070625a0da4f0fcac090fa95186e0b865a1bbc43efb78fd2ee805f7bf9cd8986"}, - {file = "web3-6.10.0.tar.gz", hash = "sha256:ea89f8a6ee74b74c3ff21954eafe00ec914365adb904c6c374f559bc46d4a61c"}, + {file = "web3-6.11.0-py3-none-any.whl", hash = "sha256:44e79da6a4765eacf137f2f388e37aa0c1e24a93bdfb462cffe9441d1be3d509"}, + {file = "web3-6.11.0.tar.gz", hash = "sha256:050dea52ae73d787272e7ecba7249f096595938c90cce1a384c20375c6b0f720"}, ] [[package]] diff --git a/pyproject.toml b/pyproject.toml index 51db16fec..9af7fefae 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,7 +1,7 @@ [project] name = "dipdup" description = "Modular framework for creating selective indexers and featureful backends for dapps" -version = "7.0.1" +version = "7.0.2" license = { text = "MIT" } authors = [ { name = "Lev Gorodetskii", email = "dipdup@drsr.io" }, diff --git a/requirements.dev.txt b/requirements.dev.txt index 1975bbc49..aeefa50c9 100644 --- a/requirements.dev.txt +++ b/requirements.dev.txt @@ -17,7 +17,7 @@ asyncpg==0.28.0 attrs==23.1.0 babel==2.13.0 bitarray==2.8.2 -black==23.9.1 +black==23.10.0 certifi==2023.7.22 chardet==5.2.0 charset-normalizer==3.3.0 @@ -55,12 +55,12 @@ lru-dict==1.2.0 MarkupSafe==2.1.3 msgpack==1.0.7 multidict==6.0.4 -mypy==1.5.1 +mypy==1.6.0 mypy-extensions==1.0.0 numpy==1.26.0 openapi-schema-validator==0.4.4 openapi-spec-validator==0.5.7 -orjson==3.9.7 +orjson==3.9.9 packaging==23.2 parsimonious==0.9.0 pathable==0.4.3 @@ -96,8 +96,8 @@ rfc3339-validator==0.1.4 rlp==3.0.0 ruamel-yaml==0.17.35 ruamel-yaml-clib==0.2.8 -ruff==0.0.292 -sentry-sdk==1.31.0 +ruff==0.1.0 +sentry-sdk==1.32.0 setuptools==68.2.2 six==1.16.0 sniffio==1.3.0 @@ -123,7 +123,7 @@ typing-extensions==4.8.0 tzlocal==5.1 urllib3==2.0.6 watchdog==3.0.0 -web3==6.10.0 +web3==6.11.0 websocket-client==1.6.4 websockets==10.4 yarl==1.9.2 diff --git a/requirements.txt b/requirements.txt index f39b277f0..b5472b087 100644 --- a/requirements.txt +++ b/requirements.txt @@ -15,7 +15,7 @@ asyncclick==8.1.3.4 asyncpg==0.28.0 attrs==23.1.0 bitarray==2.8.2 -black==23.9.1 +black==23.10.0 certifi==2023.7.22 chardet==5.2.0 charset-normalizer==3.3.0 @@ -51,7 +51,7 @@ mypy-extensions==1.0.0 numpy==1.26.0 openapi-schema-validator==0.4.4 openapi-spec-validator==0.5.7 -orjson==3.9.7 +orjson==3.9.9 packaging==23.2 parsimonious==0.9.0 pathable==0.4.3 @@ -79,7 +79,7 @@ rfc3339-validator==0.1.4 rlp==3.0.0 ruamel-yaml==0.17.35 ruamel-yaml-clib==0.2.8 -sentry-sdk==1.31.0 +sentry-sdk==1.32.0 setuptools==68.2.2 six==1.16.0 sniffio==1.3.0 @@ -92,6 +92,6 @@ tortoise-orm==0.20.0 typing-extensions==4.8.0 tzlocal==5.1 urllib3==2.0.6 -web3==6.10.0 +web3==6.11.0 websockets==10.4 yarl==1.9.2 diff --git a/scripts/dump_schema.py b/scripts/dump_schema.py index b85b43d2b..55cc32f69 100644 --- a/scripts/dump_schema.py +++ b/scripts/dump_schema.py @@ -2,7 +2,7 @@ from pathlib import Path import orjson -from dc_schema import get_schema # type: ignore[import] +from dc_schema import get_schema # type: ignore[import-not-found] from dipdup.config import DipDupConfig diff --git a/src/demo_uniswap/models/repo.py b/src/demo_uniswap/models/repo.py index b0d64779b..736fe328d 100644 --- a/src/demo_uniswap/models/repo.py +++ b/src/demo_uniswap/models/repo.py @@ -2,7 +2,7 @@ from typing import Any from typing import cast -from lru import LRU # type: ignore[import] +from lru import LRU # type: ignore[import-not-found] import demo_uniswap.models as models from dipdup.config.evm import EvmContractConfig @@ -32,7 +32,7 @@ def save_pending_position(self, idx: str, position: dict[str, Any]) -> None: self._pending_positions[idx] = position def get_pending_position(self, idx: str) -> dict[str, Any] | None: - return self._pending_positions.get(idx, None) + return self._pending_positions.get(idx, None) # type: ignore[no-any-return] async def get_ctx_factory(ctx: HandlerContext) -> models.Factory: diff --git a/src/dipdup/cli.py b/src/dipdup/cli.py index a2a32ce8a..9d49d9657 100644 --- a/src/dipdup/cli.py +++ b/src/dipdup/cli.py @@ -507,7 +507,9 @@ async def schema_wipe(ctx: click.Context, immune: bool, force: bool) -> None: conn = get_connection() await wipe_schema( conn=conn, - schema_name=config.database.schema_name, + schema_name=config.database.path + if isinstance(config.database, SqliteDatabaseConfig) + else config.database.schema_name, immune_tables=immune_tables, ) diff --git a/src/dipdup/database.py b/src/dipdup/database.py index 76f1c6c10..cff3cf138 100644 --- a/src/dipdup/database.py +++ b/src/dipdup/database.py @@ -14,8 +14,8 @@ from typing import Any from typing import cast -import asyncpg.exceptions # type: ignore[import] -import sqlparse # type: ignore[import] +import asyncpg.exceptions # type: ignore[import-untyped] +import sqlparse # type: ignore[import-untyped] from tortoise import Tortoise from tortoise.backends.asyncpg.client import AsyncpgDBClient from tortoise.backends.base.client import BaseDBAsyncClient @@ -194,44 +194,54 @@ async def generate_schema( conn: SupportedClient, name: str, ) -> None: - if isinstance(conn, AsyncpgClient): - await pg_create_schema(conn, name) + if isinstance(conn, SqliteClient): + await Tortoise.generate_schemas() + elif isinstance(conn, AsyncpgClient): + await _pg_create_schema(conn, name) + await Tortoise.generate_schemas() + await _pg_create_functions(conn) + await _pg_create_views(conn) + else: + raise NotImplementedError - await Tortoise.generate_schemas() - if isinstance(conn, AsyncpgClient): - # NOTE: Create a view for monitoring head status - sql_path = Path(__file__).parent / 'sql' / 'dipdup_head_status.sql' - # TODO: Configurable interval - await execute_sql(conn, sql_path, HEAD_STATUS_TIMEOUT) +async def _pg_create_functions(conn: AsyncpgClient) -> None: + for fn in ( + 'dipdup_approve.sql', + 'dipdup_wipe.sql', + ): + sql_path = Path(__file__).parent / 'sql' / fn + await execute_sql(conn, sql_path) -async def _wipe_schema_postgres( +async def _pg_create_views(conn: AsyncpgClient) -> None: + sql_path = Path(__file__).parent / 'sql' / 'dipdup_head_status.sql' + # TODO: Configurable interval + await execute_sql(conn, sql_path, HEAD_STATUS_TIMEOUT) + + +async def _pg_wipe_schema( conn: AsyncpgClient, schema_name: str, immune_tables: set[str], ) -> None: immune_schema_name = f'{schema_name}_immune' - # NOTE: Create a truncate_schema function to trigger cascade deletion - sql_path = Path(__file__).parent / 'sql' / 'truncate_schema.sql' - await execute_sql(conn, sql_path, schema_name, immune_schema_name) - # NOTE: Move immune tables to a separate schema - it's free! if immune_tables: - await pg_create_schema(conn, immune_schema_name) + await _pg_create_schema(conn, immune_schema_name) for table in immune_tables: - await pg_move_table(conn, table, schema_name, immune_schema_name) + await _pg_move_table(conn, table, schema_name, immune_schema_name) - await conn.execute_script(f"SELECT truncate_schema('{schema_name}')") + await conn.execute_script(f"SELECT dipdup_wipe('{schema_name}')") if immune_tables: for table in immune_tables: - await pg_move_table(conn, table, immune_schema_name, schema_name) - await pg_drop_schema(conn, immune_schema_name) + await _pg_move_table(conn, table, immune_schema_name, schema_name) + await _pg_drop_schema(conn, immune_schema_name) -async def _wipe_schema_sqlite( +async def _sqlite_wipe_schema( conn: SqliteClient, path: str, immune_tables: set[str], @@ -245,10 +255,10 @@ async def _wipe_schema_sqlite( await conn.execute_script(f'ATTACH DATABASE "{immune_path}" AS {namespace}') # NOTE: Copy immune tables to the new database. - master_query = 'SELECT name, type FROM sqlite_master' + master_query = 'SELECT name FROM sqlite_master WHERE type = "table"' result = await conn.execute_query(master_query) - for name, type_ in result[1]: - if type_ != 'table' or name not in immune_tables: + for name in result[1]: + if name not in immune_tables: # type: ignore[comparison-overlap] continue expr = f'CREATE TABLE {namespace}.{name} AS SELECT * FROM {name}' @@ -271,23 +281,23 @@ async def wipe_schema( """Truncate schema preserving immune tables. Executes in a transaction""" async with conn._in_transaction() as conn: if isinstance(conn, SqliteClient): - await _wipe_schema_sqlite(conn, schema_name, immune_tables) + await _sqlite_wipe_schema(conn, schema_name, immune_tables) elif isinstance(conn, AsyncpgClient): - await _wipe_schema_postgres(conn, schema_name, immune_tables) + await _pg_wipe_schema(conn, schema_name, immune_tables) else: raise NotImplementedError -async def pg_create_schema(conn: AsyncpgClient, name: str) -> None: +async def _pg_create_schema(conn: AsyncpgClient, name: str) -> None: """Create PostgreSQL schema if not exists""" await conn.execute_script(f'CREATE SCHEMA IF NOT EXISTS {name}') -async def pg_drop_schema(conn: AsyncpgClient, name: str) -> None: +async def _pg_drop_schema(conn: AsyncpgClient, name: str) -> None: await conn.execute_script(f'DROP SCHEMA IF EXISTS {name}') -async def pg_move_table(conn: AsyncpgClient, name: str, schema: str, new_schema: str) -> None: +async def _pg_move_table(conn: AsyncpgClient, name: str, schema: str, new_schema: str) -> None: """Move table from one schema to another""" await conn.execute_script(f'ALTER TABLE {schema}.{name} SET SCHEMA {new_schema}') diff --git a/src/dipdup/datasources/evm_subsquid.py b/src/dipdup/datasources/evm_subsquid.py index 1a866c558..e51ed26c1 100644 --- a/src/dipdup/datasources/evm_subsquid.py +++ b/src/dipdup/datasources/evm_subsquid.py @@ -7,7 +7,7 @@ from io import BytesIO from typing import Any -import pyarrow.ipc # type: ignore[import] +import pyarrow.ipc # type: ignore[import-untyped] from dipdup.config import HttpConfig from dipdup.config.evm_subsquid import SubsquidDatasourceConfig diff --git a/src/dipdup/datasources/tezos_tzkt.py b/src/dipdup/datasources/tezos_tzkt.py index ee4a891da..00fd4d479 100644 --- a/src/dipdup/datasources/tezos_tzkt.py +++ b/src/dipdup/datasources/tezos_tzkt.py @@ -713,14 +713,18 @@ async def get_transactions( params = self._get_request_params( first_level=first_level, last_level=last_level, - offset=offset, + # NOTE: This is intentional + offset=None, limit=limit, select=TRANSACTION_OPERATION_FIELDS, values=True, - cursor=True, sort='level', status='applied', ) + # TODO: TzKT doesn't support sort+cr currently + if offset is not None: + params['id.gt'] = offset + if addresses and not code_hashes: params[f'{field}.in'] = ','.join(addresses) elif code_hashes and not addresses: diff --git a/src/dipdup/indexes/tezos_tzkt_operations/matcher.py b/src/dipdup/indexes/tezos_tzkt_operations/matcher.py index d559810cc..a0b35f87e 100644 --- a/src/dipdup/indexes/tezos_tzkt_operations/matcher.py +++ b/src/dipdup/indexes/tezos_tzkt_operations/matcher.py @@ -213,6 +213,8 @@ def match_operation_subgroup( transaction = handler[2][-1] if isinstance(transaction, TzktOperationData): id_list.append(transaction.id) + elif isinstance(transaction, TzktOrigination): + id_list.append(transaction.data.id) elif isinstance(transaction, TzktTransaction): id_list.append(transaction.data.id) else: diff --git a/src/dipdup/indexes/tezos_tzkt_operations/parser.py b/src/dipdup/indexes/tezos_tzkt_operations/parser.py index 96c41ad75..f822d7fab 100644 --- a/src/dipdup/indexes/tezos_tzkt_operations/parser.py +++ b/src/dipdup/indexes/tezos_tzkt_operations/parser.py @@ -27,7 +27,7 @@ T = TypeVar('T', Hashable, type[BaseModel]) -def extract_root_outer_type(storage_type: type[BaseModel]) -> T: +def extract_root_outer_type(storage_type: type[BaseModel]) -> T: # type: ignore[type-var] """Extract Pydantic root type""" root_field = storage_type.model_fields['root'] if not root_field.is_required(): diff --git a/src/dipdup/performance.py b/src/dipdup/performance.py index 9d6296db8..05ab7041a 100644 --- a/src/dipdup/performance.py +++ b/src/dipdup/performance.py @@ -27,7 +27,7 @@ from typing import cast from async_lru import alru_cache -from lru import LRU # type: ignore[import] +from lru import LRU # type: ignore[import-not-found] from dipdup.exceptions import FrameworkException @@ -42,7 +42,7 @@ @asynccontextmanager async def with_pprofile(name: str) -> AsyncIterator[None]: try: - import pprofile # type: ignore[import] + import pprofile # type: ignore[import-untyped] _logger.warning('Full profiling is enabled, this will affect performance') except ImportError: diff --git a/src/dipdup/project.py b/src/dipdup/project.py index 20963564a..65625d53b 100644 --- a/src/dipdup/project.py +++ b/src/dipdup/project.py @@ -97,7 +97,7 @@ def prompt_anyof( default: int, ) -> tuple[int, str]: """Ask user to choose one of options; returns index and value""" - import survey # type: ignore[import] + import survey # type: ignore[import-untyped] table = tabulate( zip(options, comments, strict=True), diff --git a/src/dipdup/projects/demo_uniswap/models/repo.py.j2 b/src/dipdup/projects/demo_uniswap/models/repo.py.j2 index d739e3cde..e2e774ce5 100644 --- a/src/dipdup/projects/demo_uniswap/models/repo.py.j2 +++ b/src/dipdup/projects/demo_uniswap/models/repo.py.j2 @@ -2,7 +2,7 @@ from decimal import Decimal from typing import Any from typing import cast -from lru import LRU # type: ignore[import] +from lru import LRU # type: ignore[import-untyped] import {{ project.package }}.models as models from dipdup.config.evm import EvmContractConfig diff --git a/src/dipdup/scheduler.py b/src/dipdup/scheduler.py index 3778b3429..b1555cd07 100644 --- a/src/dipdup/scheduler.py +++ b/src/dipdup/scheduler.py @@ -4,13 +4,13 @@ from functools import partial from typing import Any -from apscheduler.events import EVENT_JOB_ERROR # type: ignore[import] +from apscheduler.events import EVENT_JOB_ERROR # type: ignore[import-untyped] from apscheduler.events import EVENT_JOB_EXECUTED from apscheduler.events import JobEvent -from apscheduler.job import Job # type: ignore[import] -from apscheduler.schedulers.asyncio import AsyncIOScheduler # type: ignore[import] -from apscheduler.triggers.cron import CronTrigger # type: ignore[import] -from apscheduler.triggers.interval import IntervalTrigger # type: ignore[import] +from apscheduler.job import Job # type: ignore[import-untyped] +from apscheduler.schedulers.asyncio import AsyncIOScheduler # type: ignore[import-untyped] +from apscheduler.triggers.cron import CronTrigger # type: ignore[import-untyped] +from apscheduler.triggers.interval import IntervalTrigger # type: ignore[import-untyped] from dipdup.config import JobConfig from dipdup.context import DipDupContext diff --git a/src/dipdup/sql/dipdup_approve.sql b/src/dipdup/sql/dipdup_approve.sql new file mode 100644 index 000000000..5691779e9 --- /dev/null +++ b/src/dipdup/sql/dipdup_approve.sql @@ -0,0 +1,7 @@ +CREATE OR REPLACE FUNCTION dipdup_approve(schema_name VARCHAR) RETURNS void AS $$ +BEGIN + UPDATE dipdup_index SET config_hash = null; + UPDATE dipdup_schema SET reindex = null, hash = null; + RETURN; +END; +$$ LANGUAGE plpgsql; diff --git a/src/dipdup/sql/truncate_schema.sql b/src/dipdup/sql/dipdup_wipe.sql similarity index 86% rename from src/dipdup/sql/truncate_schema.sql rename to src/dipdup/sql/dipdup_wipe.sql index 01f19f7ac..965fc951a 100644 --- a/src/dipdup/sql/truncate_schema.sql +++ b/src/dipdup/sql/dipdup_wipe.sql @@ -1,5 +1,5 @@ -- source of inspiration: https://stackoverflow.com/a/11462481 -CREATE OR REPLACE FUNCTION truncate_schema(schema_name VARCHAR) RETURNS void AS $$ +CREATE OR REPLACE FUNCTION dipdup_wipe(schema_name VARCHAR) RETURNS void AS $$ DECLARE rec RECORD; BEGIN @@ -63,14 +63,6 @@ BEGIN WHEN others THEN END; END LOOP; - -- BEGIN - -- CREATE EXTENSION IF NOT EXISTS pgcrypto; - -- CREATE EXTENSION IF NOT EXISTS timescaledb; - -- EXCEPTION - -- WHEN OTHERS THEN - -- NULL; - -- END; - RETURN; END; $$ LANGUAGE plpgsql; diff --git a/tests/profile_abi_decoding.py b/tests/profile_abi_decoding.py index 483fd16ed..ace6b6b9d 100644 --- a/tests/profile_abi_decoding.py +++ b/tests/profile_abi_decoding.py @@ -1,7 +1,7 @@ import time from pathlib import Path -import pprofile # type: ignore[import] +import pprofile # type: ignore[import-untyped] from dipdup.indexes.evm_subsquid_events.matcher import decode_event_data from dipdup.package import EventAbiExtra diff --git a/tests/test_demos.py b/tests/test_demos.py index 57ea16d67..93d810cb9 100644 --- a/tests/test_demos.py +++ b/tests/test_demos.py @@ -4,6 +4,7 @@ from collections.abc import AsyncIterator from collections.abc import Awaitable from collections.abc import Callable +from contextlib import AbstractAsyncContextManager from contextlib import AsyncExitStack from contextlib import asynccontextmanager from decimal import Decimal @@ -13,6 +14,7 @@ import pytest +from dipdup.database import get_connection from dipdup.database import tortoise_wrapper from dipdup.exceptions import FrameworkException from dipdup.models.tezos_tzkt import TzktOperationType @@ -21,50 +23,58 @@ @asynccontextmanager -async def run_dipdup_demo(config: str, package: str, cmd: str = 'run') -> AsyncIterator[Path]: - config_path = CONFIGS_PATH / config - dipdup_pkg_path = SRC_PATH / 'dipdup' - demo_pkg_path = SRC_PATH / package - sqlite_config_path = Path(__file__).parent / 'configs' / 'sqlite.yaml' - - with tempfile.TemporaryDirectory() as tmp_root_path: +async def tmp_project(config_path: Path, package: str, exists: bool) -> AsyncIterator[tuple[Path, dict[str, str]]]: + with tempfile.TemporaryDirectory() as tmp_package_path: # NOTE: Symlink configs, packages and executables - tmp_config_path = Path(tmp_root_path) / 'dipdup.yaml' + tmp_config_path = Path(tmp_package_path) / 'dipdup.yaml' os.symlink(config_path, tmp_config_path) - tmp_bin_path = Path(tmp_root_path) / 'bin' + tmp_bin_path = Path(tmp_package_path) / 'bin' tmp_bin_path.mkdir() for executable in ('dipdup', 'datamodel-codegen'): if (executable_path := which(executable)) is None: raise FrameworkException(f'Executable `{executable}` not found') os.symlink(executable_path, tmp_bin_path / executable) - tmp_dipdup_pkg_path = Path(tmp_root_path) / 'dipdup' - os.symlink(dipdup_pkg_path, tmp_dipdup_pkg_path) + os.symlink( + SRC_PATH / 'dipdup', + Path(tmp_package_path) / 'dipdup', + ) # NOTE: Ensure that `run` uses existing package and `init` creates a new one - if cmd == 'run': - tmp_demo_pkg_path = Path(tmp_root_path) / package - os.symlink(demo_pkg_path, tmp_demo_pkg_path) + if exists: + os.symlink( + SRC_PATH / package, + Path(tmp_package_path) / package, + ) # NOTE: Prepare environment env = { **os.environ, 'PATH': str(tmp_bin_path), - 'PYTHONPATH': str(tmp_root_path), + 'PYTHONPATH': str(tmp_package_path), 'DIPDUP_TEST': '1', } - subprocess.run( - f'dipdup -c {tmp_config_path} -c {sqlite_config_path} {cmd}', - cwd=tmp_root_path, - check=True, - shell=True, - env=env, - capture_output=True, - ) + yield Path(tmp_package_path), env - yield Path(tmp_root_path) + +async def run_in_tmp( + tmp_path: Path, + env: dict[str, str], + *cmd: str, +) -> None: + sqlite_config_path = Path(__file__).parent / 'configs' / 'sqlite.yaml' + tmp_config_path = Path(tmp_path) / 'dipdup.yaml' + + subprocess.run( + f'dipdup -c {tmp_config_path} -c {sqlite_config_path} {" ".join(cmd)}', + cwd=tmp_path, + check=True, + shell=True, + env=env, + capture_output=True, + ) async def assert_run_token() -> None: @@ -246,21 +256,71 @@ async def assert_run_dao() -> None: @pytest.mark.parametrize(test_args, test_params) -async def test_demos( +async def test_run_init( config: str, package: str, cmd: str, assert_fn: Callable[[], Awaitable[None]], ) -> None: + config_path = CONFIGS_PATH / config async with AsyncExitStack() as stack: - tmp_root_path = await stack.enter_async_context( - run_dipdup_demo(config, package, cmd), + tmp_package_path, env = await stack.enter_async_context( + tmp_project( + config_path, + package, + exists=cmd != 'init', + ), ) + await run_in_tmp(tmp_package_path, env, cmd) await stack.enter_async_context( tortoise_wrapper( - f'sqlite://{tmp_root_path}/db.sqlite3', + f'sqlite://{tmp_package_path}/db.sqlite3', f'{package}.models', ) ) await assert_fn() + + +async def _count_tables() -> int: + conn = get_connection() + _, res = await conn.execute_query('SELECT count(name) FROM sqlite_master WHERE type = "table";') + return int(res[0][0]) + + +async def test_schema() -> None: + package = 'demo_token' + config_path = CONFIGS_PATH / f'{package}.yml' + + async with AsyncExitStack() as stack: + tmp_package_path, env = await stack.enter_async_context( + tmp_project( + config_path, + package, + exists=True, + ), + ) + + def tortoise() -> AbstractAsyncContextManager[None]: + return tortoise_wrapper( + f'sqlite://{tmp_package_path}/db.sqlite3', + f'{package}.models', + ) + + async with tortoise(): + conn = get_connection() + assert (await _count_tables()) == 0 + + await run_in_tmp(tmp_package_path, env, 'schema', 'init') + + async with tortoise(): + conn = get_connection() + assert (await _count_tables()) == 10 + await conn.execute_script('CREATE TABLE test (id INTEGER PRIMARY KEY);') + assert (await _count_tables()) == 11 + + await run_in_tmp(tmp_package_path, env, 'schema', 'wipe', '--force') + + async with tortoise(): + conn = get_connection() + assert (await _count_tables()) == 0 diff --git a/tests/test_hasura.py b/tests/test_hasura.py index cc5e5b792..ea233c607 100644 --- a/tests/test_hasura.py +++ b/tests/test_hasura.py @@ -9,7 +9,7 @@ import pytest from aiohttp import web from aiohttp.pytest_plugin import AiohttpClient -from docker.client import DockerClient # type: ignore[import] +from docker.client import DockerClient # type: ignore[import-untyped] from tortoise import Tortoise from dipdup.config import DipDupConfig diff --git a/tests/test_introspection.py b/tests/test_introspection.py index defd556ff..dd5a70b9b 100644 --- a/tests/test_introspection.py +++ b/tests/test_introspection.py @@ -18,8 +18,8 @@ def test_list_simple_args() -> None: assert get_list_elt_type(list[str]) == str assert get_list_elt_type(list[int]) == int assert get_list_elt_type(list[bool]) == bool - assert get_list_elt_type(list[str | None]) == str | None # type: ignore[comparison-overlap] - assert get_list_elt_type(list[str | int]) == str | int # type: ignore[comparison-overlap] + assert get_list_elt_type(list[str | None]) == str | None + assert get_list_elt_type(list[str | int]) == str | int assert get_list_elt_type(list[tuple[str]]) == tuple[str] assert get_list_elt_type(list[list[str]]) == list[str] assert get_list_elt_type(list[dict[str, str]]) == dict[str, str] @@ -30,8 +30,8 @@ class Class: ... assert get_list_elt_type(list[Class]) == Class - assert get_list_elt_type(list[Class | None]) == Class | None # type: ignore[comparison-overlap] - assert get_list_elt_type(list[Class | int]) == Class | int # type: ignore[comparison-overlap] + assert get_list_elt_type(list[Class | None]) == Class | None + assert get_list_elt_type(list[Class | int]) == Class | int assert get_list_elt_type(list[tuple[Class]]) == tuple[Class] assert get_list_elt_type(list[list[Class]]) == list[Class] assert get_list_elt_type(list[dict[str, Class]]) == dict[str, Class] @@ -47,7 +47,7 @@ class SomethingElse(BaseModel): class OptionalList(BaseModel): root: list[str] | None - assert get_list_elt_type(ListOfMapsStorage) == int | dict[str, str] # type: ignore[comparison-overlap] + assert get_list_elt_type(ListOfMapsStorage) == int | dict[str, str] with pytest.raises(IntrospectionError): get_list_elt_type(OptionalList) @@ -60,8 +60,8 @@ def test_dict_simple_args() -> None: assert get_dict_value_type(dict[str, str]) == str assert get_dict_value_type(dict[str, int]) == int assert get_dict_value_type(dict[str, bool]) == bool - assert get_dict_value_type(dict[str, str | None]) == str | None # type: ignore[comparison-overlap] - assert get_dict_value_type(dict[str, str | int]) == str | int # type: ignore[comparison-overlap] + assert get_dict_value_type(dict[str, str | None]) == str | None + assert get_dict_value_type(dict[str, str | int]) == str | int assert get_dict_value_type(dict[str, tuple[str]]) == tuple[str] assert get_dict_value_type(dict[str, list[str]]) == list[str] assert get_dict_value_type(dict[str, dict[str, str]]) == dict[str, str] @@ -72,8 +72,8 @@ class Class: ... assert get_dict_value_type(dict[str, Class]) == Class - assert get_dict_value_type(dict[str, Class | None]) == Class | None # type: ignore[comparison-overlap] - assert get_dict_value_type(dict[str, Class | int]) == Class | int # type: ignore[comparison-overlap] + assert get_dict_value_type(dict[str, Class | None]) == Class | None + assert get_dict_value_type(dict[str, Class | int]) == Class | int assert get_dict_value_type(dict[str, tuple[Class]]) == tuple[Class] assert get_dict_value_type(dict[str, list[Class]]) == list[Class] assert get_dict_value_type(dict[str, dict[str, Class]]) == dict[str, Class] @@ -89,7 +89,7 @@ class SomethingElse(RootModel[Any]): class OptionalDict(RootModel[Any]): root: dict[str, str] | None - assert get_dict_value_type(DictOfMapsStorage) == int | dict[str, str] # type: ignore[comparison-overlap] + assert get_dict_value_type(DictOfMapsStorage) == int | dict[str, str] with pytest.raises(IntrospectionError): get_dict_value_type(OptionalDict) @@ -108,8 +108,8 @@ class Storage(BaseModel): assert get_dict_value_type(Storage, 'plain_str') == str assert get_dict_value_type(Storage, 'list_str') == list[str] assert get_dict_value_type(Storage, 'dict_of_lists') == dict[str, list[str]] - assert get_dict_value_type(Storage, 'optional_str') == str | None # type: ignore[comparison-overlap] - assert get_dict_value_type(Storage, 'union_arg') == str | int # type: ignore[comparison-overlap] + assert get_dict_value_type(Storage, 'optional_str') == str | None + assert get_dict_value_type(Storage, 'union_arg') == str | int def test_is_array() -> None: @@ -147,6 +147,6 @@ class OptionalStr(RootModel[Any]): class ListOfMapsStorage(RootModel[Any]): root: list[int | dict[str, str]] - assert extract_root_outer_type(OptionalStr) == str | None # type: ignore[comparison-overlap] + assert extract_root_outer_type(OptionalStr) == str | None # FIXME: left operand type: "Type[BaseModel]", right operand type: "Type[list[Any]]" assert extract_root_outer_type(ListOfMapsStorage) == list[int | dict[str, str]] # type: ignore[comparison-overlap]