/* * Copyright (c) 1999, 2024, Oracle and/or its affiliates. All rights reserved. * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER. * * This code is free software; you can redistribute it and/or modify it * under the terms of the GNU General Public License version 2 only, as * published by the Free Software Foundation. * * This code is distributed in the hope that it will be useful, but WITHOUT * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or * FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License * version 2 for more details (a copy is included in the LICENSE file that * accompanied this code). * * You should have received a copy of the GNU General Public License version * 2 along with this work; if not, write to the Free Software Foundation, * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA. * * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA * or visit www.oracle.com if you need additional information or have any * questions. */ /* * @test * @bug 8048819 * @summary This test stressful verifies the assertion of "The secret keys generated * by all involved parties should be the same." for javax.crypto.KeyAgreement * @run main/othervm -Djdk.crypto.KeyAgreement.legacyKDF=true SameDHKeyStressTest */ import java.security.AlgorithmParameterGenerator; import java.security.InvalidAlgorithmParameterException; import java.security.InvalidKeyException; import java.security.Key; import java.security.KeyPair; import java.security.KeyPairGenerator; import java.security.NoSuchAlgorithmException; import java.security.NoSuchProviderException; import java.security.spec.AlgorithmParameterSpec; import java.util.Arrays; import javax.crypto.KeyAgreement; import javax.crypto.SecretKey; import javax.crypto.spec.DHGenParameterSpec; import javax.crypto.spec.DHParameterSpec; public class SameDHKeyStressTest { static final String[] ALGORITHMS = {"DH", "DiffieHellman", "dh", "diffieHELLMAN"}; static final String[] SECRET_ALOGRITHMS = {"DES", "DESede", "blowfish"}; static final int[] NUMBER_OF_PARTIES = {2, 3, 4}; static final String[] PA_NAMES = {"Alice", "Bob", "Carol", "David"}; public static void main(String args[]) { int failedCnt = 0; StringBuilder failedList = new StringBuilder("Failed List:"); for (String algorithm : ALGORITHMS) { for (int numOfParties : NUMBER_OF_PARTIES) { for (String secretAlgorithm : SECRET_ALOGRITHMS) { if (!runTest(algorithm, numOfParties, secretAlgorithm)) { failedCnt++; failedList.append("\n Altorightm = ").append(algorithm). append(" Number of Parties = ").append(numOfParties). append(" Secret Algorithm = ").append(secretAlgorithm); } } } } //end of for loop if (failedCnt > 0) { System.out.println(failedList); throw new RuntimeException("SameDHKeyStressTest Failed"); } } public static boolean runTest(String algo, int numParties, String secretAlgo) { KAParticipant[] parties = new KAParticipant[numParties]; Key[] keyArchives = new Key[numParties]; try { // generate AlogirhtmParameterSpec AlgorithmParameterGenerator apg = AlgorithmParameterGenerator.getInstance("DH", System.getProperty("test.provider.name", "SunJCE")); AlgorithmParameterSpec aps = new DHGenParameterSpec(512, 64); apg.init(aps); DHParameterSpec spec = apg.generateParameters(). getParameterSpec(DHParameterSpec.class); //initilize all KeyAgreement participants for (int i = 0; i < numParties; i++) { parties[i] = new KAParticipant(PA_NAMES[i], algo); parties[i].initialize(spec); keyArchives[i] = parties[i].getPublicKey(); } // Do all phases in the KeyAgreement for all participants Key[] keyBuffer = new Key[numParties]; boolean lastPhase = false; for (int j = 0; j < numParties - 1; j++) { if (j == numParties - 2) { lastPhase = true; } for (int k = 0; k < numParties; k++) { if (k == numParties - 1) { keyBuffer[k] = parties[k].doPhase(keyArchives[0], lastPhase); } else { keyBuffer[k] = parties[k].doPhase(keyArchives[k + 1], lastPhase); } } System.arraycopy(keyBuffer, 0, keyArchives, 0, numParties); } //Comparison: The secret keys generated by all involved parties should be the same SecretKey[] sKeys = new SecretKey[numParties]; for (int n = 0; n < numParties; n++) { sKeys[n] = parties[n].generateSecret(secretAlgo); } for (int q = 0; q < numParties - 1; q++) { if (!Arrays.equals(sKeys[q].getEncoded(), sKeys[q + 1].getEncoded())) { return false; } } return true; } catch (Exception ex) { ex.printStackTrace(); return false; } } } class KAParticipant { private String name = null; private String algorithm = null; private KeyPairGenerator keyGen = null; private KeyPair keys = null; private KeyAgreement ka = null; public KAParticipant(String pName, String algo) throws NoSuchAlgorithmException, NoSuchProviderException { name = pName; algorithm = algo; keyGen = KeyPairGenerator.getInstance(algo, System.getProperty("test.provider.name", "SunJCE")); ka = KeyAgreement.getInstance(algo, System.getProperty("test.provider.name", "SunJCE")); } public void initialize(AlgorithmParameterSpec spec) throws InvalidAlgorithmParameterException, InvalidKeyException { keyGen.initialize(spec); keys = keyGen.generateKeyPair(); ka.init(keys.getPrivate()); } public Key doPhase(Key key, boolean lastPhase) throws InvalidKeyException { return ka.doPhase(key, lastPhase); } public Key getPublicKey() { return keys.getPublic(); } public byte[] generateSecret() { return ka.generateSecret(); } public SecretKey generateSecret(String algo) throws java.lang.IllegalStateException, java.security.NoSuchAlgorithmException, java.security.InvalidKeyException { return ka.generateSecret(algo); } }