Multipart and Form Handling
Multipart is an HTTP content type for messages (requests or responses) composed of multiple parts.
Each part is itself similar to an HTTP message in that it has its own body and headers. A common use
for multipart requests is for user-submitted forms, especially ones that include files.
Browsers also support (and default to) another encoding for forms, application/x-www-form-urlencoded
.
This encoding is simpler, but is not often used for binary data.
This page demonstrates the usage of forms in http4s and includes a scala-cli example at the end.
We'll start by defining imports for our examples:
import org.http4s.client.Client
import cats.effect._
import cats.syntax.all._
import org.http4s._
import org.http4s.dsl.io._
import org.http4s.headers._
import org.http4s.multipart._
import org.http4s.implicits._
Because this documentation is running in mdoc we need an implicit IORuntime
to let us run our IO
values explicitly with .unsafeRunSync()
.
In real code you should construct your whole program in IO
and assign it to run
in IOApp
.
import cats.effect.unsafe.IORuntime
implicit val runtime: IORuntime = IORuntime.global
UrlForm
To handle application/x-www-form-urlencoded
messages, http4s provides UrlForm
and respective EntityEncoder
and
EntityDecoder
. The following example shows the client sending a form request and a server parsing it:
val urlRoutes = HttpRoutes
.of[IO] { case request @ POST -> Root / "url-form" =>
request.as[UrlForm].flatMap { form =>
val name = form.values
.collectFirst { case ("name", values) => values }
.flatMap(_.headOption)
.getOrElse("")
Ok(s"Hello, $name")
}
}
val urlClient = Client.fromHttpApp(urlRoutes.orNotFound)
val urlRequest = Request[IO](
method = POST,
uri = uri"http://example/url-form",
).withEntity(UrlForm("name" -> "Duncan", "version" -> "4"))
urlClient.expect[String](urlRequest)
.unsafeRunSync()
// res0: String = "Hello, Duncan"
Multipart form
http4s also supports multipart forms, although their usage is a bit more involved. A multipart body is represented
with a Multipart[_]
value. There's an EntityDecoder
for Multipart[_]
, so parsing a request body works as expected:
HttpRoutes.of[IO] {
case request @ POST -> Root / "multipart-form" =>
request.as[Multipart[IO]].flatMap(multipart => ???)
}
However, this approach buffers the contents in memory and so it's unsuitable if large requests are expected. The size of the body can be controlled using the EntityLimiter middleware, and it's advisable to use it generally, even if no uploads are expected.
An alternative that handles large request bodies better is EntityDecoder.mixedMultipartResource
which will stream
the body into files on disk if it reaches a specified size threshold. This method returns a Resource
wrapping an EntityDecoder
,
the resource is meant to be allocated in the scope of a single request as it cleans up any temporary files that
were created while decoding. Usage is as follows:
HttpRoutes.of[IO] {
case request @ POST -> Root / "multipart-form" =>
EntityDecoder.mixedMultipartResource[IO]().use(decoder =>
request.decodeWith(decoder, strict = true)(multipart => ???)
)
}
Creating a multipart request also differs slightly from other content types. Each part in a multipart request body is
surrounded by a boundary, which is a bit of text used by the server to distinguish between the different parts. This
text is also sent in the header of the request. Currently, the EntityEncoder
can't define headers by inspecting the body,
and as such we have to define the boundary header in the request explicitly. Additionally, the boundary is randomly-generated,
which is an effect. We can call methods on the Multiparts
companion object to get a Multiparts[_]
instance, which is
a builder of multipart requests. This instance can be shared. This is an example of the creation of a multipart request:
Multiparts.forSync[IO].flatMap(multiparts =>
multiparts.multipart(
Vector( // a multipart request with two parts
Part.fileData[IO]( // there are also overloads for fileData that read directly from a file
name = "picture",
filename = "sunset.jpg",
entityBody = fs2.Stream.range[IO, Int](0, 100).map(_.toByte),
headers = `Content-Type`(MediaType.image.jpeg)
),
Part.formData[IO](name = "description", value = "A sunset")
)
)
)
.map(multipartRequest =>
Request[IO](
method = Method.POST,
uri = uri"http://example.com/",
headers = multipartRequest.headers // set the headers related to this multipart request
).withEntity(multipartRequest)
)
Here's a full example with a client and a server:
val mpRoutes = HttpRoutes
.of[IO] { case request @ POST -> Root / "multipart-form" =>
EntityDecoder.mixedMultipartResource[IO]().use(decoder =>
request.decodeWith(decoder, strict = true) { multipart =>
val picture = multipart.parts.find(_.name.contains("picture"))
val pictureSize = picture.traverse(_.body.compile.count).map(_.getOrElse(0L))
val description = multipart.parts.find(_.name.contains("description"))
val descriptionText = description.traverse(_.bodyText.compile.string).map(_.getOrElse(""))
(pictureSize, descriptionText)
.flatMapN((size, description) =>
Ok(s"This is a $size byte file, with the description '$description'")
)
}
)
}
val mpClient = Client.fromHttpApp(mpRoutes.orNotFound)
val mpRequest = Multiparts.forSync[IO].flatMap(multiparts =>
multiparts.multipart(
Vector(
Part.fileData[IO](
name = "picture",
filename = "sunset.jpg",
entityBody = fs2.Stream.range[IO, Int](0, 100).map(_.toByte),
headers = `Content-Type`(MediaType.image.jpeg)
),
Part.formData[IO](name = "description", value = "A sunset")
)
)
)
.map(multipartRequest =>
Request[IO](
method = Method.POST,
uri = uri"http://example.com/multipart-form",
headers = multipartRequest.headers
)
.withEntity(multipartRequest)
)
mpRequest.flatMap(mpClient.expect[String](_))
.unsafeRunSync()
// res4: String = "This is a 100 byte file, with the description 'A sunset'"
Like UrlForm, browsers can also submit forms in a multipart request, to use this encoding the enctype
attribute is
used in the form
element, like this: <form method="post" enctype="multipart/form-data">
.
Streaming uploads
The usage of multipart is somewhat convoluted, in part because one expects a fixed-size sequence of parts when processing
a request (notice that parts
is a Vector
, not a Stream
), this means that http4s has to get to
the end of the request so that it knows all the parts. But this isn't the only way to upload files (although it is the
only way to do it in pure HTML). For a simpler form of upload a client could send a request with a Stream[F, Byte]
as the entity. See Streaming.
This alternative allows the server to work in a fully streaming fashion, although it's obviously missing the description
from our previous example and any other form of metadata. The other parts could be put into the query string or headers (taking into account their encoding and size limitations)
or in subsequent requests. This type of request with a binary payload can also be created in Javascript, and thus can
be initiated from the browser.
Scala-cli example
You can try this self-contained example using scala-cli and pointing your
browser to http://localhost:8089/. It includes a page with a form and the endpoint receiving the submission.
To run this code create a file (it should have the .scala
extension) with the following contents and run
scala-cli file.scala
.
//> using scala 2.13
//> using dep org.http4s::http4s-ember-client::0.23.29
//> using dep org.http4s::http4s-ember-server::0.23.29
//> using dep org.http4s::http4s-dsl::0.23.29
import cats.effect._
import cats.syntax.all._
import com.comcast.ip4s._
import org.http4s._
import org.http4s.dsl.io._
import org.http4s.ember.server.EmberServerBuilder
import org.http4s.headers._
object Main extends IOApp.Simple {
val routes = HttpRoutes.of[IO] {
case GET -> Root / "form" =>
Ok(
"""
|<form method="post">
| <label for="name">Name</label>
| <input id="name" name="name" />
| <button>Submit</button>
|</form>
|""".stripMargin
).map(_.withContentType(`Content-Type`(MediaType.text.html)))
case req @ POST -> Root / "form" =>
req.as[UrlForm].flatMap { form =>
Ok(
form.values
.map { case (k, v) => s"$k: ${v.mkString_(",")}" }
.toList
.mkString_("\n")
)
}
}
def run: IO[Unit] =
EmberServerBuilder
.default[IO]
.withPort(port"8089")
.withHttpApp(routes.orNotFound)
.build
.useForever
}