PT-2026-44919 · Maven · Land.Oras:Oras-Java-Sdk

Published

2026-05-19

·

Updated

2026-05-19

CVSS v3.1

8.1

High

VectorAV:N/AC:L/PR:N/UI:R/S:U/C:N/I:H/A:H

Summary

The pullArtifact methods in Registry and OCILayout use the org.opencontainers.image.title annotation from a pulled manifest as a filename, resolving it against the caller supplied output directory without normalization or a containment check. A manifest publisher can set this annotation to a path that escapes the output directory, causing the SDK to write the layer's blob anywhere the JVM process can write.

Details

Two call sites are affected.
src/main/java/land/oras/Registry.java, pullLayer (reached from Registry.pullArtifact):
java
Path targetPath = path.resolve(layer.getAnnotations().get(Const.ANNOTATION TITLE));
...
Files.copy(is, targetPath, StandardCopyOption.REPLACE EXISTING);
src/main/java/land/oras/OCILayout.java, OCILayout.pullArtifact:
java
Files.copy(blobPath, path.resolve(layer.getAnnotations().get(Const.ANNOTATION TITLE)));
The annotation comes from the remote manifest. Path.resolve treats an absolute argument as a full override of the base, and follows .. segments upward, so the annotation controls the destination. REPLACE EXISTING overwrites files that exist at that destination.
The unpack branch of pullLayer (taken when the layer carries io.deis.oras.content.unpack=true) is not affected, because it dispatches through ArchiveUtils.untar / unzip, which apply outputPath.startsWith(normalizedTarget) after normalization. The non unpack branch and OCILayout.pullArtifact lack the equivalent check.
fetchBlob(ContainerRef, Path) is not affected. The caller passes the destination path and the title annotation is not consulted.

Fix

Path traversal

Found an issue in the description? Have something to add? Feel free to write us 👾

Weakness Enumeration

Related Identifiers

GHSA-XM96-GFJX-JCRC

Affected Products

Land.Oras:Oras-Java-Sdk