korolev.web.Request.scala Maven / Gradle / Ivy
/*
* Copyright 2017-2020 Aleksey Fomkin
*
* Licensed under the Apache License, Version 2.0 (the "License");
* you may not use this file except in compliance with the License.
* You may obtain a copy of the License at
*
* http://www.apache.org/licenses/LICENSE-2.0
*
* 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 korolev.web
import java.net.{URLDecoder, URLEncoder}
import java.nio.charset.StandardCharsets
/**
* @param body Should be handled before response be given
*/
final case class Request[Body](method: Request.Method,
pq: PathAndQuery,
headers: Seq[(String, String)],
contentLength: Option[Long],
body: Body,
renderedCookie: String = null)
extends Request.Head {
private lazy val parsedCookie = {
def d(s: String) = URLDecoder.decode(s.trim, "UTF-8")
if (renderedCookie == null || renderedCookie.isEmpty) Map.empty[String, String]
else
renderedCookie
.split(';')
.map(_.split('='))
.collect {
case Array(k, v) => (d(k), d(v))
case Array(k) => (d(k), "")
}
.toMap
}
def param(name: String): Option[String] =
pq.param(name)
def cookie(name: String): Option[String] =
parsedCookie.get(name)
def header(header: String): Option[String] =
headers.collectFirst {
case (k, v) if k.equalsIgnoreCase(header) => v
}
def withParam(name: String, value: String): Request[Body] = {
copy(pq = pq.withParam(name, value))
}
def withCookie(name: String, value: String): Request[Body] = {
val ek = URLEncoder.encode(name, "UTF-8")
val ev = URLEncoder.encode(value, "UTF-8")
if (renderedCookie == null || renderedCookie.isEmpty) {
copy(renderedCookie = s"$ek=$ev")
} else {
copy(renderedCookie = s"${renderedCookie};$ek=$ev")
}
}
def withHeader(key: String, value: String): Request[Body] =
copy(headers = (key, value) +: headers)
def withHeader(header: (String, String)): Request[Body] =
copy(headers = header +: headers)
def withHeaders(xs: (String, String)*): Request[Body] =
copy(headers = xs ++: headers)
}
object Request {
sealed trait Head {
def method: Method
def pq: PathAndQuery
def param(name: String): Option[String]
def cookie(name: String): Option[String]
def header(header: String): Option[String]
}
sealed abstract class Method(val value: String)
object Method {
final val All = Set(Post, Get, Put, Delete, Options, Head, Trace, Connect)
def fromString(method: String): Method =
method match {
case "POST" => Post
case "GET" => Get
case "PUT" => Put
case "DELETE" => Delete
case "OPTIONS" => Options
case "HEAD" => Head
case "TRACE" => Trace
case "CONNECT" => Connect
case _ => Unknown(method)
}
case object Post extends Method("POST")
case object Get extends Method("GET")
case object Put extends Method("PUT")
case object Delete extends Method("DELETE")
case object Options extends Method("OPTIONS")
case object Head extends Method("HEAD")
case object Trace extends Method("TRACE")
case object Connect extends Method("CONNECT")
case class Unknown(override val value: String) extends Method(value)
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy