eth-portfolio-temp 0.2.12__cp313-cp313-win32.whl

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of eth-portfolio-temp might be problematic. Click here for more details.

Files changed (83) hide show
  1. eth_portfolio/__init__.py +25 -0
  2. eth_portfolio/_argspec.cp313-win32.pyd +0 -0
  3. eth_portfolio/_argspec.py +42 -0
  4. eth_portfolio/_cache.py +121 -0
  5. eth_portfolio/_config.cp313-win32.pyd +0 -0
  6. eth_portfolio/_config.py +4 -0
  7. eth_portfolio/_db/__init__.py +0 -0
  8. eth_portfolio/_db/decorators.py +147 -0
  9. eth_portfolio/_db/entities.py +311 -0
  10. eth_portfolio/_db/utils.py +604 -0
  11. eth_portfolio/_decimal.py +156 -0
  12. eth_portfolio/_decorators.py +84 -0
  13. eth_portfolio/_exceptions.py +67 -0
  14. eth_portfolio/_ledgers/__init__.py +0 -0
  15. eth_portfolio/_ledgers/address.py +938 -0
  16. eth_portfolio/_ledgers/portfolio.py +327 -0
  17. eth_portfolio/_loaders/__init__.py +33 -0
  18. eth_portfolio/_loaders/_nonce.cp313-win32.pyd +0 -0
  19. eth_portfolio/_loaders/_nonce.py +196 -0
  20. eth_portfolio/_loaders/balances.cp313-win32.pyd +0 -0
  21. eth_portfolio/_loaders/balances.py +94 -0
  22. eth_portfolio/_loaders/token_transfer.py +217 -0
  23. eth_portfolio/_loaders/transaction.py +240 -0
  24. eth_portfolio/_loaders/utils.cp313-win32.pyd +0 -0
  25. eth_portfolio/_loaders/utils.py +68 -0
  26. eth_portfolio/_shitcoins.cp313-win32.pyd +0 -0
  27. eth_portfolio/_shitcoins.py +329 -0
  28. eth_portfolio/_stableish.cp313-win32.pyd +0 -0
  29. eth_portfolio/_stableish.py +42 -0
  30. eth_portfolio/_submodules.py +73 -0
  31. eth_portfolio/_utils.py +225 -0
  32. eth_portfolio/_ydb/__init__.py +0 -0
  33. eth_portfolio/_ydb/token_transfers.py +145 -0
  34. eth_portfolio/address.py +397 -0
  35. eth_portfolio/buckets.py +194 -0
  36. eth_portfolio/constants.cp313-win32.pyd +0 -0
  37. eth_portfolio/constants.py +82 -0
  38. eth_portfolio/portfolio.py +661 -0
  39. eth_portfolio/protocols/__init__.py +67 -0
  40. eth_portfolio/protocols/_base.py +108 -0
  41. eth_portfolio/protocols/convex.py +17 -0
  42. eth_portfolio/protocols/dsr.py +51 -0
  43. eth_portfolio/protocols/lending/README.md +6 -0
  44. eth_portfolio/protocols/lending/__init__.py +50 -0
  45. eth_portfolio/protocols/lending/_base.py +57 -0
  46. eth_portfolio/protocols/lending/compound.py +187 -0
  47. eth_portfolio/protocols/lending/liquity.py +110 -0
  48. eth_portfolio/protocols/lending/maker.py +104 -0
  49. eth_portfolio/protocols/lending/unit.py +46 -0
  50. eth_portfolio/protocols/liquity.py +16 -0
  51. eth_portfolio/py.typed +0 -0
  52. eth_portfolio/structs/__init__.py +43 -0
  53. eth_portfolio/structs/modified.py +69 -0
  54. eth_portfolio/structs/structs.py +637 -0
  55. eth_portfolio/typing/__init__.py +1447 -0
  56. eth_portfolio/typing/balance/single.py +176 -0
  57. eth_portfolio__mypyc.cp313-win32.pyd +0 -0
  58. eth_portfolio_scripts/__init__.py +20 -0
  59. eth_portfolio_scripts/_args.py +26 -0
  60. eth_portfolio_scripts/_logging.py +15 -0
  61. eth_portfolio_scripts/_portfolio.py +194 -0
  62. eth_portfolio_scripts/_utils.py +106 -0
  63. eth_portfolio_scripts/balances.cp313-win32.pyd +0 -0
  64. eth_portfolio_scripts/balances.py +52 -0
  65. eth_portfolio_scripts/docker/.grafana/dashboards/Portfolio/Balances.json +1962 -0
  66. eth_portfolio_scripts/docker/.grafana/dashboards/dashboards.yaml +10 -0
  67. eth_portfolio_scripts/docker/.grafana/datasources/datasources.yml +11 -0
  68. eth_portfolio_scripts/docker/__init__.cp313-win32.pyd +0 -0
  69. eth_portfolio_scripts/docker/__init__.py +16 -0
  70. eth_portfolio_scripts/docker/check.cp313-win32.pyd +0 -0
  71. eth_portfolio_scripts/docker/check.py +56 -0
  72. eth_portfolio_scripts/docker/docker-compose.yaml +61 -0
  73. eth_portfolio_scripts/docker/docker_compose.cp313-win32.pyd +0 -0
  74. eth_portfolio_scripts/docker/docker_compose.py +78 -0
  75. eth_portfolio_scripts/main.py +119 -0
  76. eth_portfolio_scripts/py.typed +1 -0
  77. eth_portfolio_scripts/victoria/__init__.py +73 -0
  78. eth_portfolio_scripts/victoria/types.py +38 -0
  79. eth_portfolio_temp-0.2.12.dist-info/METADATA +25 -0
  80. eth_portfolio_temp-0.2.12.dist-info/RECORD +83 -0
  81. eth_portfolio_temp-0.2.12.dist-info/WHEEL +5 -0
  82. eth_portfolio_temp-0.2.12.dist-info/entry_points.txt +2 -0
  83. eth_portfolio_temp-0.2.12.dist-info/top_level.txt +3 -0
@@ -0,0 +1,10 @@
1
+ apiVersion: 1
2
+
3
+ providers:
4
+ - name: 'Portfolio dashboards'
5
+ folder: 'Portfolio'
6
+ type: file
7
+ disableDeletion: true
8
+ editable: false
9
+ options:
10
+ path: /etc/grafana/provisioning/dashboards/Portfolio
@@ -0,0 +1,11 @@
1
+ apiVersion: 1
2
+
3
+ datasources:
4
+ - name: 'PROMETHEUS'
5
+ type: 'prometheus'
6
+ access: 'proxy'
7
+ org_id: 1
8
+ url: 'http://victoria-metrics:8428'
9
+ is_default: true
10
+ version: 1
11
+ editable: false
@@ -0,0 +1,16 @@
1
+ from eth_portfolio_scripts.docker.check import check_docker, check_docker_compose, check_system
2
+ from eth_portfolio_scripts.docker.docker_compose import build, down, ensure_containers, up, stop
3
+
4
+ __all__ = [
5
+ # commands
6
+ "build",
7
+ "up",
8
+ "down",
9
+ "stop",
10
+ # decorators
11
+ "ensure_containers",
12
+ # checks
13
+ "check_docker",
14
+ "check_docker_compose",
15
+ "check_system",
16
+ ]
@@ -0,0 +1,56 @@
1
+ from functools import lru_cache
2
+ from subprocess import CalledProcessError, check_output
3
+
4
+
5
+ def check_docker() -> None:
6
+ """
7
+ Check that docker is installed on the user's system.
8
+
9
+ Raises:
10
+ RuntimeError: If docker is not installed.
11
+ """
12
+ try:
13
+ check_output(["docker", "--version"])
14
+ print("docker found!")
15
+ except (CalledProcessError, FileNotFoundError):
16
+ print("checking your computer for docker")
17
+ raise RuntimeError(
18
+ "Docker is not installed. You must install Docker before using dao-treasury."
19
+ ) from None
20
+
21
+
22
+ def check_docker_compose() -> None:
23
+ """
24
+ Check that docker-compose is installed on the user's system.
25
+
26
+ Raises:
27
+ RuntimeError: If docker-compose is not installed.
28
+ """
29
+ try:
30
+ check_output(["docker-compose", "--version"])
31
+ print("docker-compose found!")
32
+ except (CalledProcessError, FileNotFoundError):
33
+ print("checking your computer for docker-compose")
34
+ try:
35
+ check_output(["docker", "compose", "--version"])
36
+ print("docker compose found!")
37
+ except (CalledProcessError, FileNotFoundError):
38
+ print("docker-compose not found, checking your computer for docker compose")
39
+ raise RuntimeError(
40
+ "Docker Compose is not installed. You must install Docker Compose before using dao-treasury."
41
+ ) from None
42
+
43
+
44
+ @lru_cache(maxsize=None)
45
+ def check_system() -> None:
46
+ """
47
+ Check that docker and docker-compose is installed on the user's system.
48
+
49
+ Raises:
50
+ RuntimeError: If docker-compose is not installed.
51
+ """
52
+ check_docker()
53
+ check_docker_compose()
54
+
55
+
56
+ __all__ = ["check_docker", "check_docker_compose", "check_system"]
@@ -0,0 +1,61 @@
1
+ networks:
2
+ eth_portfolio:
3
+
4
+ services:
5
+ grafana:
6
+ image: grafana/grafana:12.2.1
7
+ ports:
8
+ - 127.0.0.1:${GRAFANA_PORT:-3000}:3000
9
+ environment:
10
+ - GF_SECURITY_ADMIN_USER=${GF_SECURITY_ADMIN_USER:-admin}
11
+ - GF_SECURITY_ADMIN_PASSWORD=${GF_SECURITY_ADMIN_PASSWORD:-admin}
12
+ - GF_AUTH_ANONYMOUS_ENABLED=true
13
+ - GF_DASHBOARDS_DEFAULT_HOME_DASHBOARD_PATH=/etc/grafana/provisioning/dashboards/portfolio/Balances.json
14
+ - GF_SERVER_ROOT_URL
15
+ - GF_RENDERING_SERVER_URL=http://renderer:8091/render
16
+ - GF_RENDERING_CALLBACK_URL=http://grafana:3000/
17
+ - GF_LOG_FILTERS=rendering:debug
18
+ #- GF_INSTALL_PLUGINS=volkovlabs-variable-panel
19
+ #command: >
20
+ # sh -c "grafana-cli plugins install frser-sqlite-datasource && /run.sh"
21
+ volumes:
22
+ - ./.grafana/:/etc/grafana/provisioning/
23
+ networks:
24
+ - eth_portfolio
25
+ restart: always
26
+
27
+ renderer:
28
+ platform: linux/amd64
29
+ image: grafana/grafana-image-renderer:latest
30
+ ports:
31
+ - 127.0.0.1:${RENDERER_PORT:-8091}:8091
32
+ environment:
33
+ - ENABLE_METRICS=true
34
+ - HTTP_PORT=8091
35
+ networks:
36
+ - eth_portfolio
37
+ restart: always
38
+
39
+ vmagent:
40
+ image: victoriametrics/vmagent:heads-public-single-node-0-g52eb9c99e
41
+ command:
42
+ - "-remoteWrite.url=http://victoria-metrics:8428/api/v1/write"
43
+ depends_on:
44
+ - victoria-metrics
45
+ networks:
46
+ - eth_portfolio
47
+ restart: always
48
+
49
+ victoria-metrics:
50
+ image: victoriametrics/victoria-metrics:v1.129.0
51
+ volumes:
52
+ - ~/.eth-portfolio/data/victoria/:/victoria-metrics-data
53
+ command:
54
+ - '-memory.allowedBytes=3GB'
55
+ - "-retentionPeriod=10y"
56
+ - "-search.disableAutoCacheReset=true"
57
+ ports:
58
+ - 127.0.0.1:${VICTORIA_PORT:-8428}:8428
59
+ networks:
60
+ - eth_portfolio
61
+ restart: always
@@ -0,0 +1,78 @@
1
+ import logging
2
+ from functools import wraps
3
+ from importlib import resources
4
+ from os import path
5
+ from subprocess import CalledProcessError, check_output
6
+ from typing import Callable, Final, Iterable, List, Tuple, TypeVar
7
+
8
+ from typing_extensions import ParamSpec
9
+
10
+ from eth_portfolio_scripts.docker.check import check_system
11
+
12
+
13
+ logger: Final = logging.getLogger(__name__)
14
+
15
+ compose_file: Final = str(
16
+ resources.files("eth_portfolio_scripts").joinpath("docker/docker-compose.yaml")
17
+ )
18
+
19
+
20
+ def up(*services: str) -> None:
21
+ build(*services)
22
+ print("starting the infra containers...")
23
+ _exec_command(["up", "-d", *services])
24
+
25
+
26
+ def down() -> None:
27
+ _exec_command(["down"])
28
+
29
+
30
+ def build(*services: str) -> None:
31
+ print("building the grafana containers")
32
+ _exec_command(["build", *services])
33
+
34
+
35
+ def stop(container_name: str) -> None:
36
+ """
37
+ Stop the specified container if it is running.
38
+ Defaults to stopping the 'renderer' container.
39
+ """
40
+ print(f"stopping the {container_name} container...")
41
+ _exec_command(["stop", container_name])
42
+
43
+
44
+ _P = ParamSpec("_P")
45
+ _T = TypeVar("_T")
46
+
47
+
48
+ def ensure_containers(fn: Callable[_P, _T]) -> Callable[_P, _T]:
49
+ @wraps(fn)
50
+ async def compose_wrap(*args: _P.args, **kwargs: _P.kwargs) -> _T:
51
+ # register shutdown sequence
52
+ # TODO: argument to leave them up
53
+ # NOTE: do we need both this and the finally?
54
+ # signal.signal(signal.SIGINT, down)
55
+
56
+ # start Grafana containers
57
+ up()
58
+
59
+ try:
60
+ # attempt to run `fn`
61
+ return await fn(*args, **kwargs)
62
+ finally:
63
+ # stop and remove containers
64
+ # down()
65
+ pass
66
+
67
+ return compose_wrap
68
+
69
+
70
+ def _exec_command(command: List[str], *, compose_options: Tuple[str, ...] = ()) -> None:
71
+ check_system()
72
+ try:
73
+ check_output(["docker", "compose", *compose_options, "-f", compose_file, *command])
74
+ except (CalledProcessError, FileNotFoundError) as e:
75
+ try:
76
+ check_output(["docker-compose", *compose_options, "-f", compose_file, *command])
77
+ except (CalledProcessError, FileNotFoundError) as _e:
78
+ raise RuntimeError(f"Error occurred while running {' '.join(command)}: {_e}") from _e
@@ -0,0 +1,119 @@
1
+ import asyncio
2
+ from argparse import ArgumentParser
3
+ from os import environ
4
+
5
+ import brownie
6
+
7
+ from eth_portfolio_scripts import docker, logger
8
+ from eth_portfolio_scripts._args import add_infra_port_args
9
+ from eth_portfolio_scripts.balances import export_balances
10
+
11
+
12
+ parser = ArgumentParser(description="eth-portfolio")
13
+
14
+ subparsers = parser.add_subparsers(title="Commands", dest="command", required=True)
15
+
16
+ export_parser = subparsers.add_parser("export", help="Export a specific dataset for your portfolio")
17
+ export_parser.add_argument("target", help="Choose an exporter to run")
18
+ add_infra_port_args(export_parser)
19
+ export_parser.set_defaults(func=export_balances)
20
+
21
+ infra_parser = subparsers.add_parser(
22
+ "infra", help="Start the docker containers required to run any eth-portfolio service"
23
+ )
24
+ infra_parser.add_argument("cmd", help="What do you want to do?")
25
+ add_infra_port_args(infra_parser)
26
+ infra_parser.add_argument(
27
+ "--start-renderer",
28
+ action="store_true",
29
+ help="If set, starts the renderer container in addition to the default containers. By default, only grafana, victoria-metrics, and vmagent are started.",
30
+ )
31
+
32
+ export_parser.add_argument(
33
+ "--wallet",
34
+ type=str,
35
+ help="The address of a wallet to export. You can pass multiple, ie. `--wallet 0x123 0x234 0x345`",
36
+ required=True,
37
+ nargs="+",
38
+ )
39
+ export_parser.add_argument(
40
+ "--network",
41
+ type=str,
42
+ help="The brownie network identifier for the rpc you wish to use. default: mainnet",
43
+ default="mainnet",
44
+ )
45
+ export_parser.add_argument(
46
+ "--label",
47
+ type=str,
48
+ help='The label for this portfolio, if you want one. Defaults to "MyPortfolio".',
49
+ default="My Portfolio",
50
+ )
51
+ export_parser.add_argument(
52
+ "--interval",
53
+ type=str,
54
+ help="The time interval between datapoints. default: 6h",
55
+ default="6h",
56
+ )
57
+ export_parser.add_argument(
58
+ "--concurrency",
59
+ type=int,
60
+ help="The max number of historical blocks to export concurrently. default: 40",
61
+ default=40,
62
+ )
63
+ export_parser.add_argument(
64
+ "--first-tx-block",
65
+ type=int,
66
+ help=(
67
+ "The block of your portfolio's first transaction, if known. "
68
+ "This value, if provided, allows us to speed up processing of your data by limiting the block range we need to query. "
69
+ "If not provided, the whole blockchain will be scanned."
70
+ ),
71
+ default=0,
72
+ )
73
+ export_parser.add_argument(
74
+ "--export-start-block",
75
+ type=int,
76
+ help="The first block in the range you wish to export.",
77
+ default=0,
78
+ )
79
+ export_parser.add_argument(
80
+ "--daemon",
81
+ action="store_true",
82
+ help="TODO: If True, starts a daemon process instead of running in your terminal. Not currently supported.",
83
+ )
84
+
85
+ args = parser.parse_args()
86
+
87
+ if hasattr(args, "network"):
88
+ environ["BROWNIE_NETWORK_ID"] = args.network
89
+
90
+ environ["GRAFANA_PORT"] = str(args.grafana_port)
91
+ environ["RENDERER_PORT"] = str(args.renderer_port)
92
+ environ["VICTORIA_PORT"] = str(args.victoria_port)
93
+
94
+
95
+ # TODO: run forever arg
96
+ def main():
97
+ command = args.command
98
+ if command == "infra":
99
+ if args.cmd == "start":
100
+ # Start the backend containers
101
+ if getattr(args, "start_renderer", False):
102
+ docker.up()
103
+ else:
104
+ docker.up("grafana", "victoria-metrics", "vmagent")
105
+ elif args.cmd == "stop":
106
+ docker.down()
107
+ else:
108
+ raise ValueError(f"{args.target} is not a valid command")
109
+
110
+ else:
111
+ # The user's command is `export`
112
+ if args.target == "balances":
113
+ asyncio.run(args.func(args))
114
+ else:
115
+ raise ValueError(f"{args.target} is not a valid command")
116
+
117
+
118
+ if __name__ == "__main__":
119
+ brownie.project.run(__file__)
@@ -0,0 +1 @@
1
+
@@ -0,0 +1,73 @@
1
+ from gzip import compress
2
+ from logging import getLogger
3
+ from os import environ
4
+ from typing import Final, List
5
+
6
+ import a_sync
7
+ from aiohttp import ClientError, ClientSession, ServerDisconnectedError
8
+ from msgspec import json
9
+
10
+ from eth_portfolio_scripts.victoria import types
11
+ from eth_portfolio_scripts.victoria.types import Metric
12
+
13
+
14
+ BASE_URL: Final = environ.get("VM_URL", "http://127.0.0.1:8430")
15
+
16
+ # this will be populated later
17
+ session: ClientSession = None # type: ignore [assignment]
18
+
19
+ logger: Final = getLogger("eth_portfolio.victoria")
20
+
21
+ encode: Final = json.encode
22
+ decode: Final = json.decode
23
+
24
+
25
+ class VictoriaMetricsError(ValueError): ...
26
+
27
+
28
+ def get_session() -> ClientSession:
29
+ sesh = session
30
+ if sesh is None:
31
+ sesh = ClientSession(BASE_URL, raise_for_status=True)
32
+ __set_session(sesh)
33
+ return sesh
34
+
35
+
36
+ async def get(url: str) -> bytes:
37
+ session = get_session()
38
+ while True:
39
+ try:
40
+ async with session.get(url=url, headers={"Connection": "close"}) as response:
41
+ return await response.read()
42
+ except ServerDisconnectedError:
43
+ continue
44
+
45
+
46
+ @a_sync.Semaphore(2)
47
+ async def post_data(metrics_to_export: List["Metric"]) -> None:
48
+ """Post all metrics at once."""
49
+ data = compress(b"\n".join(encode(metric) for metric in metrics_to_export))
50
+ attempts = 0
51
+ session = get_session()
52
+ while True:
53
+ try:
54
+ async with session.post(
55
+ "/api/v1/import",
56
+ headers={"Connection": "close", "Content-Encoding": "gzip"},
57
+ data=data,
58
+ ):
59
+ logger.debug(f"posted {len(data)} datas")
60
+ return
61
+ except ClientError as e:
62
+ attempts += 1
63
+ logger.debug("You had a ClientError: %s", e)
64
+ if attempts >= 10:
65
+ raise e
66
+
67
+
68
+ def __set_session(sesh: ClientSession) -> None:
69
+ global session
70
+ session = sesh
71
+
72
+
73
+ __all__ = ["Metric", "get", "post_data"]
@@ -0,0 +1,38 @@
1
+ """
2
+ These 2 helpers enable us to decode only the relevant data in the json response and discard the rest
3
+ They're in a separate file so mypyc doesn't try to compile them
4
+ """
5
+
6
+ from typing import List, Tuple, TypedDict, final
7
+
8
+ from eth_typing import ChecksumAddress
9
+ from msgspec import Raw, Struct
10
+
11
+
12
+ @final
13
+ class Metric(TypedDict):
14
+ param: str
15
+ wallet: ChecksumAddress
16
+ token_address: ChecksumAddress
17
+ token: str
18
+ bucket: str
19
+ network: str
20
+ __name__: str
21
+
22
+
23
+ @final
24
+ class PrometheusItem(TypedDict):
25
+ metric: Metric
26
+ values: List[float]
27
+ timestamps: List[float]
28
+
29
+
30
+ @final
31
+ class Data(Struct):
32
+ result: Tuple[Raw, ...]
33
+
34
+
35
+ @final
36
+ class Response(Struct):
37
+ status: str
38
+ data: Data
@@ -0,0 +1,25 @@
1
+ Metadata-Version: 2.4
2
+ Name: eth_portfolio_temp
3
+ Version: 0.2.12
4
+ Summary: eth-portfolio makes it easy to analyze your portfolio.
5
+ Home-page: https://github.com/BobTheBuidler/eth-portfolio
6
+ Author: BobTheBuidler
7
+ Author-email: bobthebuidlerdefi@gmail.com
8
+ Requires-Python: >=3.10,<3.14
9
+ Requires-Dist: checksum_dict>=2.1.7
10
+ Requires-Dist: dank_mids>=4.20.154
11
+ Requires-Dist: eth-brownie<1.23,>=1.22.0.dev0
12
+ Requires-Dist: eth_retry<1,>=0.3.4
13
+ Requires-Dist: evmspec>=0.4.1
14
+ Requires-Dist: ez-a-sync>=0.32.27
15
+ Requires-Dist: faster-eth-utils
16
+ Requires-Dist: numpy<3
17
+ Requires-Dist: pandas<3,>=1.4.3
18
+ Requires-Dist: typed-envs>=0.0.9
19
+ Requires-Dist: ypricemagic<5,>=4.10.0
20
+ Dynamic: author
21
+ Dynamic: author-email
22
+ Dynamic: home-page
23
+ Dynamic: requires-dist
24
+ Dynamic: requires-python
25
+ Dynamic: summary
@@ -0,0 +1,83 @@
1
+ eth_portfolio__mypyc.cp313-win32.pyd,sha256=vVC94m1qsef8uoQ9-GDlCoUUqIXB66YKYCQO4fNk6oc,260608
2
+ eth_portfolio/__init__.py,sha256=5GmHYzhUX2gOC4MScgXnqS8x2MawbpWOuWPmqlvCotA,525
3
+ eth_portfolio/_argspec.cp313-win32.pyd,sha256=ImBMGTrGZHvemt350-PHcBA4pmnzYTf-7u5E5Q7Kgls,9216
4
+ eth_portfolio/_argspec.py,sha256=au8ycJ56-7UzbQVzYUOoMX9sToH0QNi2tS1O820xB3A,1637
5
+ eth_portfolio/_cache.py,sha256=OntwbiDOrBV5JzHHDzJlxcEVrmKK5GG0uXeLeoDDbpA,4926
6
+ eth_portfolio/_config.cp313-win32.pyd,sha256=Nd2pfFYcNRMqQSil6--1iagJ53taSN2ViZWqu3pJFh4,9216
7
+ eth_portfolio/_config.py,sha256=BSOvFS4D0oqgVTtBbtW2g2kpL8Mk6xiP0PipfXs_91A,102
8
+ eth_portfolio/_decimal.py,sha256=DwLUg8pzeTIREG2sHgRK48hoWmjelWNvpEKj10Ra7to,5002
9
+ eth_portfolio/_decorators.py,sha256=AvB-q69MvaFe0Ykh1os6hSewcUujURQ9onqobMaONOM,2899
10
+ eth_portfolio/_exceptions.py,sha256=xb5VmCtBtXy9WXu_1ofoG4j9BJsMQXMVydCJNUulrNY,2500
11
+ eth_portfolio/_shitcoins.cp313-win32.pyd,sha256=vXHwkex9YngjVrS9iGQagqxecXKst5P7CqLYxjLmFzk,9216
12
+ eth_portfolio/_shitcoins.py,sha256=OZLQWpuNegNz_GXPod1yMZMLiryKY0ZL_fpq1HppnGM,17296
13
+ eth_portfolio/_stableish.cp313-win32.pyd,sha256=w5dzl6RZ4khiyMR8zIRvDKxiHAcMhRXsRPfY23zsOIM,9216
14
+ eth_portfolio/_stableish.py,sha256=8pdlYe2YcZ8rTzdCzg-ihRQLPtSFXBNDjtnvuTgPRcQ,2063
15
+ eth_portfolio/_submodules.py,sha256=zne-2YyVFoKHwkLuHx8cdz5lpcwwSDw1edJ9v8G4c1A,2263
16
+ eth_portfolio/_utils.py,sha256=iGck5SKUwpfnuZvLnKDR1A00obZNKx622Y9GY_TYiZA,7886
17
+ eth_portfolio/address.py,sha256=RrFIWSx6jSITwh16Hqs6W6S9fz1KEaebQEk1vqL3HwI,14536
18
+ eth_portfolio/buckets.py,sha256=c5_RLfMyJbnK_PBx5mmdL3YlIX67dD6LqnBxiAr52N0,6896
19
+ eth_portfolio/constants.cp313-win32.pyd,sha256=4mqpw8aZhJLyd16iOlxqGP1XhcmWMprPNnZDNrYcZTA,9216
20
+ eth_portfolio/constants.py,sha256=rUXWFaCGrrF_qzIdDvVzakQJsLnrGtFNn2EgRbZh8fQ,3741
21
+ eth_portfolio/portfolio.py,sha256=FSJ1_FG120dN5vWSPyhbHfyuFHdmFqTHJ5_Ci-4Rts4,24909
22
+ eth_portfolio/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
23
+ eth_portfolio/_db/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
24
+ eth_portfolio/_db/decorators.py,sha256=RS-RiuOAgbrMKQpCwwAtP3lPRs8vKM6aPhh9LP4SwtI,5308
25
+ eth_portfolio/_db/entities.py,sha256=1Z1AsR400SJYzDrDGxUJt1HuSKB3ErvxIytFtju7UHU,10345
26
+ eth_portfolio/_db/utils.py,sha256=3P4EVQ5cHLbDrAnBESAIKhoDpJLXLS8IDJNnsh7T2Sc,21586
27
+ eth_portfolio/_ledgers/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
28
+ eth_portfolio/_ledgers/address.py,sha256=wght92UXf6UIO9GNRzdMy3H43Em7kUk9ArtBsRF-xjw,33830
29
+ eth_portfolio/_ledgers/portfolio.py,sha256=rAr5S1tksgLPUOK1J1FbIWgvTQNzTqrRp8oukiP2BTo,13427
30
+ eth_portfolio/_loaders/__init__.py,sha256=YrBeRuA48cN2Bg2R2nXiNt1CsUIG_7N6ENIpY6aISPo,1735
31
+ eth_portfolio/_loaders/_nonce.cp313-win32.pyd,sha256=J7_r1hzcS4LXEriPF6Cr28lRQiEFPlXN6CcWl3LpFQY,9216
32
+ eth_portfolio/_loaders/_nonce.py,sha256=bA-5fZG8ALuADgQGrnoxt6p6j7QEwI6vIGnZRF1Zx7M,6481
33
+ eth_portfolio/_loaders/balances.cp313-win32.pyd,sha256=vo9NdCij4CC5rauLQrxnSPH2SfHtRPoQHpLwFVcILz4,9216
34
+ eth_portfolio/_loaders/balances.py,sha256=_v-x1M3lzDHPt8svHmCcpxZKw0BaGHfZimhkwgphY8A,3226
35
+ eth_portfolio/_loaders/token_transfer.py,sha256=itaWVDyotG0EYPb4o94UOpxYvTdE6YIuUR2pzpXAnwI,8748
36
+ eth_portfolio/_loaders/transaction.py,sha256=oOG3cFdLAa10k2hpN17GO7Wtq-rpUJ1oo7hLAKDjbDc,9393
37
+ eth_portfolio/_loaders/utils.cp313-win32.pyd,sha256=4LP_ZJ4n9Bjox9hXzr29vEGbThxxHNQfMWRYz-nxyZc,9216
38
+ eth_portfolio/_loaders/utils.py,sha256=gJvIikinjCPj-elQVep55VTya9slSARApx-FJKYwnkE,2333
39
+ eth_portfolio/_ydb/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
40
+ eth_portfolio/_ydb/token_transfers.py,sha256=HUTeWShcQD4IzocTHmBOreml0jbVttLcaed1fNiJ2qg,5297
41
+ eth_portfolio/protocols/__init__.py,sha256=k1e4MI8KPALIg7awaVA9LbBuEjqxWZF7uGgxysQ4y90,2623
42
+ eth_portfolio/protocols/_base.py,sha256=nuMnnltg4HZ0Z5Nl83ofwAsTE3A9jLYhDZFkRhAKIIY,3793
43
+ eth_portfolio/protocols/convex.py,sha256=az8FhxLtXf0WKo25Ke0IDRGF8x1yMlffoCRz9aG2fGE,535
44
+ eth_portfolio/protocols/dsr.py,sha256=jn_4niDINXQxqu9Tfb77vsfbhv4oIRRFKsKvnMgEO4I,1806
45
+ eth_portfolio/protocols/liquity.py,sha256=Ptiem5zd1op6qmZj2d6PmcgSqrKCLhlTw3CmIwII1Dc,656
46
+ eth_portfolio/protocols/lending/README.md,sha256=0C941RnyfQwT1DE9bvxwLP-gBTHlQcQysF-75FXHINo,498
47
+ eth_portfolio/protocols/lending/__init__.py,sha256=4jKDPx3sb7Iyi9kOdJRpM6LJy_GhUpviLIbpV_6noYU,1693
48
+ eth_portfolio/protocols/lending/_base.py,sha256=_1zXroYHlliLP685xiRfw8I_tQOoZyaPlnxA_gzLytU,2311
49
+ eth_portfolio/protocols/lending/compound.py,sha256=1aoksFSvIIsk-cNDs2-Kdwl-RtIdTzh3QgpksXG1kEs,7439
50
+ eth_portfolio/protocols/lending/liquity.py,sha256=bJ5TKD-WTVn3S1tlbSGGVFCoKFvgStHVOzKBM2LbsRY,4294
51
+ eth_portfolio/protocols/lending/maker.py,sha256=zqi1E5yQ_QDFOk5OMR4gB5_Ag74RMIkp6u3TyX-P3fg,4073
52
+ eth_portfolio/protocols/lending/unit.py,sha256=OQGDZWdsTOUL4jOKJ1xef6wXhWLHnsRFJjAzn0XSBmM,2011
53
+ eth_portfolio/structs/__init__.py,sha256=x-9CdKe8XMukp8Kjr_lD49ohrSd0iGEvvsMAHtAPrLI,1486
54
+ eth_portfolio/structs/modified.py,sha256=QIuFh-u5VTHe0oborn3oHAiAGD0wqhUQW7XFJVZu2KI,1853
55
+ eth_portfolio/structs/structs.py,sha256=TDJ4u3E_H8uG7iX-C1Pux8TesY2zXRhQUJxmN7cp4cc,20490
56
+ eth_portfolio/typing/__init__.py,sha256=v_KbOgqTKcR4y-05Bel_mg6uAV-IsGurGR-jDW46SMg,58763
57
+ eth_portfolio/typing/balance/single.py,sha256=NmtWXqCWMIkaoyvfL67Rh0wWurLf55fVaDkyUxZ27oY,6501
58
+ eth_portfolio_scripts/__init__.py,sha256=DIBnQmjmhmNL1lO9Lq4QptrZmC7u6_N3p9zRjcZ9vbY,281
59
+ eth_portfolio_scripts/_args.py,sha256=M33vPkja62XEJeCZAqacNSBCqbzse07wepwyBOtkvVo,682
60
+ eth_portfolio_scripts/_logging.py,sha256=EgW8ozQZLAgt_cUgqe5BYZLMVQwB6X-0gq0T-BvqlTY,369
61
+ eth_portfolio_scripts/_portfolio.py,sha256=XP6fUfZfv2hBdW6PCBdpUyT3TRAU56u2QKWwhmHBTAE,7333
62
+ eth_portfolio_scripts/_utils.py,sha256=lj2B79G8YX21ATNp_v-onsU_kNFZF3unBclCVp3AIn8,3163
63
+ eth_portfolio_scripts/balances.cp313-win32.pyd,sha256=lvBa2f_iOqgxZsvEsde_bgD_cSTVZynhXk3IWC7aKyk,9216
64
+ eth_portfolio_scripts/balances.py,sha256=Fr8NGCt5yhvu7LnLH27oOVKcL2XwowmAVezXUOIEzd0,1640
65
+ eth_portfolio_scripts/main.py,sha256=VfwWEVOu4oxp5M-rQLGo-XlDKlG_Q7Y4cLiXMUSXeZg,3804
66
+ eth_portfolio_scripts/py.typed,sha256=frcCV1k9oG9oKj3dpUqdJg1PxRT2RSN_XKdLCPjaYaY,2
67
+ eth_portfolio_scripts/docker/__init__.cp313-win32.pyd,sha256=Xua6qxx1Hkc1h9cggKio9ANTuJ2n4oq5sVDUqfZQ1Dw,9216
68
+ eth_portfolio_scripts/docker/__init__.py,sha256=R27uZPLUEK2--fb9IrxrDUk6R5_IVFJSv7s_ia7p5Xk,409
69
+ eth_portfolio_scripts/docker/check.cp313-win32.pyd,sha256=uWd4L4yBVY_hPdPRcQZ-qBp3Q9oTnb6Wjcjo5ifgV14,9216
70
+ eth_portfolio_scripts/docker/check.py,sha256=JFxUxGRSz20EgjKqCLBkCqR4N1ihzO_3ykU3jPzp_UU,1801
71
+ eth_portfolio_scripts/docker/docker-compose.yaml,sha256=SpvQSpll0mTgMogA_UnNQ66xcjDC_Tnzu8AElsikzDg,1870
72
+ eth_portfolio_scripts/docker/docker_compose.cp313-win32.pyd,sha256=PClwA52dwVb9yUrH7pEclZYPJ09v_8e7yaOqyxz0BlU,9216
73
+ eth_portfolio_scripts/docker/docker_compose.py,sha256=tewgCWZP6g9sVFHbQnGC1au-bTnvuvXMZgTKKgpmeOc,2304
74
+ eth_portfolio_scripts/docker/.grafana/dashboards/dashboards.yaml,sha256=wktTI-OAdF_khhbciZFo4Gt2V9bUjbe7GLqwdzTKf0U,212
75
+ eth_portfolio_scripts/docker/.grafana/dashboards/Portfolio/Balances.json,sha256=XGMV8e4tDak53e9bmymwAB4uqZmAIcU5JlRT3OiwTeU,70750
76
+ eth_portfolio_scripts/docker/.grafana/datasources/datasources.yml,sha256=8PPH_QDhfbRRh3IidskW46rifJejloa1a9I1KCw2FTk,199
77
+ eth_portfolio_scripts/victoria/__init__.py,sha256=R0VvKiAC0e57zZNihcCptVkFO5CBHIbp2trFYuyY01M,2038
78
+ eth_portfolio_scripts/victoria/types.py,sha256=KNq8aIiNXeiDnCKL7xycmouo0YeKI-sbQkIcTymcSYk,745
79
+ eth_portfolio_temp-0.2.12.dist-info/METADATA,sha256=wxDoS5z6zTKNpxIvfaqR0XhBgnghIfyiJ0ySdis8Kss,800
80
+ eth_portfolio_temp-0.2.12.dist-info/WHEEL,sha256=0ABLuJ37exXk5N_efmYNs2NU9NK1K2Qlod_6bYkofEA,97
81
+ eth_portfolio_temp-0.2.12.dist-info/entry_points.txt,sha256=yqoC6X3LU1NA_-oJ6mloEYEPNmS-0hPS9OtEwgIeDGU,66
82
+ eth_portfolio_temp-0.2.12.dist-info/top_level.txt,sha256=4MlbY-Yj8oGBGL8piXiO4SOpk2gZFF9ZXVTObTZOzqM,57
83
+ eth_portfolio_temp-0.2.12.dist-info/RECORD,,
@@ -0,0 +1,5 @@
1
+ Wheel-Version: 1.0
2
+ Generator: setuptools (80.9.0)
3
+ Root-Is-Purelib: false
4
+ Tag: cp313-cp313-win32
5
+
@@ -0,0 +1,2 @@
1
+ [console_scripts]
2
+ eth-portfolio = eth_portfolio_scripts.main:main
@@ -0,0 +1,3 @@
1
+ eth_portfolio
2
+ eth_portfolio__mypyc
3
+ eth_portfolio_scripts