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

scodec.codecs.MultiplexedCodec.scala Maven / Gradle / Ivy

/*
 * Copyright (c) 2013, Scodec
 * All rights reserved.
 *
 * Redistribution and use in source and binary forms, with or without modification,
 * are permitted provided that the following conditions are met:
 *
 * 1. Redistributions of source code must retain the above copyright notice, this
 *    list of conditions and the following disclaimer.
 *
 * 2. Redistributions in binary form must reproduce the above copyright notice,
 *    this list of conditions and the following disclaimer in the documentation
 *    and/or other materials provided with the distribution.
 *
 * 3. Neither the name of the copyright holder nor the names of its contributors
 *    may be used to endorse or promote products derived from this software without
 *    specific prior written permission.
 *
 * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" AND
 * ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE IMPLIED
 * WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE ARE
 * DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT HOLDER OR CONTRIBUTORS BE LIABLE FOR
 * ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR CONSEQUENTIAL DAMAGES
 * (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF SUBSTITUTE GOODS OR SERVICES;
 * LOSS OF USE, DATA, OR PROFITS; OR BUSINESS INTERRUPTION) HOWEVER CAUSED AND ON
 * ANY THEORY OF LIABILITY, WHETHER IN CONTRACT, STRICT LIABILITY, OR TORT
 * (INCLUDING NEGLIGENCE OR OTHERWISE) ARISING IN ANY WAY OUT OF THE USE OF THIS
 * SOFTWARE, EVEN IF ADVISED OF THE POSSIBILITY OF SUCH DAMAGE.
 */

package scodec
package codecs

import scala.collection.Factory

import scodec.bits.BitVector

/**
  * A trait that enables custom handling for encoding/decoding sequences.
  */
sealed trait MultiplexedCodec:

  def sizeBound: SizeBound = SizeBound.unknown

  /**
    * Encodes all elements of the specified sequence and combines the results using `mux`, or returns the first encountered error.
    *
    * @param enc element encoder
    * @param mux multiplexing function
    * @param seq elements to encode
    * @return
    */
  final def encode[A](enc: Encoder[A], mux: (BitVector, BitVector) => BitVector)(
      seq: collection.immutable.Seq[A]
  ): Attempt[BitVector] =
    val buf = new collection.mutable.ArrayBuffer[BitVector](seq.size)
    var failure: Err | Null = null
    seq.foreach { a =>
      if failure == null then
        enc.encode(a) match
          case Attempt.Successful(aa) => buf += aa
          case Attempt.Failure(err)   => failure = err.pushContext(buf.size.toString)
    }
    if failure == null then
      def merge(offset: Int, size: Int): BitVector = size match
        case 0 => BitVector.empty
        case 1 => buf(offset)
        case _ =>
          val half = size / 2
          mux(merge(offset, half), merge(offset + half, half + (if size % 2 == 0 then 0 else 1)))
      Attempt.successful(merge(0, buf.size))
    else
      Attempt.failure(failure.nn) // FIXME .nn shouldn't be necessary here

  /**
    * Repeatedly decodes values of type `A` and returns a collection of the specified type.
    * Uses `deMux` repeatedly to obtain the stream of vectors to decode to a value of type `A`.
    * Terminates when the next stream to decode is empty or upon first decoding error.
    *
    * Note: For large sequences, it may be necessary to compact bits in `deMux`.
    *
    * @param dec element decoder
    * @param deMux returns `(next, rest)` tuples where `next` is input to `dec` yielding `(value, remainder)` and `remainder ++ rest` is the next input to `deMux`
    * @param buffer input bits
    * @return
    */
  final def decode[F[_], A](dec: Decoder[A], deMux: BitVector => (BitVector, BitVector))(
      buffer: BitVector
  )(using cbf: Factory[A, F[A]]): Attempt[DecodeResult[F[A]]] =
    val builder = cbf.newBuilder
    var temp = deMux(buffer)
    var count = 0
    var error: Option[Err] = None
    while temp._1.nonEmpty do
      dec.decode(temp._1) match
        case Attempt.Successful(DecodeResult(value, remainder)) =>
          builder += value
          count += 1
          temp = deMux(remainder ++ temp._2)
        case Attempt.Failure(err) =>
          error = Some(err.pushContext(count.toString))
          temp = (BitVector.empty, BitVector.empty)
    Attempt.fromErrOption(error, DecodeResult(builder.result(), temp._2))

object DeMultiplexer:

  /**
    * Returns a `(next, rest)` tuple where `next` is the prefix of the input preceding the first occurrence of `delimiter`.
    *
    * Note: The search for `delimiter` is performed at `delimiter` sized intervals.
    *
    * @param bits the input bits
    * @param delimiter the separator bits
    * @return
    */
  def delimited(bits: BitVector, delimiter: BitVector): (BitVector, BitVector) =
    delimited(bits, delimiter, 0)

  private def delimited(
      bits: BitVector,
      delimiter: BitVector,
      start: Long
  ): (BitVector, BitVector) =
    bits.indexOfSlice(delimiter, start) match
      case -1                             => (bits, BitVector.empty)
      case i if (i % delimiter.size) == 0 => (bits.take(i), bits.drop(i + delimiter.size))
      case i                              => delimited(bits, delimiter, i + delimiter.size)

private[codecs] class VectorMultiplexedCodec[A](
    mux: (BitVector, BitVector) => BitVector,
    deMux: BitVector => (BitVector, BitVector),
    codec: Codec[A]
) extends Codec[Vector[A]]
    with MultiplexedCodec:
  def encode(value: Vector[A]): Attempt[BitVector] = encode(codec, mux)(value)

  def decode(bits: BitVector): Attempt[DecodeResult[Vector[A]]] =
    decode[Vector, A](codec, deMux)(bits)

  override def toString: String = s"vectorMultiplexed($codec, $mux, $deMux)"

private[codecs] class ListMultiplexedCodec[A](
    mux: (BitVector, BitVector) => BitVector,
    deMux: BitVector => (BitVector, BitVector),
    codec: Codec[A]
) extends Codec[List[A]]
    with MultiplexedCodec:
  def encode(value: List[A]): Attempt[BitVector] = encode(codec, mux)(value)

  def decode(bits: BitVector): Attempt[DecodeResult[List[A]]] = decode[List, A](codec, deMux)(bits)

  override def toString: String = s"listMultiplexed($codec, $mux, $deMux)"




© 2015 - 2025 Weber Informatics LLC | Privacy Policy