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