From 53e93380ab1adfadf0a86d219c9216bb8bfe4174 Mon Sep 17 00:00:00 2001 From: "stainless-app[bot]" <142633134+stainless-app[bot]@users.noreply.github.com> Date: Sat, 7 Mar 2026 18:42:55 +0000 Subject: [PATCH 1/6] chore(ci): skip uploading artifacts on stainless-internal branches --- .github/workflows/ci.yml | 8 ++++++-- 1 file changed, 6 insertions(+), 2 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index 5100e6d..f8832d8 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -61,14 +61,18 @@ jobs: run: rye build - name: Get GitHub OIDC Token - if: github.repository == 'stainless-sdks/coingecko-python' + if: |- + github.repository == 'stainless-sdks/coingecko-python' && + !startsWith(github.ref, 'refs/heads/stl/') id: github-oidc uses: actions/github-script@v8 with: script: core.setOutput('github_token', await core.getIDToken()); - name: Upload tarball - if: github.repository == 'stainless-sdks/coingecko-python' + if: |- + github.repository == 'stainless-sdks/coingecko-python' && + !startsWith(github.ref, 'refs/heads/stl/') env: URL: https://pkg.stainless.com/s AUTH: ${{ steps.github-oidc.outputs.github_token }} From 5b4c5d1fec42630e39a569422baf75eb0990bb31 Mon Sep 17 00:00:00 2001 From: "stainless-app[bot]" <142633134+stainless-app[bot]@users.noreply.github.com> Date: Tue, 17 Mar 2026 11:39:43 +0000 Subject: [PATCH 2/6] fix(pydantic): do not pass `by_alias` unless set --- src/coingecko_sdk/_compat.py | 11 +++++++++-- 1 file changed, 9 insertions(+), 2 deletions(-) diff --git a/src/coingecko_sdk/_compat.py b/src/coingecko_sdk/_compat.py index 786ff42..e6690a4 100644 --- a/src/coingecko_sdk/_compat.py +++ b/src/coingecko_sdk/_compat.py @@ -2,7 +2,7 @@ from typing import TYPE_CHECKING, Any, Union, Generic, TypeVar, Callable, cast, overload from datetime import date, datetime -from typing_extensions import Self, Literal +from typing_extensions import Self, Literal, TypedDict import pydantic from pydantic.fields import FieldInfo @@ -131,6 +131,10 @@ def model_json(model: pydantic.BaseModel, *, indent: int | None = None) -> str: return model.model_dump_json(indent=indent) +class _ModelDumpKwargs(TypedDict, total=False): + by_alias: bool + + def model_dump( model: pydantic.BaseModel, *, @@ -142,6 +146,9 @@ def model_dump( by_alias: bool | None = None, ) -> dict[str, Any]: if (not PYDANTIC_V1) or hasattr(model, "model_dump"): + kwargs: _ModelDumpKwargs = {} + if by_alias is not None: + kwargs["by_alias"] = by_alias return model.model_dump( mode=mode, exclude=exclude, @@ -149,7 +156,7 @@ def model_dump( exclude_defaults=exclude_defaults, # warnings are not supported in Pydantic v1 warnings=True if PYDANTIC_V1 else warnings, - by_alias=by_alias, + **kwargs, ) return cast( "dict[str, Any]", From 7a2809e203ebeacd52704696b19c94780d9c0db6 Mon Sep 17 00:00:00 2001 From: "stainless-app[bot]" <142633134+stainless-app[bot]@users.noreply.github.com> Date: Tue, 17 Mar 2026 11:44:28 +0000 Subject: [PATCH 3/6] fix(deps): bump minimum typing-extensions version --- pyproject.toml | 2 +- 1 file changed, 1 insertion(+), 1 deletion(-) diff --git a/pyproject.toml b/pyproject.toml index 1b9f459..500fa06 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -11,7 +11,7 @@ authors = [ dependencies = [ "httpx>=0.23.0, <1", "pydantic>=1.9.0, <3", - "typing-extensions>=4.10, <5", + "typing-extensions>=4.14, <5", "anyio>=3.5.0, <5", "distro>=1.7.0, <2", "sniffio", From 4d1960abd7dfb3610749bcc303579deb787f8853 Mon Sep 17 00:00:00 2001 From: "stainless-app[bot]" <142633134+stainless-app[bot]@users.noreply.github.com> Date: Tue, 17 Mar 2026 11:47:40 +0000 Subject: [PATCH 4/6] chore(internal): tweak CI branches --- .github/workflows/ci.yml | 14 ++++++++------ 1 file changed, 8 insertions(+), 6 deletions(-) diff --git a/.github/workflows/ci.yml b/.github/workflows/ci.yml index f8832d8..1eb5c5f 100644 --- a/.github/workflows/ci.yml +++ b/.github/workflows/ci.yml @@ -1,12 +1,14 @@ name: CI on: push: - branches-ignore: - - 'generated' - - 'codegen/**' - - 'integrated/**' - - 'stl-preview-head/**' - - 'stl-preview-base/**' + branches: + - '**' + - '!integrated/**' + - '!stl-preview-head/**' + - '!stl-preview-base/**' + - '!generated' + - '!codegen/**' + - 'codegen/stl/**' pull_request: branches-ignore: - 'stl-preview-head/**' From a87e9d4bacf39d06c9becdd45dcf6a422699e987 Mon Sep 17 00:00:00 2001 From: "stainless-app[bot]" <142633134+stainless-app[bot]@users.noreply.github.com> Date: Fri, 20 Mar 2026 03:34:36 +0000 Subject: [PATCH 5/6] fix: sanitize endpoint path params --- src/coingecko_sdk/_utils/__init__.py | 1 + src/coingecko_sdk/_utils/_path.py | 127 ++++++++++++++++++ .../coins/circulating_supply_chart.py | 10 +- src/coingecko_sdk/resources/coins/coins.py | 6 +- .../resources/coins/contract/contract.py | 5 +- .../resources/coins/contract/market_chart.py | 18 ++- src/coingecko_sdk/resources/coins/history.py | 6 +- .../resources/coins/market_chart.py | 10 +- src/coingecko_sdk/resources/coins/ohlc.py | 10 +- src/coingecko_sdk/resources/coins/tickers.py | 6 +- .../resources/coins/total_supply_chart.py | 10 +- .../resources/derivatives/exchanges.py | 6 +- .../resources/exchanges/exchanges.py | 6 +- .../resources/exchanges/tickers.py | 6 +- .../resources/exchanges/volume_chart.py | 10 +- .../resources/nfts/contract/contract.py | 13 +- .../resources/nfts/contract/market_chart.py | 14 +- .../resources/nfts/market_chart.py | 6 +- src/coingecko_sdk/resources/nfts/nfts.py | 6 +- src/coingecko_sdk/resources/nfts/tickers.py | 5 +- .../resources/onchain/categories.py | 6 +- .../resources/onchain/networks/dexes.py | 10 +- .../resources/onchain/networks/new_pools.py | 6 +- .../resources/onchain/networks/pools/info.py | 10 +- .../resources/onchain/networks/pools/multi.py | 6 +- .../resources/onchain/networks/pools/ohlcv.py | 16 ++- .../resources/onchain/networks/pools/pools.py | 10 +- .../onchain/networks/pools/trades.py | 10 +- .../onchain/networks/tokens/holders_chart.py | 14 +- .../resources/onchain/networks/tokens/info.py | 5 +- .../onchain/networks/tokens/multi.py | 6 +- .../onchain/networks/tokens/ohlcv.py | 16 ++- .../onchain/networks/tokens/pools.py | 10 +- .../onchain/networks/tokens/tokens.py | 6 +- .../onchain/networks/tokens/top_holders.py | 6 +- .../onchain/networks/tokens/top_traders.py | 14 +- .../onchain/networks/tokens/trades.py | 14 +- .../onchain/networks/trending_pools.py | 6 +- .../onchain/simple/networks/token_price.py | 10 +- .../resources/public_treasury.py | 18 +-- .../resources/simple/token_price.py | 6 +- src/coingecko_sdk/resources/token_lists.py | 5 +- tests/test_utils/test_path.py | 89 ++++++++++++ 43 files changed, 443 insertions(+), 137 deletions(-) create mode 100644 src/coingecko_sdk/_utils/_path.py create mode 100644 tests/test_utils/test_path.py diff --git a/src/coingecko_sdk/_utils/__init__.py b/src/coingecko_sdk/_utils/__init__.py index dc64e29..10cb66d 100644 --- a/src/coingecko_sdk/_utils/__init__.py +++ b/src/coingecko_sdk/_utils/__init__.py @@ -1,3 +1,4 @@ +from ._path import path_template as path_template from ._sync import asyncify as asyncify from ._proxy import LazyProxy as LazyProxy from ._utils import ( diff --git a/src/coingecko_sdk/_utils/_path.py b/src/coingecko_sdk/_utils/_path.py new file mode 100644 index 0000000..4d6e1e4 --- /dev/null +++ b/src/coingecko_sdk/_utils/_path.py @@ -0,0 +1,127 @@ +from __future__ import annotations + +import re +from typing import ( + Any, + Mapping, + Callable, +) +from urllib.parse import quote + +# Matches '.' or '..' where each dot is either literal or percent-encoded (%2e / %2E). +_DOT_SEGMENT_RE = re.compile(r"^(?:\.|%2[eE]){1,2}$") + +_PLACEHOLDER_RE = re.compile(r"\{(\w+)\}") + + +def _quote_path_segment_part(value: str) -> str: + """Percent-encode `value` for use in a URI path segment. + + Considers characters not in `pchar` set from RFC 3986 §3.3 to be unsafe. + https://datatracker.ietf.org/doc/html/rfc3986#section-3.3 + """ + # quote() already treats unreserved characters (letters, digits, and -._~) + # as safe, so we only need to add sub-delims, ':', and '@'. + # Notably, unlike the default `safe` for quote(), / is unsafe and must be quoted. + return quote(value, safe="!$&'()*+,;=:@") + + +def _quote_query_part(value: str) -> str: + """Percent-encode `value` for use in a URI query string. + + Considers &, = and characters not in `query` set from RFC 3986 §3.4 to be unsafe. + https://datatracker.ietf.org/doc/html/rfc3986#section-3.4 + """ + return quote(value, safe="!$'()*+,;:@/?") + + +def _quote_fragment_part(value: str) -> str: + """Percent-encode `value` for use in a URI fragment. + + Considers characters not in `fragment` set from RFC 3986 §3.5 to be unsafe. + https://datatracker.ietf.org/doc/html/rfc3986#section-3.5 + """ + return quote(value, safe="!$&'()*+,;=:@/?") + + +def _interpolate( + template: str, + values: Mapping[str, Any], + quoter: Callable[[str], str], +) -> str: + """Replace {name} placeholders in `template`, quoting each value with `quoter`. + + Placeholder names are looked up in `values`. + + Raises: + KeyError: If a placeholder is not found in `values`. + """ + # re.split with a capturing group returns alternating + # [text, name, text, name, ..., text] elements. + parts = _PLACEHOLDER_RE.split(template) + + for i in range(1, len(parts), 2): + name = parts[i] + if name not in values: + raise KeyError(f"a value for placeholder {{{name}}} was not provided") + val = values[name] + if val is None: + parts[i] = "null" + elif isinstance(val, bool): + parts[i] = "true" if val else "false" + else: + parts[i] = quoter(str(values[name])) + + return "".join(parts) + + +def path_template(template: str, /, **kwargs: Any) -> str: + """Interpolate {name} placeholders in `template` from keyword arguments. + + Args: + template: The template string containing {name} placeholders. + **kwargs: Keyword arguments to interpolate into the template. + + Returns: + The template with placeholders interpolated and percent-encoded. + + Safe characters for percent-encoding are dependent on the URI component. + Placeholders in path and fragment portions are percent-encoded where the `segment` + and `fragment` sets from RFC 3986 respectively are considered safe. + Placeholders in the query portion are percent-encoded where the `query` set from + RFC 3986 §3.3 is considered safe except for = and & characters. + + Raises: + KeyError: If a placeholder is not found in `kwargs`. + ValueError: If resulting path contains /./ or /../ segments (including percent-encoded dot-segments). + """ + # Split the template into path, query, and fragment portions. + fragment_template: str | None = None + query_template: str | None = None + + rest = template + if "#" in rest: + rest, fragment_template = rest.split("#", 1) + if "?" in rest: + rest, query_template = rest.split("?", 1) + path_template = rest + + # Interpolate each portion with the appropriate quoting rules. + path_result = _interpolate(path_template, kwargs, _quote_path_segment_part) + + # Reject dot-segments (. and ..) in the final assembled path. The check + # runs after interpolation so that adjacent placeholders or a mix of static + # text and placeholders that together form a dot-segment are caught. + # Also reject percent-encoded dot-segments to protect against incorrectly + # implemented normalization in servers/proxies. + for segment in path_result.split("/"): + if _DOT_SEGMENT_RE.match(segment): + raise ValueError(f"Constructed path {path_result!r} contains dot-segment {segment!r} which is not allowed") + + result = path_result + if query_template is not None: + result += "?" + _interpolate(query_template, kwargs, _quote_query_part) + if fragment_template is not None: + result += "#" + _interpolate(fragment_template, kwargs, _quote_fragment_part) + + return result diff --git a/src/coingecko_sdk/resources/coins/circulating_supply_chart.py b/src/coingecko_sdk/resources/coins/circulating_supply_chart.py index c75ffea..d54d29e 100644 --- a/src/coingecko_sdk/resources/coins/circulating_supply_chart.py +++ b/src/coingecko_sdk/resources/coins/circulating_supply_chart.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -77,7 +77,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/circulating_supply_chart", + path_template("/coins/{id}/circulating_supply_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -129,7 +129,7 @@ def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/circulating_supply_chart/range", + path_template("/coins/{id}/circulating_supply_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -200,7 +200,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/circulating_supply_chart", + path_template("/coins/{id}/circulating_supply_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -252,7 +252,7 @@ async def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/circulating_supply_chart/range", + path_template("/coins/{id}/circulating_supply_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/coins.py b/src/coingecko_sdk/resources/coins/coins.py index 65de0ba..1762df2 100644 --- a/src/coingecko_sdk/resources/coins/coins.py +++ b/src/coingecko_sdk/resources/coins/coins.py @@ -48,7 +48,7 @@ AsyncTickersResourceWithStreamingResponse, ) from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from .categories import ( CategoriesResource, @@ -230,7 +230,7 @@ def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}", + path_template("/coins/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -373,7 +373,7 @@ async def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}", + path_template("/coins/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/contract/contract.py b/src/coingecko_sdk/resources/coins/contract/contract.py index c14cc61..c933d07 100644 --- a/src/coingecko_sdk/resources/coins/contract/contract.py +++ b/src/coingecko_sdk/resources/coins/contract/contract.py @@ -5,6 +5,7 @@ import httpx from ...._types import Body, Query, Headers, NotGiven, not_given +from ...._utils import path_template from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -83,7 +84,7 @@ def get( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return self._get( - f"/coins/{id}/contract/{contract_address}", + path_template("/coins/{id}/contract/{contract_address}", id=id, contract_address=contract_address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), @@ -147,7 +148,7 @@ async def get( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return await self._get( - f"/coins/{id}/contract/{contract_address}", + path_template("/coins/{id}/contract/{contract_address}", id=id, contract_address=contract_address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), diff --git a/src/coingecko_sdk/resources/coins/contract/market_chart.py b/src/coingecko_sdk/resources/coins/contract/market_chart.py index 65ad04b..f151bc6 100644 --- a/src/coingecko_sdk/resources/coins/contract/market_chart.py +++ b/src/coingecko_sdk/resources/coins/contract/market_chart.py @@ -7,7 +7,7 @@ import httpx from ...._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ...._utils import maybe_transform, async_maybe_transform +from ...._utils import path_template, maybe_transform, async_maybe_transform from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -111,7 +111,9 @@ def get( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return self._get( - f"/coins/{id}/contract/{contract_address}/market_chart", + path_template( + "/coins/{id}/contract/{contract_address}/market_chart", id=id, contract_address=contract_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -201,7 +203,9 @@ def get_range( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return self._get( - f"/coins/{id}/contract/{contract_address}/market_chart/range", + path_template( + "/coins/{id}/contract/{contract_address}/market_chart/range", id=id, contract_address=contract_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -309,7 +313,9 @@ async def get( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return await self._get( - f"/coins/{id}/contract/{contract_address}/market_chart", + path_template( + "/coins/{id}/contract/{contract_address}/market_chart", id=id, contract_address=contract_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -399,7 +405,9 @@ async def get_range( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return await self._get( - f"/coins/{id}/contract/{contract_address}/market_chart/range", + path_template( + "/coins/{id}/contract/{contract_address}/market_chart/range", id=id, contract_address=contract_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/history.py b/src/coingecko_sdk/resources/coins/history.py index a4a3487..4196184 100644 --- a/src/coingecko_sdk/resources/coins/history.py +++ b/src/coingecko_sdk/resources/coins/history.py @@ -5,7 +5,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -74,7 +74,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/history", + path_template("/coins/{id}/history", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -145,7 +145,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/history", + path_template("/coins/{id}/history", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/market_chart.py b/src/coingecko_sdk/resources/coins/market_chart.py index 8b1057e..bb49a1e 100644 --- a/src/coingecko_sdk/resources/coins/market_chart.py +++ b/src/coingecko_sdk/resources/coins/market_chart.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -107,7 +107,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/market_chart", + path_template("/coins/{id}/market_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -194,7 +194,7 @@ def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/market_chart/range", + path_template("/coins/{id}/market_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -298,7 +298,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/market_chart", + path_template("/coins/{id}/market_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -385,7 +385,7 @@ async def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/market_chart/range", + path_template("/coins/{id}/market_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/ohlc.py b/src/coingecko_sdk/resources/coins/ohlc.py index f914a77..8154177 100644 --- a/src/coingecko_sdk/resources/coins/ohlc.py +++ b/src/coingecko_sdk/resources/coins/ohlc.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -106,7 +106,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/ohlc", + path_template("/coins/{id}/ohlc", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -167,7 +167,7 @@ def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/ohlc/range", + path_template("/coins/{id}/ohlc/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -269,7 +269,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/ohlc", + path_template("/coins/{id}/ohlc", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -330,7 +330,7 @@ async def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/ohlc/range", + path_template("/coins/{id}/ohlc/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/tickers.py b/src/coingecko_sdk/resources/coins/tickers.py index a6d680c..67d2ebd 100644 --- a/src/coingecko_sdk/resources/coins/tickers.py +++ b/src/coingecko_sdk/resources/coins/tickers.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -91,7 +91,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/tickers", + path_template("/coins/{id}/tickers", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -181,7 +181,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/tickers", + path_template("/coins/{id}/tickers", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/coins/total_supply_chart.py b/src/coingecko_sdk/resources/coins/total_supply_chart.py index c8c9ca6..85f5ec1 100644 --- a/src/coingecko_sdk/resources/coins/total_supply_chart.py +++ b/src/coingecko_sdk/resources/coins/total_supply_chart.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -77,7 +77,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/total_supply_chart", + path_template("/coins/{id}/total_supply_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -129,7 +129,7 @@ def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/coins/{id}/total_supply_chart/range", + path_template("/coins/{id}/total_supply_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -200,7 +200,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/total_supply_chart", + path_template("/coins/{id}/total_supply_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -252,7 +252,7 @@ async def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/coins/{id}/total_supply_chart/range", + path_template("/coins/{id}/total_supply_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/derivatives/exchanges.py b/src/coingecko_sdk/resources/derivatives/exchanges.py index 1371ffc..419ff81 100644 --- a/src/coingecko_sdk/resources/derivatives/exchanges.py +++ b/src/coingecko_sdk/resources/derivatives/exchanges.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -134,7 +134,7 @@ def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/derivatives/exchanges/{id}", + path_template("/derivatives/exchanges/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -277,7 +277,7 @@ async def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/derivatives/exchanges/{id}", + path_template("/derivatives/exchanges/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/exchanges/exchanges.py b/src/coingecko_sdk/resources/exchanges/exchanges.py index f83fb40..3701569 100644 --- a/src/coingecko_sdk/resources/exchanges/exchanges.py +++ b/src/coingecko_sdk/resources/exchanges/exchanges.py @@ -16,7 +16,7 @@ AsyncTickersResourceWithStreamingResponse, ) from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -149,7 +149,7 @@ def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/exchanges/{id}", + path_template("/exchanges/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -306,7 +306,7 @@ async def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/exchanges/{id}", + path_template("/exchanges/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/exchanges/tickers.py b/src/coingecko_sdk/resources/exchanges/tickers.py index 0775f38..13d8063 100644 --- a/src/coingecko_sdk/resources/exchanges/tickers.py +++ b/src/coingecko_sdk/resources/exchanges/tickers.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -100,7 +100,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/exchanges/{id}/tickers", + path_template("/exchanges/{id}/tickers", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -199,7 +199,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/exchanges/{id}/tickers", + path_template("/exchanges/{id}/tickers", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/exchanges/volume_chart.py b/src/coingecko_sdk/resources/exchanges/volume_chart.py index 03a102c..abf1987 100644 --- a/src/coingecko_sdk/resources/exchanges/volume_chart.py +++ b/src/coingecko_sdk/resources/exchanges/volume_chart.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Query, Headers, NotGiven, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -74,7 +74,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/exchanges/{id}/volume_chart", + path_template("/exchanges/{id}/volume_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -118,7 +118,7 @@ def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/exchanges/{id}/volume_chart/range", + path_template("/exchanges/{id}/volume_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -186,7 +186,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/exchanges/{id}/volume_chart", + path_template("/exchanges/{id}/volume_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -230,7 +230,7 @@ async def get_range( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/exchanges/{id}/volume_chart/range", + path_template("/exchanges/{id}/volume_chart/range", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/nfts/contract/contract.py b/src/coingecko_sdk/resources/nfts/contract/contract.py index 7cd1ee4..2a4e4f9 100644 --- a/src/coingecko_sdk/resources/nfts/contract/contract.py +++ b/src/coingecko_sdk/resources/nfts/contract/contract.py @@ -5,6 +5,7 @@ import httpx from ...._types import Body, Query, Headers, NotGiven, not_given +from ...._utils import path_template from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -82,7 +83,11 @@ def get_contract_address( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return self._get( - f"/nfts/{asset_platform_id}/contract/{contract_address}", + path_template( + "/nfts/{asset_platform_id}/contract/{contract_address}", + asset_platform_id=asset_platform_id, + contract_address=contract_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), @@ -145,7 +150,11 @@ async def get_contract_address( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return await self._get( - f"/nfts/{asset_platform_id}/contract/{contract_address}", + path_template( + "/nfts/{asset_platform_id}/contract/{contract_address}", + asset_platform_id=asset_platform_id, + contract_address=contract_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), diff --git a/src/coingecko_sdk/resources/nfts/contract/market_chart.py b/src/coingecko_sdk/resources/nfts/contract/market_chart.py index 9ab5273..a6bc781 100644 --- a/src/coingecko_sdk/resources/nfts/contract/market_chart.py +++ b/src/coingecko_sdk/resources/nfts/contract/market_chart.py @@ -5,7 +5,7 @@ import httpx from ...._types import Body, Query, Headers, NotGiven, not_given -from ...._utils import maybe_transform, async_maybe_transform +from ...._utils import path_template, maybe_transform, async_maybe_transform from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -75,7 +75,11 @@ def get( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return self._get( - f"/nfts/{asset_platform_id}/contract/{contract_address}/market_chart", + path_template( + "/nfts/{asset_platform_id}/contract/{contract_address}/market_chart", + asset_platform_id=asset_platform_id, + contract_address=contract_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -141,7 +145,11 @@ async def get( if not contract_address: raise ValueError(f"Expected a non-empty value for `contract_address` but received {contract_address!r}") return await self._get( - f"/nfts/{asset_platform_id}/contract/{contract_address}/market_chart", + path_template( + "/nfts/{asset_platform_id}/contract/{contract_address}/market_chart", + asset_platform_id=asset_platform_id, + contract_address=contract_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/nfts/market_chart.py b/src/coingecko_sdk/resources/nfts/market_chart.py index 8d93a32..0c4f0d3 100644 --- a/src/coingecko_sdk/resources/nfts/market_chart.py +++ b/src/coingecko_sdk/resources/nfts/market_chart.py @@ -5,7 +5,7 @@ import httpx from ..._types import Body, Query, Headers, NotGiven, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -72,7 +72,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/nfts/{id}/market_chart", + path_template("/nfts/{id}/market_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -135,7 +135,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/nfts/{id}/market_chart", + path_template("/nfts/{id}/market_chart", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/nfts/nfts.py b/src/coingecko_sdk/resources/nfts/nfts.py index d4183bd..bab86af 100644 --- a/src/coingecko_sdk/resources/nfts/nfts.py +++ b/src/coingecko_sdk/resources/nfts/nfts.py @@ -16,7 +16,7 @@ AsyncTickersResourceWithStreamingResponse, ) from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -108,7 +108,7 @@ def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/nfts/{id}", + path_template("/nfts/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), @@ -304,7 +304,7 @@ async def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/nfts/{id}", + path_template("/nfts/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), diff --git a/src/coingecko_sdk/resources/nfts/tickers.py b/src/coingecko_sdk/resources/nfts/tickers.py index a4064e2..4728ebe 100644 --- a/src/coingecko_sdk/resources/nfts/tickers.py +++ b/src/coingecko_sdk/resources/nfts/tickers.py @@ -5,6 +5,7 @@ import httpx from ..._types import Body, Query, Headers, NotGiven, not_given +from ..._utils import path_template from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -66,7 +67,7 @@ def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/nfts/{id}/tickers", + path_template("/nfts/{id}/tickers", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), @@ -121,7 +122,7 @@ async def get( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/nfts/{id}/tickers", + path_template("/nfts/{id}/tickers", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), diff --git a/src/coingecko_sdk/resources/onchain/categories.py b/src/coingecko_sdk/resources/onchain/categories.py index 0c3e788..80964e2 100644 --- a/src/coingecko_sdk/resources/onchain/categories.py +++ b/src/coingecko_sdk/resources/onchain/categories.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -148,7 +148,7 @@ def get_pools( if not category_id: raise ValueError(f"Expected a non-empty value for `category_id` but received {category_id!r}") return self._get( - f"/onchain/categories/{category_id}/pools", + path_template("/onchain/categories/{category_id}/pools", category_id=category_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -291,7 +291,7 @@ async def get_pools( if not category_id: raise ValueError(f"Expected a non-empty value for `category_id` but received {category_id!r}") return await self._get( - f"/onchain/categories/{category_id}/pools", + path_template("/onchain/categories/{category_id}/pools", category_id=category_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/dexes.py b/src/coingecko_sdk/resources/onchain/networks/dexes.py index a7baa11..21b55cf 100644 --- a/src/coingecko_sdk/resources/onchain/networks/dexes.py +++ b/src/coingecko_sdk/resources/onchain/networks/dexes.py @@ -7,7 +7,7 @@ import httpx from ...._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ...._utils import maybe_transform, async_maybe_transform +from ...._utils import path_template, maybe_transform, async_maybe_transform from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -74,7 +74,7 @@ def get( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return self._get( - f"/onchain/networks/{network}/dexes", + path_template("/onchain/networks/{network}/dexes", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -129,7 +129,7 @@ def get_pools( if not dex: raise ValueError(f"Expected a non-empty value for `dex` but received {dex!r}") return self._get( - f"/onchain/networks/{network}/dexes/{dex}/pools", + path_template("/onchain/networks/{network}/dexes/{dex}/pools", network=network, dex=dex), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -199,7 +199,7 @@ async def get( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return await self._get( - f"/onchain/networks/{network}/dexes", + path_template("/onchain/networks/{network}/dexes", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -254,7 +254,7 @@ async def get_pools( if not dex: raise ValueError(f"Expected a non-empty value for `dex` but received {dex!r}") return await self._get( - f"/onchain/networks/{network}/dexes/{dex}/pools", + path_template("/onchain/networks/{network}/dexes/{dex}/pools", network=network, dex=dex), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/new_pools.py b/src/coingecko_sdk/resources/onchain/networks/new_pools.py index e84029a..2746bb7 100644 --- a/src/coingecko_sdk/resources/onchain/networks/new_pools.py +++ b/src/coingecko_sdk/resources/onchain/networks/new_pools.py @@ -5,7 +5,7 @@ import httpx from ...._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ...._utils import maybe_transform, async_maybe_transform +from ...._utils import path_template, maybe_transform, async_maybe_transform from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -133,7 +133,7 @@ def get_network( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return self._get( - f"/onchain/networks/{network}/new_pools", + path_template("/onchain/networks/{network}/new_pools", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -263,7 +263,7 @@ async def get_network( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return await self._get( - f"/onchain/networks/{network}/new_pools", + path_template("/onchain/networks/{network}/new_pools", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/pools/info.py b/src/coingecko_sdk/resources/onchain/networks/pools/info.py index 0a4b99a..8f2d0af 100644 --- a/src/coingecko_sdk/resources/onchain/networks/pools/info.py +++ b/src/coingecko_sdk/resources/onchain/networks/pools/info.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -77,7 +77,9 @@ def get( if not pool_address: raise ValueError(f"Expected a non-empty value for `pool_address` but received {pool_address!r}") return self._get( - f"/onchain/networks/{network}/pools/{pool_address}/info", + path_template( + "/onchain/networks/{network}/pools/{pool_address}/info", network=network, pool_address=pool_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -143,7 +145,9 @@ async def get( if not pool_address: raise ValueError(f"Expected a non-empty value for `pool_address` but received {pool_address!r}") return await self._get( - f"/onchain/networks/{network}/pools/{pool_address}/info", + path_template( + "/onchain/networks/{network}/pools/{pool_address}/info", network=network, pool_address=pool_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/pools/multi.py b/src/coingecko_sdk/resources/onchain/networks/pools/multi.py index d8b7665..18d637b 100644 --- a/src/coingecko_sdk/resources/onchain/networks/pools/multi.py +++ b/src/coingecko_sdk/resources/onchain/networks/pools/multi.py @@ -5,7 +5,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -81,7 +81,7 @@ def get_addresses( if not addresses: raise ValueError(f"Expected a non-empty value for `addresses` but received {addresses!r}") return self._get( - f"/onchain/networks/{network}/pools/multi/{addresses}", + path_template("/onchain/networks/{network}/pools/multi/{addresses}", network=network, addresses=addresses), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -160,7 +160,7 @@ async def get_addresses( if not addresses: raise ValueError(f"Expected a non-empty value for `addresses` but received {addresses!r}") return await self._get( - f"/onchain/networks/{network}/pools/multi/{addresses}", + path_template("/onchain/networks/{network}/pools/multi/{addresses}", network=network, addresses=addresses), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/pools/ohlcv.py b/src/coingecko_sdk/resources/onchain/networks/pools/ohlcv.py index a2e5c90..57132bb 100644 --- a/src/coingecko_sdk/resources/onchain/networks/pools/ohlcv.py +++ b/src/coingecko_sdk/resources/onchain/networks/pools/ohlcv.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -97,7 +97,12 @@ def get_timeframe( if not timeframe: raise ValueError(f"Expected a non-empty value for `timeframe` but received {timeframe!r}") return self._get( - f"/onchain/networks/{network}/pools/{pool_address}/ohlcv/{timeframe}", + path_template( + "/onchain/networks/{network}/pools/{pool_address}/ohlcv/{timeframe}", + network=network, + pool_address=pool_address, + timeframe=timeframe, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -193,7 +198,12 @@ async def get_timeframe( if not timeframe: raise ValueError(f"Expected a non-empty value for `timeframe` but received {timeframe!r}") return await self._get( - f"/onchain/networks/{network}/pools/{pool_address}/ohlcv/{timeframe}", + path_template( + "/onchain/networks/{network}/pools/{pool_address}/ohlcv/{timeframe}", + network=network, + pool_address=pool_address, + timeframe=timeframe, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/pools/pools.py b/src/coingecko_sdk/resources/onchain/networks/pools/pools.py index 8bef6b8..af9c4e3 100644 --- a/src/coingecko_sdk/resources/onchain/networks/pools/pools.py +++ b/src/coingecko_sdk/resources/onchain/networks/pools/pools.py @@ -39,7 +39,7 @@ AsyncTradesResourceWithStreamingResponse, ) from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -133,7 +133,7 @@ def get( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return self._get( - f"/onchain/networks/{network}/pools", + path_template("/onchain/networks/{network}/pools", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -192,7 +192,7 @@ def get_address( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return self._get( - f"/onchain/networks/{network}/pools/{address}", + path_template("/onchain/networks/{network}/pools/{address}", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -288,7 +288,7 @@ async def get( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return await self._get( - f"/onchain/networks/{network}/pools", + path_template("/onchain/networks/{network}/pools", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -347,7 +347,7 @@ async def get_address( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return await self._get( - f"/onchain/networks/{network}/pools/{address}", + path_template("/onchain/networks/{network}/pools/{address}", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/pools/trades.py b/src/coingecko_sdk/resources/onchain/networks/pools/trades.py index 5658eb6..9dc1ee5 100644 --- a/src/coingecko_sdk/resources/onchain/networks/pools/trades.py +++ b/src/coingecko_sdk/resources/onchain/networks/pools/trades.py @@ -5,7 +5,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -78,7 +78,9 @@ def get( if not pool_address: raise ValueError(f"Expected a non-empty value for `pool_address` but received {pool_address!r}") return self._get( - f"/onchain/networks/{network}/pools/{pool_address}/trades", + path_template( + "/onchain/networks/{network}/pools/{pool_address}/trades", network=network, pool_address=pool_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -153,7 +155,9 @@ async def get( if not pool_address: raise ValueError(f"Expected a non-empty value for `pool_address` but received {pool_address!r}") return await self._get( - f"/onchain/networks/{network}/pools/{pool_address}/trades", + path_template( + "/onchain/networks/{network}/pools/{pool_address}/trades", network=network, pool_address=pool_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/holders_chart.py b/src/coingecko_sdk/resources/onchain/networks/tokens/holders_chart.py index ce1af66..a09327e 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/holders_chart.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/holders_chart.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -76,7 +76,11 @@ def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return self._get( - f"/onchain/networks/{network}/tokens/{token_address}/holders_chart", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/holders_chart", + network=network, + token_address=token_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -141,7 +145,11 @@ async def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{token_address}/holders_chart", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/holders_chart", + network=network, + token_address=token_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/info.py b/src/coingecko_sdk/resources/onchain/networks/tokens/info.py index cd09778..15a884a 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/info.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/info.py @@ -5,6 +5,7 @@ import httpx from ....._types import Body, Query, Headers, NotGiven, not_given +from ....._utils import path_template from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -70,7 +71,7 @@ def get( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return self._get( - f"/onchain/networks/{network}/tokens/{address}/info", + path_template("/onchain/networks/{network}/tokens/{address}/info", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), @@ -129,7 +130,7 @@ async def get( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{address}/info", + path_template("/onchain/networks/{network}/tokens/{address}/info", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/multi.py b/src/coingecko_sdk/resources/onchain/networks/tokens/multi.py index 7f87115..c0a023b 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/multi.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/multi.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -82,7 +82,7 @@ def get_addresses( if not addresses: raise ValueError(f"Expected a non-empty value for `addresses` but received {addresses!r}") return self._get( - f"/onchain/networks/{network}/tokens/multi/{addresses}", + path_template("/onchain/networks/{network}/tokens/multi/{addresses}", network=network, addresses=addresses), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -160,7 +160,7 @@ async def get_addresses( if not addresses: raise ValueError(f"Expected a non-empty value for `addresses` but received {addresses!r}") return await self._get( - f"/onchain/networks/{network}/tokens/multi/{addresses}", + path_template("/onchain/networks/{network}/tokens/multi/{addresses}", network=network, addresses=addresses), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/ohlcv.py b/src/coingecko_sdk/resources/onchain/networks/tokens/ohlcv.py index e7489ec..c8634bc 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/ohlcv.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/ohlcv.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -98,7 +98,12 @@ def get_timeframe( if not timeframe: raise ValueError(f"Expected a non-empty value for `timeframe` but received {timeframe!r}") return self._get( - f"/onchain/networks/{network}/tokens/{token_address}/ohlcv/{timeframe}", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/ohlcv/{timeframe}", + network=network, + token_address=token_address, + timeframe=timeframe, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -195,7 +200,12 @@ async def get_timeframe( if not timeframe: raise ValueError(f"Expected a non-empty value for `timeframe` but received {timeframe!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{token_address}/ohlcv/{timeframe}", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/ohlcv/{timeframe}", + network=network, + token_address=token_address, + timeframe=timeframe, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/pools.py b/src/coingecko_sdk/resources/onchain/networks/tokens/pools.py index 340236e..cbd6886 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/pools.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/pools.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -90,7 +90,9 @@ def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return self._get( - f"/onchain/networks/{network}/tokens/{token_address}/pools", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/pools", network=network, token_address=token_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -178,7 +180,9 @@ async def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{token_address}/pools", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/pools", network=network, token_address=token_address + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/tokens.py b/src/coingecko_sdk/resources/onchain/networks/tokens/tokens.py index b99a441..a5dab5e 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/tokens.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/tokens.py @@ -47,7 +47,7 @@ AsyncTradesResourceWithStreamingResponse, ) from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from .top_holders import ( TopHoldersResource, @@ -180,7 +180,7 @@ def get_address( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return self._get( - f"/onchain/networks/{network}/tokens/{address}", + path_template("/onchain/networks/{network}/tokens/{address}", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -292,7 +292,7 @@ async def get_address( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{address}", + path_template("/onchain/networks/{network}/tokens/{address}", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/top_holders.py b/src/coingecko_sdk/resources/onchain/networks/tokens/top_holders.py index 34ebff3..b946e5e 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/top_holders.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/top_holders.py @@ -5,7 +5,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -78,7 +78,7 @@ def get( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return self._get( - f"/onchain/networks/{network}/tokens/{address}/top_holders", + path_template("/onchain/networks/{network}/tokens/{address}/top_holders", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -153,7 +153,7 @@ async def get( if not address: raise ValueError(f"Expected a non-empty value for `address` but received {address!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{address}/top_holders", + path_template("/onchain/networks/{network}/tokens/{address}/top_holders", network=network, address=address), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/top_traders.py b/src/coingecko_sdk/resources/onchain/networks/tokens/top_traders.py index 812873e..4c88632 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/top_traders.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/top_traders.py @@ -7,7 +7,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -84,7 +84,11 @@ def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return self._get( - f"/onchain/networks/{network_id}/tokens/{token_address}/top_traders", + path_template( + "/onchain/networks/{network_id}/tokens/{token_address}/top_traders", + network_id=network_id, + token_address=token_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -164,7 +168,11 @@ async def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return await self._get( - f"/onchain/networks/{network_id}/tokens/{token_address}/top_traders", + path_template( + "/onchain/networks/{network_id}/tokens/{token_address}/top_traders", + network_id=network_id, + token_address=token_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/tokens/trades.py b/src/coingecko_sdk/resources/onchain/networks/tokens/trades.py index ec849f9..8277869 100644 --- a/src/coingecko_sdk/resources/onchain/networks/tokens/trades.py +++ b/src/coingecko_sdk/resources/onchain/networks/tokens/trades.py @@ -5,7 +5,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -74,7 +74,11 @@ def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return self._get( - f"/onchain/networks/{network}/tokens/{token_address}/trades", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/trades", + network=network, + token_address=token_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -142,7 +146,11 @@ async def get( if not token_address: raise ValueError(f"Expected a non-empty value for `token_address` but received {token_address!r}") return await self._get( - f"/onchain/networks/{network}/tokens/{token_address}/trades", + path_template( + "/onchain/networks/{network}/tokens/{token_address}/trades", + network=network, + token_address=token_address, + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/networks/trending_pools.py b/src/coingecko_sdk/resources/onchain/networks/trending_pools.py index 9d4b95b..edc1052 100644 --- a/src/coingecko_sdk/resources/onchain/networks/trending_pools.py +++ b/src/coingecko_sdk/resources/onchain/networks/trending_pools.py @@ -7,7 +7,7 @@ import httpx from ...._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ...._utils import maybe_transform, async_maybe_transform +from ...._utils import path_template, maybe_transform, async_maybe_transform from ...._compat import cached_property from ...._resource import SyncAPIResource, AsyncAPIResource from ...._response import ( @@ -143,7 +143,7 @@ def get_network( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return self._get( - f"/onchain/networks/{network}/trending_pools", + path_template("/onchain/networks/{network}/trending_pools", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -282,7 +282,7 @@ async def get_network( if not network: raise ValueError(f"Expected a non-empty value for `network` but received {network!r}") return await self._get( - f"/onchain/networks/{network}/trending_pools", + path_template("/onchain/networks/{network}/trending_pools", network=network), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/onchain/simple/networks/token_price.py b/src/coingecko_sdk/resources/onchain/simple/networks/token_price.py index 43c0a57..302c132 100644 --- a/src/coingecko_sdk/resources/onchain/simple/networks/token_price.py +++ b/src/coingecko_sdk/resources/onchain/simple/networks/token_price.py @@ -5,7 +5,7 @@ import httpx from ....._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ....._utils import maybe_transform, async_maybe_transform +from ....._utils import path_template, maybe_transform, async_maybe_transform from ....._compat import cached_property from ....._resource import SyncAPIResource, AsyncAPIResource from ....._response import ( @@ -90,7 +90,9 @@ def get_addresses( if not addresses: raise ValueError(f"Expected a non-empty value for `addresses` but received {addresses!r}") return self._get( - f"/onchain/simple/networks/{network}/token_price/{addresses}", + path_template( + "/onchain/simple/networks/{network}/token_price/{addresses}", network=network, addresses=addresses + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -181,7 +183,9 @@ async def get_addresses( if not addresses: raise ValueError(f"Expected a non-empty value for `addresses` but received {addresses!r}") return await self._get( - f"/onchain/simple/networks/{network}/token_price/{addresses}", + path_template( + "/onchain/simple/networks/{network}/token_price/{addresses}", network=network, addresses=addresses + ), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/public_treasury.py b/src/coingecko_sdk/resources/public_treasury.py index dbbfec5..36f24d8 100644 --- a/src/coingecko_sdk/resources/public_treasury.py +++ b/src/coingecko_sdk/resources/public_treasury.py @@ -14,7 +14,7 @@ public_treasury_get_transaction_history_params, ) from .._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from .._utils import maybe_transform, async_maybe_transform +from .._utils import path_template, maybe_transform, async_maybe_transform from .._compat import cached_property from .._resource import SyncAPIResource, AsyncAPIResource from .._response import ( @@ -93,7 +93,7 @@ def get_coin_id( return cast( PublicTreasuryGetCoinIDResponse, self._get( - f"/{entity}/public_treasury/{coin_id}", + path_template("/{entity}/public_treasury/{coin_id}", entity=entity, coin_id=coin_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -149,7 +149,7 @@ def get_entity_id( if not entity_id: raise ValueError(f"Expected a non-empty value for `entity_id` but received {entity_id!r}") return self._get( - f"/public_treasury/{entity_id}", + path_template("/public_treasury/{entity_id}", entity_id=entity_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -202,7 +202,7 @@ def get_holding_chart( if not coin_id: raise ValueError(f"Expected a non-empty value for `coin_id` but received {coin_id!r}") return self._get( - f"/public_treasury/{entity_id}/{coin_id}/holding_chart", + path_template("/public_treasury/{entity_id}/{coin_id}/holding_chart", entity_id=entity_id, coin_id=coin_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -269,7 +269,7 @@ def get_transaction_history( if not entity_id: raise ValueError(f"Expected a non-empty value for `entity_id` but received {entity_id!r}") return self._get( - f"/public_treasury/{entity_id}/transaction_history", + path_template("/public_treasury/{entity_id}/transaction_history", entity_id=entity_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -350,7 +350,7 @@ async def get_coin_id( return cast( PublicTreasuryGetCoinIDResponse, await self._get( - f"/{entity}/public_treasury/{coin_id}", + path_template("/{entity}/public_treasury/{coin_id}", entity=entity, coin_id=coin_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -406,7 +406,7 @@ async def get_entity_id( if not entity_id: raise ValueError(f"Expected a non-empty value for `entity_id` but received {entity_id!r}") return await self._get( - f"/public_treasury/{entity_id}", + path_template("/public_treasury/{entity_id}", entity_id=entity_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -459,7 +459,7 @@ async def get_holding_chart( if not coin_id: raise ValueError(f"Expected a non-empty value for `coin_id` but received {coin_id!r}") return await self._get( - f"/public_treasury/{entity_id}/{coin_id}/holding_chart", + path_template("/public_treasury/{entity_id}/{coin_id}/holding_chart", entity_id=entity_id, coin_id=coin_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -526,7 +526,7 @@ async def get_transaction_history( if not entity_id: raise ValueError(f"Expected a non-empty value for `entity_id` but received {entity_id!r}") return await self._get( - f"/public_treasury/{entity_id}/transaction_history", + path_template("/public_treasury/{entity_id}/transaction_history", entity_id=entity_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/simple/token_price.py b/src/coingecko_sdk/resources/simple/token_price.py index b7f1416..0bfc8f5 100644 --- a/src/coingecko_sdk/resources/simple/token_price.py +++ b/src/coingecko_sdk/resources/simple/token_price.py @@ -7,7 +7,7 @@ import httpx from ..._types import Body, Omit, Query, Headers, NotGiven, omit, not_given -from ..._utils import maybe_transform, async_maybe_transform +from ..._utils import path_template, maybe_transform, async_maybe_transform from ..._compat import cached_property from ..._resource import SyncAPIResource, AsyncAPIResource from ..._response import ( @@ -116,7 +116,7 @@ def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return self._get( - f"/simple/token_price/{id}", + path_template("/simple/token_price/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, @@ -232,7 +232,7 @@ async def get_id( if not id: raise ValueError(f"Expected a non-empty value for `id` but received {id!r}") return await self._get( - f"/simple/token_price/{id}", + path_template("/simple/token_price/{id}", id=id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, diff --git a/src/coingecko_sdk/resources/token_lists.py b/src/coingecko_sdk/resources/token_lists.py index 69f8e4a..9729056 100644 --- a/src/coingecko_sdk/resources/token_lists.py +++ b/src/coingecko_sdk/resources/token_lists.py @@ -5,6 +5,7 @@ import httpx from .._types import Body, Query, Headers, NotGiven, not_given +from .._utils import path_template from .._compat import cached_property from .._resource import SyncAPIResource, AsyncAPIResource from .._response import ( @@ -67,7 +68,7 @@ def get_all_json( if not asset_platform_id: raise ValueError(f"Expected a non-empty value for `asset_platform_id` but received {asset_platform_id!r}") return self._get( - f"/token_lists/{asset_platform_id}/all.json", + path_template("/token_lists/{asset_platform_id}/all.json", asset_platform_id=asset_platform_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), @@ -123,7 +124,7 @@ async def get_all_json( if not asset_platform_id: raise ValueError(f"Expected a non-empty value for `asset_platform_id` but received {asset_platform_id!r}") return await self._get( - f"/token_lists/{asset_platform_id}/all.json", + path_template("/token_lists/{asset_platform_id}/all.json", asset_platform_id=asset_platform_id), options=make_request_options( extra_headers=extra_headers, extra_query=extra_query, extra_body=extra_body, timeout=timeout ), diff --git a/tests/test_utils/test_path.py b/tests/test_utils/test_path.py new file mode 100644 index 0000000..3a96447 --- /dev/null +++ b/tests/test_utils/test_path.py @@ -0,0 +1,89 @@ +from __future__ import annotations + +from typing import Any + +import pytest + +from coingecko_sdk._utils._path import path_template + + +@pytest.mark.parametrize( + "template, kwargs, expected", + [ + ("/v1/{id}", dict(id="abc"), "/v1/abc"), + ("/v1/{a}/{b}", dict(a="x", b="y"), "/v1/x/y"), + ("/v1/{a}{b}/path/{c}?val={d}#{e}", dict(a="x", b="y", c="z", d="u", e="v"), "/v1/xy/path/z?val=u#v"), + ("/{w}/{w}", dict(w="echo"), "/echo/echo"), + ("/v1/static", {}, "/v1/static"), + ("", {}, ""), + ("/v1/?q={n}&count=10", dict(n=42), "/v1/?q=42&count=10"), + ("/v1/{v}", dict(v=None), "/v1/null"), + ("/v1/{v}", dict(v=True), "/v1/true"), + ("/v1/{v}", dict(v=False), "/v1/false"), + ("/v1/{v}", dict(v=".hidden"), "/v1/.hidden"), # dot prefix ok + ("/v1/{v}", dict(v="file.txt"), "/v1/file.txt"), # dot in middle ok + ("/v1/{v}", dict(v="..."), "/v1/..."), # triple dot ok + ("/v1/{a}{b}", dict(a=".", b="txt"), "/v1/.txt"), # dot var combining with adjacent to be ok + ("/items?q={v}#{f}", dict(v=".", f=".."), "/items?q=.#.."), # dots in query/fragment are fine + ( + "/v1/{a}?query={b}", + dict(a="../../other/endpoint", b="a&bad=true"), + "/v1/..%2F..%2Fother%2Fendpoint?query=a%26bad%3Dtrue", + ), + ("/v1/{val}", dict(val="a/b/c"), "/v1/a%2Fb%2Fc"), + ("/v1/{val}", dict(val="a/b/c?query=value"), "/v1/a%2Fb%2Fc%3Fquery=value"), + ("/v1/{val}", dict(val="a/b/c?query=value&bad=true"), "/v1/a%2Fb%2Fc%3Fquery=value&bad=true"), + ("/v1/{val}", dict(val="%20"), "/v1/%2520"), # escapes escape sequences in input + # Query: slash and ? are safe, # is not + ("/items?q={v}", dict(v="a/b"), "/items?q=a/b"), + ("/items?q={v}", dict(v="a?b"), "/items?q=a?b"), + ("/items?q={v}", dict(v="a#b"), "/items?q=a%23b"), + ("/items?q={v}", dict(v="a b"), "/items?q=a%20b"), + # Fragment: slash and ? are safe + ("/docs#{v}", dict(v="a/b"), "/docs#a/b"), + ("/docs#{v}", dict(v="a?b"), "/docs#a?b"), + # Path: slash, ? and # are all encoded + ("/v1/{v}", dict(v="a/b"), "/v1/a%2Fb"), + ("/v1/{v}", dict(v="a?b"), "/v1/a%3Fb"), + ("/v1/{v}", dict(v="a#b"), "/v1/a%23b"), + # same var encoded differently by component + ( + "/v1/{v}?q={v}#{v}", + dict(v="a/b?c#d"), + "/v1/a%2Fb%3Fc%23d?q=a/b?c%23d#a/b?c%23d", + ), + ("/v1/{val}", dict(val="x?admin=true"), "/v1/x%3Fadmin=true"), # query injection + ("/v1/{val}", dict(val="x#admin"), "/v1/x%23admin"), # fragment injection + ], +) +def test_interpolation(template: str, kwargs: dict[str, Any], expected: str) -> None: + assert path_template(template, **kwargs) == expected + + +def test_missing_kwarg_raises_key_error() -> None: + with pytest.raises(KeyError, match="org_id"): + path_template("/v1/{org_id}") + + +@pytest.mark.parametrize( + "template, kwargs", + [ + ("{a}/path", dict(a=".")), + ("{a}/path", dict(a="..")), + ("/v1/{a}", dict(a=".")), + ("/v1/{a}", dict(a="..")), + ("/v1/{a}/path", dict(a=".")), + ("/v1/{a}/path", dict(a="..")), + ("/v1/{a}{b}", dict(a=".", b=".")), # adjacent vars → ".." + ("/v1/{a}.", dict(a=".")), # var + static → ".." + ("/v1/{a}{b}", dict(a="", b=".")), # empty + dot → "." + ("/v1/%2e/{x}", dict(x="ok")), # encoded dot in static text + ("/v1/%2e./{x}", dict(x="ok")), # mixed encoded ".." in static + ("/v1/.%2E/{x}", dict(x="ok")), # mixed encoded ".." in static + ("/v1/{v}?q=1", dict(v="..")), + ("/v1/{v}#frag", dict(v="..")), + ], +) +def test_dot_segment_rejected(template: str, kwargs: dict[str, Any]) -> None: + with pytest.raises(ValueError, match="dot-segment"): + path_template(template, **kwargs) From f08b6fa4c917515087065f05c67db1491a5b7ea9 Mon Sep 17 00:00:00 2001 From: "stainless-app[bot]" <142633134+stainless-app[bot]@users.noreply.github.com> Date: Fri, 20 Mar 2026 03:34:56 +0000 Subject: [PATCH 6/6] release: 1.13.1 --- .release-please-manifest.json | 2 +- CHANGELOG.md | 16 ++++++++++++++++ pyproject.toml | 2 +- src/coingecko_sdk/_version.py | 2 +- 4 files changed, 19 insertions(+), 3 deletions(-) diff --git a/.release-please-manifest.json b/.release-please-manifest.json index f94eeca..734ad79 100644 --- a/.release-please-manifest.json +++ b/.release-please-manifest.json @@ -1,3 +1,3 @@ { - ".": "1.13.0" + ".": "1.13.1" } \ No newline at end of file diff --git a/CHANGELOG.md b/CHANGELOG.md index 458ba75..61a7b88 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,5 +1,21 @@ # Changelog +## 1.13.1 (2026-03-20) + +Full Changelog: [v1.13.0...v1.13.1](https://github.com/coingecko/coingecko-python/compare/v1.13.0...v1.13.1) + +### Bug Fixes + +* **deps:** bump minimum typing-extensions version ([7a2809e](https://github.com/coingecko/coingecko-python/commit/7a2809e203ebeacd52704696b19c94780d9c0db6)) +* **pydantic:** do not pass `by_alias` unless set ([5b4c5d1](https://github.com/coingecko/coingecko-python/commit/5b4c5d1fec42630e39a569422baf75eb0990bb31)) +* sanitize endpoint path params ([a87e9d4](https://github.com/coingecko/coingecko-python/commit/a87e9d4bacf39d06c9becdd45dcf6a422699e987)) + + +### Chores + +* **ci:** skip uploading artifacts on stainless-internal branches ([53e9338](https://github.com/coingecko/coingecko-python/commit/53e93380ab1adfadf0a86d219c9216bb8bfe4174)) +* **internal:** tweak CI branches ([4d1960a](https://github.com/coingecko/coingecko-python/commit/4d1960abd7dfb3610749bcc303579deb787f8853)) + ## 1.13.0 (2026-02-25) Full Changelog: [v1.12.0...v1.13.0](https://github.com/coingecko/coingecko-python/compare/v1.12.0...v1.13.0) diff --git a/pyproject.toml b/pyproject.toml index 500fa06..7daa5fd 100644 --- a/pyproject.toml +++ b/pyproject.toml @@ -1,6 +1,6 @@ [project] name = "coingecko_sdk" -version = "1.13.0" +version = "1.13.1" description = "The official Python library for the coingecko API" dynamic = ["readme"] license = "Apache-2.0" diff --git a/src/coingecko_sdk/_version.py b/src/coingecko_sdk/_version.py index a04e2ca..64f38c7 100644 --- a/src/coingecko_sdk/_version.py +++ b/src/coingecko_sdk/_version.py @@ -1,4 +1,4 @@ # File generated from our OpenAPI spec by Stainless. See CONTRIBUTING.md for details. __title__ = "coingecko_sdk" -__version__ = "1.13.0" # x-release-please-version +__version__ = "1.13.1" # x-release-please-version