Message.java

  1. /*******************************************************************************

  2.  * Original from https://github.com/cose-wg/COSE-JAVA Commit 1a20373
  3.  *
  4.  * Copyright (c) 2016, Jim Schaad
  5.  * Copyright (c) 2018, Ludwig Seitz, RISE SICS
  6.  * Copyright (c) 2018, Rikard Höglund, RISE SICS
  7.  * All rights reserved.

  8.  * Redistribution and use in source and binary forms, with or without
  9.  * modification, are permitted provided that the following conditions are met:

  10.  * Redistributions of source code must retain the above copyright notice,
  11.  * this list of conditions and the following disclaimer.

  12.  * Redistributions in binary form must reproduce the above copyright notice,
  13.  * this list of conditions and the following disclaimer in the documentation
  14.  * and/or other materials provided with the distribution.

  15.  * Neither the name of COSE-JAVA nor the names of its
  16.  * contributors may be used to endorse or promote products derived from
  17.  * this software without specific prior written permission.

  18.  * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS"
  19.  * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE
  20.  * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR
  21.  * PURPOSE ARE
  22.  * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE
  23.  * FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL
  24.  * DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR
  25.  * SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER
  26.  * CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY,
  27.  * OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF
  28.  * THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
  29.  *
  30.  * Contributors:
  31.  *    Achim Kraus (Bosch Software Innovations GmbH) - update to cbor 4.0.0
  32.  *                                                    align with cose 1.0
  33.  *                                                    commit 629912b94ea80c4c6
  34.  ******************************************************************************/
  35. package org.eclipse.californium.cose;

  36. import com.upokecenter.cbor.CBORObject;
  37. import com.upokecenter.cbor.CBORType;

  38. import java.util.ArrayList;
  39. import java.util.List;

  40. import org.eclipse.californium.elements.util.StandardCharsets;

  41. /**
  42.  * The Message class provides a common class that all of the COSE message classes
  43.  * inherit from.  It provides the function used for decoding all of the known
  44.  * messages.
  45.  *
  46.  * @author jimsch
  47.  */
  48. public abstract class Message extends Attribute {
  49.     /**
  50.      * Is the tag identifying the message emitted?
  51.      */
  52.     protected boolean emitTag = true;
  53.    
  54.     /**
  55.      * Is the content emitted as part of the message?
  56.      */
  57.     protected boolean emitContent = true;
  58.    
  59.     /**
  60.      * What message tag identifies this message?
  61.     */
  62.     protected MessageTag messageTag = MessageTag.Unknown;
  63.    
  64.     /**
  65.      * What is the plain text content of the message.
  66.      */
  67.     protected byte[] rgbContent = null;
  68.  
  69.     /**
  70.      * Decode a COSE message object.  This function assumes that the message
  71.      * has a leading CBOR tag to identify the message type.  If this is not
  72.      * true then use {#link DecodeFromBytes(byte[], MessageTag)}.
  73.      *
  74.      * @param rgbData byte stream to be decoded
  75.      * @return the decoded message object
  76.      * @throws CoseException on a decode failure
  77.      */
  78.     public static Message DecodeFromBytes(byte[] rgbData) throws CoseException {
  79.         return DecodeFromBytes(rgbData, MessageTag.Unknown);
  80.     }
  81.    
  82.     /**
  83.      * Decode a COSE message object. Use a value of {@code MessageTag.Unknown}
  84.      * to decode a generic structure with tagging.  Use a specific value if
  85.      * the tagging is absent or if a known structure is passed in.
  86.      *
  87.      * @param rgbData byte stream to be decoded
  88.      * @param defaultTag assumed message type to be decoded
  89.      * @return the decoded message object
  90.      * @throws CoseException on a decode failure.
  91.      */
  92.     public static Message DecodeFromBytes(byte[] rgbData, MessageTag defaultTag) throws CoseException {
  93.         CBORObject messageObject = CBORObject.DecodeFromBytes(rgbData);
  94.        
  95.         if (messageObject.getType() != CBORType.Array)  throw new CoseException("Message is not a COSE security Message");
  96.        
  97.         if (messageObject.isTagged()) {
  98.             if (messageObject.getTagCount() != 1) throw new CoseException("Malformed message - too many tags");
  99.            
  100.             if (defaultTag == MessageTag.Unknown) {
  101.                 defaultTag = MessageTag.FromInt(messageObject.getMostInnerTag().ToInt32Unchecked());
  102.             }
  103.             else if (defaultTag != MessageTag.FromInt(messageObject.getMostInnerTag().ToInt32Unchecked())) {
  104.                 throw new CoseException("Passed in tag does not match actual tag");
  105.             }
  106.         }
  107.        
  108.         Message msg;
  109.        
  110.         switch (defaultTag) {
  111.             case Unknown: // Unknown
  112.                 throw new CoseException("Message was not tagged and no default tagging option given");
  113.        
  114.             case Encrypt:
  115.             msg = new EncryptMessage();
  116.             break;

  117.         case Encrypt0:
  118.             msg = new Encrypt0Message();
  119.             break;

  120.             case MAC:
  121.             msg = new MACMessage();
  122.             break;

  123.         case MAC0:
  124.             msg = new MAC0Message();
  125.             break;

  126.             case Sign1:
  127.             msg = new Sign1Message();
  128.             break;

  129.             case Sign:
  130.             msg = new SignMessage();
  131.             break;
  132.                
  133.             default:
  134.                 throw new CoseException("Message is not recognized as a COSE security Object");
  135.         }
  136.    
  137.         msg.DecodeFromCBORObject(messageObject);
  138.         return msg;
  139.        
  140.     }

  141.     /**
  142.      * Encode the message to a byte array.  This function will force cryptographic operations to be executed as needed.
  143.      *
  144.      * @return byte encoded object
  145.      * @throws CoseException Internal COSE Exception
  146.      */
  147.     public byte[] EncodeToBytes() throws CoseException {
  148.         return EncodeToCBORObject().EncodeToBytes();
  149.     }

  150.     /**
  151.      * Given a CBOR tree, parse the message.  This is an abstract function that is implemented for each different supported COSE message.
  152.      *
  153.      * @param messageObject CBORObject to be converted to a message.
  154.      * @throws CoseException Internal COSE Exception
  155.      */
  156.    
  157.     protected abstract void DecodeFromCBORObject(CBORObject messageObject) throws CoseException;
  158.    
  159.     /**
  160.      * Encode the COSE message object to a CBORObject tree.  This function call will force cryptographic operations to be executed as needed.
  161.      * This is an internal function, as such it does not add the tag on the front and is implemented on a per message object.
  162.      *
  163.      * @return CBORObject representing the message.
  164.      * @throws CoseException Internal COSE Exception
  165.      */
  166.     protected abstract CBORObject EncodeCBORObject() throws CoseException;
  167.    
  168.     /**
  169.      * Encode the COSE message object to a CBORObject tree.  This function call will force cryptographic operations to be executed as needed.
  170.      *
  171.      * @return CBORObject representing the message.
  172.      * @throws CoseException Internal COSE Exception
  173.      */
  174.     public CBORObject EncodeToCBORObject() throws CoseException {
  175.         CBORObject obj;
  176.        
  177.         obj = EncodeCBORObject();
  178.        
  179.         if (emitTag) {
  180.             obj = CBORObject.FromObjectAndTag(obj, messageTag.value);
  181.         }
  182.        
  183.         return obj;
  184.     }

  185.     /**
  186.      * Return the content bytes of the message
  187.      *
  188.      * @return bytes of the content
  189.      */
  190.     public byte[] GetContent() {
  191.         return rgbContent;
  192.     }
  193.    
  194.     /**
  195.      * Does the message current have content?
  196.      *
  197.      * @return true if it has content
  198.      */
  199.     public boolean HasContent() {
  200.         return rgbContent != null;
  201.     }
  202.    
  203.     /**
  204.      * Set the content bytes of the message.  If the message was transmitted with
  205.      * detached content, this must be called before doing cryptographic processing on the message.
  206.      *
  207.      * @param rgbData bytes to set as the content
  208.      */
  209.     public void SetContent(byte[] rgbData) {
  210.         rgbContent = rgbData;
  211.     }
  212.    
  213.     /**
  214.      * Set the content bytes as a text string.  The string will be encoded using UTF8 into a byte string.
  215.      *
  216.      * @param strData string to set as the content
  217.      */
  218.     public void SetContent(String strData) {
  219.         rgbContent = strData.getBytes(StandardCharsets.UTF_8);
  220.     }

  221.     List<CounterSign> counterSignList = new ArrayList<CounterSign>();
  222.     CounterSign1 counterSign1;

  223.     public void addCountersignature(CounterSign countersignature) {
  224.         counterSignList.add(countersignature);
  225.     }

  226.     public List<CounterSign> getCountersignerList() {
  227.         return counterSignList;
  228.     }

  229.     public CounterSign1 getCountersign1() {
  230.         return counterSign1;
  231.     }

  232.     public void setCountersign1(CounterSign1 value) {
  233.         counterSign1 = value;
  234.     }

  235.     protected void ProcessCounterSignatures() throws CoseException {
  236.         if (!counterSignList.isEmpty()) {
  237.             if (counterSignList.size() == 1) {
  238.                 counterSignList.get(0).sign(rgbProtected, rgbContent);
  239.                 addAttribute(HeaderKeys.CounterSignature, counterSignList.get(0).EncodeToCBORObject(),
  240.                         Attribute.UNPROTECTED);
  241.             } else {
  242.                 CBORObject list = CBORObject.NewArray();
  243.                 for (CounterSign sig : counterSignList) {
  244.                     sig.sign(rgbProtected, rgbContent);
  245.                     list.Add(sig.EncodeToCBORObject());
  246.                 }
  247.                 addAttribute(HeaderKeys.CounterSignature, list, Attribute.UNPROTECTED);
  248.             }
  249.         }

  250.         if (counterSign1 != null) {
  251.             counterSign1.sign(rgbProtected, rgbContent);
  252.             addAttribute(HeaderKeys.CounterSignature0, counterSign1.EncodeToCBORObject(), Attribute.UNPROTECTED);
  253.         }
  254.     }

  255.     public boolean validate(CounterSign1 countersignature) throws CoseException {
  256.         return countersignature.validate(rgbProtected, rgbContent);
  257.     }

  258.     public boolean validate(CounterSign countersignature) throws CoseException {
  259.         return countersignature.validate(rgbProtected, rgbContent);
  260.     }
  261. }