jdk/src/java.base/share/classes/java/security/PKCS12Attribute.java
author coleenp
Wed, 30 Aug 2017 19:18:22 -0400
changeset 47098 e704f55561c3
parent 25859 3317bb8137f4
child 47002 e82e52e3e566
permissions -rw-r--r--
8164207: Checking missing load-acquire in relation to _pd_set in dictionary.cpp Summary: Use load_acquire for accessing DictionaryEntry::_pd_set since it's accessed outside the SystemDictionary_lock Reviewed-by: zgu, twisti, dholmes, adinn
Ignore whitespace changes - Everywhere: Within whitespace: At end of lines:
15298
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     1
/*
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     2
 * Copyright (c) 2013, Oracle and/or its affiliates. All rights reserved.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     3
 * DO NOT ALTER OR REMOVE COPYRIGHT NOTICES OR THIS FILE HEADER.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     4
 *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     5
 * This code is free software; you can redistribute it and/or modify it
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     6
 * under the terms of the GNU General Public License version 2 only, as
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     7
 * published by the Free Software Foundation.  Oracle designates this
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     8
 * particular file as subject to the "Classpath" exception as provided
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
     9
 * by Oracle in the LICENSE file that accompanied this code.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    10
 *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    11
 * This code is distributed in the hope that it will be useful, but WITHOUT
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    12
 * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    13
 * FITNESS FOR A PARTICULAR PURPOSE.  See the GNU General Public License
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    14
 * version 2 for more details (a copy is included in the LICENSE file that
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    15
 * accompanied this code).
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    16
 *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    17
 * You should have received a copy of the GNU General Public License version
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    18
 * 2 along with this work; if not, write to the Free Software Foundation,
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    19
 * Inc., 51 Franklin St, Fifth Floor, Boston, MA 02110-1301 USA.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    20
 *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    21
 * Please contact Oracle, 500 Oracle Parkway, Redwood Shores, CA 94065 USA
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    22
 * or visit www.oracle.com if you need additional information or have any
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    23
 * questions.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    24
 */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    25
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    26
package java.security;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    27
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    28
import java.io.IOException;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    29
import java.math.BigInteger;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    30
import java.util.Arrays;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    31
import java.util.regex.Pattern;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    32
import sun.security.util.*;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    33
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    34
/**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    35
 * An attribute associated with a PKCS12 keystore entry.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    36
 * The attribute name is an ASN.1 Object Identifier and the attribute
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    37
 * value is a set of ASN.1 types.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    38
 *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    39
 * @since 1.8
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    40
 */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    41
public final class PKCS12Attribute implements KeyStore.Entry.Attribute {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    42
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    43
    private static final Pattern COLON_SEPARATED_HEX_PAIRS =
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    44
        Pattern.compile("^[0-9a-fA-F]{2}(:[0-9a-fA-F]{2})+$");
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    45
    private String name;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    46
    private String value;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    47
    private byte[] encoded;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    48
    private int hashValue = -1;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    49
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    50
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    51
     * Constructs a PKCS12 attribute from its name and value.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    52
     * The name is an ASN.1 Object Identifier represented as a list of
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    53
     * dot-separated integers.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    54
     * A string value is represented as the string itself.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    55
     * A binary value is represented as a string of colon-separated
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    56
     * pairs of hexadecimal digits.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    57
     * Multi-valued attributes are represented as a comma-separated
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    58
     * list of values, enclosed in square brackets. See
15303
d365a2901ccb 8006863: javadoc cleanup for 8005408
vinnie
parents: 15298
diff changeset
    59
     * {@link Arrays#toString(java.lang.Object[])}.
15298
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    60
     * <p>
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    61
     * A string value will be DER-encoded as an ASN.1 UTF8String and a
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    62
     * binary value will be DER-encoded as an ASN.1 Octet String.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    63
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    64
     * @param name the attribute's identifier
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    65
     * @param value the attribute's value
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    66
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    67
     * @exception NullPointerException if {@code name} or {@code value}
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    68
     *     is {@code null}
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    69
     * @exception IllegalArgumentException if {@code name} or
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    70
     *     {@code value} is incorrectly formatted
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    71
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    72
    public PKCS12Attribute(String name, String value) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    73
        if (name == null || value == null) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    74
            throw new NullPointerException();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    75
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    76
        // Validate name
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    77
        ObjectIdentifier type;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    78
        try {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    79
            type = new ObjectIdentifier(name);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    80
        } catch (IOException e) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    81
            throw new IllegalArgumentException("Incorrect format: name", e);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    82
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    83
        this.name = name;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    84
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    85
        // Validate value
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    86
        int length = value.length();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    87
        String[] values;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    88
        if (value.charAt(0) == '[' && value.charAt(length - 1) == ']') {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    89
            values = value.substring(1, length - 1).split(", ");
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    90
        } else {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    91
            values = new String[]{ value };
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    92
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    93
        this.value = value;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    94
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    95
        try {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    96
            this.encoded = encode(type, values);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    97
        } catch (IOException e) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    98
            throw new IllegalArgumentException("Incorrect format: value", e);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
    99
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   100
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   101
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   102
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   103
     * Constructs a PKCS12 attribute from its ASN.1 DER encoding.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   104
     * The DER encoding is specified by the following ASN.1 definition:
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   105
     * <pre>
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   106
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   107
     * Attribute ::= SEQUENCE {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   108
     *     type   AttributeType,
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   109
     *     values SET OF AttributeValue
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   110
     * }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   111
     * AttributeType ::= OBJECT IDENTIFIER
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   112
     * AttributeValue ::= ANY defined by type
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   113
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   114
     * </pre>
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   115
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   116
     * @param encoded the attribute's ASN.1 DER encoding. It is cloned
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   117
     *     to prevent subsequent modificaion.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   118
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   119
     * @exception NullPointerException if {@code encoded} is
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   120
     *     {@code null}
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   121
     * @exception IllegalArgumentException if {@code encoded} is
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   122
     *     incorrectly formatted
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   123
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   124
    public PKCS12Attribute(byte[] encoded) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   125
        if (encoded == null) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   126
            throw new NullPointerException();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   127
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   128
        this.encoded = encoded.clone();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   129
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   130
        try {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   131
            parse(encoded);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   132
        } catch (IOException e) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   133
            throw new IllegalArgumentException("Incorrect format: encoded", e);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   134
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   135
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   136
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   137
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   138
     * Returns the attribute's ASN.1 Object Identifier represented as a
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   139
     * list of dot-separated integers.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   140
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   141
     * @return the attribute's identifier
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   142
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   143
    @Override
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   144
    public String getName() {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   145
        return name;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   146
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   147
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   148
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   149
     * Returns the attribute's ASN.1 DER-encoded value as a string.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   150
     * An ASN.1 DER-encoded value is returned in one of the following
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   151
     * {@code String} formats:
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   152
     * <ul>
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   153
     * <li> the DER encoding of a basic ASN.1 type that has a natural
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   154
     *      string representation is returned as the string itself.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   155
     *      Such types are currently limited to BOOLEAN, INTEGER,
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   156
     *      OBJECT IDENTIFIER, UTCTime, GeneralizedTime and the
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   157
     *      following six ASN.1 string types: UTF8String,
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   158
     *      PrintableString, T61String, IA5String, BMPString and
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   159
     *      GeneralString.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   160
     * <li> the DER encoding of any other ASN.1 type is not decoded but
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   161
     *      returned as a binary string of colon-separated pairs of
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   162
     *      hexadecimal digits.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   163
     * </ul>
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   164
     * Multi-valued attributes are represented as a comma-separated
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   165
     * list of values, enclosed in square brackets. See
15303
d365a2901ccb 8006863: javadoc cleanup for 8005408
vinnie
parents: 15298
diff changeset
   166
     * {@link Arrays#toString(java.lang.Object[])}.
15298
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   167
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   168
     * @return the attribute value's string encoding
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   169
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   170
    @Override
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   171
    public String getValue() {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   172
        return value;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   173
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   174
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   175
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   176
     * Returns the attribute's ASN.1 DER encoding.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   177
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   178
     * @return a clone of the attribute's DER encoding
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   179
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   180
    public byte[] getEncoded() {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   181
        return encoded.clone();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   182
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   183
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   184
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   185
     * Compares this {@code PKCS12Attribute} and a specified object for
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   186
     * equality.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   187
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   188
     * @param obj the comparison object
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   189
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   190
     * @return true if {@code obj} is a {@code PKCS12Attribute} and
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   191
     * their DER encodings are equal.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   192
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   193
    @Override
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   194
    public boolean equals(Object obj) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   195
        if (this == obj) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   196
            return true;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   197
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   198
        if (!(obj instanceof PKCS12Attribute)) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   199
            return false;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   200
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   201
        return Arrays.equals(encoded, ((PKCS12Attribute) obj).getEncoded());
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   202
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   203
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   204
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   205
     * Returns the hashcode for this {@code PKCS12Attribute}.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   206
     * The hash code is computed from its DER encoding.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   207
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   208
     * @return the hash code
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   209
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   210
    @Override
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   211
    public int hashCode() {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   212
        if (hashValue == -1) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   213
            Arrays.hashCode(encoded);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   214
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   215
        return hashValue;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   216
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   217
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   218
    /**
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   219
     * Returns a string representation of this {@code PKCS12Attribute}.
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   220
     *
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   221
     * @return a name/value pair separated by an 'equals' symbol
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   222
     */
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   223
    @Override
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   224
    public String toString() {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   225
        return (name + "=" + value);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   226
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   227
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   228
    private byte[] encode(ObjectIdentifier type, String[] values)
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   229
            throws IOException {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   230
        DerOutputStream attribute = new DerOutputStream();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   231
        attribute.putOID(type);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   232
        DerOutputStream attrContent = new DerOutputStream();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   233
        for (String value : values) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   234
            if (COLON_SEPARATED_HEX_PAIRS.matcher(value).matches()) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   235
                byte[] bytes =
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   236
                    new BigInteger(value.replace(":", ""), 16).toByteArray();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   237
                if (bytes[0] == 0) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   238
                    bytes = Arrays.copyOfRange(bytes, 1, bytes.length);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   239
                }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   240
                attrContent.putOctetString(bytes);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   241
            } else {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   242
                attrContent.putUTF8String(value);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   243
            }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   244
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   245
        attribute.write(DerValue.tag_Set, attrContent);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   246
        DerOutputStream attributeValue = new DerOutputStream();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   247
        attributeValue.write(DerValue.tag_Sequence, attribute);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   248
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   249
        return attributeValue.toByteArray();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   250
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   251
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   252
    private void parse(byte[] encoded) throws IOException {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   253
        DerInputStream attributeValue = new DerInputStream(encoded);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   254
        DerValue[] attrSeq = attributeValue.getSequence(2);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   255
        ObjectIdentifier type = attrSeq[0].getOID();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   256
        DerInputStream attrContent =
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   257
            new DerInputStream(attrSeq[1].toByteArray());
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   258
        DerValue[] attrValueSet = attrContent.getSet(1);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   259
        String[] values = new String[attrValueSet.length];
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   260
        String printableString;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   261
        for (int i = 0; i < attrValueSet.length; i++) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   262
            if (attrValueSet[i].tag == DerValue.tag_OctetString) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   263
                values[i] = Debug.toString(attrValueSet[i].getOctetString());
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   264
            } else if ((printableString = attrValueSet[i].getAsString())
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   265
                != null) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   266
                values[i] = printableString;
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   267
            } else if (attrValueSet[i].tag == DerValue.tag_ObjectId) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   268
                values[i] = attrValueSet[i].getOID().toString();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   269
            } else if (attrValueSet[i].tag == DerValue.tag_GeneralizedTime) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   270
                values[i] = attrValueSet[i].getGeneralizedTime().toString();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   271
            } else if (attrValueSet[i].tag == DerValue.tag_UtcTime) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   272
                values[i] = attrValueSet[i].getUTCTime().toString();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   273
            } else if (attrValueSet[i].tag == DerValue.tag_Integer) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   274
                values[i] = attrValueSet[i].getBigInteger().toString();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   275
            } else if (attrValueSet[i].tag == DerValue.tag_Boolean) {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   276
                values[i] = String.valueOf(attrValueSet[i].getBoolean());
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   277
            } else {
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   278
                values[i] = Debug.toString(attrValueSet[i].getDataBytes());
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   279
            }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   280
        }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   281
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   282
        this.name = type.toString();
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   283
        this.value = values.length == 1 ? values[0] : Arrays.toString(values);
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   284
    }
06867bfe82ac 8005408: KeyStore API enhancements
vinnie
parents:
diff changeset
   285
}