Help language development. Donate to The Perl Foundation

HTTP::Request::FormData cpan:HANENKAMP last updated on 2018-03-14



HTTP::Request::FormData - handler to expedite the handling of multipart/form-data requests


use HTTP::UserAgent;
use HTTP::Request::Common;

my $ua =;

my $fd =;
$fd.add-part('username', 'alice');
$fd.add-part('avatar', 'alice.png'.IO, Content-Type => 'image/png');

my $req = POST(
    Content-Type => $fd.content-type,
    content => $fd.content,

my $res = $ua.request($req);


This provides a structured object for use with HTTP::Request that serializes into a nice neat content buffer.

I wrote this as an expedient work-around to some encoding issues I have with the way HTTP::Request works. This may be a better way to solve that problem than the API ported form LWP in Perl 5, but I am not making a judgement in that regard. I was merely attempting to expedite a solution and sharing it with the hope that it might be useful to someone else as well.

That said, this module aims at correctly rendering multipart/form-data content according to the specification laid out in RFC 7578.


method new

multi method new() returns HTTP::Request::FormData:D

Constructs a new, empty form data object.

method add-part

multi method add-part(Str:D $name, IO::Path $file, Str :$content-type, Str :$filename)
multi method add-part(Str:D $name, IO::Handle $file, Str :$content-type, Str :$filename)
multi method add-part(Str:D $name, $value, Str :$content-type, Str :$filename)

These methods append a new part to the end of the object. Three methods of adding parts are provided.

Passing an IO::Path will result in the given file being slurped automatically. The part will be read as binary data unless you specify a $content-type that starts with "text/".

Similarly, an IO::Handle may be passed and the contents will be slurped from here as well. This allows you control over whether to set the binary flag or not yourself.

Finally, you can pass an explicit value, which can be pretty much any string, blob, or other cool value you like.

In each case, you can specify the $filename to use for that part. Note that the $filename will be inferred from the given path if an IO::Path is passed.

method parts

method parts() returns Seq:D

Returns a Seq of the parts. Each part is returned as a HTTP::Request::FormData::Part object, which provides the following accessors:

method boundary

method boundary() returns Str:D

This is an accessor that allows you to see what the boundary will be used to separate the parts when rendered. This boundary meets the requirements of RFC 7578, which stipulates that the boundary must not be found in the content contained within any of the parts. This means that if you add a new part, the boundary may have to change.

For this reason, the add-part method will throw an exception if you attempt to add a part after calling this method to select a boundary.

method content-type

method content-type() returns Str:D

This is a handy helper for generating a MIME type with the boundary parameter like this:

use HTTP::Request::Common;

my $fd =;
$fd.add-part('name', 'alice');

my $req = POST(
    Content-Type => $fd.content-type,
    content => $fd.content,

It is essentially equivalent to:

qq<multipart/formdata; boundary=$fd.boundary()>

As this calls boundary(), calls to add-part will throw an exception after the first call to this method is made.

method content

method content() returns Blob:D

This will return the serialized data associated with this form data object. Each part will have it's headers and content rendered in order and separated by the boundary as specified in RFC 7578.