Skip to content

Commit

Permalink
draft/wip
Browse files Browse the repository at this point in the history
  • Loading branch information
git-hyagi committed Mar 28, 2024
1 parent ef0a194 commit 070bac1
Show file tree
Hide file tree
Showing 18 changed files with 587 additions and 101 deletions.
6 changes: 6 additions & 0 deletions pulp_container/app/__init__.py
Original file line number Diff line number Diff line change
@@ -1,5 +1,11 @@
from pulpcore.plugin import PulpPluginAppConfig

#import debugpy
#try:
# debugpy.listen(('0.0.0.0',5678))
# debugpy.wait_for_client()
#except:
# pass

class PulpContainerPluginAppConfig(PulpPluginAppConfig):
"""Entry point for the container plugin."""
Expand Down
2 changes: 1 addition & 1 deletion pulp_container/app/content.py
Original file line number Diff line number Diff line change
Expand Up @@ -8,7 +8,7 @@
app.add_routes(
[
web.get(
r"/pulp/container/{path:.+}/{content:(blobs|manifests)}/sha256:{digest:.+}",
r"/pulp/container/{path:.+}/{content:(blobs|manifests|config-blobs)}/sha256:{digest:.+}",
registry.get_by_digest,
)
]
Expand Down
63 changes: 63 additions & 0 deletions pulp_container/app/migrations/0039_configblob_and_more.py
Original file line number Diff line number Diff line change
@@ -0,0 +1,63 @@
# Generated by Django 4.2.10 on 2024-03-21 19:41

from django.db import migrations, models
import django.db.models.deletion


class Migration(migrations.Migration):

dependencies = [
('container', '0038_add_manifest_metadata_fields'),
]

operations = [
migrations.CreateModel(
name='ConfigBlob',
fields=[
('content_ptr', models.OneToOneField(auto_created=True, on_delete=django.db.models.deletion.CASCADE, parent_link=True, primary_key=True, serialize=False, to='core.content')),
('data', models.TextField()),
('architecture', models.TextField()),
('os', models.TextField()),
('rootfs', models.JSONField(default=dict)),
('digest', models.TextField(db_index=True)),
('created', models.DateTimeField()),
('author', models.TextField(blank=True, default='')),
('os_version', models.TextField(blank=True, default='')),
('os_features', models.TextField(blank=True, default='')),
('variant', models.TextField(blank=True, default='')),
('config', models.JSONField(default=dict)),
('history', models.JSONField(default=dict)),
],
options={
'default_related_name': '%(app_label)s_%(model_name)s',
'unique_together': {('digest',)},
},
bases=('core.content',),
),
migrations.AddField(
model_name='containerpushrepository',
name='pending_config_blobs',
field=models.ManyToManyField(to='container.configblob'),
),
migrations.AddField(
model_name='containerrepository',
name='pending_config_blobs',
field=models.ManyToManyField(to='container.configblob'),
),
migrations.AlterField(
model_name='manifest',
name='config_blob',
field=models.ForeignKey(null=True, on_delete=django.db.models.deletion.CASCADE, related_name='config_blob', to='container.configblob'),
),
migrations.CreateModel(
name='ConfigBlobManifest',
fields=[
('id', models.AutoField(auto_created=True, primary_key=True, serialize=False, verbose_name='ID')),
('manifest', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='config_blob_manifests', to='container.manifest')),
('manifest_blob', models.ForeignKey(on_delete=django.db.models.deletion.CASCADE, related_name='manifest_blobs', to='container.configblob')),
],
options={
'unique_together': {('manifest', 'manifest_blob')},
},
),
]
20 changes: 19 additions & 1 deletion pulp_container/app/modelresource.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@

from pulp_container.app.models import (
Blob,
ConfigBlob,
ContainerRepository,
ContainerPushRepository,
Manifest,
Expand Down Expand Up @@ -60,6 +61,22 @@ class Meta:
exclude = RepositoryResource.Meta.exclude + ("manifest_signing_service",)


class ConfigBlobResource(BaseContentResource):
"""
Resource for import/export of configblob entities
"""

def set_up_queryset(self):
"""
:return: ConfigBlobs specific to a specified repo-version.
"""
return ConfigBlob.objects.filter(pk__in=self.repo_version.content).order_by("content_ptr_id")

class Meta:
model = ConfigBlob
import_id_fields = model.natural_key_fields()


class BlobResource(BaseContentResource):
"""
Resource for import/export of blob entities
Expand Down Expand Up @@ -89,7 +106,7 @@ class ManifestResource(BaseContentResource):
config_blob = fields.Field(
column_name="config_blob",
attribute="config_blob",
widget=widgets.ForeignKeyWidget(Blob, field="digest"),
widget=widgets.ForeignKeyWidget(ConfigBlob, field="digest"),
)

def set_up_queryset(self):
Expand Down Expand Up @@ -179,6 +196,7 @@ class Meta:


IMPORT_ORDER = [
ConfigBlobResource,
BlobResource,
ManifestResource,
ManifestListManifestResource,
Expand Down
82 changes: 74 additions & 8 deletions pulp_container/app/models.py
Original file line number Diff line number Diff line change
Expand Up @@ -38,6 +38,49 @@
logger = getLogger(__name__)


class ConfigBlob(Content):
"""
The manifest for image configuration.
Fields:
created (models.DateTimeField): An combined date and time at which the image was created.
author (models.TextField): Name and/or email of the person or entity which created and is
responsible for maintaining the image.
architecture (models.TextField): The platform architecture.
os (models.TextField): The platform OS name.
os_version (models.TextField): The platform OS version.
os_features (models.TextField): The platform OS features.
variant (models.TextField): The platform variant.
config (models.ForeignKey):
rootfs (models.ForeignKey):
history (models.ForeignKey):
"""

PROTECTED_FROM_RECLAIM = False

TYPE = "config-blob"

data = models.TextField()
architecture = models.TextField()
os = models.TextField()
rootfs = models.JSONField(default=dict)
digest = models.TextField(db_index=True)

created = models.DateTimeField()
author = models.TextField(default="", blank=True)
os_version = models.TextField(default="", blank=True)
os_features = models.TextField(default="", blank=True)
variant = models.TextField(default="", blank=True)

config = models.JSONField(default=dict)
history = models.JSONField(default=dict)

class Meta:
default_related_name = "%(app_label)s_%(model_name)s"
unique_together = ("digest",)


class Blob(Content):
"""
A blob defined within a manifest.
Expand Down Expand Up @@ -107,7 +150,7 @@ class Manifest(Content):

blobs = models.ManyToManyField(Blob, through="BlobManifest")
config_blob = models.ForeignKey(
Blob, related_name="config_blob", null=True, on_delete=models.CASCADE
ConfigBlob, related_name="config_blob", null=True, on_delete=models.CASCADE
)

# Order matters for through fields, (source, target)
Expand Down Expand Up @@ -137,11 +180,14 @@ def init_annotations(self, manifest_data=None):
return bool(self.annotations)

def init_labels(self):
if self.config_blob:
config_artifact = self.config_blob._artifacts.get()

config_data, _ = get_content_data(config_artifact)
self.labels = config_data.get("config", {}).get("Labels") or {}
if self.config_blob and "Labels" in self.config_blob.config.keys():
#if isinstance(self.config_blob.config["Labels"],str):
# self.labels = json.loads(self.config_blob.config)["Labels"]
#else:
# self.labels = self.config_blob.config["Labels"] or {}
self.labels = self.config_blob.config["Labels"] or {}
else:
self.labels = {}

return bool(self.labels)

Expand Down Expand Up @@ -203,6 +249,22 @@ class Meta:
unique_together = ("manifest", "manifest_blob")


class ConfigBlobManifest(models.Model):
"""
Many-to-many relationship between ConfigBlobs and Manifests.
"""

manifest = models.ForeignKey(
Manifest, related_name="config_blob_manifests", on_delete=models.CASCADE
)
manifest_blob = models.ForeignKey(
ConfigBlob, related_name="manifest_blobs", on_delete=models.CASCADE
)

class Meta:
unique_together = ("manifest", "manifest_blob")


class ManifestListManifest(models.Model):
"""
The manifest referenced by a manifest list.
Expand Down Expand Up @@ -577,14 +639,15 @@ class ContainerRepository(
"""

TYPE = "container"
CONTENT_TYPES = [Blob, Manifest, Tag, ManifestSignature]
CONTENT_TYPES = [Blob, ConfigBlob, Manifest, Tag, ManifestSignature]
REMOTE_TYPES = [ContainerRemote]
PUSH_ENABLED = False

manifest_signing_service = models.ForeignKey(
ManifestSigningService, on_delete=models.SET_NULL, null=True
)
pending_blobs = models.ManyToManyField(Blob)
pending_config_blobs = models.ManyToManyField(ConfigBlob)
pending_manifests = models.ManyToManyField(Manifest)

class Meta:
Expand Down Expand Up @@ -617,6 +680,7 @@ def remove_pending_content(self, repository_version):
base_version=repository_version.base_version
).values_list("pk")
self.pending_blobs.remove(*Blob.objects.filter(pk__in=added_content))
self.pending_config_blobs.remove(*ConfigBlob.objects.filter(pk__in=added_content))
self.pending_manifests.remove(*Manifest.objects.filter(pk__in=added_content))


Expand All @@ -635,13 +699,14 @@ class ContainerPushRepository(Repository, AutoAddObjPermsMixin):
"""

TYPE = "container-push"
CONTENT_TYPES = [Blob, Manifest, Tag, ManifestSignature]
CONTENT_TYPES = [Blob, ConfigBlob, Manifest, Tag, ManifestSignature]
PUSH_ENABLED = True

manifest_signing_service = models.ForeignKey(
ManifestSigningService, on_delete=models.SET_NULL, null=True
)
pending_blobs = models.ManyToManyField(Blob)
pending_config_blobs = models.ManyToManyField(ConfigBlob)
pending_manifests = models.ManyToManyField(Manifest)

class Meta:
Expand Down Expand Up @@ -671,6 +736,7 @@ def remove_pending_content(self, repository_version):
base_version=repository_version.base_version
).values_list("pk")
self.pending_blobs.remove(*Blob.objects.filter(pk__in=added_content))
self.pending_config_blobs.remove(*ConfigBlob.objects.filter(pk__in=added_content))
self.pending_manifests.remove(*Manifest.objects.filter(pk__in=added_content))


Expand Down
4 changes: 4 additions & 0 deletions pulp_container/app/redirects.py
Original file line number Diff line number Diff line change
Expand Up @@ -4,6 +4,7 @@
from django.shortcuts import redirect

from pulp_container.app.exceptions import ManifestNotFound
from pulp_container.app.models import ConfigBlob
from pulp_container.app.utils import get_accepted_media_types
from pulp_container.constants import BLOB_CONTENT_TYPE, MEDIA_TYPE

Expand Down Expand Up @@ -58,6 +59,9 @@ def issue_blob_redirect(self, blob):
"""
Issue a redirect for the passed blob.
"""
if isinstance(blob, ConfigBlob):
return self.redirect_to_content_app("config-blobs", blob.digest)

return self.redirect_to_content_app("blobs", blob.digest)


Expand Down
Loading

0 comments on commit 070bac1

Please sign in to comment.