PkiQbftExtraDataCodec.java

/*
 * Copyright ConsenSys AG.
 *
 * 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
 *
 * http://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.
 *
 * SPDX-License-Identifier: Apache-2.0
 */
package org.hyperledger.besu.consensus.qbft.pki;

import org.hyperledger.besu.consensus.common.bft.BftExtraData;
import org.hyperledger.besu.consensus.qbft.QbftExtraDataCodec;
import org.hyperledger.besu.ethereum.rlp.BytesValueRLPInput;
import org.hyperledger.besu.ethereum.rlp.BytesValueRLPOutput;
import org.hyperledger.besu.ethereum.rlp.RLPInput;

import java.util.List;

import org.apache.tuweni.bytes.Bytes;

/**
 * The PkiQbftExtraData encoding format is different from the "regular" QbftExtraData encoding. We
 * have an extra bytes element in the end of the list.
 */
public class PkiQbftExtraDataCodec extends QbftExtraDataCodec {

  /** The constant QBFT_EXTRA_DATA_LIST_SIZE. */
  public static final int QBFT_EXTRA_DATA_LIST_SIZE = 5;

  @Override
  public BftExtraData decodeRaw(final Bytes input) {
    if (input.isEmpty()) {
      throw new IllegalArgumentException("Invalid Bytes supplied - Bft Extra Data required.");
    }

    final BftExtraData bftExtraData = super.decodeRaw(input);

    final RLPInput rlpInput = new BytesValueRLPInput(input, false);

    final Bytes cms;
    final List<RLPInput> elements = rlpInput.readList(RLPInput::readAsRlp);
    if (elements.size() > QBFT_EXTRA_DATA_LIST_SIZE) {
      final RLPInput cmsElement = elements.get(elements.size() - 1);
      cms = cmsElement.readBytes();
    } else {
      cms = Bytes.EMPTY;
    }

    return new PkiQbftExtraData(bftExtraData, cms);
  }

  @Override
  protected Bytes encode(final BftExtraData bftExtraData, final EncodingType encodingType) {
    return encode(bftExtraData, encodingType, true);
  }

  private Bytes encode(
      final BftExtraData bftExtraData, final EncodingType encodingType, final boolean includeCms) {
    final Bytes encoded = super.encode(bftExtraData, encodingType);
    if (!(bftExtraData instanceof PkiQbftExtraData) || !includeCms) {
      return encoded;
    }

    final BytesValueRLPOutput rlpOutput = new BytesValueRLPOutput();
    rlpOutput.startList();
    // Read through extraData RLP list elements and write them to the new RLP output
    new BytesValueRLPInput(encoded, false)
        .readList(RLPInput::readAsRlp).stream()
            .map(RLPInput::raw)
            .forEach(rlpOutput::writeRLPBytes);
    rlpOutput.writeBytes(((PkiQbftExtraData) bftExtraData).getCms());
    rlpOutput.endList();

    return rlpOutput.encoded();
  }

  /**
   * Encode without cms.
   *
   * @param bftExtraData the bft extra data
   * @return the bytes
   */
  public Bytes encodeWithoutCms(final BftExtraData bftExtraData) {
    return encode(bftExtraData, EncodingType.ALL, false);
  }
}