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

harness.csv.CsvHeaderDecoder.scala Maven / Gradle / Ivy

package harness.csv

import cats.data.NonEmptyList
import cats.syntax.either.*
import cats.syntax.option.*
import cats.syntax.traverse.*
import harness.core.{StringDecoder, Zip}

trait CsvHeaderDecoder[T] private { self =>

  protected def buildImpl(headers: IArray[Option[String]]): Either[String, CsvHeaderDecoder.AppliedCsvHeaderDecoder[T]]

  final def decode(lines: List[IArray[Option[String]]]): Either[String, List[T]] =
    lines match {
      // TODO (KR) :
      case head :: tail =>
        self.buildImpl(head).flatMap { decoder =>
          tail.zipWithIndex.traverse { (line, idx) => decoder.decodeImpl(line, idx + 2) }
        }
      case Nil => "Missing header row".asLeft
    }

  final def optional: CsvHeaderDecoder[Option[T]] =
    self.buildImpl(_).map(_.optional)

  final def ++[T2](other: CsvHeaderDecoder[T2])(implicit zip: Zip[T, T2]): CsvHeaderDecoder[zip.Out] = { headers =>
    for {
      t <- self.buildImpl(headers)
      t2 <- other.buildImpl(headers)
    } yield t ++ t2
  }

  final def map[T2](f: T => T2): CsvHeaderDecoder[T2] =
    self.buildImpl(_).map(_.map(f))

  final def emap[T2](f: T => Either[String, T2]): CsvHeaderDecoder[T2] =
    self.buildImpl(_).map(_.emap(f))

}
object CsvHeaderDecoder {

  inline def apply[T](implicit decoder: CsvHeaderDecoder[T]): CsvHeaderDecoder[T] = decoder

  def decode[T](csv: String)(implicit decoder: CsvHeaderDecoder[T]): Either[String, List[T]] =
    Parser.parse(csv).flatMap(decoder.decode)

  def cell[T](header: String, f: String => Either[String, T]): CsvHeaderDecoder[T] =
    _.zipWithIndex.find(_._1.contains(header)) match {
      case Some((_, cell)) =>
        new AppliedCsvHeaderDecoder[T] {
          override protected val minSize: Int = cell
          override protected val headerIndexes: Set[Int] = Set(cell)
          override def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, T] =
            line(cell) match {
              case Some(value) => f(value)
              case None        => s"Error for header '$header' on line $lineNo in cell ${cell + 1}  : unexpected null".asLeft
            }
        }.asRight
      case None => s"Could not find header '$header'".asLeft
    }

  def cell[T](header: String)(implicit decoder: StringDecoder[T]): CsvHeaderDecoder[T] =
    CsvHeaderDecoder.cell(header, decoder.decode)

  def multiOptHeaderDecoder[T](header: String)(implicit decoder: StringDecoder[T]): CsvHeaderDecoder[NonEmptyList[Option[T]]] = { headers =>
    NonEmptyList.fromList(headers.zipWithIndex.filter { (h, _) => h.contains(header) }.map(_._2).toList) match {
      case Some(indexes) =>
        new AppliedCsvHeaderDecoder[NonEmptyList[Option[T]]] {
          override protected val minSize: Int = indexes.toList.max
          override protected val headerIndexes: Set[Int] = indexes.toList.toSet
          override def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, NonEmptyList[Option[T]]] =
            indexes.traverse { line(_).traverse(decoder.decode) }
        }.asRight
      case None => s"Did not find any headers with value '$header'".asLeft
    }
  }

  implicit def optional[T](implicit decoder: CsvHeaderDecoder[T]): CsvHeaderDecoder[Option[T]] =
    decoder.optional

  trait AppliedCsvHeaderDecoder[T] { self =>

    protected val minSize: Int
    protected val headerIndexes: Set[Int]
    def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, T]

    final def optional: CsvHeaderDecoder.AppliedCsvHeaderDecoder[Option[T]] =
      new AppliedCsvHeaderDecoder[Option[T]] {
        override protected val minSize: Int = self.minSize
        override protected val headerIndexes: Set[Int] = self.headerIndexes
        override def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, Option[T]] =
          if (self.headerIndexes.forall(line(_).isEmpty)) None.asRight
          else self.decodeImpl(line, lineNo).map(_.some)
      }

    final def ++[T2](other: CsvHeaderDecoder.AppliedCsvHeaderDecoder[T2])(implicit zip: Zip[T, T2]): CsvHeaderDecoder.AppliedCsvHeaderDecoder[zip.Out] =
      new AppliedCsvHeaderDecoder[zip.Out] {
        override protected val minSize: Int = self.minSize.max(other.minSize)
        override protected val headerIndexes: Set[Int] = self.headerIndexes ++ other.headerIndexes
        override def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, zip.Out] =
          for {
            t <- self.decodeImpl(line, lineNo)
            t2 <- other.decodeImpl(line, lineNo)
          } yield zip.zip(t, t2)
      }

    final def map[T2](f: T => T2): CsvHeaderDecoder.AppliedCsvHeaderDecoder[T2] =
      new AppliedCsvHeaderDecoder[T2] {
        override protected val minSize: Int = self.minSize
        override protected val headerIndexes: Set[Int] = self.headerIndexes
        override def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, T2] =
          self.decodeImpl(line, lineNo).map(f)
      }

    final def emap[T2](f: T => Either[String, T2]): CsvHeaderDecoder.AppliedCsvHeaderDecoder[T2] =
      new AppliedCsvHeaderDecoder[T2] {
        override protected val minSize: Int = self.minSize
        override protected val headerIndexes: Set[Int] = self.headerIndexes
        override def decodeImpl(line: IArray[Option[String]], lineNo: Int): Either[String, T2] =
          self.decodeImpl(line, lineNo).flatMap(f)
      }

  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy