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.
- pulp_certguard/app/__init__.py +1 -1
- pulp_file/app/__init__.py +1 -1
- pulp_file/pytest_plugin.py +22 -104
- pulp_file/tests/functional/api/test_bad_sync.py +62 -2
- pulpcore/app/apps.py +1 -1
- pulpcore/app/entrypoint.py +4 -1
- pulpcore/app/files.py +15 -5
- pulpcore/app/netutil.py +20 -0
- pulpcore/app/serializers/content.py +1 -1
- pulpcore/app/serializers/fields.py +10 -0
- pulpcore/content/__init__.py +2 -1
- pulpcore/content/entrypoint.py +4 -1
- pulpcore/download/base.py +9 -2
- {pulpcore-3.80.2.dist-info → pulpcore-3.81.0.dist-info}/METADATA +21 -21
- {pulpcore-3.80.2.dist-info → pulpcore-3.81.0.dist-info}/RECORD +19 -18
- {pulpcore-3.80.2.dist-info → pulpcore-3.81.0.dist-info}/WHEEL +0 -0
- {pulpcore-3.80.2.dist-info → pulpcore-3.81.0.dist-info}/entry_points.txt +0 -0
- {pulpcore-3.80.2.dist-info → pulpcore-3.81.0.dist-info}/licenses/LICENSE +0 -0
- {pulpcore-3.80.2.dist-info → pulpcore-3.81.0.dist-info}/top_level.txt +0 -0
pulp_certguard/app/__init__.py
CHANGED
pulp_file/app/__init__.py
CHANGED
pulp_file/pytest_plugin.py
CHANGED
|
@@ -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,
|
|
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":
|
|
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(
|
|
79
|
-
fixture_dir =
|
|
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 /
|
|
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
|
|
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
pulpcore/app/entrypoint.py
CHANGED
|
@@ -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(
|
|
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
|
-
|
|
43
|
-
|
|
44
|
-
|
|
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
|
-
|
|
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):
|
pulpcore/app/netutil.py
ADDED
|
@@ -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 =
|
|
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)
|
pulpcore/content/__init__.py
CHANGED
|
@@ -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])
|
pulpcore/content/entrypoint.py
CHANGED
|
@@ -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(
|
|
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
|
-
|
|
204
|
-
|
|
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.
|
|
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
|
|
25
|
-
Requires-Dist: aiofiles
|
|
26
|
-
Requires-Dist: aiohttp<3.
|
|
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.
|
|
30
|
-
Requires-Dist: click
|
|
31
|
-
Requires-Dist: cryptography<
|
|
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
|
|
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,>=
|
|
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.
|
|
47
|
+
Requires-Dist: jq<1.10.0,>=1.6.0
|
|
48
48
|
Requires-Dist: PyOpenSSL<26.0
|
|
49
|
-
Requires-Dist: opentelemetry-api<1.
|
|
50
|
-
Requires-Dist: opentelemetry-sdk<1.
|
|
51
|
-
Requires-Dist: opentelemetry-exporter-otlp-proto-http<1.
|
|
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.
|
|
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
|
|
56
|
+
Requires-Dist: pyparsing<3.3,>=3.1.0
|
|
57
57
|
Requires-Dist: python-gnupg<=0.5.4,>=0.5
|
|
58
|
-
Requires-Dist: PyYAML
|
|
59
|
-
Requires-Dist: redis<5.
|
|
60
|
-
Requires-Dist: tablib<3.6.0
|
|
61
|
-
Requires-Dist: url-normalize
|
|
62
|
-
Requires-Dist: uuid6<=
|
|
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.
|
|
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=
|
|
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=
|
|
52
|
-
pulp_file/app/__init__.py,sha256=
|
|
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=
|
|
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=
|
|
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=
|
|
117
|
-
pulpcore/app/files.py,sha256=
|
|
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=
|
|
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=
|
|
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=
|
|
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=
|
|
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=
|
|
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.
|
|
533
|
-
pulpcore-3.
|
|
534
|
-
pulpcore-3.
|
|
535
|
-
pulpcore-3.
|
|
536
|
-
pulpcore-3.
|
|
537
|
-
pulpcore-3.
|
|
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,,
|
|
File without changes
|
|
File without changes
|
|
File without changes
|
|
File without changes
|