Skip to content

Commit 64d39b6

Browse files
committed
Add REST load credentials support
1 parent 55887b4 commit 64d39b6

2 files changed

Lines changed: 65 additions & 0 deletions

File tree

pyiceberg/catalog/rest/__init__.py

Lines changed: 32 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -147,6 +147,7 @@ class Endpoints:
147147
create_table: str = "namespaces/{namespace}/tables"
148148
register_table: str = "namespaces/{namespace}/register"
149149
load_table: str = "namespaces/{namespace}/tables/{table}"
150+
load_credentials: str = "namespaces/{namespace}/tables/{table}/credentials"
150151
update_table: str = "namespaces/{namespace}/tables/{table}"
151152
drop_table: str = "namespaces/{namespace}/tables/{table}"
152153
table_exists: str = "namespaces/{namespace}/tables/{table}"
@@ -181,6 +182,7 @@ class Capability:
181182
V1_DELETE_TABLE = Endpoint(http_method=HttpMethod.DELETE, path=f"{API_PREFIX}/{Endpoints.drop_table}")
182183
V1_RENAME_TABLE = Endpoint(http_method=HttpMethod.POST, path=f"{API_PREFIX}/{Endpoints.rename_table}")
183184
V1_REGISTER_TABLE = Endpoint(http_method=HttpMethod.POST, path=f"{API_PREFIX}/{Endpoints.register_table}")
185+
V1_LOAD_CREDENTIALS = Endpoint(http_method=HttpMethod.GET, path=f"{API_PREFIX}/{Endpoints.load_credentials}")
184186

185187
V1_LIST_VIEWS = Endpoint(http_method=HttpMethod.GET, path=f"{API_PREFIX}/{Endpoints.list_views}")
186188
V1_LOAD_VIEW = Endpoint(http_method=HttpMethod.GET, path=f"{API_PREFIX}/{Endpoints.load_view}")
@@ -293,6 +295,10 @@ class TableResponse(IcebergBaseModel):
293295
storage_credentials: list[StorageCredential] = Field(alias="storage-credentials", default_factory=list)
294296

295297

298+
class LoadCredentialsResponse(IcebergBaseModel):
299+
storage_credentials: list[StorageCredential] = Field(alias="storage-credentials")
300+
301+
296302
class ViewResponse(IcebergBaseModel):
297303
metadata_location: str | None = Field(alias="metadata-location", default=None)
298304
metadata: ViewMetadata
@@ -545,6 +551,32 @@ def _fetch_scan_tasks(self, identifier: str | Identifier, plan_task: str) -> Sca
545551

546552
return ScanTasks.model_validate_json(response.text)
547553

554+
@retry(**_RETRY_ARGS)
555+
def _load_credentials(
556+
self,
557+
identifier: str | Identifier,
558+
) -> LoadCredentialsResponse:
559+
"""Load raw vended storage credentials for a table."""
560+
self._check_endpoint(Capability.V1_LOAD_CREDENTIALS)
561+
response = self._session.get(
562+
self.url(Endpoints.load_credentials, prefixed=True, **self._split_identifier_for_path(identifier)),
563+
)
564+
try:
565+
response.raise_for_status()
566+
except HTTPError as exc:
567+
_handle_non_200_response(exc, {404: NoSuchTableError})
568+
569+
return LoadCredentialsResponse.model_validate_json(response.text)
570+
571+
def load_credentials(
572+
self,
573+
identifier: str | Identifier,
574+
location: str,
575+
) -> Properties:
576+
"""Load vended storage credentials and return the best match for a location."""
577+
credentials_response = self._load_credentials(identifier)
578+
return self._resolve_storage_credentials(credentials_response.storage_credentials, location)
579+
548580
def plan_scan(self, identifier: str | Identifier, request: PlanTableScanRequest) -> list[FileScanTask]:
549581
"""Plan a table scan and return FileScanTasks.
550582

tests/catalog/test_rest.py

Lines changed: 33 additions & 0 deletions
Original file line numberDiff line numberDiff line change
@@ -104,6 +104,7 @@
104104
Capability.V1_DELETE_TABLE,
105105
Capability.V1_RENAME_TABLE,
106106
Capability.V1_REGISTER_TABLE,
107+
Capability.V1_LOAD_CREDENTIALS,
107108
Capability.V1_LIST_VIEWS,
108109
Capability.V1_LOAD_VIEW,
109110
Capability.V1_VIEW_EXISTS,
@@ -3147,6 +3148,38 @@ def test_load_table_with_storage_credentials(rest_mock: Mocker, example_table_me
31473148
assert table.io.properties["s3.session-token"] == "vended-token"
31483149

31493150

3151+
def test_load_credentials_with_longest_prefix(rest_mock: Mocker) -> None:
3152+
rest_mock.get(
3153+
f"{TEST_URI}v1/namespaces/fokko/tables/table/credentials",
3154+
json={
3155+
"storage-credentials": [
3156+
{
3157+
"prefix": "s3://warehouse/database/",
3158+
"config": {"s3.access-key-id": "short-prefix-key"},
3159+
},
3160+
{
3161+
"prefix": "s3://warehouse/database/table",
3162+
"config": {
3163+
"s3.access-key-id": "long-prefix-key",
3164+
"s3.secret-access-key": "long-prefix-secret",
3165+
},
3166+
},
3167+
],
3168+
},
3169+
status_code=200,
3170+
request_headers=TEST_HEADERS,
3171+
)
3172+
catalog = RestCatalog("rest", uri=TEST_URI, token=TEST_TOKEN)
3173+
3174+
credentials = catalog.load_credentials(
3175+
("fokko", "table"),
3176+
"s3://warehouse/database/table/data/file.parquet",
3177+
)
3178+
3179+
assert credentials == {"s3.access-key-id": "long-prefix-key", "s3.secret-access-key": "long-prefix-secret"}
3180+
assert rest_mock.last_request.url == f"{TEST_URI}v1/namespaces/fokko/tables/table/credentials"
3181+
3182+
31503183
def test_load_table_without_storage_credentials(
31513184
rest_mock: Mocker, example_table_metadata_with_snapshot_v1_rest_json: dict[str, Any]
31523185
) -> None:

0 commit comments

Comments
 (0)