eth-portfolio-temp 0.2.2.dev0__cp312-cp312-macosx_11_0_arm64.whl → 0.2.4.dev0__cp312-cp312-macosx_11_0_arm64.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.

Binary file
eth_portfolio/_utils.py CHANGED
@@ -136,10 +136,11 @@ async def _get_price(token: AddressOrContract, block: Optional[int] = None) -> _
136
136
  elif isinstance(e.exception, NonStandardERC20):
137
137
  # Can't get symbol for handling like other excs
138
138
  logger.warning(f"NonStandardERC20 while fetching price for {token}")
139
- elif isinstance(e.exception, PriceError) and token not in SUPPRESS_ERROR_LOGS:
140
- logger.warning(
141
- f"PriceError while fetching price for {await _describe_err(token, block)}"
142
- )
139
+ elif isinstance(e.exception, PriceError):
140
+ if token not in SUPPRESS_ERROR_LOGS:
141
+ logger.warning(
142
+ f"PriceError while fetching price for {await _describe_err(token, block)}"
143
+ )
143
144
  else:
144
145
  logger.warning(f"{e} while fetching price for {await _describe_err(token, block)}")
145
146
  logger.warning(e, exc_info=True)
@@ -37,7 +37,7 @@ from eth_portfolio.typing import Addresses, PortfolioBalances
37
37
  logger = logging.getLogger(__name__)
38
38
 
39
39
 
40
- class PortfolioWallets(Iterable[PortfolioAddress], Dict[Address, PortfolioAddress]): # type: ignore [misc]
40
+ class PortfolioWallets(Iterable[PortfolioAddress], Dict[Address, PortfolioAddress]): # type: ignore [metaclass]
41
41
  """
42
42
  A container that holds all :class:`~eth_portfolio.address.PortfolioAddress` objects for a specific :class:`~eth_portfolio.Portfolio`.
43
43
 
@@ -1,3 +1,4 @@
1
+ import asyncio
1
2
  from datetime import datetime, timezone
2
3
  from logging import getLogger
3
4
  from math import floor
@@ -5,10 +6,11 @@ from typing import Awaitable, Callable, Final, Iterator, List, Optional, Tuple
5
6
 
6
7
  import a_sync
7
8
  import eth_retry
9
+ import y
8
10
  from a_sync.functools import cached_property_unsafe as cached_property
9
11
  from eth_typing import BlockNumber, ChecksumAddress
10
12
  from msgspec import ValidationError, json
11
- from y import ERC20, Network, NonStandardERC20, get_block_at_timestamp
13
+ from y import ERC20, Network, NonStandardERC20
12
14
  from y.constants import CHAINID
13
15
  from y.time import NoBlockFound
14
16
 
@@ -34,6 +36,14 @@ log_debug: Final = logger.debug
34
36
  log_error: Final = logger.error
35
37
 
36
38
 
39
+ async def get_block_at_timestamp(dt: datetime) -> BlockNumber:
40
+ while True:
41
+ try:
42
+ return y.get_block_at_timestamp(dt, sync=False)
43
+ except NoBlockFound:
44
+ await asyncio.sleep(10)
45
+
46
+
37
47
  class ExportablePortfolio(Portfolio):
38
48
  """Adds methods to export full portoflio data."""
39
49
 
@@ -77,17 +87,12 @@ class ExportablePortfolio(Portfolio):
77
87
  async def export_snapshot(self, dt: datetime) -> None:
78
88
  log_debug("checking data at %s for %s", dt, self.label)
79
89
  try:
80
- if not await self.data_exists(dt, sync=False):
81
- while True:
82
- try:
83
- block = await get_block_at_timestamp(dt, sync=False)
84
- except NoBlockFound:
85
- pass
86
- else:
87
- break
88
- log_debug("block at %s: %s", dt, block)
89
- data = await self.get_data_for_export(block, dt, sync=False)
90
- await victoria.post_data(data)
90
+ if await self.data_exists(dt, sync=False):
91
+ return
92
+ block = await get_block_at_timestamp(dt)
93
+ log_debug("block at %s: %s", dt, block)
94
+ data = await self.get_data_for_export(block, dt, sync=False)
95
+ await victoria.post_data(data)
91
96
  except Exception as e:
92
97
  log_error("Error processing %s:", dt, exc_info=True)
93
98
 
@@ -1,18 +1,18 @@
1
1
  import re
2
2
  from asyncio import Task, create_task, sleep
3
3
  from datetime import datetime, timedelta, timezone
4
- from typing import Any, AsyncGenerator, Dict, List, Optional
4
+ from typing import Any, AsyncGenerator, Dict, Final, List, Optional
5
5
 
6
6
  from brownie import chain
7
7
 
8
8
 
9
- def parse_timedelta(value: str) -> timedelta:
10
- regex = re.compile(r"(\d+)([dhms]?)")
11
- result = regex.findall(value)
9
+ timedelta_pattern: Final = re.compile(r"(\d+)([dhms]?)")
10
+
12
11
 
12
+ def parse_timedelta(value: str) -> timedelta:
13
13
  days, hours, minutes, seconds = 0, 0, 0, 0
14
14
 
15
- for val, unit in result:
15
+ for val, unit in timedelta_pattern.findall(value):
16
16
  val = int(val)
17
17
  if unit == "d":
18
18
  days = val
@@ -58,10 +58,24 @@ async def aiter_timestamps(
58
58
 
59
59
  timestamp = start
60
60
 
61
+ timestamps = []
61
62
  while timestamp <= datetime.now(tz=timezone.utc):
62
- yield timestamp
63
+ timestamps.append(timestamp)
63
64
  timestamp = timestamp + interval
64
65
 
66
+ # cycle between yielding earliest, latest, and middle from `timestamps` until complete
67
+ while timestamps:
68
+ # yield the earliest timestamp
69
+ yield timestamps.pop(0)
70
+ # yield the most recent timestamp if there is one
71
+ if timestamps:
72
+ yield timestamps.pop(-1)
73
+ # yield the most middle timestamp if there is one
74
+ if timestamps:
75
+ yield timestamps.pop(len(timestamps) // 2)
76
+
77
+ del timestamps
78
+
65
79
  while run_forever:
66
80
  while timestamp > datetime.now(tz=timezone.utc):
67
81
  await _get_waiter(timestamp)
@@ -34,9 +34,8 @@ async def export_balances(args: Namespace) -> None:
34
34
  )
35
35
 
36
36
  if export_start_block := args.export_start_block or args.first_tx_block:
37
- start = datetime.fromtimestamp(
38
- await dank_mids.eth.get_block_timestamp(args.export_start_block), tz=timezone.utc
39
- )
37
+ start_ts = await dank_mids.eth.get_block_timestamp(export_start_block)
38
+ start = datetime.fromtimestamp(start_ts, tz=timezone.utc)
40
39
  else:
41
40
  start = None
42
41
 
@@ -1,11 +1,11 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: eth_portfolio_temp
3
- Version: 0.2.2.dev0
3
+ Version: 0.2.4.dev0
4
4
  Summary: eth-portfolio makes it easy to analyze your portfolio.
5
5
  Home-page: https://github.com/BobTheBuidler/eth-portfolio
6
6
  Author: BobTheBuidler
7
7
  Author-email: bobthebuidlerdefi@gmail.com
8
- Requires-Python: >=3.10,<3.13
8
+ Requires-Python: >=3.10,<3.14
9
9
  Requires-Dist: checksum_dict>=2.1.7
10
10
  Requires-Dist: dank_mids>=4.20.154
11
11
  Requires-Dist: eth-brownie<1.23,>=1.21.0
@@ -16,7 +16,7 @@ Requires-Dist: faster-eth-utils
16
16
  Requires-Dist: numpy<3
17
17
  Requires-Dist: pandas<3,>=1.4.3
18
18
  Requires-Dist: typed-envs>=0.0.9
19
- Requires-Dist: ypricemagic<5,>=4.6.28.dev0
19
+ Requires-Dist: ypricemagic<5,>=4.6.29.dev0
20
20
  Dynamic: author
21
21
  Dynamic: author-email
22
22
  Dynamic: home-page
@@ -1,37 +1,37 @@
1
- 295eace8438df6ec133b__mypyc.cpython-312-darwin.so,sha256=vA2Qcv4ZOzgle7x5oOD9BfAP5SX2mG1DSZi8TW_Ka3Y,670224
2
- eth_portfolio_temp-0.2.2.dev0.dist-info/RECORD,,
3
- eth_portfolio_temp-0.2.2.dev0.dist-info/WHEEL,sha256=V1loQ6TpxABu1APUg0MoTRBOzSKT5xVc3skizX-ovCU,136
4
- eth_portfolio_temp-0.2.2.dev0.dist-info/entry_points.txt,sha256=yqoC6X3LU1NA_-oJ6mloEYEPNmS-0hPS9OtEwgIeDGU,66
5
- eth_portfolio_temp-0.2.2.dev0.dist-info/top_level.txt,sha256=CmTLi8Lvs3RulV5ZNdziaDVvF_WRsgRpM17GSMECxrg,64
6
- eth_portfolio_temp-0.2.2.dev0.dist-info/METADATA,sha256=O5Y4EOxRYl5K1a1VJU_lRk_Ak1bEn-shaJLH3whBfB4,779
1
+ 295eace8438df6ec133b__mypyc.cpython-312-darwin.so,sha256=kyGyeCqcW1RRirilmG0g8_QsT7QHaBBVtd_RECks9gI,693872
2
+ eth_portfolio_temp-0.2.4.dev0.dist-info/RECORD,,
3
+ eth_portfolio_temp-0.2.4.dev0.dist-info/WHEEL,sha256=V1loQ6TpxABu1APUg0MoTRBOzSKT5xVc3skizX-ovCU,136
4
+ eth_portfolio_temp-0.2.4.dev0.dist-info/entry_points.txt,sha256=yqoC6X3LU1NA_-oJ6mloEYEPNmS-0hPS9OtEwgIeDGU,66
5
+ eth_portfolio_temp-0.2.4.dev0.dist-info/top_level.txt,sha256=CmTLi8Lvs3RulV5ZNdziaDVvF_WRsgRpM17GSMECxrg,64
6
+ eth_portfolio_temp-0.2.4.dev0.dist-info/METADATA,sha256=0kQAF2-wNUm6w-Nsltq8QzF3j4ojBziq1LGqhE8Vhio,779
7
7
  eth_portfolio/address.py,sha256=LvBh4Vp2DBC3gQ0WD-TZ6jfe9s6FZbET1krYG9_KMAA,14139
8
8
  eth_portfolio/_cache.py,sha256=PYkR-yExjuwd_t7swv4euxL2daJ-VFwh3EMb3MciRJE,4745
9
- eth_portfolio/_shitcoins.cpython-312-darwin.so,sha256=8E-SSQhmn303pRwzqzv7TsVKZgHcMpQfyso6QzhjzKk,50656
9
+ eth_portfolio/_shitcoins.cpython-312-darwin.so,sha256=BgB-vRMfGmWucGCk6u86-D7DM7IflxzKaj6gc_o8RlA,50656
10
10
  eth_portfolio/_argspec.py,sha256=VzUZkbDkmOSgNUZBGbGblqtxBfDcmBAB89dY2OX0j-U,1595
11
- eth_portfolio/constants.cpython-312-darwin.so,sha256=k7GvKWShocbw5hW2iM-gv_3EV867CgeUpSF-Xpv0VC4,50656
11
+ eth_portfolio/constants.cpython-312-darwin.so,sha256=iXTacaX-Aj_hX7XuQjJDzt4rrNJFjaS6mffQaE7caPQ,50656
12
12
  eth_portfolio/_submodules.py,sha256=J8ht9bAAvblUXqwOGN7UIOXUPxrTiq7m1RWiB0-ObaE,2190
13
13
  eth_portfolio/constants.py,sha256=LS9H2P_Qfcreb6z6NknErxvq6OAtIHbHykXy0spol9E,3659
14
14
  eth_portfolio/__init__.py,sha256=0sO4cSJaLYwJfnfVOJRFw7p5_Lzhsr95YuJ1aNQM83A,500
15
15
  eth_portfolio/_decimal.py,sha256=nuUl6DZHCtrbBN7snwvs4LLaoJ5k7mn3dt4qmlFCkTU,4731
16
16
  eth_portfolio/buckets.py,sha256=QwdqWzW9iWBmRcJ2aktRtL-obvtBnND179krY5ISbhY,6695
17
17
  eth_portfolio/_decorators.py,sha256=_ZSurUFEIwZRiMFMhLcIXkD-Ey1CqfBqGaE24hLzOuA,2815
18
- eth_portfolio/_stableish.cpython-312-darwin.so,sha256=W1Csmt2pUVm_V39QrR2OFzyjYdEdiQOd7GYX6uBDQ8M,50656
18
+ eth_portfolio/_stableish.cpython-312-darwin.so,sha256=FXi7U77INM7FzU_wc9vMpS5w8xSNeGT0yXq4blLOiaU,50656
19
19
  eth_portfolio/_stableish.py,sha256=VTv69Q91AHxbNbbY0LB4PFwKEseHdkE4_6fLPKH1uW0,2021
20
20
  eth_portfolio/py.typed,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
21
21
  eth_portfolio/_shitcoins.py,sha256=FrqttZzQdheTAabdIK0oVHayqHF0UkLQLEiMz8njQe0,15563
22
22
  eth_portfolio/_config.py,sha256=mw-OA3M8rUA2hqy5wNt30AZRHkWFgR8K2sJg0o5Baxg,98
23
23
  eth_portfolio/_exceptions.py,sha256=bw3IdXhqrWxeFqYLm7pJZqRHKZas_cCg8A5Ih1WQEqQ,2433
24
- eth_portfolio/portfolio.py,sha256=KljK3FZQMtRQLs1D-HHgCN-DVoZ6ZzEdmclmF3__oz0,24243
25
- eth_portfolio/_utils.py,sha256=10In2lr0QmfiavyeA4Z8wDTuujRVmjcc0cd-o71JEBg,7742
26
- eth_portfolio/_config.cpython-312-darwin.so,sha256=Arcck4z1bb1u-ZH1-HkR63vUGOxs7kxDEbrphklbG8A,50640
27
- eth_portfolio/_argspec.cpython-312-darwin.so,sha256=3ifL37aZqGjy4EfYYS3BGlQ50kn1bgRw-LCnsry1kp8,50640
28
- eth_portfolio/_loaders/balances.cpython-312-darwin.so,sha256=dVoUy2Vias1RA1WZsyHfXdDzjLNVtKFCJjZok1K1ALM,50656
24
+ eth_portfolio/portfolio.py,sha256=MP_0Y6fXw8TNoyoFCPiu-MUA44jnMm0L_33gec6MjKU,24248
25
+ eth_portfolio/_utils.py,sha256=Ab3m6i79U-tnTrc_b_kTmOO4xKjbsX9slf6l0sXzf4c,7770
26
+ eth_portfolio/_config.cpython-312-darwin.so,sha256=X_wW0TwSeN89PHnC282JU75E_LuzAeMcevz2EaYGPb4,50640
27
+ eth_portfolio/_argspec.cpython-312-darwin.so,sha256=tvrzT5L6FbygoVLVNUyM2xRd_kuKo7MOET6i6kjmgL0,50640
28
+ eth_portfolio/_loaders/balances.cpython-312-darwin.so,sha256=qxhGZZNlqajCfDilZ80g0w9EUMap8gSwlPcbEhmN0nk,50656
29
29
  eth_portfolio/_loaders/transaction.py,sha256=FEYdoNDCgD6-mqZT9MuyYIEkdhG3iUwyNeMmqjpMGJw,9146
30
- eth_portfolio/_loaders/utils.cpython-312-darwin.so,sha256=sitetWKzh8FJrTyRfoIrQztOLQlnvfVdASzHLYSN-rg,50648
30
+ eth_portfolio/_loaders/utils.cpython-312-darwin.so,sha256=4tS6AM2A3AYb15aA2tbS6CJ6o_AO5taXg000ROOooU4,50648
31
31
  eth_portfolio/_loaders/__init__.py,sha256=lb45_0ak32Z7N3-Nx1CAoRKiZ1_w-_YGbmSCNuunro8,1702
32
32
  eth_portfolio/_loaders/_nonce.py,sha256=cx0EjTAhVU2EudMKSO-7-t1jl3y4dYn6as4XiY6KyS8,6285
33
33
  eth_portfolio/_loaders/utils.py,sha256=my3Wg2Ip5gSWuzAWcoLQQEmVmAPhIHbBzQQc0iP1iCA,2258
34
- eth_portfolio/_loaders/_nonce.cpython-312-darwin.so,sha256=aDpftnI9_8QGn1MmlMq71LaiBAUk2mHuuyXpPvGzMB8,50656
34
+ eth_portfolio/_loaders/_nonce.cpython-312-darwin.so,sha256=ANNX_EmJ8cEgiLrd4ou4EG2KOdq6cRckZH0u2fopKb4,50656
35
35
  eth_portfolio/_loaders/balances.py,sha256=BTWfkJIoSraUMe94Wuj8NPyg5EO0OByIjbXu7j6PZEo,3132
36
36
  eth_portfolio/_loaders/token_transfer.py,sha256=MEw95D2iKBYa_9vIatCHJGe7zEJuYUEC8P1mTjjRXlo,8524
37
37
  eth_portfolio/_ydb/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
@@ -61,21 +61,21 @@ eth_portfolio/protocols/lending/README.md,sha256=OhZfsW8e-aD-q02g0maG9QGgW0Iietk
61
61
  eth_portfolio/protocols/lending/compound.py,sha256=bqnIevm7NWhk6nMfkMZUKlQuVWr-tXnuntX-4De1YZM,7252
62
62
  eth_portfolio/protocols/lending/unit.py,sha256=7oRvIkoKUm8IOnDv59pILSSJWFNNjwr4CR1gg2psW9M,1965
63
63
  eth_portfolio_scripts/_logging.py,sha256=B_rQMYt_1PhpwCOLBRpkKK6M1ljcF0wAIgqfPIsFUGU,354
64
- eth_portfolio_scripts/balances.cpython-312-darwin.so,sha256=-6RMffp9NVGZ89etBwEc6t7G_CXAd7mp45lYMEQaqL4,50656
64
+ eth_portfolio_scripts/balances.cpython-312-darwin.so,sha256=V4vs0icBG4N82nIM4gKJ13Iq1ACXKHwFJqfHNo-cQ3w,50656
65
65
  eth_portfolio_scripts/_args.py,sha256=k6J6XkRe1VuN1DiyGuXLCR7NBSvzH5jnVChfzodKuB8,656
66
66
  eth_portfolio_scripts/__init__.py,sha256=TC5c4WZdzSHhTIBYuwzrAyzFuGzBmHiDX_g6ghO09jQ,261
67
- eth_portfolio_scripts/_portfolio.py,sha256=36n8DE9OfHvdF0ez12yzWpxI7at6_eo3eno7V_YYu3A,6932
67
+ eth_portfolio_scripts/_portfolio.py,sha256=NUW2wl1MhKa6Y1AW-e7NIXPMTtTRRVttzSKUdVjJm9M,6955
68
68
  eth_portfolio_scripts/py.typed,sha256=AbpHGcgLb-kRsJGnwFEktk7uzpZOCcBY74-YBdrKVGs,1
69
- eth_portfolio_scripts/balances.py,sha256=m0w2SbFChvOAnxUIKavgdnaq-DS01KimzDqmFV5apJs,1587
69
+ eth_portfolio_scripts/balances.py,sha256=QvpOdXnH1V-tOuEmAOPxH3MrincDExlIP1eEJ2IG2Lk,1588
70
70
  eth_portfolio_scripts/main.py,sha256=vafXy4JvNWrP203AgCRcSKHMsOSCC7VVj9zBd_BFYZU,3685
71
- eth_portfolio_scripts/_utils.py,sha256=qyct1Mwf2-1DYj7BUWt2bocUfGKMap1l9sxUfHAjER0,2538
72
- eth_portfolio_scripts/docker/__init__.cpython-312-darwin.so,sha256=Xa3_BKLx4Y4SpGCsy7hGU9ju4bcSSmecRPawJU-lNyE,50656
71
+ eth_portfolio_scripts/_utils.py,sha256=-JiZLpSa8P2_ZFXw1AfpVmTO6X65yPeVVdnrDpNFzNQ,3057
72
+ eth_portfolio_scripts/docker/__init__.cpython-312-darwin.so,sha256=YPSq2vYAvmVn-a_EXKCBRt_PenuVCKRpBuialce0jcI,50656
73
73
  eth_portfolio_scripts/docker/docker-compose.yaml,sha256=pQ4eesSfMtcb9muDFT5TFDqZty15BoC2Rrd8HY0LS04,1809
74
74
  eth_portfolio_scripts/docker/docker_compose.py,sha256=K7_Y_8NeN5s0pb9WUamgfi9OppjpaU0Mlz2nDzewcRA,2226
75
75
  eth_portfolio_scripts/docker/check.py,sha256=E6_Ar4dr6HfVAgpwoDPcObIolikHKQr8VFkH01sOuIM,1745
76
76
  eth_portfolio_scripts/docker/__init__.py,sha256=ZXSIYcjp94c0O9o39BDGWju3YrCMLl4gtov7YChc9Ig,393
77
- eth_portfolio_scripts/docker/docker_compose.cpython-312-darwin.so,sha256=uJjmC3zY2F6ogFK-s23WwfDlDNvVUaXhjxG5vn_hpSo,50712
78
- eth_portfolio_scripts/docker/check.cpython-312-darwin.so,sha256=PTf2NqLxiEdcC6ePMBVIPfWw85ZyXPHsBghuAOjNoGg,50664
77
+ eth_portfolio_scripts/docker/docker_compose.cpython-312-darwin.so,sha256=8O8p6KqOT--uZINXQ2y-_PDXOS906F8bvvWIY27Yzbw,50712
78
+ eth_portfolio_scripts/docker/check.cpython-312-darwin.so,sha256=qauw3tjb-IxYSDzQVIkHWzAW1QAgC0g4-SeV3z1b8do,50664
79
79
  eth_portfolio_scripts/docker/.grafana/datasources/datasources.yml,sha256=pBE_0Nh_J7d9Fiy3Xu6vuac_HWCBcFsJJSV-ryjQR1Y,188
80
80
  eth_portfolio_scripts/docker/.grafana/dashboards/dashboards.yaml,sha256=MynNDOk69IihoYdd2bL7j8CnRb2Co4gdqW7T4m6AaEU,202
81
81
  eth_portfolio_scripts/docker/.grafana/dashboards/Portfolio/Balances.json,sha256=dBmjogLJRuixCHWSs4ROE0_FXb-PUbqWBHEdLoP-1MU,68788