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

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