> pair =
+ getKeystoresForReading(alias);
+ Certificate[] chain = null;
+
+ try {
+ String entryAlias = pair.getKey();
+ for (KeyStore keystore : pair.getValue()) {
+ chain = keystore.getCertificateChain(entryAlias);
+ if (chain != null) {
+ break;
+ }
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return chain;
+ }
+
+ /**
+ * Returns the certificate associated with the given alias.
+ *
+ * If the given alias name identifies a
+ * trusted certificate entry, the certificate associated with that
+ * entry is returned. If the given alias name identifies a
+ * key entry, the first element of the certificate chain of that
+ * entry is returned, or null if that entry does not have a certificate
+ * chain.
+ *
+ * @param alias the alias name
+ *
+ * @return the certificate, or null if the given alias does not exist or
+ * does not contain a certificate.
+ */
+ public Certificate engineGetCertificate(String alias) {
+
+ AbstractMap.SimpleEntry> pair =
+ getKeystoresForReading(alias);
+ Certificate cert = null;
+
+ try {
+ String entryAlias = pair.getKey();
+ for (KeyStore keystore : pair.getValue()) {
+ cert = keystore.getCertificate(entryAlias);
+ if (cert != null) {
+ break;
+ }
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return cert;
+ }
+
+ /**
+ * Returns the creation date of the entry identified by the given alias.
+ *
+ * @param alias the alias name
+ *
+ * @return the creation date of this entry, or null if the given alias does
+ * not exist
+ */
+ public Date engineGetCreationDate(String alias) {
+
+ AbstractMap.SimpleEntry> pair =
+ getKeystoresForReading(alias);
+ Date date = null;
+
+ try {
+ String entryAlias = pair.getKey();
+ for (KeyStore keystore : pair.getValue()) {
+ date = keystore.getCreationDate(entryAlias);
+ if (date != null) {
+ break;
+ }
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return date;
+ }
+
+ /**
+ * Assigns the given private key to the given alias, protecting
+ * it with the given password as defined in PKCS8.
+ *
+ * The given java.security.PrivateKey key
must
+ * be accompanied by a certificate chain certifying the
+ * corresponding public key.
+ *
+ *
If the given alias already exists, the keystore information
+ * associated with it is overridden by the given key and certificate
+ * chain.
+ *
+ * @param alias the alias name
+ * @param key the private key to be associated with the alias
+ * @param password the password to protect the key
+ * @param chain the certificate chain for the corresponding public
+ * key (only required if the given key is of type
+ * java.security.PrivateKey
).
+ *
+ * @exception KeyStoreException if the given key is not a private key,
+ * cannot be protected, or this operation fails for some other reason
+ */
+ public void engineSetKeyEntry(String alias, Key key, char[] password,
+ Certificate[] chain)
+ throws KeyStoreException
+ {
+ AbstractMap.SimpleEntry> pair =
+ getKeystoreForWriting(alias);
+
+ if (pair == null) {
+ throw new KeyStoreException("Error setting key entry for '" +
+ alias + "'");
+ }
+ String entryAlias = pair.getKey();
+ Map.Entry keystore = pair.getValue();
+ keystore.getValue().setKeyEntry(entryAlias, key, password, chain);
+ }
+
+ /**
+ * Assigns the given key (that has already been protected) to the given
+ * alias.
+ *
+ * If the protected key is of type
+ * java.security.PrivateKey
, it must be accompanied by a
+ * certificate chain certifying the corresponding public key. If the
+ * underlying keystore implementation is of type jks
,
+ * key
must be encoded as an
+ * EncryptedPrivateKeyInfo
as defined in the PKCS #8 standard.
+ *
+ *
If the given alias already exists, the keystore information
+ * associated with it is overridden by the given key (and possibly
+ * certificate chain).
+ *
+ * @param alias the alias name
+ * @param key the key (in protected format) to be associated with the alias
+ * @param chain the certificate chain for the corresponding public
+ * key (only useful if the protected key is of type
+ * java.security.PrivateKey
).
+ *
+ * @exception KeyStoreException if this operation fails.
+ */
+ public void engineSetKeyEntry(String alias, byte[] key,
+ Certificate[] chain)
+ throws KeyStoreException
+ {
+ AbstractMap.SimpleEntry> pair =
+ getKeystoreForWriting(alias);
+
+ if (pair == null) {
+ throw new KeyStoreException(
+ "Error setting protected key entry for '" + alias + "'");
+ }
+ String entryAlias = pair.getKey();
+ Map.Entry keystore = pair.getValue();
+ keystore.getValue().setKeyEntry(entryAlias, key, chain);
+ }
+
+ /**
+ * Assigns the given certificate to the given alias.
+ *
+ * If the given alias already exists in this keystore and identifies a
+ * trusted certificate entry, the certificate associated with it is
+ * overridden by the given certificate.
+ *
+ * @param alias the alias name
+ * @param cert the certificate
+ *
+ * @exception KeyStoreException if the given alias already exists and does
+ * not identify a trusted certificate entry, or this operation
+ * fails for some other reason.
+ */
+ public void engineSetCertificateEntry(String alias, Certificate cert)
+ throws KeyStoreException
+ {
+ AbstractMap.SimpleEntry> pair =
+ getKeystoreForWriting(alias);
+
+ if (pair == null) {
+ throw new KeyStoreException("Error setting certificate entry for '"
+ + alias + "'");
+ }
+ String entryAlias = pair.getKey();
+ Map.Entry keystore = pair.getValue();
+ keystore.getValue().setCertificateEntry(entryAlias, cert);
+ }
+
+ /**
+ * Deletes the entry identified by the given alias from this keystore.
+ *
+ * @param alias the alias name
+ *
+ * @exception KeyStoreException if the entry cannot be removed.
+ */
+ public void engineDeleteEntry(String alias) throws KeyStoreException
+ {
+ AbstractMap.SimpleEntry> pair =
+ getKeystoreForWriting(alias);
+
+ if (pair == null) {
+ throw new KeyStoreException("Error deleting entry for '" + alias +
+ "'");
+ }
+ String entryAlias = pair.getKey();
+ Map.Entry keystore = pair.getValue();
+ keystore.getValue().deleteEntry(entryAlias);
+ }
+
+ /**
+ * Lists all the alias names of this keystore.
+ *
+ * @return enumeration of the alias names
+ */
+ public Enumeration engineAliases() {
+ final Iterator> iterator =
+ keystores.entrySet().iterator();
+
+ return new Enumeration() {
+ private int index = 0;
+ private Map.Entry keystoresEntry = null;
+ private String prefix = null;
+ private Enumeration aliases = null;
+
+ public boolean hasMoreElements() {
+ try {
+ if (aliases == null) {
+ if (iterator.hasNext()) {
+ keystoresEntry = iterator.next();
+ prefix = keystoresEntry.getKey() +
+ entryNameSeparator;
+ aliases = keystoresEntry.getValue().aliases();
+ } else {
+ return false;
+ }
+ }
+ if (aliases.hasMoreElements()) {
+ return true;
+ } else {
+ if (iterator.hasNext()) {
+ keystoresEntry = iterator.next();
+ prefix = keystoresEntry.getKey() +
+ entryNameSeparator;
+ aliases = keystoresEntry.getValue().aliases();
+ } else {
+ return false;
+ }
+ }
+ } catch (KeyStoreException e) {
+ return false;
+ }
+
+ return aliases.hasMoreElements();
+ }
+
+ public String nextElement() {
+ if (hasMoreElements()) {
+ return prefix + aliases.nextElement();
+ }
+ throw new NoSuchElementException();
+ }
+ };
+ }
+
+ /**
+ * Checks if the given alias exists in this keystore.
+ *
+ * @param alias the alias name
+ *
+ * @return true if the alias exists, false otherwise
+ */
+ public boolean engineContainsAlias(String alias) {
+
+ AbstractMap.SimpleEntry> pair =
+ getKeystoresForReading(alias);
+
+ try {
+ String entryAlias = pair.getKey();
+ for (KeyStore keystore : pair.getValue()) {
+ if (keystore.containsAlias(entryAlias)) {
+ return true;
+ }
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return false;
+ }
+
+ /**
+ * Retrieves the number of entries in this keystore.
+ *
+ * @return the number of entries in this keystore
+ */
+ public int engineSize() {
+
+ int size = 0;
+ try {
+ for (KeyStore keystore : keystores.values()) {
+ size += keystore.size();
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return size;
+ }
+
+ /**
+ * Returns true if the entry identified by the given alias is a
+ * key entry, and false otherwise.
+ *
+ * @return true if the entry identified by the given alias is a
+ * key entry, false otherwise.
+ */
+ public boolean engineIsKeyEntry(String alias) {
+
+ AbstractMap.SimpleEntry> pair =
+ getKeystoresForReading(alias);
+
+ try {
+ String entryAlias = pair.getKey();
+ for (KeyStore keystore : pair.getValue()) {
+ if (keystore.isKeyEntry(entryAlias)) {
+ return true;
+ }
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return false;
+ }
+
+ /**
+ * Returns true if the entry identified by the given alias is a
+ * trusted certificate entry, and false otherwise.
+ *
+ * @return true if the entry identified by the given alias is a
+ * trusted certificate entry, false otherwise.
+ */
+ public boolean engineIsCertificateEntry(String alias) {
+
+ AbstractMap.SimpleEntry> pair =
+ getKeystoresForReading(alias);
+
+ try {
+ String entryAlias = pair.getKey();
+ for (KeyStore keystore : pair.getValue()) {
+ if (keystore.isCertificateEntry(entryAlias)) {
+ return true;
+ }
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ return false;
+ }
+
+ /*
+ * Returns a keystore entry alias and a list of target keystores.
+ * When the supplied alias prefix identifies a keystore then that single
+ * keystore is returned. When no alias prefix is supplied then all the
+ * keystores are returned.
+ */
+ private AbstractMap.SimpleEntry>
+ getKeystoresForReading(String alias) {
+
+ String[] splits = alias.split(this.entryNameSeparatorRegEx, 2);
+ if (splits.length == 2) { // prefixed alias
+ KeyStore keystore = keystores.get(splits[0]);
+ if (keystore != null) {
+ return new AbstractMap.SimpleEntry<>(splits[1],
+ (Collection) Collections.singleton(keystore));
+ }
+ } else if (splits.length == 1) { // unprefixed alias
+ // Check all keystores for the first occurrence of the alias
+ return new AbstractMap.SimpleEntry<>(alias, keystores.values());
+ }
+ return new AbstractMap.SimpleEntry<>("",
+ (Collection) Collections.emptyList());
+ }
+
+ /*
+ * Returns a keystore entry alias and a single target keystore.
+ * An alias prefix must be supplied.
+ */
+ private
+ AbstractMap.SimpleEntry>
+ getKeystoreForWriting(String alias) {
+
+ String[] splits = alias.split(this.entryNameSeparator, 2);
+ if (splits.length == 2) { // prefixed alias
+ KeyStore keystore = keystores.get(splits[0]);
+ if (keystore != null) {
+ return new AbstractMap.SimpleEntry<>(splits[1],
+ new AbstractMap.SimpleEntry<>(splits[0], keystore));
+ }
+ }
+ return null;
+ }
+
+ /**
+ * Returns the (alias) name of the first keystore entry whose certificate
+ * matches the given certificate.
+ *
+ * This method attempts to match the given certificate with each
+ * keystore entry. If the entry being considered
+ * is a trusted certificate entry, the given certificate is
+ * compared to that entry's certificate. If the entry being considered is
+ * a key entry, the given certificate is compared to the first
+ * element of that entry's certificate chain (if a chain exists).
+ *
+ * @param cert the certificate to match with.
+ *
+ * @return the (alias) name of the first entry with matching certificate,
+ * or null if no such entry exists in this keystore.
+ */
+ public String engineGetCertificateAlias(Certificate cert) {
+
+ try {
+
+ String alias = null;
+ for (KeyStore keystore : keystores.values()) {
+ if ((alias = keystore.getCertificateAlias(cert)) != null) {
+ break;
+ }
+ }
+ return alias;
+
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+ }
+
+ /**
+ * Stores this keystore to the given output stream, and protects its
+ * integrity with the given password.
+ *
+ * @param stream the output stream to which this keystore is written.
+ * @param password the password to generate the keystore integrity check
+ *
+ * @exception IOException if there was an I/O problem with data
+ * @exception NoSuchAlgorithmException if the appropriate data integrity
+ * algorithm could not be found
+ * @exception CertificateException if any of the certificates included in
+ * the keystore data could not be stored
+ */
+ public void engineStore(OutputStream stream, char[] password)
+ throws IOException, NoSuchAlgorithmException, CertificateException
+ {
+ // Support storing to a stream only when a single keystore has been
+ // configured
+ try {
+ if (keystores.size() == 1) {
+ keystores.values().iterator().next().store(stream, password);
+ return;
+ }
+ } catch (KeyStoreException e) {
+ throw new IllegalStateException(e);
+ }
+
+ throw new UnsupportedOperationException(
+ "This keystore must be stored using a " +
+ "KeyStore.DomainLoadStoreParameter");
+ }
+
+ @Override
+ public void engineStore(KeyStore.LoadStoreParameter param)
+ throws IOException, NoSuchAlgorithmException, CertificateException
+ {
+ if (param instanceof KeyStore.DomainLoadStoreParameter) {
+ KeyStore.DomainLoadStoreParameter domainParameter =
+ (KeyStore.DomainLoadStoreParameter) param;
+ List builders = getBuilders(
+ domainParameter.getConfiguration(),
+ domainParameter.getProtectionParams());
+
+ for (KeyStoreBuilderComponents builder : builders) {
+
+ try {
+
+ KeyStore.ProtectionParameter pp = builder.protection;
+ if (!(pp instanceof KeyStore.PasswordProtection)) {
+ throw new KeyStoreException(
+ new IllegalArgumentException("ProtectionParameter" +
+ " must be a KeyStore.PasswordPartection"));
+ }
+ char[] password =
+ ((KeyStore.PasswordProtection) builder.protection)
+ .getPassword();
+
+ // Store the keystores
+ KeyStore keystore = keystores.get(builder.name);
+ keystore.store(new FileOutputStream(builder.file),
+ password);
+
+ } catch (KeyStoreException e) {
+ throw new IOException(e);
+ }
+ }
+ } else {
+ throw new UnsupportedOperationException(
+ "This keystore must be stored using a " +
+ "KeyStore.DomainLoadStoreParameter");
+ }
+ }
+
+ /**
+ * Loads the keystore from the given input stream.
+ *
+ * If a password is given, it is used to check the integrity of the
+ * keystore data. Otherwise, the integrity of the keystore is not checked.
+ *
+ * @param stream the input stream from which the keystore is loaded
+ * @param password the (optional) password used to check the integrity of
+ * the keystore.
+ *
+ * @exception IOException if there is an I/O or format problem with the
+ * keystore data
+ * @exception NoSuchAlgorithmException if the algorithm used to check
+ * the integrity of the keystore cannot be found
+ * @exception CertificateException if any of the certificates in the
+ * keystore could not be loaded
+ */
+ public void engineLoad(InputStream stream, char[] password)
+ throws IOException, NoSuchAlgorithmException, CertificateException
+ {
+ // Support loading from a stream only for a JKS or default type keystore
+ try {
+ KeyStore keystore = null;
+
+ try {
+ keystore = KeyStore.getInstance("JKS");
+ keystore.load(stream, password);
+
+ } catch (Exception e) {
+ // Retry
+ if (!"JKS".equalsIgnoreCase(DEFAULT_KEYSTORE_TYPE)) {
+ keystore = KeyStore.getInstance(DEFAULT_KEYSTORE_TYPE);
+ keystore.load(stream, password);
+ } else {
+ throw e;
+ }
+ }
+ String keystoreName = DEFAULT_STREAM_PREFIX + streamCounter++;
+ keystores.put(keystoreName, keystore);
+
+ } catch (Exception e) {
+ throw new UnsupportedOperationException(
+ "This keystore must be loaded using a " +
+ "KeyStore.DomainLoadStoreParameter");
+ }
+ }
+
+ @Override
+ public void engineLoad(KeyStore.LoadStoreParameter param)
+ throws IOException, NoSuchAlgorithmException, CertificateException
+ {
+ if (param instanceof KeyStore.DomainLoadStoreParameter) {
+ KeyStore.DomainLoadStoreParameter domainParameter =
+ (KeyStore.DomainLoadStoreParameter) param;
+ List builders = getBuilders(
+ domainParameter.getConfiguration(),
+ domainParameter.getProtectionParams());
+
+ for (KeyStoreBuilderComponents builder : builders) {
+
+ try {
+ // Load the keystores (file-based and non-file-based)
+ if (builder.file != null) {
+ keystores.put(builder.name,
+ KeyStore.Builder.newInstance(builder.type,
+ builder.provider, builder.file,
+ builder.protection)
+ .getKeyStore());
+ } else {
+ keystores.put(builder.name,
+ KeyStore.Builder.newInstance(builder.type,
+ builder.provider, builder.protection)
+ .getKeyStore());
+ }
+ } catch (KeyStoreException e) {
+ throw new IOException(e);
+ }
+ }
+ } else {
+ throw new UnsupportedOperationException(
+ "This keystore must be loaded using a " +
+ "KeyStore.DomainLoadStoreParameter");
+ }
+ }
+
+ /*
+ * Parse a keystore domain configuration file and associated collection
+ * of keystore passwords to create a collection of KeyStore.Builder.
+ */
+ private List getBuilders(URI configuration,
+ Map passwords)
+ throws IOException {
+
+ PolicyParser parser = new PolicyParser(true); // expand properties
+ Collection domains = null;
+ List builders = new ArrayList<>();
+ String uriDomain = configuration.getFragment();
+
+ try (InputStreamReader configurationReader =
+ new InputStreamReader(
+ PolicyUtil.getInputStream(configuration.toURL()), "UTF-8")) {
+ parser.read(configurationReader);
+ domains = parser.getDomainEntries();
+
+ } catch (MalformedURLException mue) {
+ throw new IOException(mue);
+
+ } catch (PolicyParser.ParsingException pe) {
+ throw new IOException(pe);
+ }
+
+ for (PolicyParser.DomainEntry domain : domains) {
+ Map domainProperties = domain.getProperties();
+
+ if (uriDomain != null &&
+ (!uriDomain.equalsIgnoreCase(domain.getName()))) {
+ continue; // skip this domain
+ }
+
+ if (domainProperties.containsKey(ENTRY_NAME_SEPARATOR)) {
+ this.entryNameSeparator =
+ domainProperties.get(ENTRY_NAME_SEPARATOR);
+ // escape any regex meta characters
+ char ch = 0;
+ StringBuilder s = new StringBuilder();
+ for (int i = 0; i < this.entryNameSeparator.length(); i++) {
+ ch = this.entryNameSeparator.charAt(i);
+ if (REGEX_META.indexOf(ch) != -1) {
+ s.append('\\');
+ }
+ s.append(ch);
+ }
+ this.entryNameSeparatorRegEx = s.toString();
+ }
+
+ Collection keystores =
+ domain.getEntries();
+ for (PolicyParser.KeyStoreEntry keystore : keystores) {
+ String keystoreName = keystore.getName();
+ Map properties =
+ new HashMap<>(domainProperties);
+ properties.putAll(keystore.getProperties());
+
+ String keystoreType = DEFAULT_KEYSTORE_TYPE;
+ if (properties.containsKey(KEYSTORE_TYPE)) {
+ keystoreType = properties.get(KEYSTORE_TYPE);
+ }
+
+ Provider keystoreProvider = null;
+ if (properties.containsKey(KEYSTORE_PROVIDER_NAME)) {
+ String keystoreProviderName =
+ properties.get(KEYSTORE_PROVIDER_NAME);
+ keystoreProvider =
+ Security.getProvider(keystoreProviderName);
+ if (keystoreProvider == null) {
+ throw new IOException("Error locating JCE provider: " +
+ keystoreProviderName);
+ }
+ }
+
+ File keystoreFile = null;
+ if (properties.containsKey(KEYSTORE_URI)) {
+ String uri = properties.get(KEYSTORE_URI);
+
+ try {
+ if (uri.startsWith("file://")) {
+ keystoreFile = new File(new URI(uri));
+ } else {
+ keystoreFile = new File(uri);
+ }
+
+ } catch (URISyntaxException | IllegalArgumentException e) {
+ throw new IOException(
+ "Error processing keystore property: " +
+ "keystoreURI=\"" + uri + "\"", e);
+ }
+ }
+
+ KeyStore.ProtectionParameter keystoreProtection = null;
+ if (passwords.containsKey(keystoreName)) {
+ keystoreProtection = passwords.get(keystoreName);
+
+ } else if (properties.containsKey(KEYSTORE_PASSWORD_ENV)) {
+ String env = properties.get(KEYSTORE_PASSWORD_ENV);
+ String pwd = System.getenv(env);
+ if (pwd != null) {
+ keystoreProtection =
+ new KeyStore.PasswordProtection(pwd.toCharArray());
+ } else {
+ throw new IOException(
+ "Error processing keystore property: " +
+ "keystorePasswordEnv=\"" + env + "\"");
+ }
+ } else {
+ keystoreProtection = new KeyStore.PasswordProtection(null);
+ }
+
+ builders.add(new KeyStoreBuilderComponents(keystoreName,
+ keystoreType, keystoreProvider, keystoreFile,
+ keystoreProtection));
+ }
+ break; // skip other domains
+ }
+ if (builders.isEmpty()) {
+ throw new IOException("Error locating domain configuration data " +
+ "for: " + configuration);
+ }
+
+ return builders;
+ }
+
+/*
+ * Utility class that holds the components used to construct a KeyStore.Builder
+ */
+class KeyStoreBuilderComponents {
+ String name;
+ String type;
+ Provider provider;
+ File file;
+ KeyStore.ProtectionParameter protection;
+
+ KeyStoreBuilderComponents(String name, String type, Provider provider,
+ File file, KeyStore.ProtectionParameter protection) {
+ this.name = name;
+ this.type = type;
+ this.provider = provider;
+ this.file = file;
+ this.protection = protection;
+ }
+}
+}
diff --git a/src/share/classes/sun/security/provider/PolicyParser.java b/src/share/classes/sun/security/provider/PolicyParser.java
index b5247c76f9a61813499461f1a94ffa48e7d72322..b13345f7c559fb166dff1180d32573314a6aba45 100644
--- a/src/share/classes/sun/security/provider/PolicyParser.java
+++ b/src/share/classes/sun/security/provider/PolicyParser.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 1997, 2012, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 1997, 2013, 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
@@ -32,12 +32,7 @@ import java.net.URL;
import java.security.GeneralSecurityException;
import java.security.Principal;
import java.text.MessageFormat;
-import java.util.Enumeration;
-import java.util.Hashtable;
-import java.util.Iterator;
-import java.util.LinkedList;
-import java.util.Vector;
-import java.util.StringTokenizer;
+import java.util.*;
import javax.security.auth.x500.X500Principal;
import sun.security.util.Debug;
@@ -97,6 +92,7 @@ public class PolicyParser {
private Vector grantEntries;
+ private Map domainEntries;
// Convenience variables for parsing
private static final Debug debug = Debug.getInstance("parser",
@@ -195,9 +191,10 @@ public class PolicyParser {
*/
lookahead = st.nextToken();
+ GrantEntry ge = null;
while (lookahead != StreamTokenizer.TT_EOF) {
if (peek("grant")) {
- GrantEntry ge = parseGrantEntry();
+ ge = parseGrantEntry();
// could be null if we couldn't expand a property
if (ge != null)
add(ge);
@@ -209,6 +206,24 @@ public class PolicyParser {
// only one keystore passwordURL per policy file, others will be
// ignored
parseStorePassURL();
+ } else if (ge == null && keyStoreUrlString == null &&
+ storePassURL == null && peek("domain")) {
+ if (domainEntries == null) {
+ domainEntries = new TreeMap<>();
+ }
+ DomainEntry de = parseDomainEntry();
+ if (de != null) {
+ String domainName = de.getName();
+ if (!domainEntries.containsKey(domainName)) {
+ domainEntries.put(domainName, de);
+ } else {
+ MessageFormat form =
+ new MessageFormat(ResourcesMgr.getString(
+ "duplicate.keystore.domain.name"));
+ Object[] source = {domainName};
+ throw new ParsingException(form.format(source));
+ }
+ }
} else {
// error?
}
@@ -304,6 +319,10 @@ public class PolicyParser {
return grantEntries.elements();
}
+ public Collection getDomainEntries() {
+ return domainEntries.values();
+ }
+
/**
* write out the policy
*/
@@ -633,6 +652,67 @@ public class PolicyParser {
return e;
}
+ /**
+ * parse a domain entry
+ */
+ private DomainEntry parseDomainEntry()
+ throws ParsingException, IOException
+ {
+ boolean ignoreEntry = false;
+ DomainEntry domainEntry;
+ String name = null;
+ Map properties = new HashMap<>();
+
+ match("domain");
+ name = match("domain name");
+
+ while(!peek("{")) {
+ // get the domain properties
+ properties = parseProperties("{");
+ }
+ match("{");
+ domainEntry = new DomainEntry(name, properties);
+
+ while(!peek("}")) {
+
+ match("keystore");
+ name = match("keystore name");
+ // get the keystore properties
+ if (!peek("}")) {
+ properties = parseProperties(";");
+ }
+ match(";");
+ domainEntry.add(new KeyStoreEntry(name, properties));
+ }
+ match("}");
+
+ return (ignoreEntry == true) ? null : domainEntry;
+ }
+
+ /*
+ * Return a collection of domain properties or keystore properties.
+ */
+ private Map parseProperties(String terminator)
+ throws ParsingException, IOException {
+
+ Map properties = new HashMap<>();
+ String key;
+ String value;
+ while (!peek(terminator)) {
+ key = match("property name");
+ match("=");
+
+ try {
+ value = expand(match("quoted string"));
+ } catch (PropertyExpander.ExpandException peee) {
+ throw new IOException(peee.getLocalizedMessage());
+ }
+ properties.put(key.toLowerCase(), value);
+ }
+
+ return properties;
+ }
+
// package-private: used by PolicyFile for static policy
static String[] parseExtDirs(String codebase, int start) {
@@ -708,6 +788,10 @@ public class PolicyParser {
if (expect.equalsIgnoreCase("*"))
found = true;
break;
+ case ';':
+ if (expect.equalsIgnoreCase(";"))
+ found = true;
+ break;
default:
}
@@ -739,6 +823,11 @@ public class PolicyParser {
} else if (expect.equalsIgnoreCase("principal type")) {
value = st.sval;
lookahead = st.nextToken();
+ } else if (expect.equalsIgnoreCase("domain name") ||
+ expect.equalsIgnoreCase("keystore name") ||
+ expect.equalsIgnoreCase("property name")) {
+ value = st.sval;
+ lookahead = st.nextToken();
} else {
throw new ParsingException(st.lineno(), expect,
st.sval);
@@ -788,6 +877,12 @@ public class PolicyParser {
else
throw new ParsingException(st.lineno(), expect, "*");
break;
+ case '=':
+ if (expect.equalsIgnoreCase("="))
+ lookahead = st.nextToken();
+ else
+ throw new ParsingException(st.lineno(), expect, "=");
+ break;
default:
throw new ParsingException(st.lineno(), expect,
new String(new char[] {(char)lookahead}));
@@ -1185,6 +1280,108 @@ public class PolicyParser {
}
}
+ /**
+ * Each domain entry in the keystore domain configuration file is
+ * represented by a DomainEntry object.
+ */
+ static class DomainEntry {
+ private final String name;
+ private final Map properties;
+ private final Map entries;
+
+ DomainEntry(String name, Map properties) {
+ this.name = name;
+ this.properties = properties;
+ entries = new HashMap<>();
+ }
+
+ String getName() {
+ return name;
+ }
+
+ Map getProperties() {
+ return properties;
+ }
+
+ Collection getEntries() {
+ return entries.values();
+ }
+
+ void add(KeyStoreEntry entry) throws ParsingException {
+ String keystoreName = entry.getName();
+ if (!entries.containsKey(keystoreName)) {
+ entries.put(keystoreName, entry);
+ } else {
+ MessageFormat form = new MessageFormat(ResourcesMgr.getString(
+ "duplicate.keystore.name"));
+ Object[] source = {keystoreName};
+ throw new ParsingException(form.format(source));
+ }
+ }
+
+ @Override
+ public String toString() {
+ StringBuilder s =
+ new StringBuilder("\ndomain ").append(name);
+
+ if (properties != null) {
+ for (Map.Entry property :
+ properties.entrySet()) {
+ s.append("\n ").append(property.getKey()).append('=')
+ .append(property.getValue());
+ }
+ }
+ s.append(" {\n");
+
+ if (entries != null) {
+ for (KeyStoreEntry entry : entries.values()) {
+ s.append(entry).append("\n");
+ }
+ }
+ s.append("}");
+
+ return s.toString();
+ }
+ }
+
+ /**
+ * Each keystore entry in the keystore domain configuration file is
+ * represented by a KeyStoreEntry object.
+ */
+
+ static class KeyStoreEntry {
+ private final String name;
+ private final Map properties;
+
+ KeyStoreEntry(String name, Map properties) {
+ this.name = name;
+ this.properties = properties;
+ }
+
+ String getName() {
+ return name;
+ }
+
+ Map getProperties() {
+ return properties;
+ }
+
+ @Override
+ public String toString() {
+ StringBuilder s = new StringBuilder("\n keystore ").append(name);
+ if (properties != null) {
+ for (Map.Entry property :
+ properties.entrySet()) {
+ s.append("\n ").append(property.getKey()).append('=')
+ .append(property.getValue());
+ }
+ }
+ s.append(";");
+
+ return s.toString();
+ }
+ }
+
public static class ParsingException extends GeneralSecurityException {
private static final long serialVersionUID = -4330692689482574072L;
diff --git a/src/share/classes/sun/security/provider/Sun.java b/src/share/classes/sun/security/provider/Sun.java
index 20edc86b2757e4ffdd11d9b5c6395c3d682de788..4af2be508642eb06aab369973eca20c5a870d243 100644
--- a/src/share/classes/sun/security/provider/Sun.java
+++ b/src/share/classes/sun/security/provider/Sun.java
@@ -1,5 +1,5 @@
/*
- * Copyright (c) 1996, 2011, Oracle and/or its affiliates. All rights reserved.
+ * Copyright (c) 1996, 2013, 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
@@ -40,13 +40,14 @@ public final class Sun extends Provider {
private static final String INFO = "SUN " +
"(DSA key/parameter generation; DSA signing; SHA-1, MD5 digests; " +
- "SecureRandom; X.509 certificates; JKS keystore; PKIX CertPathValidator; " +
+ "SecureRandom; X.509 certificates; JKS & DKS keystores; " +
+ "PKIX CertPathValidator; " +
"PKIX CertPathBuilder; LDAP, Collection CertStores, JavaPolicy Policy; " +
"JavaLoginConfig Configuration)";
public Sun() {
/* We are the SUN provider */
- super("SUN", 1.7, INFO);
+ super("SUN", 1.8, INFO);
// if there is no security manager installed, put directly into
// the provider. Otherwise, create a temporary map and use a
diff --git a/src/share/classes/sun/security/provider/SunEntries.java b/src/share/classes/sun/security/provider/SunEntries.java
index 3876acbf3be7193f09c26f308a3ee53f7ad16aac..daa4e96760c91b24f72d915a962d426c62d200a9 100644
--- a/src/share/classes/sun/security/provider/SunEntries.java
+++ b/src/share/classes/sun/security/provider/SunEntries.java
@@ -208,6 +208,7 @@ final class SunEntries {
map.put("KeyStore.JKS", "sun.security.provider.JavaKeyStore$JKS");
map.put("KeyStore.CaseExactJKS",
"sun.security.provider.JavaKeyStore$CaseExactJKS");
+ map.put("KeyStore.DKS", "sun.security.provider.DomainKeyStore$DKS");
/*
* Policy
diff --git a/src/share/classes/sun/security/util/Resources.java b/src/share/classes/sun/security/util/Resources.java
index ef073b0a1fd3fd4802fe1af197a17ead1ac53d18..50028264d1408b2b899f20bbd661100159be9e35 100644
--- a/src/share/classes/sun/security/util/Resources.java
+++ b/src/share/classes/sun/security/util/Resources.java
@@ -127,6 +127,8 @@ public class Resources extends java.util.ListResourceBundle {
{"multiple.Codebase.expressions",
"multiple Codebase expressions"},
{"multiple.SignedBy.expressions","multiple SignedBy expressions"},
+ {"duplicate.keystore.domain.name","duplicate keystore domain name: {0}"},
+ {"duplicate.keystore.name","duplicate keystore name: {0}"},
{"SignedBy.has.empty.alias","SignedBy has empty alias"},
{"can.not.specify.Principal.with.a.wildcard.class.without.a.wildcard.name",
"can not specify Principal with a wildcard class without a wildcard name"},
diff --git a/test/sun/security/provider/KeyStore/DKSTest.java b/test/sun/security/provider/KeyStore/DKSTest.java
new file mode 100644
index 0000000000000000000000000000000000000000..c2b6baf3a42a6a171f90106210172a291672b7dc
--- /dev/null
+++ b/test/sun/security/provider/KeyStore/DKSTest.java
@@ -0,0 +1,203 @@
+/*
+ * Copyright (c) 2013, 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.
+ */
+
+/*
+ * see ./DKSTest.sh
+ */
+
+import java.io.*;
+import java.net.*;
+import java.security.*;
+import java.security.KeyStore;
+import java.security.cert.*;
+import java.security.cert.Certificate;
+import java.util.*;
+
+// Load and store entries in domain keystores
+
+public class DKSTest {
+
+ private static final String TEST_SRC = System.getProperty("test.src");
+ private static final String CERT = TEST_SRC + "/../../pkcs12/trusted.pem";
+ private static final String CONFIG = "file://" + TEST_SRC + "/domains.cfg";
+ private static final Map PASSWORDS =
+ new HashMap() {{
+ put("keystore",
+ new KeyStore.PasswordProtection("test123".toCharArray()));
+ put("policy_keystore",
+ new KeyStore.PasswordProtection(
+ "Alias.password".toCharArray()));
+ put("pw_keystore",
+ new KeyStore.PasswordProtection("test12".toCharArray()));
+ put("eckeystore1",
+ new KeyStore.PasswordProtection("password".toCharArray()));
+ put("eckeystore2",
+ new KeyStore.PasswordProtection("password".toCharArray()));
+ put("truststore",
+ new KeyStore.PasswordProtection("changeit".toCharArray()));
+ put("empty",
+ new KeyStore.PasswordProtection("passphrase".toCharArray()));
+ }};
+
+ public static void main(String[] args) throws Exception {
+ try {
+ main0();
+ } finally {
+ // cleanup
+ new File(TEST_SRC + "/empty.jks").delete();
+ new File(TEST_SRC + "/Alias.keystore_tmp").delete();
+ new File(TEST_SRC + "/pw.jks_tmp").delete();
+ new File(TEST_SRC + "/secp256r1server-secp384r1ca.p12_tmp").delete();
+ new File(TEST_SRC + "/sect193r1server-rsa1024ca.p12_tmp").delete();
+ }
+ }
+
+ private static void main0() throws Exception {
+ /*
+ * domain keystore: system
+ */
+ URI config = new URI(CONFIG + "#system");
+ int cacertsCount;
+ int expected;
+ KeyStore keystore = KeyStore.getInstance("DKS");
+ // load entries
+ keystore.load(
+ new KeyStore.DomainLoadStoreParameter(config, PASSWORDS));
+ cacertsCount = expected = keystore.size();
+ System.out.println("\nLoading domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ checkEntries(keystore, expected);
+
+ /*
+ * domain keystore: system_plus
+ */
+ config = new URI(CONFIG + "#system_plus");
+ expected = cacertsCount + 1;
+ keystore = KeyStore.getInstance("DKS");
+ // load entries
+ keystore.load(
+ new KeyStore.DomainLoadStoreParameter(config, PASSWORDS));
+ System.out.println("\nLoading domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ checkEntries(keystore, expected);
+
+ /*
+ * domain keystore: system_env
+ */
+ config = new URI(CONFIG + "#system_env");
+ expected = 1 + cacertsCount;
+ keystore = KeyStore.getInstance("DKS");
+ // load entries
+ keystore.load(
+ new KeyStore.DomainLoadStoreParameter(config,
+ Collections.emptyMap()));
+ System.out.println("\nLoading domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ checkEntries(keystore, expected);
+
+ /*
+ * domain keystore: empty
+ */
+ KeyStore empty = KeyStore.getInstance("JKS");
+ empty.load(null, null);
+
+ try (OutputStream outStream =
+ new FileOutputStream(TEST_SRC + "/empty.jks")) {
+ empty.store(outStream, "passphrase".toCharArray());
+ }
+ config = new URI(CONFIG + "#empty");
+ expected = 0;
+ keystore = KeyStore.getInstance("DKS");
+ // load entries
+ keystore.load(
+ new KeyStore.DomainLoadStoreParameter(config, PASSWORDS));
+ System.out.println("\nLoading domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ checkEntries(keystore, expected);
+
+ /*
+ * domain keystore: keystores
+ */
+ config = new URI(CONFIG + "#keystores");
+ expected = 2 + 1 + 1 + 1;
+ keystore = KeyStore.getInstance("DKS");
+ // load entries
+ keystore.load(
+ new KeyStore.DomainLoadStoreParameter(config, PASSWORDS));
+ System.out.println("\nLoading domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ checkEntries(keystore, expected);
+ // set a new trusted certificate entry
+ Certificate cert = loadCertificate(CERT);
+ String alias = "pw_keystore tmp-cert";
+ System.out.println("Setting new trusted certificate entry: " + alias);
+ keystore.setEntry(alias,
+ new KeyStore.TrustedCertificateEntry(cert), null);
+ expected++;
+ // store entries
+ config = new URI(CONFIG + "#keystores_tmp");
+ System.out.println("Storing domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ keystore.store(
+ new KeyStore.DomainLoadStoreParameter(config, PASSWORDS));
+ keystore = KeyStore.getInstance("DKS");
+ // reload entries
+ keystore.load(
+ new KeyStore.DomainLoadStoreParameter(config, PASSWORDS));
+ System.out.println("Reloading domain keystore: " + config + "\t[" +
+ expected + " entries]");
+ checkEntries(keystore, expected);
+ // get the new trusted certificate entry
+ System.out.println("Getting new trusted certificate entry: " + alias);
+ if (!keystore.isCertificateEntry(alias)) {
+ throw new Exception("Error: cannot retrieve certificate entry: " +
+ alias);
+ }
+ keystore.setEntry(alias,
+ new KeyStore.TrustedCertificateEntry(cert), null);
+ }
+
+ private static void checkEntries(KeyStore keystore, int expected)
+ throws Exception {
+ int i = 0;
+ for (String alias : Collections.list(keystore.aliases())) {
+ System.out.print(".");
+ i++;
+ }
+ System.out.println();
+ if (expected != i) {
+ throw new Exception("Error: unexpected entry count in keystore: " +
+ "loaded=" + i + ", expected=" + expected);
+ }
+ }
+
+ private static Certificate loadCertificate(String certFile)
+ throws Exception {
+ X509Certificate cert = null;
+ try (FileInputStream certStream = new FileInputStream(certFile)) {
+ CertificateFactory factory =
+ CertificateFactory.getInstance("X.509");
+ return factory.generateCertificate(certStream);
+ }
+ }
+}
diff --git a/test/sun/security/provider/KeyStore/DKSTest.sh b/test/sun/security/provider/KeyStore/DKSTest.sh
new file mode 100644
index 0000000000000000000000000000000000000000..b789e41414011e91f9755f2359c75831fe8f4132
--- /dev/null
+++ b/test/sun/security/provider/KeyStore/DKSTest.sh
@@ -0,0 +1,84 @@
+#! /bin/sh
+
+#
+# Copyright (c) 2013, 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 8007755
+# @summary Support the logical grouping of keystores
+
+# set a few environment variables so that the shell-script can run stand-alone
+# in the source directory
+if [ "${TESTSRC}" = "" ] ; then
+ TESTSRC="."
+fi
+
+if [ "${TESTCLASSES}" = "" ] ; then
+ TESTCLASSES="."
+fi
+
+if [ "${TESTJAVA}" = "" ] ; then
+ echo "TESTJAVA not set. Test cannot execute."
+ echo "FAILED!!!"
+ exit 1
+fi
+
+if [ "${COMPILEJAVA}" = "" ]; then
+ COMPILEJAVA="${TESTJAVA}"
+fi
+
+# set platform-dependent variables
+OS=`uname -s`
+case "$OS" in
+ SunOS )
+ PS=":"
+ FS="/"
+ ;;
+ Linux )
+ PS=":"
+ FS="/"
+ ;;
+ Darwin )
+ PS=":"
+ FS="/"
+ ;;
+ CYGWIN* )
+ PS=";"
+ FS="/"
+ ;;
+ Windows* )
+ PS=";"
+ FS="\\"
+ ;;
+ * )
+ echo "Unrecognized system!"
+ exit 1;
+ ;;
+esac
+
+${COMPILEJAVA}${FS}bin${FS}javac -d . ${TESTSRC}${FS}DKSTest.java
+
+KEYSTORE_PWD=test12 TRUSTSTORE_PWD=changeit \
+ ${TESTJAVA}${FS}bin${FS}java ${TESTVMOPTS} -Dtest.src=${TESTSRC} DKSTest
+
+exit $status
diff --git a/test/sun/security/provider/KeyStore/domains.cfg b/test/sun/security/provider/KeyStore/domains.cfg
new file mode 100644
index 0000000000000000000000000000000000000000..203d9d007cad284aa08e617ba8708e4185a12d70
--- /dev/null
+++ b/test/sun/security/provider/KeyStore/domains.cfg
@@ -0,0 +1,65 @@
+// domain containing a single keystore
+domain system {
+ keystore truststore
+ keystoreType="JKS"
+ keystoreURI="${java.home}/lib/security/cacerts";
+};
+
+// domain containing two JKS keystores
+domain system_plus {
+ keystore truststore
+ keystoreType="JKS"
+ keystoreURI="${java.home}/lib/security/cacerts";
+ keystore pw_keystore
+ keystoreType="JKS"
+ keystoreURI="${test.src}/pw.jks";
+};
+
+// domain containing a mixture of keystores
+domain keystores
+ keystoreType="PKCS12" {
+ keystore policy_keystore
+ keystoreType="JKS"
+ keystoreURI="${test.src}/../PolicyFile/Alias.keystore";
+ keystore pw_keystore
+ keystoreType="CaseExactJKS"
+ keystoreURI="${test.src}/pw.jks";
+ keystore eckeystore1
+ keystoreURI="${test.src}/../../pkcs11/ec/pkcs12/sect193r1server-rsa1024ca.p12";
+ keystore eckeystore2
+ keystoreURI="${test.src}/../../pkcs11/ec/pkcs12/secp256r1server-secp384r1ca.p12";
+};
+
+// domain containing a mixture of keystores
+domain keystores_tmp
+ keystoreType="PKCS12" {
+ keystore policy_keystore
+ keystoreType="JKS"
+ keystoreURI="${test.src}/Alias.keystore_tmp";
+ keystore pw_keystore
+ keystoreType="CaseExactJKS"
+ keystoreURI="${test.src}/pw.jks_tmp";
+ keystore eckeystore1
+ keystoreURI="${test.src}/sect193r1server-rsa1024ca.p12_tmp";
+ keystore eckeystore2
+ keystoreURI="${test.src}/secp256r1server-secp384r1ca.p12_tmp";
+};
+
+// domain where passwords are supplied via environment variables
+domain system_env
+ keystoreType="JKS"
+ keystorePasswordEnv="KEYSTORE_PWD" {
+ keystore env_keystore
+ keystoreURI="${test.src}/pw.jks";
+ keystore env_truststore
+ keystoreURI="${java.home}/lib/security/cacerts"
+ keystorePasswordEnv="TRUSTSTORE_PWD";
+};
+
+// empty domain
+domain empty
+ keystoreType="JKS"
+ keystoreProviderName="SUN" {
+ keystore empty
+ keystoreURI="${test.src}/empty.jks";
+};
diff --git a/test/sun/security/tools/keytool/AltProviderPath.sh b/test/sun/security/tools/keytool/AltProviderPath.sh
index 067a5eb7bf09de173b32e7df075887b1ad26c5d2..82bc65787abde6e1f8af4fa90ecdda72c7bdf43b 100644
--- a/test/sun/security/tools/keytool/AltProviderPath.sh
+++ b/test/sun/security/tools/keytool/AltProviderPath.sh
@@ -73,7 +73,7 @@ ${TESTJAVA}${FS}bin${FS}keytool -genkey -v -alias dummyTestCA \
-keyalg "RSA" -keysize 1024 -sigalg "ShA1WithRSA" \
-dname "cn=Dummy Test CA, ou=JSN, o=JavaSoft, c=US" -validity 3650 \
-keypass storepass -keystore keystoreCA.dks -storepass storepass \
- -storetype "dks" -provider "org.test.dummy.DummyProvider" \
+ -storetype "dummyks" -provider "org.test.dummy.DummyProvider" \
-providerPath ${TESTCLASSES}
if [ $? -ne 0 ]; then
@@ -82,7 +82,7 @@ fi
#Change keystore password
${TESTJAVA}${FS}bin${FS}keytool -storepasswd -new storepass2 \
- -keystore keystoreCA.dks -storetype "dks" -storepass storepass \
+ -keystore keystoreCA.dks -storetype "dummyks" -storepass storepass \
-provider "org.test.dummy.DummyProvider" -providerPath ${TESTCLASSES}
if [ $? -ne 0 ]; then
@@ -93,7 +93,7 @@ fi
#Change keystore key password
${TESTJAVA}${FS}bin${FS}keytool -keypasswd -alias "dummyTestCA" \
-keypass storepass -new keypass -keystore keystoreCA.dks \
- -storetype "dks" -storepass storepass2 \
+ -storetype "dummyks" -storepass storepass2 \
-provider "org.test.dummy.DummyProvider" -providerPath ${TESTCLASSES}
if [ $? -ne 0 ]; then
@@ -102,7 +102,7 @@ fi
#Export certificate
${TESTJAVA}${FS}bin${FS}keytool -v -export -rfc -alias "dummyTestCA" \
- -file "dummyTestCA.der" -keystore keystoreCA.dks -storetype "dks" \
+ -file "dummyTestCA.der" -keystore keystoreCA.dks -storetype "dummyks" \
-storepass storepass2 -provider "org.test.dummy.DummyProvider" \
-providerPath ${TESTCLASSES}
@@ -112,7 +112,7 @@ fi
#list keystore
${TESTJAVA}${FS}bin${FS}keytool -v -list -keystore keystoreCA.dks \
- -storetype "dks" -storepass storepass2 \
+ -storetype "dummyks" -storepass storepass2 \
-provider "org.test.dummy.DummyProvider" -providerPath ${TESTCLASSES}
if [ $? -ne 0 ]; then
diff --git a/test/sun/security/tools/keytool/DummyProvider.java b/test/sun/security/tools/keytool/DummyProvider.java
index ae714702051cde564bb3a5d09fbc52510bcceac5..0c7ce10d5873d3b76937825c1627037e3aa8c110 100644
--- a/test/sun/security/tools/keytool/DummyProvider.java
+++ b/test/sun/security/tools/keytool/DummyProvider.java
@@ -40,7 +40,7 @@ public class DummyProvider extends Provider {
//
// KeyStore
//
- put("KeyStore.DKS", "sun.security.provider.JavaKeyStore$JKS");
+ put("KeyStore.DummyKS", "sun.security.provider.JavaKeyStore$JKS");
//
// Signature engines