Skip to content
Merged
Show file tree
Hide file tree
Changes from all commits
Commits
File filter

Filter by extension

Filter by extension

Conversations
Failed to load comments.
Loading
Jump to
Jump to file
Failed to load files.
Loading
Diff view
Diff view
58 changes: 58 additions & 0 deletions Sources/ContainerizationOCI/Client/RegistryClient+Referrers.swift
Original file line number Diff line number Diff line change
@@ -0,0 +1,58 @@
//===----------------------------------------------------------------------===//
// Copyright © 2026 Apple Inc. and the Containerization project authors.
//
// Licensed under the Apache License, Version 2.0 (the "License");
// you may not use this file except in compliance with the License.
// You may obtain a copy of the License at
//
// https://www.apache.org/licenses/LICENSE-2.0
//
// Unless required by applicable law or agreed to in writing, software
// distributed under the License is distributed on an "AS IS" BASIS,
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
// See the License for the specific language governing permissions and
// limitations under the License.
//===----------------------------------------------------------------------===//

import AsyncHTTPClient
import ContainerizationError
import Foundation
import NIOFoundationCompat

extension RegistryClient {
/// Query the OCI referrers API for artifacts that reference a given manifest digest.
///
/// Implements `GET /v2/{name}/referrers/{digest}` from the OCI Distribution Spec v1.1.
///
/// - Parameters:
/// - name: The repository name (e.g., "library/ubuntu").
/// - digest: The digest of the subject manifest (e.g., "sha256:abc123...").
/// - artifactType: Optional filter to return only referrers with a matching artifactType.
/// - Returns: An `Index` whose `manifests` array contains descriptors of referring artifacts.
/// Returns an empty index if the registry does not support the referrers API.
public func referrers(name: String, digest: String, artifactType: String? = nil) async throws -> Index {
var components = base
components.path = "/v2/\(name)/referrers/\(digest)"

if let artifactType {
components.queryItems = [URLQueryItem(name: "artifactType", value: artifactType)]
}

let headers = [("Accept", MediaTypes.index)]

return try await request(components: components, method: .GET, headers: headers) { response in
if response.status == .notFound {
return Index(schemaVersion: 2, manifests: [])
}

guard response.status == .ok else {
let url = components.url?.absoluteString ?? "unknown"
let reason = await ErrorResponse.fromResponseBody(response.body)?.jsonString
throw Error.invalidStatus(url: url, response.status, reason: reason)
}

let buffer = try await response.body.collect(upTo: self.bufferSize)
return try JSONDecoder().decode(Index.self, from: buffer)
}
}
}
8 changes: 7 additions & 1 deletion Sources/ContainerizationOCI/Descriptor.swift
Original file line number Diff line number Diff line change
Expand Up @@ -42,15 +42,21 @@ public struct Descriptor: Codable, Sendable, Equatable {
/// This should only be used when referring to a manifest.
public var platform: Platform?

/// artifactType specifies the IANA media type of the artifact.
///
/// Used in referrers API responses to indicate the type of each referring artifact.
public let artifactType: String?

public init(
mediaType: String, digest: String, size: Int64, urls: [String]? = nil, annotations: [String: String]? = nil,
platform: Platform? = nil
platform: Platform? = nil, artifactType: String? = nil
) {
self.mediaType = mediaType
self.digest = digest
self.size = size
self.urls = urls
self.annotations = annotations
self.platform = platform
self.artifactType = artifactType
}
}
12 changes: 11 additions & 1 deletion Sources/ContainerizationOCI/Index.swift
Original file line number Diff line number Diff line change
Expand Up @@ -34,14 +34,22 @@ public struct Index: Codable, Sendable {
/// annotations contains arbitrary metadata for the image index.
public var annotations: [String: String]?

/// `subject` references another manifest this index is an artifact of.
public let subject: Descriptor?

/// `artifactType` specifies the IANA media type of the artifact this index represents.
public let artifactType: String?

public init(
schemaVersion: Int = 2, mediaType: String = MediaTypes.index, manifests: [Descriptor],
annotations: [String: String]? = nil
annotations: [String: String]? = nil, subject: Descriptor? = nil, artifactType: String? = nil
) {
self.schemaVersion = schemaVersion
self.mediaType = mediaType
self.manifests = manifests
self.annotations = annotations
self.subject = subject
self.artifactType = artifactType
}

public init(from decoder: Decoder) throws {
Expand All @@ -50,5 +58,7 @@ public struct Index: Codable, Sendable {
self.mediaType = try container.decodeIfPresent(String.self, forKey: .mediaType) ?? ""
self.manifests = try container.decode([Descriptor].self, forKey: .manifests)
self.annotations = try container.decodeIfPresent([String: String].self, forKey: .annotations)
self.subject = try container.decodeIfPresent(Descriptor.self, forKey: .subject)
self.artifactType = try container.decodeIfPresent(String.self, forKey: .artifactType)
}
}
10 changes: 9 additions & 1 deletion Sources/ContainerizationOCI/Manifest.swift
Original file line number Diff line number Diff line change
Expand Up @@ -36,14 +36,22 @@ public struct Manifest: Codable, Sendable {
/// `annotations` contains arbitrary metadata for the image manifest.
public let annotations: [String: String]?

/// `subject` references another manifest this manifest is an artifact of.
public let subject: Descriptor?

/// `artifactType` specifies the IANA media type of the artifact this manifest represents.
public let artifactType: String?

public init(
schemaVersion: Int = 2, mediaType: String = MediaTypes.imageManifest, config: Descriptor, layers: [Descriptor],
annotations: [String: String]? = nil
annotations: [String: String]? = nil, subject: Descriptor? = nil, artifactType: String? = nil
) {
self.schemaVersion = schemaVersion
self.mediaType = mediaType
self.config = config
self.layers = layers
self.annotations = annotations
self.subject = subject
self.artifactType = artifactType
}
}
99 changes: 99 additions & 0 deletions Tests/ContainerizationOCITests/OCIImageTests.swift
Original file line number Diff line number Diff line change
Expand Up @@ -16,6 +16,7 @@

//

import Foundation
import Testing

@testable import ContainerizationOCI
Expand All @@ -36,6 +37,30 @@ struct OCITests {

#expect(descriptor.platform?.architecture == "arm64")
#expect(descriptor.platform?.os == "linux")
#expect(descriptor.artifactType == nil)
}

@Test func descriptorWithArtifactType() throws {
let testArtifactType = "application/vnd.example.test.v1+json"
let descriptor = ContainerizationOCI.Descriptor(
mediaType: MediaTypes.imageManifest,
digest: "sha256:abc123",
size: 1234,
artifactType: testArtifactType
)
#expect(descriptor.artifactType == testArtifactType)

let data = try JSONEncoder().encode(descriptor)
let decoded = try JSONDecoder().decode(ContainerizationOCI.Descriptor.self, from: data)
#expect(decoded.artifactType == testArtifactType)
}

@Test func descriptorWithoutArtifactTypeDecodesAsNil() throws {
let json = """
{"mediaType":"application/vnd.oci.descriptor.v1+json","digest":"sha256:abc","size":0}
"""
let decoded = try JSONDecoder().decode(ContainerizationOCI.Descriptor.self, from: json.data(using: .utf8)!)
#expect(decoded.artifactType == nil)
}

@Test func index() {
Expand All @@ -47,6 +72,37 @@ struct OCITests {

let index = ContainerizationOCI.Index(schemaVersion: 1, manifests: descriptors)
#expect(index.manifests.count == 5)
#expect(index.subject == nil)
#expect(index.artifactType == nil)
}

@Test func indexWithSubjectAndArtifactType() throws {
let testArtifactType = "application/vnd.example.test.v1+json"
let subject = ContainerizationOCI.Descriptor(mediaType: MediaTypes.imageManifest, digest: "sha256:subject", size: 512)
let index = ContainerizationOCI.Index(
schemaVersion: 2,
manifests: [],
subject: subject,
artifactType: testArtifactType
)
#expect(index.subject?.digest == "sha256:subject")
#expect(index.artifactType == testArtifactType)

let data = try JSONEncoder().encode(index)
let decoded = try JSONDecoder().decode(ContainerizationOCI.Index.self, from: data)
#expect(decoded.subject?.digest == "sha256:subject")
#expect(decoded.artifactType == testArtifactType)
}

@Test func indexDecodesWithoutNewFields() throws {
let json = """
{"schemaVersion":2,"manifests":[{"mediaType":"application/vnd.oci.descriptor.v1+json","digest":"sha256:abc","size":10}]}
"""
let decoded = try JSONDecoder().decode(ContainerizationOCI.Index.self, from: json.data(using: .utf8)!)
#expect(decoded.schemaVersion == 2)
#expect(decoded.manifests.count == 1)
#expect(decoded.subject == nil)
#expect(decoded.artifactType == nil)
}

@Test func manifests() {
Expand All @@ -61,5 +117,48 @@ struct OCITests {
let manifest = ContainerizationOCI.Manifest(schemaVersion: 1, config: config, layers: descriptors)
#expect(manifest.config.digest == "123")
#expect(manifest.layers.count == 5)
#expect(manifest.subject == nil)
#expect(manifest.artifactType == nil)
}

@Test func manifestWithSubjectAndArtifactType() throws {
let testArtifactType = "application/vnd.example.test.v1+json"
let config = ContainerizationOCI.Descriptor(mediaType: MediaTypes.emptyJSON, digest: "sha256:empty", size: 2)
let subject = ContainerizationOCI.Descriptor(mediaType: MediaTypes.imageManifest, digest: "sha256:target", size: 1234)
let layer = ContainerizationOCI.Descriptor(
mediaType: testArtifactType,
digest: "sha256:meta",
size: 89,
annotations: ["org.opencontainers.image.title": "metadata.json"]
)

let manifest = ContainerizationOCI.Manifest(
config: config,
layers: [layer],
subject: subject,
artifactType: testArtifactType
)
#expect(manifest.subject?.digest == "sha256:target")
#expect(manifest.artifactType == testArtifactType)
#expect(manifest.layers[0].annotations?["org.opencontainers.image.title"] == "metadata.json")

let data = try JSONEncoder().encode(manifest)
let decoded = try JSONDecoder().decode(ContainerizationOCI.Manifest.self, from: data)
#expect(decoded.subject?.digest == "sha256:target")
#expect(decoded.artifactType == testArtifactType)
}

@Test func manifestDecodesWithoutNewFields() throws {
let json = """
{
"schemaVersion": 2,
"config": {"mediaType":"application/vnd.oci.empty.v1+json","digest":"sha256:abc","size":2},
"layers": []
}
"""
let decoded = try JSONDecoder().decode(ContainerizationOCI.Manifest.self, from: json.data(using: .utf8)!)
#expect(decoded.schemaVersion == 2)
#expect(decoded.subject == nil)
#expect(decoded.artifactType == nil)
}
}