All Downloads are FREE. Search and download functionalities are using the official Maven repository.

com.authlete.cose.COSERecipient Maven / Gradle / Ivy

The newest version!
/*
 * Copyright (C) 2023 Authlete, Inc.
 *
 * Licensed under the Apache License, Version 2.0 (the "License");
 * you may not use this file except in compliance with the License.
 * You may obtain a copy of the License at
 *
 *     https://www.apache.org/licenses/LICENSE-2.0
 *
 * Unless required by applicable law or agreed to in writing, software
 * distributed under the License is distributed on an "AS IS" BASIS,
 * WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
 * See the License for the specific language governing permissions and
 * limitations under the License.
 */
package com.authlete.cose;


import java.util.ArrayList;
import java.util.List;
import com.authlete.cbor.CBORByteArray;
import com.authlete.cbor.CBORItem;
import com.authlete.cbor.CBORItemList;
import com.authlete.cbor.CBORNull;
import com.authlete.cbor.CBORizer;


/**
 * COSE_recipient
 *
 * 

* {@code COSE_recipient} is defined in 5.1. Enveloped * COSE Structure of RFC 9052 as follows. *

* *
 * COSE_recipient = [
 *     Headers,
 *     ciphertext : bstr / nil,
 *     ? recipients : [+COSE_recipient]
 * ]
 * 
* * @since 1.1 * * @see RFC 9052, 5.1. Enveloped COSE Structure */ public class COSERecipient extends COSEObject { /** * A constructor with a protected header, an unprotected header and * a cipher text. * * @param protectedHeader * A protected header. Must not be null. * * @param unprotectedHeader * An unprotected header. Must not be null. * * @param ciphertext * A cipher text. Must be either {@link CBORByteArray} or * {@link CBORNull}. */ public COSERecipient( COSEProtectedHeader protectedHeader, COSEUnprotectedHeader unprotectedHeader, CBORItem ciphertext) { super(protectedHeader, unprotectedHeader, ciphertext); } /** * A constructor with a protected header, an unprotected header, a * cipher text and recipients. * * @param protectedHeader * A protected header. Must not be null. * * @param unprotectedHeader * An unprotected header. Must not be null. * * @param ciphertext * A cipher text. Must be either {@link CBORByteArray} or * {@link CBORNull}. * * @param recipients * Recipients. Must not be null. Items in the list must be * instances of {@link COSERecipient}. At least one element * must be contained. */ public COSERecipient( COSEProtectedHeader protectedHeader, COSEUnprotectedHeader unprotectedHeader, CBORItem ciphertext, CBORItemList recipients) { super(protectedHeader, unprotectedHeader, ciphertext, recipients); validateRecipients(recipients); } private static void validateRecipients(CBORItemList recipients) { if (recipients == null) { throw new IllegalArgumentException( "A recipient list given to COSERecipient's constructor must not be null."); } List items = recipients.getItems(); if (items == null || items.size() == 0) { throw new IllegalArgumentException( "A recipient list given to COSERecipient's constructor must not be empty."); } for (CBORItem item : items) { if (item instanceof COSERecipient) { continue; } throw new IllegalArgumentException( "Items in the recipient list given to COSERecipient's constructor must be instances of COSERecipient."); } } /** * Get the cipher text. * *

* The type of the value is {@link CBORByteArray} or {@link CBORNull}. *

* * @return * The cipher text. */ public CBORItem getCiphertext() { return getItems().get(2); } /** * Get the list of recipients. Items in the list are {@link COSERecipient} * instances. * * @return * The list of recipients. {@code null} is returned if this * instance was created without recipients. */ public CBORItemList getRecipients() { if (getItems().size() <= 3) { return null; } return (CBORItemList)getItems().get(3); } /** * Interpret the given CBOR data item as a recipient and build an instance * of {@link COSERecipient} from it. * * @param item * A CBOR data item that represents a recipient. It must be a CBOR * array. * * @return * A recipient. * * @throws COSEException * The given CBOR data item does not conform to the requirements * of recipient. */ public static COSERecipient build(CBORItem item) throws COSEException { // Build the protected header, the unprotected header and the content. List common = COSEObject.buildCommon(item, "COSE_recipient"); // If the COSE recipient does not have the optional 'recipients'. if (common.size() == 3) { // Build a COSERecipient instance without recipients. return new COSERecipient( (COSEProtectedHeader)common.get(0), (COSEUnprotectedHeader)common.get(1), common.get(2)); } // The fourth element of COSE_recipient, which is a list of recipients. CBORItem fourth = common.get(3); // If the fourth element is not a CBOR array. if (!(fourth instanceof CBORItemList)) { throw new COSEException( "The fourth element (recipients) of COSE_recipient must be a CBOR array."); } // Recipients CBORItemList recipients = buildRecipients((CBORItemList)fourth); try { // Build a COSERecipient instance with recipients. return new COSERecipient( (COSEProtectedHeader)common.get(0), (COSEUnprotectedHeader)common.get(1), common.get(2), recipients); } catch (Exception cause) { throw new COSEException(cause.getMessage(), cause); } } private static CBORItemList buildRecipients(CBORItemList list) throws COSEException { List items = list.getItems(); if (items == null || items.size() == 0) { throw new COSEException( "A list of recipients in COSE_recipient must contain at least one recipient."); } List recipients = new ArrayList<>(); for (CBORItem item : items) { // Interpret the CBOR data item as a COSE_recipient. ; recursive call COSERecipient recipient = build(item); // Add the recipient. recipients.add(recipient); } return new CBORItemList(recipients); } /** * Build a {@code COSERecipient} instance from the given object list. * * @param list * A object list that represents {@code COSE_recipient}. * * @return * A {@code COSERecipient} instance built from the given object list. * * @throws COSEException */ public static COSERecipient build(List list) throws COSEException { return build(new CBORizer().cborizeCollection(list)); } }