aboutsummaryrefslogtreecommitdiff
path: root/src/main/model/csr/Values.java
blob: 5c1e212d16a251c200b2acd7431fd065295ead28 (plain)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
package model.csr;

import model.asn1.ASN1Object;
import model.asn1.Encodable;
import model.asn1.Tag;
import model.asn1.exceptions.ParseException;
import model.asn1.parsing.BytesReader;

import java.util.ArrayList;
import java.util.Arrays;
import java.util.List;
import java.util.stream.Stream;

/**
 * Represents a CSR attribute values list.
 * <pre>
 *   Attribute { ATTRIBUTE:IOSet } ::= SEQUENCE {
 *       type   ATTRIBUTE.&id({IOSet}),
 *       values SET SIZE(1..MAX) OF ATTRIBUTE.&Type({IOSet}{@type})
 *   }
 * </pre>
 * Values can be none or any length. Parsing and decoding the values are handled in specific types.
 */
public class Values extends ASN1Object {
    private final ASN1Object[] array;

    /**
     * EFFECT: Initialize the list with the given tag, parentTag, and array. For tag and parentTag, consult
     * {@link ASN1Object}.
     * REQUIRES: All elements in the array shall be the same ASN.1 type.
     */
    public Values(Tag tag, Tag parentTag, ASN1Object[] array) {
        super(tag, parentTag);
        this.array = array;
    }

    /**
     * EFFECT: Parse the list from input DER bytes. For details on parsing, refer to {@link ASN1Object}.
     *   Throws {@link ParseException} for invalid input.
     * MODIFIES: this, encoded
     */
    public Values(BytesReader encoded, boolean hasParentTag) throws ParseException {
        super(encoded, hasParentTag);
        final List<ASN1Object> list = new ArrayList<>();
        for (int i = 0; i < getLength();) {
            int index = encoded.getIndex();
            final ASN1Object value = ASN1Object.parse(encoded, false);
            list.add(value);
            index = encoded.getIndex() - index;
            i += index;
        }
        this.array = list.toArray(new ASN1Object[0]);
    }

    /**
     * EFFECTS: Encode the SET OF into DER, keep order. Values will be encoded one-by-one.
     */
    @Override
    public Byte[] encodeValueDER() {
        return Stream.of(array)
                .map(Encodable::encodeDER)
                .flatMap(Arrays::stream)
                .toArray(Byte[]::new);
    }

    public ASN1Object[] getArray() {
        return array;
    }
}