Skip to content

Commit 13d852a

Browse files
committedJan 31, 2025
8349017: Update ML tests to verify against ACVP 1.1.0.38 version
Reviewed-by: weijun
1 parent 137ad5d commit 13d852a

File tree

8 files changed

+122
-3175
lines changed

8 files changed

+122
-3175
lines changed
 

‎test/jdk/sun/security/provider/acvp/Launcher.java

+82-41
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2024, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -20,13 +20,19 @@
2020
* or visit www.oracle.com if you need additional information or have any
2121
* questions.
2222
*/
23+
24+
import jdk.test.lib.artifacts.Artifact;
25+
import jdk.test.lib.artifacts.ArtifactResolver;
26+
import jdk.test.lib.artifacts.ArtifactResolverException;
2327
import jdk.test.lib.json.JSONValue;
2428
import jtreg.SkippedException;
2529

26-
import java.nio.file.Files;
30+
import java.io.InputStream;
2731
import java.nio.file.Path;
2832
import java.security.Provider;
2933
import java.security.Security;
34+
import java.util.zip.ZipEntry;
35+
import java.util.zip.ZipFile;
3036

3137
/*
3238
* @test
@@ -35,19 +41,17 @@
3541
* @modules java.base/sun.security.provider
3642
*/
3743

38-
/// This test runs on `internalProjection.json`-style files generated
39-
/// by NIST's ACVP Server. See [https://github.com/usnistgov/ACVP-Server].
40-
///
41-
/// The files are either put into the `data` directory or another
42-
/// directory specified by the `test.acvp.data` test property.
43-
/// The test walks through the directory recursively and looks for
44-
/// file names equal to or ending with `internalProjection.json` and
45-
/// runs tests on them.
44+
/// This test runs on `internalProjection.json`-style files generated by NIST's
45+
/// ACVP Server ([GitHub repository](https://github.com/usnistgov/ACVP-Server)).
46+
/// These files are included in ZIP archives available under the
47+
/// [tags section](https://github.com/usnistgov/ACVP-Server/tags)
48+
/// of the repository.
4649
///
47-
/// Set the `test.acvp.alg` test property to only test the specified algorithm.
50+
/// The zip archive is either hosted on artifactory server or
51+
/// specified with local path to the test.
52+
/// The test looks for test data files in the archive listed with `TEST_FILES`.
4853
///
49-
/// Sample files can be downloaded from
50-
/// [https://github.com/usnistgov/ACVP-Server/tree/master/gen-val/json-files].
54+
/// These tests are currently compatible with ACVP version 1.1.0.38.
5155
///
5256
/// By default, the test uses system-preferred implementations.
5357
/// If you want to test a specific provider, set the
@@ -58,19 +62,30 @@
5862
/// [https://github.com/usnistgov/ACVP?tab=readme-ov-file#supported-algorithms].
5963
///
6064
/// Example:
65+
///
66+
/// Run locally with ArtifactResolver
6167
/// ```
62-
/// jtreg -Dtest.acvp.provider=SunJCE \
63-
/// -Dtest.acvp.alg=ML-KEM \
64-
/// -Dtest.acvp.data=/path/to/json-files/ \
65-
/// -jdk:/path/to/jdk Launcher.java
68+
/// jtreg -Djdk.test.lib.artifacts.ACVP-Server=<path-to-archive-file>
6669
/// ```
67-
public class Launcher {
70+
/// OR host the zip archive on artifactory server.
71+
///
6872

69-
private static final String ONLY_ALG
70-
= System.getProperty("test.acvp.alg");
73+
public class Launcher {
7174

7275
private static final Provider PROVIDER;
7376

77+
private static final String ACVP_BUNDLE_LOC = "jpg.tests.jdk";
78+
private static final String ACVP_BUNDLE_NAME = "ACVP-Server";
79+
private static final String ACVP_BUNDLE_VERSION = "1.1.0.38";
80+
// Zip archive entry name, do not update to use File.separator
81+
private static final String[] TEST_FILES = {
82+
"gen-val/json-files/ML-DSA-keyGen-FIPS204/internalProjection.json",
83+
"gen-val/json-files/ML-DSA-sigGen-FIPS204/internalProjection.json",
84+
"gen-val/json-files/ML-DSA-sigVer-FIPS204/internalProjection.json",
85+
"gen-val/json-files/ML-KEM-encapDecap-FIPS203/internalProjection.json",
86+
"gen-val/json-files/ML-KEM-keyGen-FIPS203/internalProjection.json"
87+
};
88+
7489
private static int count = 0;
7590
private static int invalidTest = 0;
7691
private static int unsupportedTest = 0;
@@ -91,24 +106,26 @@ public class Launcher {
91106

92107
public static void main(String[] args) throws Exception {
93108

94-
var testDataProp = System.getProperty("test.acvp.data");
95-
Path dataPath = testDataProp != null
96-
? Path.of(testDataProp)
97-
: Path.of(System.getProperty("test.src"), "data");
98-
System.out.println("Data path: " + dataPath);
109+
Path archivePath = fetchACVPServerTests(ACVP_SERVER_TESTS.class);
110+
System.out.println("Data path: " + archivePath);
99111

100112
if (PROVIDER != null) {
101113
System.out.println("Provider: " + PROVIDER.getName());
102114
}
103-
if (ONLY_ALG != null) {
104-
System.out.println("Algorithm: " + ONLY_ALG);
105-
}
106115

107-
try (var stream = Files.walk(dataPath)) {
108-
stream.filter(Files::isRegularFile)
109-
.filter(p -> p.getFileName().toString()
110-
.endsWith("internalProjection.json"))
111-
.forEach(Launcher::run);
116+
// Read test data files from zip archive
117+
try (ZipFile zf = new ZipFile(archivePath.toFile())) {
118+
for (String testFile : TEST_FILES) {
119+
// Zip archive entry name, do not update to use File.separator
120+
String fullEntryName = ACVP_BUNDLE_NAME + "-" + ACVP_BUNDLE_VERSION + "/" + testFile;
121+
System.out.println("Find and test with: " + fullEntryName);
122+
ZipEntry ze = zf.getEntry(fullEntryName);
123+
if (ze != null) {
124+
run(zf.getInputStream(ze));
125+
} else {
126+
throw new RuntimeException("Entry not found: " + fullEntryName);
127+
}
128+
}
112129
}
113130

114131
if (count > 0) {
@@ -117,25 +134,25 @@ public static void main(String[] args) throws Exception {
117134
System.out.println("Invalid tests: " + invalidTest);
118135
System.out.println("Unsupported tests: " + unsupportedTest);
119136
} else {
120-
throw new SkippedException("No supported test found");
137+
throw new RuntimeException("No supported test found");
138+
}
139+
140+
if (invalidTest != 0 || unsupportedTest != 0){
141+
throw new RuntimeException("Invalid or Unsupported tests found");
121142
}
122143
}
123144

124-
static void run(Path test) {
145+
static void run(InputStream test) {
125146
try {
126147
JSONValue kat;
127-
try {
128-
kat = JSONValue.parse(Files.readString(test));
148+
try (test) {
149+
kat = JSONValue.parse(new String(test.readAllBytes()));
129150
} catch (Exception e) {
130151
System.out.println("Warning: cannot parse " + test + ". Skipped");
131152
invalidTest++;
132153
return;
133154
}
134155
var alg = kat.get("algorithm").asString();
135-
if (ONLY_ALG != null && !alg.equals(ONLY_ALG)) {
136-
return;
137-
}
138-
System.out.println(">>> Testing " + test + "...");
139156
switch (alg) {
140157
case "ML-DSA" -> {
141158
ML_DSA_Test.run(kat, PROVIDER);
@@ -160,4 +177,28 @@ static void run(Path test) {
160177
throw new RuntimeException(e);
161178
}
162179
}
180+
181+
private static Path fetchACVPServerTests(Class<?> clazz) {
182+
try {
183+
return ArtifactResolver.resolve(clazz).entrySet().stream()
184+
.findAny().get().getValue();
185+
} catch (ArtifactResolverException e) {
186+
Throwable cause = e.getCause();
187+
if (cause == null) {
188+
throw new SkippedException("Cannot resolve artifact, "
189+
+ "please check if JIB jar is present in classpath.", e);
190+
}
191+
192+
throw new SkippedException("Fetch artifact failed: " + clazz, e);
193+
}
194+
}
195+
196+
@Artifact(
197+
organization = ACVP_BUNDLE_LOC,
198+
name = ACVP_BUNDLE_NAME,
199+
revision = ACVP_BUNDLE_VERSION,
200+
extension = "zip",
201+
unpack = false)
202+
private static class ACVP_SERVER_TESTS {
203+
}
163204
}

‎test/jdk/sun/security/provider/acvp/ML_DSA_Test.java

+40-14
Original file line numberDiff line numberDiff line change
@@ -1,5 +1,5 @@
11
/*
2-
* Copyright (c) 2024, Oracle and/or its affiliates. All rights reserved.
2+
* Copyright (c) 2024, 2025, Oracle and/or its affiliates. All rights reserved.
33
* DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
44
*
55
* This code is free software; you can redistribute it and/or modify it
@@ -36,10 +36,6 @@ public class ML_DSA_Test {
3636

3737
public static void run(JSONValue kat, Provider provider) throws Exception {
3838

39-
// We only have ML-DSA test for internal functions, which
40-
// is equivalent to the FIP 204 draft.
41-
ML_DSA_Impls.version = ML_DSA_Impls.Version.DRAFT;
42-
4339
var mode = kat.get("mode").asString();
4440
switch (mode) {
4541
case "keyGen" -> keyGenTest(kat, provider);
@@ -50,7 +46,7 @@ public static void run(JSONValue kat, Provider provider) throws Exception {
5046
}
5147

5248
static NamedParameterSpec genParams(String pname) {
53-
return switch (pname) {
49+
return switch (pname) {
5450
case "ML-DSA-44" -> NamedParameterSpec.ML_DSA_44;
5551
case "ML-DSA-65" -> NamedParameterSpec.ML_DSA_65;
5652
case "ML-DSA-87" -> NamedParameterSpec.ML_DSA_87;
@@ -89,9 +85,23 @@ static void sigGenTest(JSONValue kat, Provider p) throws Exception {
8985
: Signature.getInstance("ML-DSA", p);
9086
for (var t : kat.get("testGroups").asArray()) {
9187
var pname = t.get("parameterSet").asString();
92-
var det = Boolean.parseBoolean(t.get("deterministic").asString());
9388
System.out.println(">> " + pname + " sign");
89+
var det = Boolean.parseBoolean(t.get("deterministic").asString());
90+
if (t.get("signatureInterface").asString().equals("internal")) {
91+
ML_DSA_Impls.version = ML_DSA_Impls.Version.DRAFT;
92+
} else {
93+
ML_DSA_Impls.version = ML_DSA_Impls.Version.FINAL;
94+
}
95+
if (t.get("externalMu").asString().equals("true")) {
96+
continue; // Not supported
97+
}
9498
for (var c : t.get("tests").asArray()) {
99+
var cstr = c.get("context");
100+
var ctxt = cstr == null ? new byte[0] : toByteArray(cstr.asString());
101+
var hashAlg = c.get("hashAlg").asString();
102+
if (!hashAlg.equals("none") || ctxt.length != 0) {
103+
continue; // Not supported
104+
}
95105
System.out.print(Integer.parseInt(c.get("tcId").asString()) + " ");
96106
var sk = new PrivateKey() {
97107
public String getAlgorithm() { return pname; }
@@ -103,8 +113,7 @@ static void sigGenTest(JSONValue kat, Provider p) throws Exception {
103113
s.initSign(sk, sr);
104114
s.update(toByteArray(c.get("message").asString()));
105115
var sig = s.sign();
106-
Asserts.assertEqualsByteArray(
107-
toByteArray(c.get("signature").asString()), sig);
116+
Asserts.assertEqualsByteArray(toByteArray(c.get("signature").asString()), sig);
108117
}
109118
System.out.println();
110119
}
@@ -116,14 +125,31 @@ static void sigVerTest(JSONValue kat, Provider p) throws Exception {
116125
: Signature.getInstance("ML-DSA", p);
117126
for (var t : kat.get("testGroups").asArray()) {
118127
var pname = t.get("parameterSet").asString();
119-
var pk = new PublicKey() {
120-
public String getAlgorithm() { return pname; }
121-
public String getFormat() { return "RAW"; }
122-
public byte[] getEncoded() { return toByteArray(t.get("pk").asString()); }
123-
};
124128
System.out.println(">> " + pname + " verify");
129+
130+
if (t.get("signatureInterface").asString().equals("internal")) {
131+
ML_DSA_Impls.version = ML_DSA_Impls.Version.DRAFT;
132+
} else {
133+
ML_DSA_Impls.version = ML_DSA_Impls.Version.FINAL;
134+
}
135+
136+
if (t.get("externalMu").asString().equals("true")) {
137+
continue; // Not supported
138+
}
139+
125140
for (var c : t.get("tests").asArray()) {
141+
var cstr = c.get("context");
142+
var ctxt = cstr == null ? new byte[0] : toByteArray(cstr.asString());
143+
var hashAlg = c.get("hashAlg").asString();
144+
if (!hashAlg.equals("none") || ctxt.length != 0) {
145+
continue; // Not supported
146+
}
126147
System.out.print(c.get("tcId").asString() + " ");
148+
var pk = new PublicKey() {
149+
public String getAlgorithm() { return pname; }
150+
public String getFormat() { return "RAW"; }
151+
public byte[] getEncoded() { return toByteArray(c.get("pk").asString()); }
152+
};
127153
// Only ML-DSA sigVer has negative tests
128154
var expected = Boolean.parseBoolean(c.get("testPassed").asString());
129155
var actual = true;

‎test/jdk/sun/security/provider/acvp/data/ML-DSA-keyGen-FIPS204/internalProjection.json

-555
This file was deleted.

‎test/jdk/sun/security/provider/acvp/data/ML-DSA-sigGen-FIPS204/internalProjection.json

-507
This file was deleted.

‎test/jdk/sun/security/provider/acvp/data/ML-DSA-sigVer-FIPS204/internalProjection.json

-396
This file was deleted.

‎test/jdk/sun/security/provider/acvp/data/ML-KEM-encapDecap-FIPS203/internalProjection.json

-1,023
This file was deleted.

‎test/jdk/sun/security/provider/acvp/data/ML-KEM-keyGen-FIPS203/internalProjection.json

-630
This file was deleted.

‎test/jdk/sun/security/provider/acvp/data/acvp.md

-9
This file was deleted.

0 commit comments

Comments
 (0)
Please sign in to comment.