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

io.github.paoloboni.http.QueryParamsConverter.scala Maven / Gradle / Ivy

There is a newer version: 1.6.1
Show newest version
/*
 * Copyright (c) 2022 Paolo Boni
 *
 * Permission is hereby granted, free of charge, to any person obtaining a copy of
 * this software and associated documentation files (the "Software"), to deal in
 * the Software without restriction, including without limitation the rights to
 * use, copy, modify, merge, publish, distribute, sublicense, and/or sell copies of
 * the Software, and to permit persons to whom the Software is furnished to do so,
 * subject to the following conditions:
 *
 * The above copyright notice and this permission notice shall be included in all
 * copies or substantial portions of the Software.
 *
 * THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR
 * IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, FITNESS
 * FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR
 * COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER
 * IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN
 * CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
 */

package io.github.paoloboni.http

import io.github.paoloboni.http
import sttp.model.QueryParams

import scala.deriving._
import scala.compiletime._
import io.github.paoloboni.http.StringConverter

trait QueryParamsConverter[T]:
  def to(t: T): QueryParams

object QueryParamsConverter:
  def apply[T: QueryParamsConverter]: QueryParamsConverter[T] = summon

  inline private def label[A]: String = constValue[A].asInstanceOf[String]

  inline private def summonStringConverter[T]: StringConverter[T] = summonFrom {
    case converter: StringConverter[T] => converter
    case _ =>
      new StringConverter[T] {
        override def to(t: T): String = t.toString
      }
  }

  inline def processCases[T](
      s: Mirror.SumOf[T]
  )(labels: List[String], queryParamsConverters: List[QueryParamsConverter[_]])(x: T): QueryParams = {
    val ord                  = s.ordinal(x)
    val queryParamsConverter = queryParamsConverters(ord)
    val params               = queryParamsConverter.asInstanceOf[QueryParamsConverter[T]].to(x)
    params.param("type", labels(ord))
  }

  inline def summonStringConverters[A <: Tuple]: List[StringConverter[_]] =
    inline erasedValue[A] match
      case _: EmptyTuple => Nil
      case _: (t *: ts)  => summonStringConverter[t] :: summonStringConverters[ts]

  inline def summonQueryParamsConverters[A <: Tuple]: List[QueryParamsConverter[_]] =
    inline erasedValue[A] match
      case _: EmptyTuple => Nil
      case _: (t *: ts)  => summonInline[QueryParamsConverter[t]] :: summonQueryParamsConverters[ts]

  inline def summonAllLabels[A <: Tuple]: List[String] =
    inline erasedValue[A] match
      case _: EmptyTuple => Nil
      case _: (t *: ts)  => label[t] :: summonAllLabels[ts]

  inline def processElems[T](stringConverters: List[StringConverter[_]])(t: T): QueryParams = {
    val pElem = t.asInstanceOf[Product]
    (pElem.productElementNames.toList zip pElem.productIterator.toList zip stringConverters)
      .map { case ((label, elem), converter) =>
        label -> converter.asInstanceOf[StringConverter[Any]].to(elem)
      }
      .foldRight(QueryParams()) {
        case ((label, value), params) if value != "" => params.param(label, value)
        case ((label, value), params)                => params
      }
  }

  inline given derived[T](using m: Mirror.Of[T]): QueryParamsConverter[T] = {
    lazy val stringConverters = summonStringConverters[m.MirroredElemTypes]
    lazy val labels           = summonAllLabels[m.MirroredElemLabels]
    new QueryParamsConverter[T] {
      override def to(t: T): QueryParams = {
        inline m match {
          case s: Mirror.SumOf[T] =>
            val queryParamsConverters = summonQueryParamsConverters[m.MirroredElemTypes]
            processCases[T](s)(labels, queryParamsConverters)(t)
          case p: Mirror.ProductOf[T] => processElems(stringConverters)(t)
        }
      }
    }
  }

  extension [T](t: T) def toQueryParams(using converter: QueryParamsConverter[T]): QueryParams = converter.to(t)




© 2015 - 2025 Weber Informatics LLC | Privacy Policy