/*
 * Copyright (c) 2023, 2025, 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 8301154 8309214
 * @summary test cert chain deletion logic w/ NSS PKCS11 KeyStore
 * @library /test/lib ..
 * @run testng/othervm CertChainRemoval
 */
import jdk.test.lib.SecurityTools;
import java.io.File;
import java.io.FileInputStream;
import java.nio.file.Path;

import java.security.Key;
import java.security.KeyStore;
import java.security.KeyStoreException;
import java.security.Provider;
import java.security.cert.Certificate;
import java.util.Arrays;
import java.util.Enumeration;

import jtreg.SkippedException;
import org.testng.SkipException;
import org.testng.annotations.BeforeClass;
import org.testng.annotations.Test;


public class CertChainRemoval extends PKCS11Test {

    private static final Path TEST_DATA_PATH = Path.of(BASE)
            .resolve("CertChainRemoval");
    private static final String DIR = TEST_DATA_PATH.toString();

    private record KeyStoreInfo(File file, String type, char[] passwd) {}

    private static final KeyStoreInfo TEMP = new KeyStoreInfo(
            new File(DIR, "temp.ks"),
            "JKS",
            new char[] { 'c', 'h', 'a', 'n', 'g', 'e', 'i', 't' });

    private static final KeyStoreInfo PKCS11KS = new KeyStoreInfo(
            null,
            "PKCS11",
            new char[] { 't', 'e', 's', 't', '1', '2' });

    @BeforeClass
    public void setUp() throws Exception {
        copyNssCertKeyToClassesDir();
        setCommonSystemProps();
        // if temp keystore already exists; skip the creation
        if (!TEMP.file.exists()) {
            createKeyStore(TEMP);
        }
        System.setProperty("CUSTOM_P11_CONFIG",
                TEST_DATA_PATH.resolve("p11-nss.txt").toString());
    }

    @Test
    public void test() throws Exception {
        try {
            main(new CertChainRemoval());
        } catch (SkippedException se) {
            throw new SkipException("One or more tests are skipped");
        }
    }

    private static void printKeyStore(String header, KeyStore ks)
            throws KeyStoreException {
        System.out.println(header);
        Enumeration enu = ks.aliases();
        int count = 0;
        while (enu.hasMoreElements()) {
            count++;
            System.out.println("Entry# " + count +
                    " = " + (String)enu.nextElement());
        }
        System.out.println("========");
    }

    private static void checkEntry(KeyStore ks, String alias,
            Certificate[] expChain) throws KeyStoreException {
        Certificate c = ks.getCertificate(alias);
        Certificate[] chain = ks.getCertificateChain(alias);
        if (expChain == null) {
            if (c != null || (chain != null && chain.length != 0)) {
                throw new RuntimeException("Fail: " + alias + " not removed");
            }
        } else {
            if (!c.equals(expChain[0]) || !Arrays.equals(chain, expChain)) {
                System.out.println("expChain: " + expChain.length);
                System.out.println("actualChain: " + chain.length);
                throw new RuntimeException("Fail: " + alias +
                        " chain check diff");
            }
        }
    }

    public void main(Provider p) throws Exception {
        KeyStore sunks = KeyStore.getInstance(TEMP.type, "SUN");
        sunks.load(new FileInputStream(TEMP.file), TEMP.passwd);
        printKeyStore("Starting with: ", sunks);

        KeyStore p11ks;
        try {
            p11ks = KeyStore.getInstance(PKCS11KS.type, p);
            p11ks.load(null, PKCS11KS.passwd);
            printKeyStore("Initial PKCS11 KeyStore: ", p11ks);
        } catch (Exception e) {
            throw new SkippedException("Skip test, due to " + e, e);
        }

        // get the necessary keys from the temp keystore
        Key pk1PrivKey = sunks.getKey("pk1", TEMP.passwd);
        Certificate pk1Cert = sunks.getCertificate("pk1");
        Key caPrivKey = sunks.getKey("ca1", TEMP.passwd);
        Certificate ca1Cert = sunks.getCertificate("ca1");
        Key rootPrivKey = sunks.getKey("root", TEMP.passwd);
        Certificate rootCert = sunks.getCertificate("root");

        Certificate[] pk1Chain = { pk1Cert, ca1Cert, rootCert };
        Certificate[] ca1Chain = { ca1Cert, rootCert };
        Certificate[] rootChain = { rootCert };

        // populate keystore with "pk1" and "ca", then delete "pk1"
        System.out.println("Add pk1, ca1 and root, then delete pk1");
        p11ks.setKeyEntry("pk1", pk1PrivKey, null, pk1Chain);
        p11ks.setKeyEntry("ca1", caPrivKey, null, ca1Chain);
        p11ks.setKeyEntry("root", rootPrivKey, null, rootChain);
        p11ks.deleteEntry("pk1");

        // reload the keystore
        p11ks.store(null, PKCS11KS.passwd);
        p11ks.load(null, PKCS11KS.passwd);
        printKeyStore("Reload#1: ca1 / root", p11ks);

        // should only have "ca1" and "root"
        checkEntry(p11ks, "pk1", null);
        checkEntry(p11ks, "ca1", ca1Chain);
        checkEntry(p11ks, "root", rootChain);

        // now add "pk1" and delete "ca1"
        System.out.println("Now add pk1 and delete ca1");
        p11ks.setKeyEntry("pk1", pk1PrivKey, null, pk1Chain);
        p11ks.deleteEntry("ca1");

        // reload the keystore
        p11ks.store(null, PKCS11KS.passwd);
        p11ks.load(null, PKCS11KS.passwd);
        printKeyStore("Reload#2: pk1 / root", p11ks);

        // should only have "pk1" and "root" now
        checkEntry(p11ks, "pk1", pk1Chain);
        checkEntry(p11ks, "ca1", null);
        checkEntry(p11ks, "root", rootChain);

        // now delete "root"
        System.out.println("Now delete root");
        p11ks.deleteEntry("root");

        // reload the keystore
        p11ks.store(null, PKCS11KS.passwd);
        p11ks.load(null, PKCS11KS.passwd);
        printKeyStore("Reload#3: pk1", p11ks);

        // should only have "pk1" now
        checkEntry(p11ks, "pk1", pk1Chain);
        checkEntry(p11ks, "ca1", null);
        checkEntry(p11ks, "root", null);

        // now delete "pk1"
        System.out.println("Now delete pk1");
        p11ks.deleteEntry("pk1");

        // reload the keystore
        p11ks.store(null, PKCS11KS.passwd);
        p11ks.load(null, PKCS11KS.passwd);
        printKeyStore("Reload#4: ", p11ks);

        // should have nothing now
        checkEntry(p11ks, "pk1", null);
        checkEntry(p11ks, "ca1", null);
        checkEntry(p11ks, "root", null);

        System.out.println("Test Passed");
    }

    private static void createKeyStore(KeyStoreInfo ksi) throws Exception {
        System.out.println("Creating keypairs and storing them into " +
            ksi.file.getAbsolutePath());
        String keyGenOptions = " -keyalg RSA -keysize 2048 ";
        String keyStoreOptions = " -keystore " + ksi.file.getAbsolutePath() +
                " -storetype " + ksi.type + " -storepass " +
                new String(ksi.passwd);

        String[] aliases = { "ROOT", "CA1", "PK1" };
        for (String n : aliases) {
            SecurityTools.keytool("-genkeypair -alias " + n +
                " -dname CN=" + n + keyGenOptions + keyStoreOptions);
            String issuer = switch (n) {
                case "CA1"-> "ROOT";
                case "PK1"-> "CA1";
                default-> null;
            };
            if (issuer != null) {
                // export CSR and issue the cert using the issuer
                SecurityTools.keytool("-certreq -alias " + n +
                    " -file tmp.req" + keyStoreOptions);
                SecurityTools.keytool("-gencert -alias " + issuer +
                    " -infile tmp.req -outfile tmp.cert -validity 3650" +
                    keyStoreOptions);
                SecurityTools.keytool("-importcert -alias " + n +
                    " -file tmp.cert" + keyStoreOptions);
            }
        }
    }

}
