pulpcore 3.80.2__py3-none-any.whl → 3.81.0__py3-none-any.whl

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

Potentially problematic release.


This version of pulpcore might be problematic. Click here for more details.

@@ -6,6 +6,6 @@ class PulpCertGuardPluginAppConfig(PulpPluginAppConfig):
6
6
 
7
7
  name = "pulp_certguard.app"
8
8
  label = "certguard"
9
- version = "3.80.2"
9
+ version = "3.81.0"
10
10
  python_package_name = "pulpcore"
11
11
  domain_compatible = True
pulp_file/app/__init__.py CHANGED
@@ -8,6 +8,6 @@ class PulpFilePluginAppConfig(PulpPluginAppConfig):
8
8
 
9
9
  name = "pulp_file.app"
10
10
  label = "file"
11
- version = "3.80.2"
11
+ version = "3.81.0"
12
12
  python_package_name = "pulpcore"
13
13
  domain_compatible = True
@@ -1,12 +1,7 @@
1
1
  import os
2
2
  import uuid
3
- import subprocess
4
- from collections import defaultdict
5
- from pathlib import Path
6
3
 
7
- import aiofiles
8
4
  import pytest
9
- from aiohttp import web
10
5
 
11
6
  from pulpcore.tests.functional.utils import BindingsNamespace, generate_iso, generate_manifest
12
7
 
@@ -37,9 +32,9 @@ def file_random_content_unit(file_content_unit_with_name_factory):
37
32
 
38
33
 
39
34
  @pytest.fixture
40
- def file_content_unit_with_name_factory(file_bindings, random_artifact, monitor_task):
35
+ def file_content_unit_with_name_factory(file_bindings, random_artifact_factory, monitor_task):
41
36
  def _file_content_unit_with_name_factory(name):
42
- artifact_attrs = {"artifact": random_artifact.pulp_href, "relative_path": name}
37
+ artifact_attrs = {"artifact": random_artifact_factory().pulp_href, "relative_path": name}
43
38
  return file_bindings.ContentFilesApi.read(
44
39
  monitor_task(
45
40
  file_bindings.ContentFilesApi.create(**artifact_attrs).task
@@ -74,14 +69,13 @@ def file_distribution_factory(file_bindings, gen_object_with_cleanup):
74
69
  return _file_distribution_factory
75
70
 
76
71
 
77
- @pytest.fixture
78
- def file_fixtures_root(tmp_path):
79
- fixture_dir = tmp_path / "fixtures"
80
- fixture_dir.mkdir()
72
+ @pytest.fixture(scope="class")
73
+ def file_fixtures_root(tmp_path_factory):
74
+ fixture_dir = tmp_path_factory.mktemp("fixtures")
81
75
  return fixture_dir
82
76
 
83
77
 
84
- @pytest.fixture
78
+ @pytest.fixture(scope="class")
85
79
  def write_3_iso_file_fixture_data_factory(file_fixtures_root):
86
80
  def _write_3_iso_file_fixture_data_factory(name, overwrite=False, seed=None):
87
81
  file_fixtures_root.joinpath(name).mkdir(exist_ok=overwrite)
@@ -96,12 +90,12 @@ def write_3_iso_file_fixture_data_factory(file_fixtures_root):
96
90
  return _write_3_iso_file_fixture_data_factory
97
91
 
98
92
 
99
- @pytest.fixture
93
+ @pytest.fixture(scope="class")
100
94
  def basic_manifest_path(write_3_iso_file_fixture_data_factory):
101
95
  return write_3_iso_file_fixture_data_factory("basic")
102
96
 
103
97
 
104
- @pytest.fixture
98
+ @pytest.fixture(scope="class")
105
99
  def copy_manifest_only_factory(file_fixtures_root):
106
100
  def _copy_manifest_only(name):
107
101
  file_fixtures_root.joinpath(f"{name}-manifest").mkdir()
@@ -114,12 +108,12 @@ def copy_manifest_only_factory(file_fixtures_root):
114
108
  return _copy_manifest_only
115
109
 
116
110
 
117
- @pytest.fixture
111
+ @pytest.fixture(scope="class")
118
112
  def basic_manifest_only_path(copy_manifest_only_factory):
119
113
  return copy_manifest_only_factory("basic")
120
114
 
121
115
 
122
- @pytest.fixture
116
+ @pytest.fixture(scope="class")
123
117
  def large_manifest_path(file_fixtures_root):
124
118
  one_megabyte = 1048576
125
119
  file_fixtures_root.joinpath("large").mkdir()
@@ -128,7 +122,7 @@ def large_manifest_path(file_fixtures_root):
128
122
  return "/large/PULP_MANIFEST"
129
123
 
130
124
 
131
- @pytest.fixture
125
+ @pytest.fixture(scope="class")
132
126
  def range_header_manifest_path(file_fixtures_root):
133
127
  """A path to a File repository manifest that contains 8 unique files each 4mb in size."""
134
128
  one_megabyte = 1048576
@@ -147,7 +141,7 @@ def range_header_manifest_path(file_fixtures_root):
147
141
  return "/range/PULP_MANIFEST"
148
142
 
149
143
 
150
- @pytest.fixture
144
+ @pytest.fixture(scope="class")
151
145
  def manifest_path_with_commas(file_fixtures_root):
152
146
  file_fixtures_root.joinpath("comma_test").mkdir()
153
147
  file_fixtures_root.joinpath("comma_test/comma,folder").mkdir()
@@ -161,15 +155,15 @@ def manifest_path_with_commas(file_fixtures_root):
161
155
  return "/comma_test/PULP_MANIFEST"
162
156
 
163
157
 
164
- @pytest.fixture
158
+ @pytest.fixture(scope="class")
165
159
  def invalid_manifest_path(file_fixtures_root, basic_manifest_path):
166
- file_path_to_corrupt = file_fixtures_root / Path("basic/1.iso")
160
+ file_path_to_corrupt = file_fixtures_root / "basic/1.iso"
167
161
  with open(file_path_to_corrupt, "w") as f:
168
162
  f.write("this is not the right data")
169
163
  return basic_manifest_path
170
164
 
171
165
 
172
- @pytest.fixture
166
+ @pytest.fixture(scope="class")
173
167
  def duplicate_filename_paths(write_3_iso_file_fixture_data_factory):
174
168
  return (
175
169
  write_3_iso_file_fixture_data_factory("file"),
@@ -177,24 +171,24 @@ def duplicate_filename_paths(write_3_iso_file_fixture_data_factory):
177
171
  )
178
172
 
179
173
 
180
- @pytest.fixture
174
+ @pytest.fixture(scope="class")
181
175
  def file_fixture_server_ssl_client_cert_req(
182
176
  ssl_ctx_req_client_auth, file_fixtures_root, gen_fixture_server
183
177
  ):
184
178
  return gen_fixture_server(file_fixtures_root, ssl_ctx_req_client_auth)
185
179
 
186
180
 
187
- @pytest.fixture
181
+ @pytest.fixture(scope="class")
188
182
  def file_fixture_server_ssl(ssl_ctx, file_fixtures_root, gen_fixture_server):
189
183
  return gen_fixture_server(file_fixtures_root, ssl_ctx)
190
184
 
191
185
 
192
- @pytest.fixture
186
+ @pytest.fixture(scope="class")
193
187
  def file_fixture_server(file_fixtures_root, gen_fixture_server):
194
188
  return gen_fixture_server(file_fixtures_root, None)
195
189
 
196
190
 
197
- @pytest.fixture
191
+ @pytest.fixture(scope="class")
198
192
  def file_remote_factory(file_fixture_server, file_bindings, gen_object_with_cleanup):
199
193
  def _file_remote_factory(
200
194
  manifest_path=None, url=None, policy="immediate", pulp_domain=None, **body
@@ -213,7 +207,7 @@ def file_remote_factory(file_fixture_server, file_bindings, gen_object_with_clea
213
207
  return _file_remote_factory
214
208
 
215
209
 
216
- @pytest.fixture
210
+ @pytest.fixture(scope="class")
217
211
  def file_remote_ssl_factory(
218
212
  file_fixture_server_ssl,
219
213
  file_bindings,
@@ -235,7 +229,7 @@ def file_remote_ssl_factory(
235
229
  return _file_remote_ssl_factory
236
230
 
237
231
 
238
- @pytest.fixture
232
+ @pytest.fixture(scope="class")
239
233
  def file_remote_client_cert_req_factory(
240
234
  file_fixture_server_ssl_client_cert_req,
241
235
  file_bindings,
@@ -290,83 +284,7 @@ def file_publication_factory(file_bindings, gen_object_with_cleanup):
290
284
  return _file_publication_factory
291
285
 
292
286
 
293
- @pytest.fixture
294
- def gen_bad_response_fixture_server(gen_threaded_aiohttp_server):
295
- """
296
- This server will perform 3 bad responses for each file requested.
297
-
298
- 1st response will be incomplete, sending only half of the data.
299
- 2nd response will have corrupted data, with one byte changed.
300
- 3rd response will return error 429.
301
- 4th response will be correct.
302
- """
303
-
304
- def _gen_fixture_server(fixtures_root, ssl_ctx):
305
- record = []
306
- num_requests = defaultdict(int)
307
-
308
- async def handler(request):
309
- nonlocal num_requests
310
- record.append(request)
311
- relative_path = request.raw_path[1:] # Strip off leading "/"
312
- file_path = Path(fixtures_root) / Path(relative_path)
313
- # Max retries is 3. So on fourth request, send full data
314
- num_requests[relative_path] += 1
315
- if "PULP_MANIFEST" in relative_path or num_requests[relative_path] % 4 == 0:
316
- return web.FileResponse(file_path)
317
-
318
- # On third request send 429 error, TooManyRequests
319
- if num_requests[relative_path] % 4 == 3:
320
- raise web.HTTPTooManyRequests
321
-
322
- size = file_path.stat().st_size
323
- response = web.StreamResponse(headers={"content-length": f"{size}"})
324
- await response.prepare(request)
325
- async with aiofiles.open(file_path, "rb") as f:
326
- # Send only partial content causing aiohttp.ClientPayloadError if request num == 1
327
- chunk = await f.read(size // 2)
328
- await response.write(chunk)
329
- # Send last chunk with modified last byte if request num == 2
330
- if num_requests[relative_path] % 4 == 2:
331
- chunk2 = await f.read()
332
- await response.write(chunk2[:-1])
333
- await response.write(bytes([chunk2[-1] ^ 1]))
334
- else:
335
- request.transport.close()
336
-
337
- return response
338
-
339
- app = web.Application()
340
- app.add_routes([web.get("/{tail:.*}", handler)])
341
- return gen_threaded_aiohttp_server(app, ssl_ctx, record)
342
-
343
- return _gen_fixture_server
344
-
345
-
346
- @pytest.fixture
347
- def bad_response_fixture_server(file_fixtures_root, gen_bad_response_fixture_server):
348
- return gen_bad_response_fixture_server(file_fixtures_root, None)
349
-
350
-
351
- @pytest.fixture
352
- def wget_recursive_download_on_host():
353
- def _wget_recursive_download_on_host(url, destination):
354
- subprocess.check_output(
355
- [
356
- "wget",
357
- "--recursive",
358
- "--no-parent",
359
- "--no-host-directories",
360
- "--directory-prefix",
361
- destination,
362
- url,
363
- ]
364
- )
365
-
366
- return _wget_recursive_download_on_host
367
-
368
-
369
- @pytest.fixture
287
+ @pytest.fixture(scope="class")
370
288
  def generate_server_and_remote(
371
289
  file_bindings, gen_fixture_server, file_fixtures_root, gen_object_with_cleanup
372
290
  ):
@@ -1,7 +1,9 @@
1
+ from collections import defaultdict
2
+
1
3
  import pytest
2
4
  import uuid
3
-
4
- from collections import defaultdict
5
+ import aiofiles
6
+ from aiohttp import web
5
7
 
6
8
  from pulpcore.client.pulp_file import RepositorySyncURL
7
9
 
@@ -28,6 +30,64 @@ def perform_sync(
28
30
  yield _perform_sync
29
31
 
30
32
 
33
+ @pytest.fixture(scope="class")
34
+ def gen_bad_response_fixture_server(gen_threaded_aiohttp_server):
35
+ """
36
+ This server will perform 3 bad responses for each file requested.
37
+
38
+ 1st response will be incomplete, sending only half of the data.
39
+ 2nd response will have corrupted data, with one byte changed.
40
+ 3rd response will return error 429.
41
+ 4th response will be correct.
42
+ """
43
+
44
+ def _gen_fixture_server(fixtures_root, ssl_ctx):
45
+ record = []
46
+ num_requests = defaultdict(int)
47
+
48
+ async def handler(request):
49
+ nonlocal num_requests
50
+ record.append(request)
51
+ relative_path = request.raw_path[1:] # Strip off leading "/"
52
+ file_path = fixtures_root / relative_path
53
+ # Max retries is 3. So on fourth request, send full data
54
+ num_requests[relative_path] += 1
55
+ if "PULP_MANIFEST" in relative_path or num_requests[relative_path] % 4 == 0:
56
+ return web.FileResponse(file_path)
57
+
58
+ # On third request send 429 error, TooManyRequests
59
+ if num_requests[relative_path] % 4 == 3:
60
+ raise web.HTTPTooManyRequests
61
+
62
+ size = file_path.stat().st_size
63
+ response = web.StreamResponse(headers={"content-length": f"{size}"})
64
+ await response.prepare(request)
65
+ async with aiofiles.open(file_path, "rb") as f:
66
+ # Send only partial content causing aiohttp.ClientPayloadError if request num == 1
67
+ chunk = await f.read(size // 2)
68
+ await response.write(chunk)
69
+ # Send last chunk with modified last byte if request num == 2
70
+ if num_requests[relative_path] % 4 == 2:
71
+ chunk2 = await f.read()
72
+ await response.write(chunk2[:-1])
73
+ await response.write(bytes([chunk2[-1] ^ 1]))
74
+ else:
75
+ request.transport.close()
76
+
77
+ return response
78
+
79
+ app = web.Application()
80
+ app.add_routes([web.get("/{tail:.*}", handler)])
81
+ return gen_threaded_aiohttp_server(app, ssl_ctx, record)
82
+
83
+ return _gen_fixture_server
84
+
85
+
86
+ @pytest.fixture(scope="class")
87
+ def bad_response_fixture_server(file_fixtures_root, gen_bad_response_fixture_server):
88
+ return gen_bad_response_fixture_server(file_fixtures_root, None)
89
+
90
+
31
91
  @pytest.mark.parallel
32
92
  def test_bad_response_retry(bad_response_fixture_server, large_manifest_path, perform_sync):
33
93
  """
pulpcore/app/apps.py CHANGED
@@ -247,7 +247,7 @@ class PulpAppConfig(PulpPluginAppConfig):
247
247
  label = "core"
248
248
 
249
249
  # The version of this app
250
- version = "3.80.2"
250
+ version = "3.81.0"
251
251
 
252
252
  # The python package name providing this app
253
253
  python_package_name = "pulpcore"
@@ -11,6 +11,7 @@ from django.db.utils import InterfaceError, DatabaseError
11
11
  from gunicorn.workers.sync import SyncWorker
12
12
 
13
13
  from pulpcore.app.apps import pulp_plugin_configs
14
+ from pulpcore.app.netutil import has_ipv6
14
15
  from pulpcore.app.pulpcore_gunicorn_application import PulpcoreGunicornApplication
15
16
 
16
17
  logger = getLogger(__name__)
@@ -102,7 +103,9 @@ class PulpcoreApiApplication(PulpcoreGunicornApplication):
102
103
  # https://github.com/benoitc/gunicorn/blob/master/gunicorn/config.py
103
104
 
104
105
 
105
- @click.option("--bind", "-b", default=["[::]:24817"], multiple=True)
106
+ @click.option(
107
+ "--bind", "-b", default=[f"{ '[::]' if has_ipv6() else '0.0.0.0' }:24817"], multiple=True
108
+ )
106
109
  @click.option("--workers", "-w", type=int)
107
110
  # @click.option("--threads", "-w", type=int) # We don't use a threaded worker...
108
111
  @click.option("--name", "-n", "proc_name")
pulpcore/app/files.py CHANGED
@@ -1,4 +1,6 @@
1
1
  import os
2
+ import concurrent.futures
3
+ from concurrent.futures import ThreadPoolExecutor, ALL_COMPLETED
2
4
  from gettext import gettext as _
3
5
 
4
6
  from django.core.files.uploadedfile import TemporaryUploadedFile
@@ -39,9 +41,13 @@ class PulpTemporaryUploadedFile(TemporaryUploadedFile):
39
41
  instance = cls(name, "", size, "", "")
40
42
  instance.file = file
41
43
  # Default 1MB
42
- while data := file.read(1048576):
43
- for hasher in models.Artifact.DIGEST_FIELDS:
44
- instance.hashers[hasher].update(data)
44
+ with ThreadPoolExecutor(max_workers=6) as executor:
45
+ while data := file.read(1048576):
46
+ futures = [
47
+ executor.submit(instance.hashers[hasher].update, data)
48
+ for hasher in models.Artifact.DIGEST_FIELDS
49
+ ]
50
+ concurrent.futures.wait(futures, timeout=None, return_when=ALL_COMPLETED)
45
51
 
46
52
  # calling the method read() moves the file's pointer to the end of the file object,
47
53
  # thus, it is necessary to reset the file's pointer position back to 0 in case of
@@ -82,11 +88,15 @@ class HashingFileUploadHandler(TemporaryFileUploadHandler):
82
88
  self.file = PulpTemporaryUploadedFile(
83
89
  file_name, content_type, 0, charset, content_type_extra
84
90
  )
91
+ self.threadpool = ThreadPoolExecutor(max_workers=6)
85
92
 
86
93
  def receive_data_chunk(self, raw_data, start):
94
+ futures = [
95
+ self.threadpool.submit(self.file.hashers[hasher].update, raw_data)
96
+ for hasher in models.Artifact.DIGEST_FIELDS
97
+ ]
87
98
  self.file.write(raw_data)
88
- for hasher in models.Artifact.DIGEST_FIELDS:
89
- self.file.hashers[hasher].update(raw_data)
99
+ concurrent.futures.wait(futures, timeout=None, return_when=ALL_COMPLETED)
90
100
 
91
101
 
92
102
  class TemporaryDownloadedFile(TemporaryUploadedFile):
@@ -0,0 +1,20 @@
1
+ """
2
+ network utility functions
3
+ """
4
+
5
+ import socket
6
+
7
+
8
+ def has_ipv6():
9
+ """
10
+ check if host can use IPv6
11
+ """
12
+ # if python has IPv6 support we need to check if we can bind an IPv6 socket
13
+ if socket.has_ipv6:
14
+ try:
15
+ with socket.socket(socket.AF_INET6) as sock:
16
+ sock.bind(("::1", 0))
17
+ return True
18
+ except OSError:
19
+ pass
20
+ return False
@@ -11,7 +11,7 @@ from pulpcore.app.util import get_domain
11
11
 
12
12
  class NoArtifactContentSerializer(base.ModelSerializer):
13
13
  pulp_href = base.DetailIdentityField(view_name_pattern=r"contents(-.*/.*)-detail")
14
- pulp_labels = serializers.HStoreField(
14
+ pulp_labels = fields.PulpLabelsField(
15
15
  help_text=_(
16
16
  "A dictionary of arbitrary key/value pairs used to describe a specific "
17
17
  "Content instance."
@@ -435,3 +435,13 @@ def pulp_labels_validator(value):
435
435
  )
436
436
 
437
437
  return value
438
+
439
+
440
+ class PulpLabelsField(serializers.HStoreField):
441
+ """
442
+ Custom field for handling pulp labels that ensures proper dictionary format.
443
+ Converts JSON strings to dictionaries during validation.
444
+ """
445
+
446
+ def get_value(self, dictionary):
447
+ return dictionary.get(self.field_name, empty)
@@ -25,13 +25,14 @@ from pulpcore.app.models import ContentAppStatus # noqa: E402: module level not
25
25
  from pulpcore.app.util import get_worker_name # noqa: E402: module level not at top of file
26
26
 
27
27
  from .handler import Handler # noqa: E402: module level not at top of file
28
- from .instrumentation import instrumentation # noqa: E402: module level not at top of file
29
28
  from .authentication import authenticate, guid # noqa: E402: module level not at top of file
30
29
 
31
30
 
32
31
  log = logging.getLogger(__name__)
33
32
 
34
33
  if settings.OTEL_ENABLED:
34
+ from .instrumentation import instrumentation # noqa: E402: module level not at top of file
35
+
35
36
  app = web.Application(middlewares=[guid, authenticate, instrumentation()])
36
37
  else:
37
38
  app = web.Application(middlewares=[guid, authenticate])
@@ -1,4 +1,5 @@
1
1
  import click
2
+ from pulpcore.app.netutil import has_ipv6
2
3
  from pulpcore.app.pulpcore_gunicorn_application import PulpcoreGunicornApplication
3
4
  from django.conf import settings
4
5
 
@@ -19,7 +20,9 @@ class PulpcoreContentApplication(PulpcoreGunicornApplication):
19
20
  return pulpcore.content.server
20
21
 
21
22
 
22
- @click.option("--bind", "-b", default=["[::]:24816"], multiple=True)
23
+ @click.option(
24
+ "--bind", "-b", default=[f"{ '[::]' if has_ipv6() else '0.0.0.0' }:24816"], multiple=True
25
+ )
23
26
  @click.option("--workers", "-w", type=int)
24
27
  # @click.option("--threads", "-w", type=int) # We don't use a threaded worker...
25
28
  @click.option("--name", "-n", "proc_name")
pulpcore/download/base.py CHANGED
@@ -2,6 +2,8 @@ from gettext import gettext as _
2
2
 
3
3
  import asyncio
4
4
  from collections import namedtuple
5
+ import concurrent.futures
6
+ from concurrent.futures import ThreadPoolExecutor, ALL_COMPLETED
5
7
  import logging
6
8
  import os
7
9
  import tempfile
@@ -34,6 +36,8 @@ Args:
34
36
  values are header content. None when not using the HttpDownloader or sublclass.
35
37
  """
36
38
 
39
+ THREADPOOL = ThreadPoolExecutor()
40
+
37
41
 
38
42
  class BaseDownloader:
39
43
  """
@@ -200,8 +204,11 @@ class BaseDownloader:
200
204
  Args:
201
205
  data (bytes): The data to have its size and digest values recorded.
202
206
  """
203
- for algorithm in self._digests.values():
204
- algorithm.update(data)
207
+ global THREADPOOL
208
+ futures = [
209
+ THREADPOOL.submit(algorithm.update, data) for algorithm in self._digests.values()
210
+ ]
211
+ concurrent.futures.wait(futures, timeout=None, return_when=ALL_COMPLETED)
205
212
  self._size += len(data)
206
213
 
207
214
  @property
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: pulpcore
3
- Version: 3.80.2
3
+ Version: 3.81.0
4
4
  Summary: Pulp Django Application and Related Modules
5
5
  Author-email: Pulp Team <pulp-list@redhat.com>
6
6
  Project-URL: Homepage, https://pulpproject.org
@@ -21,17 +21,17 @@ Classifier: Programming Language :: Python :: 3.12
21
21
  Requires-Python: >=3.9
22
22
  Description-Content-Type: text/markdown
23
23
  License-File: LICENSE
24
- Requires-Dist: aiodns<=3.4.0,>=3.3
25
- Requires-Dist: aiofiles<24.2.0,>=22.1
26
- Requires-Dist: aiohttp<3.12,>=3.8.1
24
+ Requires-Dist: aiodns<3.6,>=3.3.0
25
+ Requires-Dist: aiofiles<=24.1.0,>=22.1
26
+ Requires-Dist: aiohttp<3.13,>=3.8.1
27
27
  Requires-Dist: asyncio-throttle<=1.0.2,>=1.0
28
28
  Requires-Dist: async-timeout<4.0.4,>=4.0.3; python_version < "3.11"
29
- Requires-Dist: backoff<2.2.2,>=2.1.2
30
- Requires-Dist: click<=8.1.8,>=8.1.0
31
- Requires-Dist: cryptography<45.0.5,>=38.0.1
29
+ Requires-Dist: backoff<2.3,>=2.1.2
30
+ Requires-Dist: click<8.2,>=8.1.0
31
+ Requires-Dist: cryptography<46.0,>=38.0.1
32
32
  Requires-Dist: Django~=4.2.0
33
33
  Requires-Dist: django-filter<=25.1,>=23.1
34
- Requires-Dist: django-guid<=3.5.1,>=3.3
34
+ Requires-Dist: django-guid<3.6,>=3.3.0
35
35
  Requires-Dist: django-import-export<3.4.0,>=2.9
36
36
  Requires-Dist: django-lifecycle<=1.2.4,>=1.0
37
37
  Requires-Dist: djangorestframework<=3.16.0,>=3.14.0
@@ -40,28 +40,28 @@ Requires-Dist: drf-access-policy<1.5.1,>=1.1.2
40
40
  Requires-Dist: drf-nested-routers<=0.94.2,>=0.93.4
41
41
  Requires-Dist: drf-spectacular==0.27.2
42
42
  Requires-Dist: dynaconf<3.3.0,>=3.2.5
43
- Requires-Dist: gunicorn<23.1.0,>=20.1
43
+ Requires-Dist: gunicorn<23.1.0,>=22.0
44
44
  Requires-Dist: importlib-metadata<=6.0.1,>=6.0.1
45
45
  Requires-Dist: jinja2<=3.1.6,>=3.1
46
46
  Requires-Dist: json_stream<2.4,>=2.3.2
47
- Requires-Dist: jq<1.9.0,>=1.6.0
47
+ Requires-Dist: jq<1.10.0,>=1.6.0
48
48
  Requires-Dist: PyOpenSSL<26.0
49
- Requires-Dist: opentelemetry-api<1.34,>=1.27.0
50
- Requires-Dist: opentelemetry-sdk<1.34,>=1.27.0
51
- Requires-Dist: opentelemetry-exporter-otlp-proto-http<1.34,>=1.27.0
49
+ Requires-Dist: opentelemetry-api<1.35,>=1.27.0
50
+ Requires-Dist: opentelemetry-sdk<1.35,>=1.27.0
51
+ Requires-Dist: opentelemetry-exporter-otlp-proto-http<1.35,>=1.27.0
52
52
  Requires-Dist: protobuf<6.0,>=4.21.1
53
- Requires-Dist: pulp-glue<0.33,>=0.18.0
53
+ Requires-Dist: pulp-glue<0.34,>=0.18.0
54
54
  Requires-Dist: pygtrie<=2.5.0,>=2.5
55
55
  Requires-Dist: psycopg[binary]<3.3,>=3.1.8
56
- Requires-Dist: pyparsing<=3.2.3,>=3.1.0
56
+ Requires-Dist: pyparsing<3.3,>=3.1.0
57
57
  Requires-Dist: python-gnupg<=0.5.4,>=0.5
58
- Requires-Dist: PyYAML<=6.0.2,>=5.1.1
59
- Requires-Dist: redis<5.2.2,>=4.3
60
- Requires-Dist: tablib<3.6.0
61
- Requires-Dist: url-normalize<=2.2.1,>=1.4.3
62
- Requires-Dist: uuid6<=2024.7.10,>=2023.5.2
58
+ Requires-Dist: PyYAML<6.1,>=5.1.1
59
+ Requires-Dist: redis<5.3,>=4.3.0
60
+ Requires-Dist: tablib<3.6,>=3.5.0
61
+ Requires-Dist: url-normalize<2.3,>=1.4.3
62
+ Requires-Dist: uuid6<=2025.0.0,>=2023.5.2
63
63
  Requires-Dist: whitenoise<6.10.0,>=5.0
64
- Requires-Dist: yarl<1.20.1,>=1.8
64
+ Requires-Dist: yarl<1.21,>=1.8
65
65
  Provides-Extra: sftp
66
66
  Requires-Dist: django-storages[sftp]==1.14.6; extra == "sftp"
67
67
  Provides-Extra: s3
@@ -1,6 +1,6 @@
1
1
  pulp_certguard/__init__.py,sha256=llnEd00PrsAretsgAOHiNKFbmvIdXe3iDVPmSaKz7gU,71
2
2
  pulp_certguard/pytest_plugin.py,sha256=qhRbChzqN2PROtD-65KuoTfKr5k9T3GPsz9daFgpqpM,852
3
- pulp_certguard/app/__init__.py,sha256=eYIaNGdyJN3i8Hyg4xdmWYHCH92mz7dc8qNvl9hlkEM,297
3
+ pulp_certguard/app/__init__.py,sha256=lADsgmmvuoqR01RdMScEKtYMgNnGT7c5pAFJ0oaUuDU,297
4
4
  pulp_certguard/app/models.py,sha256=xy5IWxf0LQxayIDmQw25Y2YhB_NrlTGvuvdY-YW7QBU,8119
5
5
  pulp_certguard/app/serializers.py,sha256=3jxWu82vU3xA578Qbyz-G4Q9Zlh3MFLGRHzX62M0RF8,1826
6
6
  pulp_certguard/app/utils.py,sha256=O6T1Npdb8fu3XqIkDJd8PQdEFJWPUeQ-i_aHXBl7MEc,816
@@ -48,8 +48,8 @@ pulp_certguard/tests/unit/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJW
48
48
  pulp_certguard/tests/unit/test_models.py,sha256=TBI0yKsrdbnJSPeBFfxSqhXK7zaNvR6qg5JehGH3Pds,229
49
49
  pulp_file/__init__.py,sha256=0vOCXofR6Eyxkg4y66esnOGPeESCe23C1cNBHj56w44,61
50
50
  pulp_file/manifest.py,sha256=1WwIOJrPSkFcmkRm7CkWifVOCoZvo_nnANgce6uuG7U,3796
51
- pulp_file/pytest_plugin.py,sha256=Fi_p-Vle_I-VYUSe4Zlg7esb_Ul5fpB8Rx9UGLK5UNQ,13281
52
- pulp_file/app/__init__.py,sha256=LS53kL09EDYbyCrSTAdukSjcbFbPIeOBExG0gOVDQhk,292
51
+ pulp_file/pytest_plugin.py,sha256=l1PvTxUi5D3uJy4SnHWNhr-otWEYNcm-kc5nSqVJg0Y,10646
52
+ pulp_file/app/__init__.py,sha256=xbejq5CE_8ckJvK4tdkH2Xppm9fMJ3rmESzRvsPxlwQ,292
53
53
  pulp_file/app/modelresource.py,sha256=v-m-_bBEsfr8wG0TI5ffx1TuKUy2-PsirhuQz4XXF-0,1063
54
54
  pulp_file/app/models.py,sha256=QsrVg_2uKqnR89sLN2Y7Zy260_nLIcUfa94uZowlmFw,4571
55
55
  pulp_file/app/replica.py,sha256=OtNWVmdFUgNTYhPttftVNQnSrnvx2_hnrJgtW_G0Vrg,1894
@@ -82,7 +82,7 @@ pulp_file/tests/functional/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJ
82
82
  pulp_file/tests/functional/api/__init__.py,sha256=IxizGLz9A_3Sh2ZGHpRlqMyZ1gIVAepBpJB474OJefE,62
83
83
  pulp_file/tests/functional/api/test_acs.py,sha256=L9MuroKKGSuxe2SzmHWCTeUUe3WSClKSn5UBiedViPg,9277
84
84
  pulp_file/tests/functional/api/test_auto_publish.py,sha256=GGcXqIZhANsMd5R2MYNtjCsqp1vHT6q13AoEv-3dBE0,4514
85
- pulp_file/tests/functional/api/test_bad_sync.py,sha256=QrYYbZA7noLor5PMoJuXj6KQ6aci_qnlxCmr3xP4oak,2057
85
+ pulp_file/tests/functional/api/test_bad_sync.py,sha256=IKdMNDL3jRDA2SiaVsJHFU7PWCIBa-kpKM-ILNTKPaw,4448
86
86
  pulp_file/tests/functional/api/test_content_labels.py,sha256=9jvG3SPQ_40OrdgEu7xHQYmNNmqWseGoTHITe6OsWLw,6798
87
87
  pulp_file/tests/functional/api/test_crud_content_unit.py,sha256=YUug-4ZCADnxYUj3aB8e8lyJ7_rrXPPfRcXW3IlBKI8,14772
88
88
  pulp_file/tests/functional/api/test_crud_remotes.py,sha256=ukQOUGssEiwYNm7A8ti5b_cv91eec8FsbGghCNgNXhs,3459
@@ -110,17 +110,18 @@ pulpcore/pytest_plugin.py,sha256=skubiEUIevVURr4LnmmVMt_ZeH5vT9mI0yiPUYerMnQ,380
110
110
  pulpcore/responses.py,sha256=mIGKmdCfTSoZxbFu4yIH1xbdLx1u5gqt3D99LTamcJg,6125
111
111
  pulpcore/app/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
112
112
  pulpcore/app/access_policy.py,sha256=5vCKy6WoHtIt1_-eS5vMaZ7CmR4G-CIpsrB8yT-d88Q,6079
113
- pulpcore/app/apps.py,sha256=rUgJ2SXznPOA50XaX9hwIhc2qCyBqdcsWlVowyn6r8A,17860
113
+ pulpcore/app/apps.py,sha256=O6TBkjY-UHFWcGeWNzEjN2xDH4He_A0x9f4_rFpmpWg,17860
114
114
  pulpcore/app/authentication.py,sha256=1LIJW6HIQQlZrliHy__jdzkDEh6Oj7xKgd0V-vRcDus,2855
115
115
  pulpcore/app/checks.py,sha256=jbfTF7nmftBbky4AQXHigpyCaGydKasvRUXsd72JZVg,1946
116
- pulpcore/app/entrypoint.py,sha256=m9kwANkh9OkhyAcWqPbrZg21IMQibLlB8_k1tkVgedg,4888
117
- pulpcore/app/files.py,sha256=uPodXYTVh7Ud-lQn8F58viSdom7TMh2X1SpoDt6XRKw,5797
116
+ pulpcore/app/entrypoint.py,sha256=YIfQpM5UxybBTasiEY5ptq--UmqPqjdIGnwmqVsDC7E,4972
117
+ pulpcore/app/files.py,sha256=BHq2T6cizPXfepQuEw_RfPXBqzcDRRVK3sedGDrgSJ4,6315
118
118
  pulpcore/app/global_access_conditions.py,sha256=Jezc1Hf0bQFaZvZFEDPpBrJmK0EvIa6zHRHHZYkez2Y,30398
119
119
  pulpcore/app/importexport.py,sha256=x1gGrHgirfMLsv92GEwBIQe12aItJJW9JH8TPij-rms,8795
120
120
  pulpcore/app/loggers.py,sha256=7tteVBBIf4W7jk4tB7QNpFGjCZueDDrPAavHj46LdJI,79
121
121
  pulpcore/app/manage.py,sha256=5YGD5ly3dJcLjX4jKIo3BBPFi_aqEPqi-CCoogV8lm8,286
122
122
  pulpcore/app/mime_types.py,sha256=xQh9gd5jHKxS0RrYqUjg98pST-Cyfcrk_Et1l2eby_w,6706
123
123
  pulpcore/app/modelresource.py,sha256=SlnKD_K-ZFtR5Gt8hpoF5odEX2y-1v2bb6UkJcOqnd8,4847
124
+ pulpcore/app/netutil.py,sha256=qO99WtWPLjAKPh_APw92Y3mUtxPH91NfeRuWLii1MLA,412
124
125
  pulpcore/app/openpgp.py,sha256=MYwCTGz7J9-Zr5aSBrz7KGWWWNC1EI-aItGb5dr7XPE,17246
125
126
  pulpcore/app/pulp_hashlib.py,sha256=NoVCO8duLz9rggPcilg0smi6fTDnsn-zS9dXgO831Pg,1327
126
127
  pulpcore/app/pulpcore_gunicorn_application.py,sha256=caqbDg9dhzECbx9Ss76biuEARhquj9gQaSL6v3XLy2w,2612
@@ -311,10 +312,10 @@ pulpcore/app/serializers/__init__.py,sha256=M1g5Si6hRi1flhQ-F1n6G-uFEzIZ7BUqHZ8i
311
312
  pulpcore/app/serializers/access_policy.py,sha256=NNtuzDW5H4RGfy5LbRFEHWDTDzXdL-Kihe9uquXnxBU,2818
312
313
  pulpcore/app/serializers/acs.py,sha256=wBbGE5YHR7dJWuicbDtiPQUJ7xEgz1zKHGkJEaMfpDU,5834
313
314
  pulpcore/app/serializers/base.py,sha256=ojWmsr2U2Mx8qpSFxqHLNQyfU2Z9q7hY1NUwVs9s3HE,21418
314
- pulpcore/app/serializers/content.py,sha256=lqfSah9Kg2i6dV-x2MGwQ-1q87GB9VOtjkQdjQKC5tQ,11967
315
+ pulpcore/app/serializers/content.py,sha256=TKEh774Q-0l1AfUeMmVxPM5lk5UiYZxgkt9NU1RjZ9M,11966
315
316
  pulpcore/app/serializers/domain.py,sha256=-xRJS_Olb1s2bqFzKamV0d_QnYO-e2iIyBJw-39uqMI,22688
316
317
  pulpcore/app/serializers/exporter.py,sha256=TxAgHDt34YUGPusawn7B8HL3bBymp46__6CnfhXSgGs,11179
317
- pulpcore/app/serializers/fields.py,sha256=NlxrK9__GVt1untO4USkd5ARltEgp_NqwG0M3St3SuQ,16181
318
+ pulpcore/app/serializers/fields.py,sha256=kWA6wYNA8dYmDJfFE_3eVUQRj4vCceqjzYPEEP53rxA,16481
318
319
  pulpcore/app/serializers/importer.py,sha256=PVSNs5U0dfAm-XlRKpMqOXK0VmUErxJauNJCNro6ONA,8200
319
320
  pulpcore/app/serializers/openpgp.py,sha256=3Svxskj_-HmOVbjay7QI82zXnKTsbtaSlZZ03CoT-MQ,8966
320
321
  pulpcore/app/serializers/orphans.py,sha256=Vhyaj0fqYT4pkiYoNjgmsy1u5BiR_aHwZm2y7rk9cbk,1967
@@ -371,13 +372,13 @@ pulpcore/app/viewsets/upload.py,sha256=Mfy9Vcm5KcqARooH4iExzoXVkL6boDddEqAnGWDWz
371
372
  pulpcore/app/viewsets/user.py,sha256=86eMawpaVrvp6ilQmb1C4j7SKpesPB5HgMovYL9rY3Q,13813
372
373
  pulpcore/cache/__init__.py,sha256=GkYD4PgIMaVL83ywfAsLBC9JNNDUpmTtbitW9zZSslk,131
373
374
  pulpcore/cache/cache.py,sha256=d8GMlvjeGG9MOMdi5_9029WpGCKH8Y5q9b2lt3wSREo,17371
374
- pulpcore/content/__init__.py,sha256=CVrhM5Ep2NFZBWOPxuyXXz7xL0bdZsmpBaOLPeA14SI,4010
375
+ pulpcore/content/__init__.py,sha256=lJDgxeIHXI__7LOQciRQSTItebgsxpoEGbd19Q7jNdI,4015
375
376
  pulpcore/content/authentication.py,sha256=lEZBkXBBBkIdtFMCSpHDD7583M0bO-zsZNYXTmpr4k8,3235
376
- pulpcore/content/entrypoint.py,sha256=svs6pEYa5bEGhWAAHpZt-uqlTuVXQ2UdW4U_LRlRHhY,2048
377
+ pulpcore/content/entrypoint.py,sha256=DiQTQzfcUiuyl37uvy6Wpa_7kr8t79ekpMHr31MDL2s,2132
377
378
  pulpcore/content/handler.py,sha256=-EsaNoTUZu2k3zCOMLA-IiD8vyYwUAJOJxstH0CuaZo,56693
378
379
  pulpcore/content/instrumentation.py,sha256=H0N0GWzvOPGGjFi6eIbGW3mcvagfnAfazccTh-BZVmE,1426
379
380
  pulpcore/download/__init__.py,sha256=s3Wh2GKdsmbUooVIR6wSvhYVIhpaTbtfR3Ar1OJhC7s,154
380
- pulpcore/download/base.py,sha256=G8jgyowvVEFCGA_KTr0CtHn0qHWXKsnv4Xpi0KSMglM,12821
381
+ pulpcore/download/base.py,sha256=4KCAYnV8jSOX078ETwlfwNZGY3xCBF9yy866tyGKAzE,13095
381
382
  pulpcore/download/factory.py,sha256=NQ1c7lqf8cCTZvhBeDaDjCE2qBAvPRzSDbtP2yN8SFk,9679
382
383
  pulpcore/download/file.py,sha256=nrPqcqB1k7MOaIhTKB2KS_Vc6O10--c-t1Tgf-Ct8fg,2235
383
384
  pulpcore/download/http.py,sha256=EVNlO10sZYzx8GmIg8ulHY-jw1V1sNXgvwS7RrU-VPE,12333
@@ -529,9 +530,9 @@ pulpcore/tests/unit/stages/test_artifactdownloader.py,sha256=qB1ANdFmNtUnljg8fCd
529
530
  pulpcore/tests/unit/stages/test_stages.py,sha256=H1a2BQLjdZlZvcb_qULp62huZ1xy6ItTcthktVyGU0w,4735
530
531
  pulpcore/tests/unit/viewsets/__init__.py,sha256=47DEQpj8HBSa-_TImW-5JCeuQeRkm5NMpJWZG3hSuFU,0
531
532
  pulpcore/tests/unit/viewsets/test_viewset_base.py,sha256=W9o3V6758bZctR6krMPPQytb0xJuF-jb4uBWTNDoD_U,4837
532
- pulpcore-3.80.2.dist-info/licenses/LICENSE,sha256=dhnHU8rJXUdAIgIjveSKAyYG_KzN5eVG-bxETIGrNW0,17988
533
- pulpcore-3.80.2.dist-info/METADATA,sha256=nEk9DQ9c12BFdDsxXq3P45UZ4ji955j6m9I64m2Ptm4,4333
534
- pulpcore-3.80.2.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
535
- pulpcore-3.80.2.dist-info/entry_points.txt,sha256=OZven4wzXzQA5b5q9MpP4HUpIPPQCSvIOvkKtNInrK0,452
536
- pulpcore-3.80.2.dist-info/top_level.txt,sha256=6h-Lm3FKQSaT_nL1KSxu_hBnzKE15bcvf_BoU-ea4CI,34
537
- pulpcore-3.80.2.dist-info/RECORD,,
533
+ pulpcore-3.81.0.dist-info/licenses/LICENSE,sha256=dhnHU8rJXUdAIgIjveSKAyYG_KzN5eVG-bxETIGrNW0,17988
534
+ pulpcore-3.81.0.dist-info/METADATA,sha256=u_SDc6wjbxK47aBe7nOmVXMF94QeFzueBlMxARrluwc,4320
535
+ pulpcore-3.81.0.dist-info/WHEEL,sha256=_zCd3N1l69ArxyTb8rzEoP9TpbYXkqRFSNOD5OuxnTs,91
536
+ pulpcore-3.81.0.dist-info/entry_points.txt,sha256=OZven4wzXzQA5b5q9MpP4HUpIPPQCSvIOvkKtNInrK0,452
537
+ pulpcore-3.81.0.dist-info/top_level.txt,sha256=6h-Lm3FKQSaT_nL1KSxu_hBnzKE15bcvf_BoU-ea4CI,34
538
+ pulpcore-3.81.0.dist-info/RECORD,,