stackit-objectstorage 1.2.0__tar.gz → 1.2.2__tar.gz

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.
Files changed (38) hide show
  1. stackit_objectstorage-1.2.2/.gitignore +267 -0
  2. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/PKG-INFO +14 -12
  3. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/pyproject.toml +48 -36
  4. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/api_client.py +22 -9
  5. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/exceptions.py +1 -1
  6. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/__init__.py +0 -1
  7. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/create_access_key_payload.py +15 -1
  8. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/error_message.py +3 -3
  9. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/http_validation_error.py +3 -3
  10. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/list_access_keys_response.py +3 -3
  11. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/list_buckets_response.py +3 -3
  12. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/list_credentials_groups_response.py +3 -3
  13. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/validation_error.py +3 -3
  14. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/rest.py +19 -3
  15. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/LICENSE.md +0 -0
  16. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/NOTICE.txt +0 -0
  17. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/README.md +0 -0
  18. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/__init__.py +0 -0
  19. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/api/__init__.py +0 -0
  20. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/api/default_api.py +0 -0
  21. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/api_response.py +0 -0
  22. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/configuration.py +0 -0
  23. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/access_key.py +0 -0
  24. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/bucket.py +0 -0
  25. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/create_access_key_response.py +0 -0
  26. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/create_bucket_response.py +0 -0
  27. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/create_credentials_group_payload.py +0 -0
  28. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/create_credentials_group_response.py +0 -0
  29. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/credentials_group.py +0 -0
  30. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/delete_access_key_response.py +0 -0
  31. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/delete_bucket_response.py +0 -0
  32. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/delete_credentials_group_response.py +0 -0
  33. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/detailed_error.py +0 -0
  34. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/get_bucket_response.py +0 -0
  35. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/location_inner.py +0 -0
  36. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/project_scope.py +0 -0
  37. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/models/project_status.py +0 -0
  38. {stackit_objectstorage-1.2.0 → stackit_objectstorage-1.2.2}/src/stackit/objectstorage/py.typed +0 -0
@@ -0,0 +1,267 @@
1
+ ### VisualStudioCode template
2
+ .vscode/*
3
+ !.vscode/settings.json
4
+ !.vscode/tasks.json
5
+ !.vscode/launch.json
6
+ !.vscode/extensions.json
7
+ !.vscode/*.code-snippets
8
+
9
+ # Local History for Visual Studio Code
10
+ .history/
11
+
12
+ # Built Visual Studio Code Extensions
13
+ *.vsix
14
+
15
+ ### JetBrains template
16
+ # Covers JetBrains IDEs: IntelliJ, RubyMine, PhpStorm, AppCode, PyCharm, CLion, Android Studio, WebStorm and Rider
17
+ # Reference: https://intellij-support.jetbrains.com/hc/en-us/articles/206544839
18
+
19
+ # User-specific stuff
20
+ .idea/**/workspace.xml
21
+ .idea/**/tasks.xml
22
+ .idea/**/usage.statistics.xml
23
+ .idea/**/dictionaries
24
+ .idea/**/shelf
25
+
26
+ # AWS User-specific
27
+ .idea/**/aws.xml
28
+
29
+ # Generated files
30
+ .idea/**/contentModel.xml
31
+
32
+ # Sensitive or high-churn files
33
+ .idea/**/dataSources/
34
+ .idea/**/dataSources.ids
35
+ .idea/**/dataSources.local.xml
36
+ .idea/**/sqlDataSources.xml
37
+ .idea/**/dynamic.xml
38
+ .idea/**/uiDesigner.xml
39
+ .idea/**/dbnavigator.xml
40
+
41
+ # Gradle
42
+ .idea/**/gradle.xml
43
+ .idea/**/libraries
44
+
45
+ # Gradle and Maven with auto-import
46
+ # When using Gradle or Maven with auto-import, you should exclude module files,
47
+ # since they will be recreated, and may cause churn. Uncomment if using
48
+ # auto-import.
49
+ # .idea/artifacts
50
+ # .idea/compiler.xml
51
+ # .idea/jarRepositories.xml
52
+ # .idea/modules.xml
53
+ # .idea/*.iml
54
+ # .idea/modules
55
+ # *.iml
56
+ # *.ipr
57
+
58
+ # CMake
59
+ cmake-build-*/
60
+
61
+ # Mongo Explorer plugin
62
+ .idea/**/mongoSettings.xml
63
+
64
+ # File-based project format
65
+ *.iws
66
+
67
+ # IntelliJ
68
+ out/
69
+
70
+ # mpeltonen/sbt-idea plugin
71
+ .idea_modules/
72
+
73
+ # JIRA plugin
74
+ atlassian-ide-plugin.xml
75
+
76
+ # Cursive Clojure plugin
77
+ .idea/replstate.xml
78
+
79
+ # SonarLint plugin
80
+ .idea/sonarlint/
81
+
82
+ # Crashlytics plugin (for Android Studio and IntelliJ)
83
+ com_crashlytics_export_strings.xml
84
+ crashlytics.properties
85
+ crashlytics-build.properties
86
+ fabric.properties
87
+
88
+ # Editor-based Rest Client
89
+ .idea/httpRequests
90
+
91
+ # Android studio 3.1+ serialized cache file
92
+ .idea/caches/build_file_checksums.ser
93
+
94
+ ### VirtualEnv template
95
+ # Virtualenv
96
+ # http://iamzed.com/2009/05/07/a-primer-on-virtualenv/
97
+ .Python
98
+ [Bb]in
99
+ [Ii]nclude
100
+ [Ll]ib
101
+ [Ll]ib64
102
+ [Ll]ocal
103
+ [Ss]cripts
104
+ pyvenv.cfg
105
+ .venv
106
+ pip-selfcheck.json
107
+
108
+ ### Python template
109
+ # Byte-compiled / optimized / DLL files
110
+ __pycache__/
111
+ *.py[cod]
112
+ *$py.class
113
+
114
+ # C extensions
115
+ *.so
116
+
117
+ # Distribution / packaging
118
+ build/
119
+ develop-eggs/
120
+ dist/
121
+ downloads/
122
+ eggs/
123
+ .eggs/
124
+ lib/
125
+ lib64/
126
+ parts/
127
+ sdist/
128
+ var/
129
+ wheels/
130
+ share/python-wheels/
131
+ *.egg-info/
132
+ .installed.cfg
133
+ *.egg
134
+ MANIFEST
135
+
136
+ # PyInstaller
137
+ # Usually these files are written by a python script from a template
138
+ # before PyInstaller builds the exe, so as to inject date/other infos into it.
139
+ *.manifest
140
+ *.spec
141
+
142
+ # Installer logs
143
+ pip-log.txt
144
+ pip-delete-this-directory.txt
145
+
146
+ # Unit test / coverage reports
147
+ htmlcov/
148
+ .tox/
149
+ .nox/
150
+ .coverage
151
+ .coverage.*
152
+ .cache
153
+ nosetests.xml
154
+ coverage.xml
155
+ *.cover
156
+ *.py,cover
157
+ .hypothesis/
158
+ .pytest_cache/
159
+ cover/
160
+
161
+ # Translations
162
+ *.mo
163
+ *.pot
164
+
165
+ # Django stuff:
166
+ *.log
167
+ local_settings.py
168
+ db.sqlite3
169
+ db.sqlite3-journal
170
+
171
+ # Flask stuff:
172
+ instance/
173
+ .webassets-cache
174
+
175
+ # Scrapy stuff:
176
+ .scrapy
177
+
178
+ # Sphinx documentation
179
+ docs/_build/
180
+
181
+ # PyBuilder
182
+ .pybuilder/
183
+ target/
184
+
185
+ # Jupyter Notebook
186
+ .ipynb_checkpoints
187
+
188
+ # IPython
189
+ profile_default/
190
+ ipython_config.py
191
+
192
+ # pyenv
193
+ # For a library or package, you might want to ignore these files since the code is
194
+ # intended to run in multiple environments; otherwise, check them in:
195
+ # .python-version
196
+
197
+ # pipenv
198
+ # According to pypa/pipenv#598, it is recommended to include Pipfile.lock in version control.
199
+ # However, in case of collaboration, if having platform-specific dependencies or dependencies
200
+ # having no cross-platform support, pipenv may install dependencies that don't work, or not
201
+ # install all needed dependencies.
202
+ #Pipfile.lock
203
+
204
+ # poetry
205
+ # Similar to Pipfile.lock, it is generally recommended to include poetry.lock in version control.
206
+ # This is especially recommended for binary packages to ensure reproducibility, and is more
207
+ # commonly ignored for libraries.
208
+ # https://python-poetry.org/docs/basic-usage/#commit-your-poetrylock-file-to-version-control
209
+ #poetry.lock
210
+
211
+ # pdm
212
+ # Similar to Pipfile.lock, it is generally recommended to include pdm.lock in version control.
213
+ #pdm.lock
214
+ # pdm stores project-wide configurations in .pdm.toml, but it is recommended to not include it
215
+ # in version control.
216
+ # https://pdm.fming.dev/#use-with-ide
217
+ .pdm.toml
218
+
219
+ # PEP 582; used by e.g. github.com/David-OConnor/pyflow and github.com/pdm-project/pdm
220
+ __pypackages__/
221
+
222
+ # Celery stuff
223
+ celerybeat-schedule
224
+ celerybeat.pid
225
+
226
+ # SageMath parsed files
227
+ *.sage.py
228
+
229
+ # Environments
230
+ .env
231
+ env/
232
+ venv/
233
+ ENV/
234
+ env.bak/
235
+ venv.bak/
236
+
237
+ # Spyder project settings
238
+ .spyderproject
239
+ .spyproject
240
+
241
+ # Rope project settings
242
+ .ropeproject
243
+
244
+ # mkdocs documentation
245
+ /site
246
+
247
+ # mypy
248
+ .mypy_cache/
249
+ .dmypy.json
250
+ dmypy.json
251
+
252
+ # Pyre type checker
253
+ .pyre/
254
+
255
+ # pytype static type analyzer
256
+ .pytype/
257
+
258
+ # Cython debug symbols
259
+ cython_debug/
260
+
261
+ # PyCharm
262
+ # JetBrains specific template is maintained in a separate JetBrains.gitignore that can
263
+ # be found at https://github.com/github/gitignore/blob/main/Global/JetBrains.gitignore
264
+ # and can be added to the global gitignore or merged into this file. For a more nuclear
265
+ # option (not recommended) you can uncomment the following to ignore the entire idea folder.
266
+ .idea/
267
+
@@ -1,10 +1,12 @@
1
- Metadata-Version: 2.3
1
+ Metadata-Version: 2.4
2
2
  Name: stackit-objectstorage
3
- Version: 1.2.0
3
+ Version: 1.2.2
4
4
  Summary: STACKIT Object Storage API
5
- Author: STACKIT Developer Tools
6
- Author-email: developer-tools@stackit.cloud
7
- Requires-Python: >=3.9,<4.0
5
+ Project-URL: Homepage, https://github.com/stackitcloud/stackit-sdk-python
6
+ Project-URL: Issues, https://github.com/stackitcloud/stackit-sdk-python/issues
7
+ Author-email: STACKIT Developer Tools <developer-tools@stackit.cloud>
8
+ License-File: LICENSE.md
9
+ License-File: NOTICE.txt
8
10
  Classifier: License :: OSI Approved :: Apache Software License
9
11
  Classifier: Operating System :: OS Independent
10
12
  Classifier: Programming Language :: Python :: 3
@@ -13,12 +15,12 @@ Classifier: Programming Language :: Python :: 3.10
13
15
  Classifier: Programming Language :: Python :: 3.11
14
16
  Classifier: Programming Language :: Python :: 3.12
15
17
  Classifier: Programming Language :: Python :: 3.13
16
- Requires-Dist: pydantic (>=2.9.2)
17
- Requires-Dist: python-dateutil (>=2.9.0.post0)
18
- Requires-Dist: requests (>=2.32.3)
19
- Requires-Dist: stackit-core (>=0.0.1a)
20
- Project-URL: Homepage, https://github.com/stackitcloud/stackit-sdk-python
21
- Project-URL: Issues, https://github.com/stackitcloud/stackit-sdk-python/issues
18
+ Classifier: Programming Language :: Python :: 3.14
19
+ Requires-Python: <4.0,>=3.9
20
+ Requires-Dist: pydantic>=2.9.2
21
+ Requires-Dist: python-dateutil>=2.9.0.post0
22
+ Requires-Dist: requests>=2.32.3
23
+ Requires-Dist: stackit-core>=0.0.1a
22
24
  Description-Content-Type: text/markdown
23
25
 
24
26
  # stackit.objectstorage
@@ -43,4 +45,4 @@ import stackit.objectstorage
43
45
 
44
46
  ## Getting Started
45
47
 
46
- [Examples](https://github.com/stackitcloud/stackit-sdk-python/tree/main/examples) for the usage of the package can be found in the [GitHub repository](https://github.com/stackitcloud/stackit-sdk-python) of the SDK.
48
+ [Examples](https://github.com/stackitcloud/stackit-sdk-python/tree/main/examples) for the usage of the package can be found in the [GitHub repository](https://github.com/stackitcloud/stackit-sdk-python) of the SDK.
@@ -1,55 +1,67 @@
1
1
  [project]
2
2
  name = "stackit-objectstorage"
3
-
4
- [tool.poetry]
5
- name = "stackit-objectstorage"
6
- version = "v1.2.0"
7
- authors = [
8
- "STACKIT Developer Tools <developer-tools@stackit.cloud>",
9
- ]
3
+ version = "v1.2.2"
10
4
  description = "STACKIT Object Storage API"
5
+ authors = [{ name = "STACKIT Developer Tools", email = "developer-tools@stackit.cloud" }]
6
+ requires-python = ">=3.9,<4.0"
11
7
  readme = "README.md"
12
- #license = "NoLicense"
13
8
  classifiers = [
14
9
  "Programming Language :: Python :: 3",
15
10
  "License :: OSI Approved :: Apache Software License",
16
11
  "Operating System :: OS Independent",
12
+ "Programming Language :: Python :: 3.9",
13
+ "Programming Language :: Python :: 3.10",
14
+ "Programming Language :: Python :: 3.11",
15
+ "Programming Language :: Python :: 3.12",
16
+ "Programming Language :: Python :: 3.13",
17
+ "Programming Language :: Python :: 3.14",
17
18
  ]
18
- packages = [
19
- { include = "stackit", from="src" }
19
+ dependencies = [
20
+ "stackit-core>=0.0.1a",
21
+ "requests>=2.32.3",
22
+ "pydantic>=2.9.2",
23
+ "python-dateutil>=2.9.0.post0",
20
24
  ]
21
25
 
22
- [tool.poetry.dependencies]
23
- python = ">=3.9,<4.0"
24
- stackit-core = ">=0.0.1a"
25
- requests = ">=2.32.3"
26
- pydantic = ">=2.9.2"
27
- python-dateutil = ">=2.9.0.post0"
28
-
29
- [tool.poetry.group.dev.dependencies]
30
- black = ">=24.8.0"
31
- pytest = ">=8.3.3"
32
- flake8 = [
33
- { version= ">=5.0.3", python="<3.12"},
34
- { version= ">=6.0.1", python=">=3.12"}
35
- ]
36
- flake8-black = ">=0.3.6"
37
- flake8-pyproject = ">=1.2.3"
38
- autoimport = ">=1.6.1"
39
- flake8-eol = ">=0.0.8"
40
- flake8-eradicate = ">=1.5.0"
41
- flake8-bandit = ">=4.1.1"
42
- flake8-bugbear = ">=23.1.14"
43
- flake8-quotes = ">=3.4.0"
44
- isort = ">=5.13.2"
45
-
46
26
  [project.urls]
47
27
  Homepage = "https://github.com/stackitcloud/stackit-sdk-python"
48
28
  Issues = "https://github.com/stackitcloud/stackit-sdk-python/issues"
49
29
 
30
+ [dependency-groups]
31
+ dev = [
32
+ "black>=24.8.0",
33
+ "pytest>=8.3.3",
34
+ "flake8>=5.0.3 ; python_full_version < '3.12'",
35
+ "flake8>=6.0.1 ; python_full_version >= '3.12'",
36
+ "flake8-black>=0.3.6",
37
+ "flake8-pyproject>=1.2.3",
38
+ "autoimport>=1.6.1",
39
+ "flake8-eol>=0.0.8",
40
+ "flake8-eradicate>=1.5.0",
41
+ "flake8-bandit>=4.1.1",
42
+ "flake8-bugbear>=23.1.14",
43
+ "flake8-quotes>=3.4.0",
44
+ "isort>=5.13.2",
45
+ ]
46
+
47
+ [tool.uv]
48
+ default-groups = "all"
49
+
50
+ [tool.uv.sources]
51
+ stackit-core = { path = "../../core" }
52
+
53
+ [tool.hatch.build.targets.sdist]
54
+ include = ["src/stackit"]
55
+
56
+ [tool.hatch.build.targets.wheel]
57
+ include = ["src/stackit"]
58
+
59
+ [tool.hatch.build.targets.wheel.sources]
60
+ "src/stackit" = "stackit"
61
+
50
62
  [build-system]
51
- requires = ["setuptools", "poetry-core"]
52
- build-backend = "poetry.core.masonry.api"
63
+ requires = ["hatchling"]
64
+ build-backend = "hatchling.build"
53
65
 
54
66
  [tool.pytest.ini_options]
55
67
  pythonpath = [
@@ -12,11 +12,13 @@
12
12
  """ # noqa: E501
13
13
 
14
14
  import datetime
15
+ import decimal
15
16
  import json
16
17
  import mimetypes
17
18
  import os
18
19
  import re
19
20
  import tempfile
21
+ import uuid
20
22
  from enum import Enum
21
23
  from typing import Dict, List, Optional, Tuple, Union
22
24
  from urllib.parse import quote
@@ -63,8 +65,10 @@ class ApiClient:
63
65
  "bool": bool,
64
66
  "date": datetime.date,
65
67
  "datetime": datetime.datetime,
68
+ "decimal": decimal.Decimal,
66
69
  "object": object,
67
70
  }
71
+ _pool = None
68
72
 
69
73
  def __init__(self, configuration, header_name=None, header_value=None, cookie=None) -> None:
70
74
  self.config: Configuration = configuration
@@ -267,7 +271,7 @@ class ApiClient:
267
271
  return_data = self.__deserialize_file(response_data)
268
272
  elif response_type is not None:
269
273
  match = None
270
- content_type = response_data.getheader("content-type")
274
+ content_type = response_data.headers.get("content-type")
271
275
  if content_type is not None:
272
276
  match = re.search(r"charset=([a-zA-Z\-\d]+)[\s;]?", content_type)
273
277
  encoding = match.group(1) if match else "utf-8"
@@ -284,7 +288,7 @@ class ApiClient:
284
288
  return ApiResponse(
285
289
  status_code=response_data.status,
286
290
  data=return_data,
287
- headers=response_data.getheaders(),
291
+ headers=response_data.headers,
288
292
  raw_data=response_data.data,
289
293
  )
290
294
 
@@ -296,6 +300,7 @@ class ApiClient:
296
300
  If obj is str, int, long, float, bool, return directly.
297
301
  If obj is datetime.datetime, datetime.date
298
302
  convert to string in iso8601 format.
303
+ If obj is decimal.Decimal return string representation.
299
304
  If obj is list, sanitize each element in the list.
300
305
  If obj is dict, return the dict.
301
306
  If obj is OpenAPI model, return the properties dict.
@@ -311,12 +316,16 @@ class ApiClient:
311
316
  return obj.get_secret_value()
312
317
  elif isinstance(obj, self.PRIMITIVE_TYPES):
313
318
  return obj
319
+ elif isinstance(obj, uuid.UUID):
320
+ return str(obj)
314
321
  elif isinstance(obj, list):
315
322
  return [self.sanitize_for_serialization(sub_obj) for sub_obj in obj]
316
323
  elif isinstance(obj, tuple):
317
324
  return tuple(self.sanitize_for_serialization(sub_obj) for sub_obj in obj)
318
325
  elif isinstance(obj, (datetime.datetime, datetime.date)):
319
326
  return obj.isoformat()
327
+ elif isinstance(obj, decimal.Decimal):
328
+ return str(obj)
320
329
 
321
330
  elif isinstance(obj, dict):
322
331
  obj_dict = obj
@@ -326,7 +335,7 @@ class ApiClient:
326
335
  # and attributes which value is not None.
327
336
  # Convert attribute name to json key in
328
337
  # model definition for request.
329
- if hasattr(obj, "to_dict") and callable(obj.to_dict):
338
+ if hasattr(obj, "to_dict") and callable(getattr(obj, "to_dict")): # noqa: B009
330
339
  obj_dict = obj.to_dict()
331
340
  else:
332
341
  obj_dict = obj.__dict__
@@ -354,7 +363,7 @@ class ApiClient:
354
363
  data = json.loads(response_text)
355
364
  except ValueError:
356
365
  data = response_text
357
- elif re.match(r"^application/(json|[\w!#$&.+-^_]+\+json)\s*(;|$)", content_type, re.IGNORECASE):
366
+ elif re.match(r"^application/(json|[\w!#$&.+\-^_]+\+json)\s*(;|$)", content_type, re.IGNORECASE):
358
367
  if response_text == "":
359
368
  data = ""
360
369
  else:
@@ -400,12 +409,14 @@ class ApiClient:
400
409
 
401
410
  if klass in self.PRIMITIVE_TYPES:
402
411
  return self.__deserialize_primitive(data, klass)
403
- elif klass == object:
412
+ elif klass is object:
404
413
  return self.__deserialize_object(data)
405
- elif klass == datetime.date:
414
+ elif klass is datetime.date:
406
415
  return self.__deserialize_date(data)
407
- elif klass == datetime.datetime:
416
+ elif klass is datetime.datetime:
408
417
  return self.__deserialize_datetime(data)
418
+ elif klass is decimal.Decimal:
419
+ return decimal.Decimal(data)
409
420
  elif issubclass(klass, Enum):
410
421
  return self.__deserialize_enum(data, klass)
411
422
  else:
@@ -553,12 +564,14 @@ class ApiClient:
553
564
  os.close(fd)
554
565
  os.remove(path)
555
566
 
556
- content_disposition = response.getheader("Content-Disposition")
567
+ content_disposition = response.headers.get("Content-Disposition")
557
568
  if content_disposition:
558
569
  m = re.search(r'filename=[\'"]?([^\'"\s]+)[\'"]?', content_disposition)
559
570
  if m is None:
560
571
  raise ValueError("Unexpected 'content-disposition' header value")
561
- filename = m.group(1)
572
+ filename = os.path.basename(m.group(1)) # Strip any directory traversal
573
+ if filename in ("", ".", ".."): # fall back to tmp filename
574
+ filename = os.path.basename(path)
562
575
  path = os.path.join(os.path.dirname(path), filename)
563
576
 
564
577
  with open(path, "wb") as f:
@@ -129,7 +129,7 @@ class ApiException(OpenApiException):
129
129
  self.body = http_resp.data.decode("utf-8")
130
130
  except Exception: # noqa: S110
131
131
  pass
132
- self.headers = http_resp.getheaders()
132
+ self.headers = http_resp.headers
133
133
 
134
134
  @classmethod
135
135
  def from_response(
@@ -12,7 +12,6 @@
12
12
  Do not edit the class manually.
13
13
  """ # noqa: E501
14
14
 
15
-
16
15
  # import models into model package
17
16
  from stackit.objectstorage.models.access_key import AccessKey
18
17
  from stackit.objectstorage.models.bucket import Bucket
@@ -15,10 +15,11 @@ from __future__ import annotations
15
15
 
16
16
  import json
17
17
  import pprint
18
+ import re # noqa: F401
18
19
  from datetime import datetime
19
20
  from typing import Any, ClassVar, Dict, List, Optional, Set
20
21
 
21
- from pydantic import BaseModel, ConfigDict, Field
22
+ from pydantic import BaseModel, ConfigDict, Field, field_validator
22
23
  from typing_extensions import Self
23
24
 
24
25
 
@@ -30,6 +31,19 @@ class CreateAccessKeyPayload(BaseModel):
30
31
  expires: Optional[datetime] = Field(default=None, description="Expiration date. Null means never expires.")
31
32
  __properties: ClassVar[List[str]] = ["expires"]
32
33
 
34
+ @field_validator("expires", mode="before")
35
+ def expires_change_year_zero_to_one(cls, value):
36
+ """Workaround which prevents year 0 issue"""
37
+ if isinstance(value, str):
38
+ # Check for year "0000" at the beginning of the string
39
+ # This assumes common date formats like YYYY-MM-DDTHH:MM:SS+00:00 or YYYY-MM-DDTHH:MM:SSZ
40
+ if value.startswith("0000-01-01T") and re.match(
41
+ r"^\d{4}-\d{2}-\d{2}T\d{2}:\d{2}:\d{2}(\+\d{2}:\d{2}|Z)$", value
42
+ ):
43
+ # Workaround: Replace "0000" with "0001"
44
+ return "0001" + value[4:] # Take "0001" and append the rest of the string
45
+ return value
46
+
33
47
  model_config = ConfigDict(
34
48
  populate_by_name=True,
35
49
  validate_assignment=True,
@@ -71,9 +71,9 @@ class ErrorMessage(BaseModel):
71
71
  # override the default output from pydantic by calling `to_dict()` of each item in detail (list)
72
72
  _items = []
73
73
  if self.detail:
74
- for _item in self.detail:
75
- if _item:
76
- _items.append(_item.to_dict())
74
+ for _item_detail in self.detail:
75
+ if _item_detail:
76
+ _items.append(_item_detail.to_dict())
77
77
  _dict["detail"] = _items
78
78
  return _dict
79
79
 
@@ -71,9 +71,9 @@ class HTTPValidationError(BaseModel):
71
71
  # override the default output from pydantic by calling `to_dict()` of each item in detail (list)
72
72
  _items = []
73
73
  if self.detail:
74
- for _item in self.detail:
75
- if _item:
76
- _items.append(_item.to_dict())
74
+ for _item_detail in self.detail:
75
+ if _item_detail:
76
+ _items.append(_item_detail.to_dict())
77
77
  _dict["detail"] = _items
78
78
  return _dict
79
79
 
@@ -72,9 +72,9 @@ class ListAccessKeysResponse(BaseModel):
72
72
  # override the default output from pydantic by calling `to_dict()` of each item in access_keys (list)
73
73
  _items = []
74
74
  if self.access_keys:
75
- for _item in self.access_keys:
76
- if _item:
77
- _items.append(_item.to_dict())
75
+ for _item_access_keys in self.access_keys:
76
+ if _item_access_keys:
77
+ _items.append(_item_access_keys.to_dict())
78
78
  _dict["accessKeys"] = _items
79
79
  return _dict
80
80
 
@@ -72,9 +72,9 @@ class ListBucketsResponse(BaseModel):
72
72
  # override the default output from pydantic by calling `to_dict()` of each item in buckets (list)
73
73
  _items = []
74
74
  if self.buckets:
75
- for _item in self.buckets:
76
- if _item:
77
- _items.append(_item.to_dict())
75
+ for _item_buckets in self.buckets:
76
+ if _item_buckets:
77
+ _items.append(_item_buckets.to_dict())
78
78
  _dict["buckets"] = _items
79
79
  return _dict
80
80
 
@@ -72,9 +72,9 @@ class ListCredentialsGroupsResponse(BaseModel):
72
72
  # override the default output from pydantic by calling `to_dict()` of each item in credentials_groups (list)
73
73
  _items = []
74
74
  if self.credentials_groups:
75
- for _item in self.credentials_groups:
76
- if _item:
77
- _items.append(_item.to_dict())
75
+ for _item_credentials_groups in self.credentials_groups:
76
+ if _item_credentials_groups:
77
+ _items.append(_item_credentials_groups.to_dict())
78
78
  _dict["credentialsGroups"] = _items
79
79
  return _dict
80
80
 
@@ -73,9 +73,9 @@ class ValidationError(BaseModel):
73
73
  # override the default output from pydantic by calling `to_dict()` of each item in loc (list)
74
74
  _items = []
75
75
  if self.loc:
76
- for _item in self.loc:
77
- if _item:
78
- _items.append(_item.to_dict())
76
+ for _item_loc in self.loc:
77
+ if _item_loc:
78
+ _items.append(_item_loc.to_dict())
79
79
  _dict["loc"] = _items
80
80
  return _dict
81
81
 
@@ -38,12 +38,17 @@ class RESTResponse(io.IOBase):
38
38
  self.data = self.response.content
39
39
  return self.data
40
40
 
41
+ @property
42
+ def headers(self):
43
+ """Returns a dictionary of response headers."""
44
+ return self.response.headers
45
+
41
46
  def getheaders(self):
42
- """Returns a dictionary of the response headers."""
47
+ """Returns a dictionary of the response headers; use ``headers`` instead."""
43
48
  return self.response.headers
44
49
 
45
50
  def getheader(self, name, default=None):
46
- """Returns a given response header."""
51
+ """Returns a given response header; use ``headers.get()`` instead."""
47
52
  return self.response.headers.get(name, default)
48
53
 
49
54
 
@@ -93,6 +98,7 @@ class RESTClientObject:
93
98
  url,
94
99
  data=request_body,
95
100
  headers=headers,
101
+ timeout=_request_timeout,
96
102
  )
97
103
  elif content_type == "application/x-www-form-urlencoded":
98
104
  r = self.session.request(
@@ -100,6 +106,7 @@ class RESTClientObject:
100
106
  url,
101
107
  params=post_params,
102
108
  headers=headers,
109
+ timeout=_request_timeout,
103
110
  )
104
111
  elif content_type == "multipart/form-data":
105
112
  # must del headers['Content-Type'], or the correct
@@ -113,6 +120,7 @@ class RESTClientObject:
113
120
  url,
114
121
  files=post_params,
115
122
  headers=headers,
123
+ timeout=_request_timeout,
116
124
  )
117
125
  # Pass a `string` parameter directly in the body to support
118
126
  # other content types than JSON when `body` argument is
@@ -123,10 +131,17 @@ class RESTClientObject:
123
131
  url,
124
132
  data=body,
125
133
  headers=headers,
134
+ timeout=_request_timeout,
126
135
  )
127
136
  elif headers["Content-Type"].startswith("text/") and isinstance(body, bool):
128
137
  request_body = "true" if body else "false"
129
- r = self.session.request(method, url, data=request_body, headers=headers)
138
+ r = self.session.request(
139
+ method,
140
+ url,
141
+ data=request_body,
142
+ headers=headers,
143
+ timeout=_request_timeout,
144
+ )
130
145
  else:
131
146
  # Cannot generate the request from given parameters
132
147
  msg = """Cannot prepare a request message for provided
@@ -140,6 +155,7 @@ class RESTClientObject:
140
155
  url,
141
156
  params={},
142
157
  headers=headers,
158
+ timeout=_request_timeout,
143
159
  )
144
160
  except requests.exceptions.SSLError as e:
145
161
  msg = "\n".join([type(e).__name__, str(e)])