You are viewing development documentation for the main branch.
This version tracks the latest commit and may be unstable.
For the current release, see the
latest version (latest).
Working with OCI
Embed OCI image layouts in component versions and access them natively from OCI registries after transfer.
In this tutorial, you’ll learn how to embed OCI artifacts as local blobs inside component versions and transfer them to OCI registries where they become natively accessible — pullable with standard OCI tools like docker pull, oras pull, or crane.
What You’ll Learn
Create an OCI image layout using the ORAS CLI
Embed an OCI image layout as a local blob in a component version
Transfer a component version with --copy-resources so that external OCI image references are internalized as local blobs
Access local blobs natively from an OCI registry by their localReference digest and media type
Estimated time: ~20 minutes
Scenario
You’re packaging a microservice as an OCM component. The component includes a container image that must travel with the component — not just as a reference, but as an embedded artifact. When the component arrives at a target registry, you want the image to be pullable directly with standard OCI tooling, without needing the OCM CLI.
OCM v2 makes this possible through its
OCI-compatible index representation. When a component version is stored in an OCI registry, native OCI artifacts (images, Helm charts, OCI image layouts) stored as local blobs are mapped to proper OCI manifests within the component version’s index. This means they can be accessed directly by digest using any OCI-compliant client.
How It Works
flowchart LR
subgraph create ["Create"]
direction TB
ORAS["ORAS CLI"] --> Layout["OCI Image Layout (.tar)"]
Layout --> Constructor["component-constructor.yaml"]
Constructor --> CTF["CTF Archive"]
end
CTF --> Transfer["ocm transfer cv<br/>--copy-resources"]
subgraph registry ["OCI Registry"]
direction TB
Index["Component Version<br/>(OCI Index)"]
Manifest["OCI Image Manifest<br/>(native access)"]
Index --> Manifest
end
Transfer --> Index
Manifest --> Pull["docker pull / oras pull<br/>by digest"]
style Pull fill:#dcfce7,color:#166534
The component version is stored as an OCI index. Local blobs with OCI media types are stored as separate OCI manifests within that index, making them natively accessible by their digest.
Access to an OCI registry (e.g. a local registry via docker run -d -p 5001:5000 registry:2)
Tutorial
In this use case, you create an OCI image layout from scratch using ORAS, embed it in a component version, and transfer it to an OCI registry where it becomes natively accessible.
The type: dir/v1 input packs the directory into a tar archive and embeds it by value as a local blob
The mediaType: application/vnd.ocm.software.oci.layout.v1+tar tells OCM this is an OCI image layout, not an opaque blob — the dir/v1 input automatically produces the tar archive expected by this media type
During ocm add cv, OCM unpacks the layout and stores the contained manifest directly — the resulting local blob has the native OCI media type (e.g. application/vnd.oci.image.manifest.v1+json), not the tar media type
Notice that the resource has access.type: LocalBlob/v1 with the native OCI manifest media type. OCM recognized the OCI image layout during ingestion, unpacked it, and stored the manifest directly. The localReference contains the digest of the embedded manifest.
Transfer to an OCI registry
Transfer the component version to an OCI registry. The --copy-resources flag ensures all local blobs are transferred:
ocm transfer cv \
--copy-resources \
./transport-archive//github.com/acme.org/native-oci-demo:1.0.0 \
<your-registry>
Replace <your-registry> with your registry address (e.g. http://localhost:5001 for a local HTTP registry, or ghcr.io/my-org/ocm for a remote HTTPS registry).
Note
For local registries running without TLS, use the http:// scheme prefix (e.g. http://localhost:5001). HTTPS registries work without a scheme prefix.
During transfer, OCM stores the OCI manifest and its layers as native OCI objects in the registry. The component version’s index references the manifest directly.
Access the image natively
After transfer, inspect the component version in the registry to find the image’s digest:
ocm get cv <your-registry>//github.com/acme.org/native-oci-demo:1.0.0 -o yaml
Look for the localReference field in the resource’s access specification — it contains the digest of the native OCI manifest:
You can pull the artifact natively using the localReference digest combined with the component version’s repository path:
# Using ORASoras pull <your-registry>/component-descriptors/github.com/acme.org/native-oci-demo@sha256:...
# Using cranecrane pull <your-registry>/component-descriptors/github.com/acme.org/native-oci-demo@sha256:... image.tar
Note
The repository path follows the pattern <registry>/<subpath>/component-descriptors/<component-name>. The digest from localReference addresses the manifest directly within the component version’s OCI index.
In this use case, you start with a component version that references an external OCI image, transfer it with --copy-resources to internalize the image as a local blob, and then access it natively from the target registry.
access.type changed from ociArtifact to LocalBlob/v1 — the image is now embedded
localReference contains the digest of the stored image manifest/index
mediaType is application/vnd.oci.image.index.v1+json (or application/vnd.oci.image.manifest.v1+json for single-platform images)
referenceName preserves the original image reference for traceability
relation remains external — this indicates the resource was originally sourced externally, even though it is now stored locally
Pull the image natively
Use the localReference digest to pull the image with standard OCI tooling. The image is stored within the component version’s repository:
# Using dockerdocker pull <your-registry>/component-descriptors/github.com/acme.org/transfer-demo@sha256:...
# Using cranecrane manifest <your-registry>/component-descriptors/github.com/acme.org/transfer-demo@sha256:...
The image is stored as a first-class OCI manifest in the registry. No OCM tooling is required to access it — any OCI-compliant client works.
In this use case, you fetch an existing OCI artifact from a remote registry using ORAS, embed it into a component version, and transfer it to a target registry.
localReference contains the digest of the stored image manifest/index — this is the value you need for native access
mediaType is application/vnd.oci.image.index.v1+json (multi-platform) or application/vnd.oci.image.manifest.v1+json (single-platform)
referenceName preserves the original image reference for traceability
Pull natively
Use the localReference digest from the previous step to pull the artifact with standard OCI tooling:
# Pull using the localReference digestoras pull <your-registry>/component-descriptors/github.com/acme.org/fetched-oci-demo@sha256:abc123...
The image is stored as a first-class OCI manifest in the registry. No OCM tooling is required to access it — any OCI-compliant client works.
How Native Access Works
Under the hood, OCM v2 stores component versions as
OCI Image Indexes. When a local blob has an OCI-native media type (image manifest, image index, or OCI image layout), it is stored as a separate OCI manifest referenced from the component version’s index — not as an opaque layer.
Non-OCI blobs (plain files, config data) are stored as layers in the descriptor manifest, accessed only through OCM tooling
Native OCI artifacts (images, Helm charts) are stored as separate manifests in the index, accessible both through OCM and directly through any OCI client
Check Your Understanding
Why does the media type matter when embedding an OCI image layout?The media type (application/vnd.ocm.software.oci.layout.v1+tar) tells OCM that the blob contains a valid OCI image layout. During ocm add cv, OCM unpacks the tar, extracts the manifests and layers, and stores them as native OCI objects. The resulting local blob has the native OCI media type (e.g. application/vnd.oci.image.manifest.v1+json). Without the correct media type, OCM would store the tar as an opaque layer that cannot be accessed natively.What is the difference between localReference and globalAccess?
localReference is a content-addressable digest that identifies the blob within the component version’s storage. It is stable across transfers — the same digest works regardless of which registry hosts the component, because it is derived from the blob content itself. It works with any OCM repository implementation (CTF archives, OCI registries). For native OCI artifacts stored in an OCI registry, you can access them directly using this digest combined with the component version’s repository path.
globalAccess is an optional, location-specific access specification that points to the artifact in a particular registry. It is not set by default — the globalAccessPolicy must be explicitly configured to enable it. Note that this reference becomes invalid after mirroring — when the component is transferred to a different registry, the globalAccess still points to the original registry. Always use localReference for stable, location-independent access.
How do I enable globalAccess references?
By default, globalAccess is not populated. To opt in, use the two-step transfer workflow with a transfer specification:
The globalAccess.imageReference provides a direct pullable reference. Note that this reference may become stale if the component is transferred to another registry using tooling that does not update the globalAccess field.
Without globalAccess, you can still access native OCI artifacts directly using the localReference digest and the component version’s repository path in the registry.
Can I use --upload-as to control how artifacts are stored?
Yes. The ocm transfer cv command supports --upload-as with two values:
--upload-as localBlob — stores OCI artifacts as local blobs within the component version (default behavior with --copy-resources)
--upload-as ociArtifact — uploads OCI artifacts as standalone OCI artifacts in the target registry, separate from the component version
Both options make the artifact natively accessible in OCI registries, but localBlob keeps the artifact within the component version’s index while ociArtifact stores it independently.