Curt Tilmes
Curt.Tilmes@nasa.gov
Philadelphia Perl Mongers
2017-06-12
curl
is a popular command line tool for transferring data with URLscurl
is a popular command line tool for transferring data with URLs
libcurl
is a C library encompassing the functionality:
libcurl is a free and easy-to-use client-side URL transfer library, supporting DICT, FILE, FTP, FTPS, Gopher, HTTP, HTTPS, IMAP, IMAPS, LDAP, LDAPS, POP3, POP3S, RTMP, RTSP, SCP, SFTP, SMTP, SMTPS, Telnet and TFTP. libcurl supports SSL certificates, HTTP POST, HTTP PUT, FTP uploading, HTTP form based upload, proxies, cookies, user+password authentication (Basic, Digest, NTLM, Negotiate, Kerberos), file transfer resume, http proxy tunneling and more!
libcurl is free, thread-safe, IPv6 compatible, feature rich, well supported, fast, thoroughly documented and is already used by many known, big and successful companies.
curl
is a popular command line tool for transferring data with URLs
libcurl
is a C library encompassing the functionality:
libcurl is a free and easy-to-use client-side URL transfer library, supporting DICT, FILE, FTP, FTPS, Gopher, HTTP, HTTPS, IMAP, IMAPS, LDAP, LDAPS, POP3, POP3S, RTMP, RTSP, SCP, SFTP, SMTP, SMTPS, Telnet and TFTP. libcurl supports SSL certificates, HTTP POST, HTTP PUT, FTP uploading, HTTP form based upload, proxies, cookies, user+password authentication (Basic, Digest, NTLM, Negotiate, Kerberos), file transfer resume, http proxy tunneling and more!
libcurl is free, thread-safe, IPv6 compatible, feature rich, well supported, fast, thoroughly documented and is already used by many known, big and successful companies.
LibCurl
is a Perl 6 module using the NativeCall capability of
Perl 6 to interface directly with libcurl
libcurl
supports two interfaces, both of which are mirrored into
LibCurl
:
Easy
"The easy interface is a synchronous, efficient, quickly used and... yes, easy interface for file transfers."
The libcurl bindings are available via a low-level interface in
LibCurl::EasyHandle
. A friendly Perl 6 class makes them available
through a high level interface as LibCurl::Easy
.
libcurl
supports two interfaces, both of which are mirrored into
LibCurl
:
Easy
"The easy interface is a synchronous, efficient, quickly used and... yes, easy interface for file transfers."
The libcurl bindings are available via a low-level interface in
LibCurl::EasyHandle
. A friendly Perl 6 class makes them available
through a high level interface as LibCurl::Easy
.
Multi
"The multi interface is the asynchronous brother in the family and it also offers multiple transfers using a single thread and more."
The multi bindings are similarly available via a low-level
interface in LibCurl::MultiHandle
, and wrapped in a high level
interface as LibCurl::Multi
.
use LibCurl::Easy;print LibCurl::Easy.new(URL => 'http://example.com').perform.content;
use LibCurl::Easy;print LibCurl::Easy.new(URL => 'http://example.com').perform.content;
Let's break it up into three phases:
1 .new() Construct a new LibCurl::Easy Object
You can pass in many options (~80 so far implemented) to control the
nature of the desired transfer. The only option that is required is
URL
. You can also add options to the handle later.
use LibCurl::Easy;print LibCurl::Easy.new(URL => 'http://example.com').perform.content;
Let's break it up into three phases:
1 .new() Construct a new LibCurl::Easy Object
You can pass in many options (~80 so far implemented) to control the
nature of the desired transfer. The only option that is required is
URL
. You can also add options to the handle later.
2 .perform() Perform the transfer
use LibCurl::Easy;print LibCurl::Easy.new(URL => 'http://example.com').perform.content;
Let's break it up into three phases:
1 .new() Construct a new LibCurl::Easy Object
You can pass in many options (~80 so far implemented) to control the
nature of the desired transfer. The only option that is required is
URL
. You can also add options to the handle later.
2 .perform() Perform the transfer
3 .content() Examine the handle after the transfer, and, in this case, return the content. This is optional. For an upload (PUT/POST, or FTP upload), you might check status.
use LibCurl::Easy;print LibCurl::Easy.new(URL => 'http://example.com').perform.content;
Let's break it up into three phases:
1 .new() Construct a new LibCurl::Easy Object
You can pass in many options (~80 so far implemented) to control the
nature of the desired transfer. The only option that is required is
URL
. You can also add options to the handle later.
2 .perform() Perform the transfer
3 .content() Examine the handle after the transfer, and, in this case, return the content. This is optional. For an upload (PUT/POST, or FTP upload), you might check status.
Most methods perform some action, then return the same handle, so you can easily chain methods as in this example.
Because those basic actions are so frequent, there are some shortcuts which perform them:
use LibCurl::HTTP :subs;print get 'http://example.com';
Because those basic actions are so frequent, there are some shortcuts which perform them:
use LibCurl::HTTP :subs;print get 'http://example.com';
There is also a version that decodes as JSON into a data structure:
use LibCurl::HTTP :subs;print jget('http://example.com/something-that-returns-json')<someval>;
my $curl = LibCurl::Easy.new(someoption => 'something');
my $curl = LibCurl::Easy.new(someoption => 'something');
.setopt()
$curl.setopt(someoption => 'something');
my $curl = LibCurl::Easy.new(someoption => 'something');
.setopt()
$curl.setopt(someoption => 'something');
FALLBACK
method:$curl.someoption('something');
my $curl = LibCurl::Easy.new(someoption => 'something');
.setopt()
$curl.setopt(someoption => 'something');
FALLBACK
method:$curl.someoption('something');
You can shortcut with :someoption
or :!someoption
for Boolean options.
These all return the handle, so you can chain them.
For the most part, these are identical to libcurl
options
CURLOPT_SOMETHING
, just remove the CURLOPT_
and lowercase.
CAinfo CApath URL accepttimeout-ms accept-encoding address-scope append autoreferer buffersize certinfo cookie cookiefile cookiejar cookielist customrequest dirlistonly failonerror followlocation forbid-reuse fresh-connect ftp-skip-pasv-ip ftp-use-eprt ftp-use-epsv ftpport header http-version httpauth httpget httpproxytunnel infilesize low-speed-limit low-speed-time maxconnects maxfilesize maxredirs max-send-speed max-recv-speed netrc nobody noprogress nosignal password post postfields postfieldsize protocols proxy proxyauth proxyport proxytype proxyuserpwd range redir-protocols referer resume-from ssl-verifyhost ssl-verifypeer timecondition timeout timeout-ms timevalue unrestricted-auth use-ssl useragent username userpwd verbose wildcardmatch
Some fun ones:
my $curl = LibCurl::Easy.new(:verbose, URL => "http://...");
When debugging always set :verbose
and you'll get a dump of the
actual HTTP transaction, very handy.
Some fun ones:
my $curl = LibCurl::Easy.new(:verbose, URL => "http://...");
When debugging always set :verbose
and you'll get a dump of the
actual HTTP transaction, very handy.
:followlocation
= automatically follow a Location:
header as part
of an HTTP 3xx response.
Some fun ones:
my $curl = LibCurl::Easy.new(:verbose, URL => "http://...");
When debugging always set :verbose
and you'll get a dump of the
actual HTTP transaction, very handy.
:followlocation
= automatically follow a Location:
header as part
of an HTTP 3xx response.
:nobody
= Doesn't transfer the body of the request, for HTTP this
is equivalent to a HEAD
request.
Some fun ones:
my $curl = LibCurl::Easy.new(:verbose, URL => "http://...");
When debugging always set :verbose
and you'll get a dump of the
actual HTTP transaction, very handy.
:followlocation
= automatically follow a Location:
header as part
of an HTTP 3xx response.
:nobody
= Doesn't transfer the body of the request, for HTTP this
is equivalent to a HEAD
request.
proxy
= Set a proxy to use for the transfer.
Some fun ones:
my $curl = LibCurl::Easy.new(:verbose, URL => "http://...");
When debugging always set :verbose
and you'll get a dump of the
actual HTTP transaction, very handy.
:followlocation
= automatically follow a Location:
header as part
of an HTTP 3xx response.
:nobody
= Doesn't transfer the body of the request, for HTTP this
is equivalent to a HEAD
request.
proxy
= Set a proxy to use for the transfer.
private
= Store any Perl object you want access to later.
There are a few special options that set headers
(useragent,
referer,
cookie), there
are some extra options for headers: Content-MD5
, Content-Type
,
Content-Length
, Host
, Accept
, Expect
, Transfer-Encoding
.
$curl.Host('somewhere.com'); # or $curl.setopt(Host => 'somewhere.com')$curl.Content-MD5('...'); # or $curl.setopt(Content-MD5 => '...')
There are a few special options that set headers
(useragent,
referer,
cookie), there
are some extra options for headers: Content-MD5
, Content-Type
,
Content-Length
, Host
, Accept
, Expect
, Transfer-Encoding
.
$curl.Host('somewhere.com'); # or $curl.setopt(Host => 'somewhere.com')$curl.Content-MD5('...'); # or $curl.setopt(Content-MD5 => '...')
You can also add any other headers you like:
$curl.set-header(X-My-Header => 'something', X-something => 'foo');
There are a few special options that set headers
(useragent,
referer,
cookie), there
are some extra options for headers: Content-MD5
, Content-Type
,
Content-Length
, Host
, Accept
, Expect
, Transfer-Encoding
.
$curl.Host('somewhere.com'); # or $curl.setopt(Host => 'somewhere.com')$curl.Content-MD5('...'); # or $curl.setopt(Content-MD5 => '...')
You can also add any other headers you like:
$curl.set-header(X-My-Header => 'something', X-something => 'foo');
Clear all except the libcurl special headers:
$curl.clear-header;
download => 'myfile'
upload => 'myfile'
send => 'something'
send => $mybuf
If you don't specify a download filename, it will stash all incoming
content in $curl.buf
.
You can also access that content decoded as a UTF-8
Str
with
$curl.content
.
You can change encoding if you want $curl.content('utf-16')
.
say $curl.getinfo('effective-url');say $curl.getinfo('response-code');say $curl.getinfo(<effective-url response-code>); # Hash with those keyssay $curl.getinfo; # Hash of all info fieldssay $curl.response-code;
say $curl.getinfo('effective-url');say $curl.getinfo('response-code');say $curl.getinfo(<effective-url response-code>); # Hash with those keyssay $curl.getinfo; # Hash of all info fieldssay $curl.response-code;
Fields currently defined are:
appconnect_time certinfo condition-unmet connect-time content-type cookielist effective-url ftp-entry-path header-size http-connectcode httpauth-avail lastsocket local-ip local-port namelookup-time num-connects os-errno pretransfer-time primary-ip primary-port proxyauth-avail redirect-url request-size response-code rtsp-client-cseq rtsp-cseq-recv rtsp-server-cseq rtsp-session-id size-download size-upload speed-download speed-upload ssl-engines total-time
After a transfer, you can also check out the headers returned by the server:
say $curl.get-header('Content-Length');say $curl.receiveheaders<Content-Length>; # Hash of all headerssay $curl.Content-Length;
Most real errors will throw an X::LibCurl
exception
The failonerror
option will force an exception on an HTTP code ≥
400 (not usually an error from the LibCurl
perspective).
You can check response code with $curl.response-code.
:verbose
option will just dump some good stuff to STDOUT
Create a debug subroutine:
sub debug(LibCurl::Easy $easy, CURL-INFO-TYPE $type, Buf $buf) {...}$curl.setopt(debugfunction => &debug);
Gets called periodically:
You can enable the simple curl
progress printing by :!noprogress
.
(Yes, this seems backwards..)
% Total % Received % Xferd Average Speed Time Time Time Current Dload Upload Total Spent Left Speed100 364M 100 364M 0 0 104M 0 0:00:03 0:00:03 --:--:-- 104M
You can also install your own progress function:
sub xferinfo(LibCurl::Easy $easy, $dltotal, $dlnow, $ultotal, $ulnow){...}$curl.setopt(xferinfofunction => &xferinfo);
my $curl = LibCurl::Easy.new(URL => 'http://...');# normal field$curl.formadd(name => 'fieldname', contents => 'something');# upload a file from disk, give optional filename or contenttype$curl.formadd(name => 'fieldname', file => 'afile.txt', filename => 'alternate.name.txt', contenttype => 'image/jpeg');# Send a Blob of contents, but as a file with a filename$curl.formadd(name => 'fieldname', buffer => 'some.file.name.txt', bufferptr => "something".encode);$curl.perform;
Construct an Easy
handle for each desired transfer, then perform
them all simultaneously.
use LibCurl::Easy;use LibCurl::Multi;my $curl1 = LibCurl::Easy.new(:verbose, :followlocation, URL => 'http://example.com', download => './myfile1.html');my $curl2 = LibCurl::Easy.new(:verbose, :followlocation, URL => 'http://example.com', download => './myfile2.html');LibCurl:Multi.new.add-handle($curl1, $curl2).perform;say $curl1.statusline;say $curl2.statusline;
use LibCurl::Easy;use LibCurl::Multi;my $curl1 = LibCurl::Easy.new(:followlocation, URL => 'http://example.com', download => 'myfile1.html');my $curl2 = LibCurl::Easy.new(:followlocation, URL => 'http://example.com', download => 'myfile2.html');sub callback(LibCurl::Easy $easy, Exception $e){ die $e if $e; say $easy.response-code; say $easy.statusline;}my $multi = LibCurl::Multi.new(callback => &callback);$multi.add-handle($curl1, $curl2);$multi.perform;
Perl 6 implementation still in development, please try it out and let me know what you like/don't like.
Multi interface is in particular could be extended to support a more perl6-ish interface.
Perl 6 implementation still in development, please try it out and let me know what you like/don't like.
Multi interface is in particular could be extended to support a more perl6-ish interface.
Slides produced with remark
curl
is a popular command line tool for transferring data with URLsKeyboard shortcuts
↑, ←, Pg Up, k | Go to previous slide |
↓, →, Pg Dn, Space, j | Go to next slide |
Home | Go to first slide |
End | Go to last slide |
Number + Return | Go to specific slide |
b / m / f | Toggle blackout / mirrored / fullscreen mode |
c | Clone slideshow |
p | Toggle presenter mode |
t | Restart the presentation timer |
?, h | Toggle this help |
Esc | Back to slideshow |