smithy4s.http.internals.MetaDecode.scala Maven / Gradle / Ivy
/*
* Copyright 2021-2024 Disney Streaming
*
* Licensed under the Tomorrow Open Source Technology License, Version 1.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* https://disneystreaming.github.io/TOST-1.0.txt
*
* 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.
*/
package smithy4s
package http
package internals
import smithy4s.http.internals.MetaDecode._
import HttpBinding._
private[http] sealed abstract class MetaDecode[+A] {
def map[B](to: A => B): MetaDecode[B] = this match {
case StringValueMetaDecode(f) => StringValueMetaDecode(f.andThen(to))
case StringCollectionMetaDecode(f) =>
StringCollectionMetaDecode(f andThen to)
case StringMapMetaDecode(f) => StringMapMetaDecode(f.andThen(to))
case StringListMapMetaDecode(f) => StringListMapMetaDecode(f.andThen(to))
case EmptyMetaDecode => EmptyMetaDecode
case StructureMetaDecode(f) => StructureMetaDecode(f.andThen(_.map(to)))
}
private[internals] def updateMetadata(
binding: HttpBinding,
fieldName: String,
maybeDefault: Option[Any]
): (Metadata, PutField) => Unit = {
// format: off
// Looks up metadata for the specific key, then applies some procedure.
// Throws an exception if the field optional is unset and the key is not
// found.
def lookupAndProcess[K, T] (m: Metadata => Map[K, T], key: K)
(process: (T, String, PutField) => Unit) :
(Metadata, PutField) => Unit = {
(metadata, putField) =>
m(metadata).get(key) match {
case Some(value) => process(value, fieldName, putField(_))
case None if maybeDefault.isDefined => putField(maybeDefault.get)
case None => throw new MetadataError.NotFound(fieldName, binding)
}
}
// format: on
def putDefault(putField: PutField) = maybeDefault match {
case Some(default) => putField(default)
case None => throw new MetadataError.NotFound(fieldName, binding)
}
(binding, this) match {
case (PathBinding(path), StringValueMetaDecode(f)) =>
lookupAndProcess(_.path, path) { (value, fieldName, putField) =>
putField(f(value))
}
case (HeaderBinding(h), StringValueMetaDecode(f)) =>
lookupAndProcess(_.headers, h) { (values, fieldName, putField) =>
if (values.size == 1) {
putField(f(values.head))
} else throw MetadataError.ArityError(fieldName, binding)
}
case (HeaderBinding(h), StringCollectionMetaDecode(f)) =>
lookupAndProcess(_.headers, h) { (values, fieldName, putField) =>
putField(f(values.iterator))
}
case (QueryBinding(h), StringValueMetaDecode(f)) =>
lookupAndProcess(_.query, h) { (values, fieldName, putField) =>
if (values.size == 1) {
putField(f(values.head))
} else throw MetadataError.ArityError(fieldName, binding)
}
case (QueryBinding(q), StringCollectionMetaDecode(f)) =>
lookupAndProcess(_.query, q) { (values, fieldName, putField) =>
putField(f(values.iterator))
}
// see https://smithy.io/2.0/spec/http-bindings.html#httpqueryparams-trait
// when targeting Map[String,String] we take the first value encountered
case (QueryParamsBinding, StringMapMetaDecode(f)) => {
(metadata, putField) =>
val iter: Iterator[(FieldName, FieldName)] = metadata.query.iterator
.map { case (k, values) =>
if (values.nonEmpty) {
k -> values.head
} else throw MetadataError.NotFound(fieldName, QueryParamsBinding)
}
if (iter.nonEmpty) putField(f(iter))
else putDefault(putField)
}
case (QueryParamsBinding, StringListMapMetaDecode(f)) => {
(metadata, putField) =>
val iter = metadata.query.iterator
.map { case (k, values) =>
k -> values.iterator
}
if (iter.nonEmpty) putField(f(iter))
else putDefault(putField)
}
case (HeaderPrefixBinding(prefix), StringMapMetaDecode(f)) => {
(metadata, putField) =>
val iter = metadata.headers.iterator
.collect {
case (k, values) if k.startsWith(prefix) =>
if (values.size == 1) {
k.toString.drop(prefix.length()) -> values.head
} else
throw MetadataError.ArityError(fieldName, HeaderBinding(k))
}
if (iter.nonEmpty) putField(f(iter))
else putDefault(putField)
}
case (HeaderPrefixBinding(prefix), StringListMapMetaDecode(f)) => {
(metadata, putField) =>
val iter = metadata.headers.iterator
.collect {
case (k, values) if k.startsWith(prefix) =>
k.toString.drop(prefix.length()) -> values.iterator
}
if (iter.nonEmpty) putField(f(iter))
else putDefault(putField)
}
case (StatusCodeBinding, StringValueMetaDecode(f)) =>
(metadata, putField) =>
metadata.statusCode match {
case None =>
sys.error(
"Status code is not available and field needs it."
)
case Some(statusCode) =>
// TODO add a specialised case for this
putField(f(statusCode.toString))
}
case _ => (metadata: Metadata, buffer) => ()
}
}
}
private[http] object MetaDecode {
type FieldName = String
case class MetaDecodeError(f: (String, HttpBinding) => MetadataError)
extends Throwable
with scala.util.control.NoStackTrace
// format: off
final case class StringValueMetaDecode[A](f: String => A) extends MetaDecode[A]
final case class StringCollectionMetaDecode[A](f: Iterator[String] => A) extends MetaDecode[A]
final case class StringMapMetaDecode[A](f: Iterator[(String, String)] => A) extends MetaDecode[A]
final case class StringListMapMetaDecode[A](f: Iterator[(String, Iterator[String])] => A) extends MetaDecode[A]
case object EmptyMetaDecode extends MetaDecode[Nothing]
final case class StructureMetaDecode[A](f: Metadata => Either[MetadataError, A]) extends MetaDecode[A]
// format: on
type PutField = Any => Unit
def emap[A, B](
fa: MetaDecode[A]
)(f: A => Either[ConstraintError, B]): MetaDecode[B] =
fa.map(a =>
f(a) match {
case Right(b) => b
case Left(error) =>
throw MetaDecodeError(
MetadataError.FailedConstraint(_, _, error.message)
)
}
)
def fromUnsafe[A](
expectedType: String
)(f: String => A): MetaDecode[A] =
StringValueMetaDecode[A] { string =>
try { f(string) }
catch {
case _: Throwable =>
throw MetaDecodeError(
MetadataError.WrongType(
_,
_,
expectedType,
string
)
)
}
}
def from[A](
expectedType: String
)(f: String => Option[A]): MetaDecode[A] =
StringValueMetaDecode[A] { string =>
f(string) match {
case Some(value) =>
value
case None =>
throw MetaDecodeError(
MetadataError.WrongType(
_,
_,
expectedType,
string
)
)
}
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy