Compare commits
216 Commits
curl-7_7_a
...
curl-7_7_1
Author | SHA1 | Date | |
---|---|---|---|
![]() |
50a53d4eec | ||
![]() |
6bd1ed034a | ||
![]() |
fa491ed910 | ||
![]() |
66a1e3df69 | ||
![]() |
28497e7ee4 | ||
![]() |
87c7f403a9 | ||
![]() |
1a2c3acb3b | ||
![]() |
b54d752783 | ||
![]() |
b1328430c9 | ||
![]() |
34efa74a59 | ||
![]() |
794d08a728 | ||
![]() |
0abc999c4d | ||
![]() |
3e65062be2 | ||
![]() |
45ffb16c2a | ||
![]() |
0b8b0b7c86 | ||
![]() |
053bf49bd2 | ||
![]() |
8b08dfed38 | ||
![]() |
ba3a3553dc | ||
![]() |
6a26104724 | ||
![]() |
8b35b89f4d | ||
![]() |
31f9d4016d | ||
![]() |
bb601731ea | ||
![]() |
9a85172896 | ||
![]() |
a0eb52bee1 | ||
![]() |
6235a8d969 | ||
![]() |
0d6a87ed7a | ||
![]() |
b6241b3c89 | ||
![]() |
1e14f8d4c7 | ||
![]() |
bc5954fe2d | ||
![]() |
02f6894af5 | ||
![]() |
76576cd1e2 | ||
![]() |
997672ba9a | ||
![]() |
ec1f42a154 | ||
![]() |
aa1c3bb46d | ||
![]() |
95f0714ff8 | ||
![]() |
c050619b36 | ||
![]() |
58085dbbf6 | ||
![]() |
546572da04 | ||
![]() |
005536cc28 | ||
![]() |
3d5b6aa3b0 | ||
![]() |
f8d883355d | ||
![]() |
c23df41d83 | ||
![]() |
8e7b261984 | ||
![]() |
08e3d034ef | ||
![]() |
2f869f89ff | ||
![]() |
0f310a5001 | ||
![]() |
ad1abee441 | ||
![]() |
669709f80e | ||
![]() |
ea409d0374 | ||
![]() |
eaaa1a1fd4 | ||
![]() |
78b4851da1 | ||
![]() |
38c47803dd | ||
![]() |
455663ba5e | ||
![]() |
efb5d9a403 | ||
![]() |
b1a5208e6b | ||
![]() |
e6dacd92ec | ||
![]() |
952b3a2c0f | ||
![]() |
721f9bca84 | ||
![]() |
ad4d5fabf8 | ||
![]() |
aa860990ad | ||
![]() |
0fa9135d9f | ||
![]() |
8f0114a4dd | ||
![]() |
5980c2977b | ||
![]() |
19f8d71508 | ||
![]() |
6f3bccd911 | ||
![]() |
96f81a5c4a | ||
![]() |
ca05d1b59c | ||
![]() |
895dc5e530 | ||
![]() |
bcc6ca6fd1 | ||
![]() |
d538241a58 | ||
![]() |
71b4b2ffa9 | ||
![]() |
65b4a63f56 | ||
![]() |
ecbee01f4b | ||
![]() |
34fed76a35 | ||
![]() |
0adf0cfde7 | ||
![]() |
d6c456db85 | ||
![]() |
36c88343d3 | ||
![]() |
2360e5ce12 | ||
![]() |
d445eac162 | ||
![]() |
e0a6d20e20 | ||
![]() |
3bb979b897 | ||
![]() |
010daec776 | ||
![]() |
e2b0ad8429 | ||
![]() |
6eed95103a | ||
![]() |
4eb2a165e8 | ||
![]() |
b7fc1e45b5 | ||
![]() |
3395a2fa9e | ||
![]() |
a564a54e21 | ||
![]() |
92186dc3d3 | ||
![]() |
7bd6507eec | ||
![]() |
d4cc810de3 | ||
![]() |
bea7bbee1b | ||
![]() |
fe64570d5d | ||
![]() |
df6ad8d8d6 | ||
![]() |
f8e1fc32de | ||
![]() |
8c6d56f1f9 | ||
![]() |
1841c8ee6a | ||
![]() |
70793595fe | ||
![]() |
28a8e1602d | ||
![]() |
cce05b9138 | ||
![]() |
72a7fd4dc7 | ||
![]() |
9a6a476cf5 | ||
![]() |
5d0efedd2d | ||
![]() |
a426818a78 | ||
![]() |
bfe413d8bd | ||
![]() |
dbbd20646f | ||
![]() |
b8fe4deb13 | ||
![]() |
332a016e3c | ||
![]() |
3738e4bdc0 | ||
![]() |
3201d2dafa | ||
![]() |
0a1e002ca4 | ||
![]() |
9195bb64d4 | ||
![]() |
11ee547a0e | ||
![]() |
147de35d41 | ||
![]() |
e16e9b91ae | ||
![]() |
f9cde0646f | ||
![]() |
195233ed5c | ||
![]() |
048e654514 | ||
![]() |
dfbd45142d | ||
![]() |
ff681f7bfd | ||
![]() |
60bbb64a81 | ||
![]() |
c622f2bb4e | ||
![]() |
cd59f13da6 | ||
![]() |
11d718bf52 | ||
![]() |
8e8846d876 | ||
![]() |
7d562bb685 | ||
![]() |
20ddd35669 | ||
![]() |
063f88cd14 | ||
![]() |
87b0b7cab9 | ||
![]() |
70d0d9d4da | ||
![]() |
4ae3bd71ea | ||
![]() |
a9390665b8 | ||
![]() |
fb7a6e3423 | ||
![]() |
cc99e3f7de | ||
![]() |
e6b40bb6ac | ||
![]() |
f2fd1b8856 | ||
![]() |
cb4efcf275 | ||
![]() |
56a27d608a | ||
![]() |
46c9075eab | ||
![]() |
d95fa648e9 | ||
![]() |
563ad213dc | ||
![]() |
0121d7d731 | ||
![]() |
8495fac1c5 | ||
![]() |
38c349f751 | ||
![]() |
542df800ab | ||
![]() |
3e88b1cac5 | ||
![]() |
d774b10afb | ||
![]() |
b449b94393 | ||
![]() |
a6cb9b08b2 | ||
![]() |
440a3101d0 | ||
![]() |
9778a5356b | ||
![]() |
de7dcdbc54 | ||
![]() |
070968abbc | ||
![]() |
e97fc2aab5 | ||
![]() |
a23ac24192 | ||
![]() |
9ee14644a7 | ||
![]() |
c576e114b9 | ||
![]() |
639a7982ba | ||
![]() |
5bbe189420 | ||
![]() |
93ff159e32 | ||
![]() |
8eb8a0a8e4 | ||
![]() |
a4af638867 | ||
![]() |
75a9a87ec2 | ||
![]() |
b5ba011110 | ||
![]() |
e9b763ff05 | ||
![]() |
ac0bad2433 | ||
![]() |
67d5c0a970 | ||
![]() |
580896d615 | ||
![]() |
11693c0faa | ||
![]() |
26cd8eda4a | ||
![]() |
8cd3f44040 | ||
![]() |
2b30bfc349 | ||
![]() |
8ec4dba599 | ||
![]() |
1efec6572e | ||
![]() |
781dd7a9bf | ||
![]() |
beb8761b22 | ||
![]() |
071c7de9fe | ||
![]() |
3e7ebcd051 | ||
![]() |
c67952fc5c | ||
![]() |
7d7c24f915 | ||
![]() |
0dc8c4d451 | ||
![]() |
9cf4434ae2 | ||
![]() |
8ccd8b6dbc | ||
![]() |
b4f70aa2c8 | ||
![]() |
f54a282ccc | ||
![]() |
2a11bdc216 | ||
![]() |
5cd4c3ed24 | ||
![]() |
147a673063 | ||
![]() |
9ce5827fc1 | ||
![]() |
97f1c93674 | ||
![]() |
e61ceaf1bd | ||
![]() |
1118612249 | ||
![]() |
a23db7b7c7 | ||
![]() |
f6b6dff46a | ||
![]() |
55b8ceac18 | ||
![]() |
bcf448ee32 | ||
![]() |
91e4da7ddb | ||
![]() |
2873c18132 | ||
![]() |
5dd0a8a63e | ||
![]() |
2103dc41f5 | ||
![]() |
2ef13230cb | ||
![]() |
9479ac6dda | ||
![]() |
4e878eae79 | ||
![]() |
1e8e90a220 | ||
![]() |
fe95c7dc34 | ||
![]() |
6dae34d5da | ||
![]() |
36c621c9df | ||
![]() |
1717963e3d | ||
![]() |
4646a1ffa9 | ||
![]() |
0cb4eba002 | ||
![]() |
5eba359b5d | ||
![]() |
07ce7539a8 | ||
![]() |
c21f848c1c | ||
![]() |
84e94fda8b | ||
![]() |
ebd6897b10 | ||
![]() |
5ab8a9d32f |
272
CHANGES
272
CHANGES
@@ -6,7 +6,277 @@
|
|||||||
|
|
||||||
History of Changes
|
History of Changes
|
||||||
|
|
||||||
** curl 7.7 DOES NOT currently WORK. **
|
Version 7.7.1
|
||||||
|
|
||||||
|
Daniel (3 April 2001)
|
||||||
|
- Puneet Pawaia pointed out two serious problems. Libcurl would attempt to
|
||||||
|
read bad memory during situations when an (ftp) connection attempt failed.
|
||||||
|
Also, the lib/Makefile.vc6 was corrected.
|
||||||
|
|
||||||
|
- More investigations in the Location: following code made me realize that
|
||||||
|
it was not clean enough to work transparantly with persistant and non-
|
||||||
|
persistant connections. I think I've fixed it now.
|
||||||
|
|
||||||
|
Daniel (29 March 2001)
|
||||||
|
- Georg Horn mailed me some corrections for the Curl::easy perl interface.
|
||||||
|
|
||||||
|
- Experimental ftps:// support added. It is basically FTP over SSL for the
|
||||||
|
control connection. It still makes all data transfers going over unencrypted
|
||||||
|
connections. Rainer Weikusat's ftpd-ssl server hack supports this and I used
|
||||||
|
that to verify the functionality.
|
||||||
|
|
||||||
|
Daniel (27 March 2001)
|
||||||
|
- Guenole Bescon discovered that if you set a CURLOPT_TIMEOUT and then tried
|
||||||
|
to get a file from a site and it fails, the SIGALRM would still be sent
|
||||||
|
after the timeout-time, quite inexpectedly!
|
||||||
|
|
||||||
|
- I added an ftp transfer example to docs/examples/ and I also wrote a tiny
|
||||||
|
example makefile that can be used as a start when building one of the
|
||||||
|
examples.
|
||||||
|
|
||||||
|
Version 7.7.1-beta1
|
||||||
|
|
||||||
|
Daniel (26 March 2001)
|
||||||
|
- Mohamed Lrhazi reported problems with 7.6.1 and persistant HTTP/1.0
|
||||||
|
connections (when the server replied a Connection: Keep-Alive) and this
|
||||||
|
problem was not properly dealt with in 7.7 either. A patch was posted to the
|
||||||
|
curl-and-php mailing list.
|
||||||
|
|
||||||
|
Daniel (24 March 2001)
|
||||||
|
- Colin Watson reported about a problem and brought a patch that corrected it,
|
||||||
|
which was about the man page and lines starting with a single quote (') in a
|
||||||
|
way that gnroff doesn't like.
|
||||||
|
|
||||||
|
Daniel (23 March 2001)
|
||||||
|
- Peter Bray reported correctly that the root makefile used make instead of
|
||||||
|
$(MAKE) for the test target.
|
||||||
|
|
||||||
|
- Corrected the Curl::easy perl interface to use curl_easy_setopt() and not
|
||||||
|
curl_setopt() which was removed in 7.7!
|
||||||
|
|
||||||
|
- SM provided updates on three documents (MANUAL, INSTALL and FAQ).
|
||||||
|
|
||||||
|
- When following a Location:, libcurl would sometimes write to the URL string
|
||||||
|
in a way it shouldn't. As the pointer is passed-in to libcurl from an
|
||||||
|
application, we can't be allowed to write to it. The particular bug report
|
||||||
|
from 'nk' that brought this up was because he had a read-only URL that then
|
||||||
|
caused a libcurl crash!
|
||||||
|
|
||||||
|
- No longer reads HEAD responses longer than to the last header. Previously,
|
||||||
|
curl would read the full reply if the connection was a "close" one.
|
||||||
|
|
||||||
|
- libcurl did re-use connections way too much. Doing "curl
|
||||||
|
http://www.{microsoft,ibm}.com" would make it re-use the connection which
|
||||||
|
made the second request return very odd results.
|
||||||
|
|
||||||
|
Daniel (22 March 2001)
|
||||||
|
- Edin Kadribasic made me aware that curl should not re-send POST requests
|
||||||
|
when following 302-redirects. I made 302 work like 303 which means curl uses
|
||||||
|
GET in the following request(s).
|
||||||
|
|
||||||
|
- libcurl now reset the "followed-location" counter on each invoke of
|
||||||
|
curl_easy_perform() as it otherwise would sum up all redirects on the same
|
||||||
|
connection and thus could reach the maxredirs counter wrongly.
|
||||||
|
|
||||||
|
- Jim Drash suggested curl_escape() should not re-encode what already looks
|
||||||
|
like an encoded sequence and I think that's a fair suggestion.
|
||||||
|
|
||||||
|
Version 7.7
|
||||||
|
|
||||||
|
Daniel (22 March 2001)
|
||||||
|
- The configure script now fails with an error message if gethostbyname_r() is
|
||||||
|
detected but it couldn't figure out how to invoke it (what amount of
|
||||||
|
arguments it is supposed to get). Reports from Andr<64>s Garc<72>a made me aware
|
||||||
|
of this need.
|
||||||
|
|
||||||
|
- Talking with Jim Drash made me finally put the curl_escape and curl_unescape
|
||||||
|
functions in the curl.h include file and write man pages for them. The
|
||||||
|
escape function was modified to use the same interface as the unescape one
|
||||||
|
had.
|
||||||
|
|
||||||
|
- No bug reports at all on the latest betas. Release time coming up.
|
||||||
|
|
||||||
|
Version 7.7-beta5
|
||||||
|
|
||||||
|
Daniel (19 March 2001)
|
||||||
|
- Georg Ottinger reported problems with using -C together with -L in the sense
|
||||||
|
that the -C info got lost when it was redirected. I could not repeat this
|
||||||
|
problem on the 7.7 branch why I leave this for the moment. Test case 39 was
|
||||||
|
added to do exactly this, and it seems to do right.
|
||||||
|
|
||||||
|
- Christian Robottom Reis reported how his 7.7 beta didn't successfully do
|
||||||
|
form posts as elegantly as 7.6.1 did. Indeed, this was a flaw in the header
|
||||||
|
engine, as HTTP 1.1 has introduced a new 100 "transient" return code for PUT
|
||||||
|
and POST operations that I need to add support for. Section 8.2.3 in RFC2616
|
||||||
|
has all the details. Seems to work now!
|
||||||
|
|
||||||
|
Daniel (16 March 2001)
|
||||||
|
- After having experienced another machine break-down, we're back.
|
||||||
|
|
||||||
|
- Georg Horn's perl interface Curl::easy is now included in the curl release
|
||||||
|
archive. The perl/ directory is now present. Please help me with docs,
|
||||||
|
examples and updates you think fit.
|
||||||
|
|
||||||
|
- Made a new php/ directory in the release archive and moved the PHP examples
|
||||||
|
into a subdirectory in there. Not much PHP info yet, but I plan to. Please
|
||||||
|
help me here as well!
|
||||||
|
|
||||||
|
- Made libcurl return error if a transfer is aborted in the middle of a
|
||||||
|
"chunk". It actually enables libcurl to discover premature transfer aborts
|
||||||
|
even if the Content-Length: size is unknown.
|
||||||
|
|
||||||
|
Daniel (15 March 2001)
|
||||||
|
- Added --connect-timeout to curl, which sets the new CURLOPT_CONNECTTIMEOUT
|
||||||
|
option in libcurl. It limits the time curl is allowed to spend in the
|
||||||
|
connection phase. This differs from -m/--max-time that limits the entire
|
||||||
|
file transfer operation. Requested by Larry Fahnoe and others.
|
||||||
|
|
||||||
|
I also updated the curl.1 and curl_easy_setopt.3 man pages and removed the
|
||||||
|
item from the TODO.
|
||||||
|
|
||||||
|
Version 7.7-beta4
|
||||||
|
|
||||||
|
Daniel (14 March 2001)
|
||||||
|
- Made curl grok IPv6 with HTTP proxies and got everything to compile nicely
|
||||||
|
again when ENABLE_IPV6 is set.
|
||||||
|
|
||||||
|
I need to remake things in the test suite. I can't test the FTP parts with
|
||||||
|
curl built for IPv6 as it uses a different set of FTP commands then!
|
||||||
|
|
||||||
|
- I fell onto a bug report on php.net (posted by Lars Torben Wilson) that was
|
||||||
|
a report meant for our project. Anyway, it said the .netrc parsing didn't
|
||||||
|
work as supposed, and as I agreed with Lars, I made the netrc parser use
|
||||||
|
getpwuid() to figure out the home directory of the effective user and try
|
||||||
|
that netrc. It still uses the environment variable HOME for those that don't
|
||||||
|
have that function or if the user doesn't return valid pwd info.
|
||||||
|
|
||||||
|
- Edin Kadribaic posted a bug report where he got a crash when a fetch with
|
||||||
|
user+password in the URL followed a Location: to a second URL (absolute,
|
||||||
|
without name+password). This bug has been around for a long while and
|
||||||
|
crashes due to a read at address zero. Fixed now. Wrote test case 38, that
|
||||||
|
tests this.
|
||||||
|
|
||||||
|
- Modified the test suite's httpserver slightly to append all client request
|
||||||
|
data to its log file so that the test script now better can verify a range
|
||||||
|
of requests and not only the last one, as it did previously.
|
||||||
|
|
||||||
|
- Updated the curl man page with --random-file and --egd-file details.
|
||||||
|
|
||||||
|
Version 7.7-beta3
|
||||||
|
|
||||||
|
Daniel (14 March 2001)
|
||||||
|
- Bj<42>rn Stenberg provided similar fixes as J<>rn did and some additional patches
|
||||||
|
for non-SSL compiles.
|
||||||
|
|
||||||
|
- I increased the interface number for libcurl as I've removed the low level
|
||||||
|
functions from the interface. I also took this opportunity to rename the
|
||||||
|
Curl_strequal function to curl_strequal and Curl_strnequal to
|
||||||
|
curl_strnequal, as they're public libcurl functions (even if they're still
|
||||||
|
undocumented).
|
||||||
|
|
||||||
|
This will make older programs not capable of using the new libcurl with
|
||||||
|
just a drop-in replacement.
|
||||||
|
|
||||||
|
- J<>rn Hartroth updated stuff for win32 compiles:
|
||||||
|
o config-win32.h was fixed for socklen_t
|
||||||
|
o lib/ssluse.c had a bad #endif placement
|
||||||
|
o lib/file.c was made to compile on win32 again
|
||||||
|
o lib/Makefile.m32 was updated with the new files
|
||||||
|
o lib/libcurl.def matches the current interface state
|
||||||
|
|
||||||
|
Daniel (13 March 2001)
|
||||||
|
- It only took an hour or so before J<>rn Hartroth found a problem in the
|
||||||
|
chunked transfer-encoding. Given his fine example-site, I could easily spot
|
||||||
|
the problem and when I re-read the spec (the part I have pasted in the top
|
||||||
|
of the http_chunks.h file), I realized I had made my state-machine slightly
|
||||||
|
wrong and didn't expect/handle the trailing CRLF that comes after the data
|
||||||
|
in each chunk (and those extra two bytes sure feel wasted).
|
||||||
|
|
||||||
|
Had to modify test case 34 to match this as well.
|
||||||
|
|
||||||
|
Version 7.7-beta2
|
||||||
|
|
||||||
|
Daniel (13 March 2001)
|
||||||
|
- Added the policy stuff to the curl_easy_setopt man page for the two supported
|
||||||
|
policies.
|
||||||
|
|
||||||
|
- Implemented some support for the CURLOPT_CLOSEPOLICY option. The policies
|
||||||
|
CURLCLOSEPOLICY_LEAST_RECENTLY_USED and CURLCLOSEPOLICY_OLDEST are now
|
||||||
|
supported, and the "least recently used" is used as default if no policy
|
||||||
|
is chosen.
|
||||||
|
|
||||||
|
Daniel (12 March 2001)
|
||||||
|
- Added CURLOPT_RANDOM_FILE and CURLOPT_EGDSOCKET to libcurl for seeding the
|
||||||
|
SSL random engine. The random seeding support was also brought to the curl
|
||||||
|
client with the new options --random-file <file> and --egd-file <file>. I
|
||||||
|
need some people to really test this to know they work as supposed. Remember
|
||||||
|
that libcurl now informs (if verbose is on) if the random seed is considered
|
||||||
|
weak (HTTPS connections).
|
||||||
|
|
||||||
|
- Made the chunked transfer-encoding engine detected bad formatted data length
|
||||||
|
and return error if so (we can't possibly extract sensible data if this is
|
||||||
|
the case). Added a test case that detects this. Number 36. Now there are 60
|
||||||
|
test cases.
|
||||||
|
|
||||||
|
- Added 5 new libcurl options to curl/curl.h that can be used to control the
|
||||||
|
persistant connection support in libcurl. They're also documented (fairly
|
||||||
|
thoroughly) in the curl_easy_setopt.3 man page. Three of them are now
|
||||||
|
implemented, although not really tested at this point... Anyway, the new
|
||||||
|
implemented options are named CURLOPT_MAXCONNECTS, CURLOPT_FRESH_CONNECT,
|
||||||
|
CURLOPT_FORBID_REUSE. The ones still left to write code for are:
|
||||||
|
CURLOPT_CLOSEPOLICY and its related option CURLOPT_CLOSEFUNCTION.
|
||||||
|
|
||||||
|
- Made curl (the actual command line tool) use the new libcurl 7.7 persistant
|
||||||
|
connection support by re-using the same curl handle for every specified file
|
||||||
|
transfer and after some more test case tweaking we have 100% test case OK.
|
||||||
|
I made some test cases return HTTP/1.0 now to make sure that works as well.
|
||||||
|
|
||||||
|
- Had to add 'Connection: close' to the headers of a bunch of test cases so
|
||||||
|
that curl behaves "old-style" since the test http server doesn't do multiple
|
||||||
|
connections... Now I get 100% test case OK.
|
||||||
|
|
||||||
|
- The curl.haxx.se site, the main curl mailing list and my personal email are
|
||||||
|
all dead today due to power blackout in the area where the main servers are
|
||||||
|
located. Horrible.
|
||||||
|
|
||||||
|
- I've made persistance work over a squid HTTP proxy. I find it disturbing
|
||||||
|
that it uses headers that aren't present in any HTTP standard though
|
||||||
|
(Proxy-Connection:) and that makes me feel that I'm now on the edge of what
|
||||||
|
the standard actually defines. I need to get this code excercised on a lot
|
||||||
|
of different HTTP proxies before I feel safe.
|
||||||
|
|
||||||
|
Now I'm facing the problem with my test suite servers (both FTP and HTTP)
|
||||||
|
not supporting persistant connections and libcurl is doing them now. I have
|
||||||
|
to fix the test servers to get all the test cases do OK.
|
||||||
|
|
||||||
|
Daniel (8 March 2001)
|
||||||
|
- Guenole Bescon reported that libcurl did output errors to stderr even if
|
||||||
|
MUTE and NOPROGRESS was set. It turned out to be a bug and happens if
|
||||||
|
there's an error and no ERRORBUFFER is set. This is now corrected.
|
||||||
|
|
||||||
|
Version 7.7-beta1
|
||||||
|
|
||||||
|
Daniel (8 March 2001)
|
||||||
|
- "Transfer-Encoding: chunked" is no longer any trouble for libcurl. I've
|
||||||
|
added two source files and I've run some test downloads that look fine.
|
||||||
|
|
||||||
|
- HTTP HEAD works too, even on 1.1 servers.
|
||||||
|
|
||||||
|
Daniel (5 March 2001)
|
||||||
|
- The current 57 test cases now pass OK. It would suggest that libcurl works
|
||||||
|
using the old-style with one connection per handle. The test suite doesn't
|
||||||
|
handle multiple connections yet so there are no test cases for this.
|
||||||
|
|
||||||
|
- I patched the telnet.c heavily to not use any global variables anymore. It
|
||||||
|
should make it a lot nicer library-wise.
|
||||||
|
|
||||||
|
- The file:// support was modified slightly to use the internal connect-first-
|
||||||
|
then-do approach.
|
||||||
|
|
||||||
|
Daniel (4 March 2001)
|
||||||
|
- More bugs erased.
|
||||||
|
|
||||||
|
Version 7.7-alpha2
|
||||||
|
|
||||||
Daniel (4 March 2001)
|
Daniel (4 March 2001)
|
||||||
- Now, there's even a basic check that a re-used connection is still alive
|
- Now, there's even a basic check that a re-used connection is still alive
|
||||||
|
2
CVS-INFO
2
CVS-INFO
@@ -10,7 +10,7 @@ memanalyze.pl is for analyzing the output generated by curl if -DMALLOCDEBUG
|
|||||||
|
|
||||||
Makefile.dist is included as the root Makefile in distribution archives
|
Makefile.dist is included as the root Makefile in distribution archives
|
||||||
|
|
||||||
perl/ is a subdirectory with various perl scripts
|
perl/contrib/ is a subdirectory with various perl scripts
|
||||||
|
|
||||||
To build after having extracted everything from CVS, do this:
|
To build after having extracted everything from CVS, do this:
|
||||||
|
|
||||||
|
@@ -8,7 +8,7 @@ EXTRA_DIST = \
|
|||||||
CHANGES LEGAL maketgz MITX.txt MPL-1.1.txt \
|
CHANGES LEGAL maketgz MITX.txt MPL-1.1.txt \
|
||||||
config-win32.h reconf packages/README Makefile.dist
|
config-win32.h reconf packages/README Makefile.dist
|
||||||
|
|
||||||
SUBDIRS = docs lib src include tests packages
|
SUBDIRS = docs lib src include tests packages perl php
|
||||||
|
|
||||||
# create a root makefile in the distribution:
|
# create a root makefile in the distribution:
|
||||||
dist-hook:
|
dist-hook:
|
||||||
@@ -17,7 +17,7 @@ dist-hook:
|
|||||||
check: test
|
check: test
|
||||||
|
|
||||||
test:
|
test:
|
||||||
@(cd tests; make quiet-test)
|
@(cd tests; $(MAKE) quiet-test)
|
||||||
|
|
||||||
#
|
#
|
||||||
# Build source and binary rpms. For rpm-3.0 and above, the ~/.rpmmacros
|
# Build source and binary rpms. For rpm-3.0 and above, the ~/.rpmmacros
|
||||||
|
@@ -30,16 +30,16 @@ ssl:
|
|||||||
make
|
make
|
||||||
|
|
||||||
borland:
|
borland:
|
||||||
cd lib; make -f Makefile.b32
|
cd lib & make -f Makefile.b32
|
||||||
cd src; make -f Makefile.b32
|
cd src & make -f Makefile.b32
|
||||||
|
|
||||||
mingw32:
|
mingw32:
|
||||||
cd lib; make -f Makefile.m32
|
cd lib & make -f Makefile.m32
|
||||||
cd src; make -f Makefile.m32
|
cd src & make -f Makefile.m32
|
||||||
|
|
||||||
mingw32-ssl:
|
mingw32-ssl:
|
||||||
cd lib; make -f Makefile.m32 SSL=1
|
cd lib & make -f Makefile.m32 SSL=1
|
||||||
cd src; make -f Makefile.m32 SSL=1
|
cd src & make -f Makefile.m32 SSL=1
|
||||||
|
|
||||||
vc:
|
vc:
|
||||||
cd lib
|
cd lib
|
||||||
|
@@ -43,3 +43,11 @@
|
|||||||
/* Define this to 'int' if ssize_t is not an available typedefed type */
|
/* Define this to 'int' if ssize_t is not an available typedefed type */
|
||||||
#undef ssize_t
|
#undef ssize_t
|
||||||
|
|
||||||
|
/* Define this to 'int' if socklen_t is not an available typedefed type */
|
||||||
|
#undef socklen_t
|
||||||
|
|
||||||
|
/* Define this as a suitable file to read random data from */
|
||||||
|
#undef RANDOM_FILE
|
||||||
|
|
||||||
|
/* Define this to your Entropy Gathering Daemon socket pathname */
|
||||||
|
#undef EGD_SOCKET
|
||||||
|
@@ -26,6 +26,9 @@
|
|||||||
/* Define this to 'int' if ssize_t is not an available typedefed type */
|
/* Define this to 'int' if ssize_t is not an available typedefed type */
|
||||||
#define ssize_t int
|
#define ssize_t int
|
||||||
|
|
||||||
|
/* Define this to 'int' if socklen_t is not an available typedefed type */
|
||||||
|
#define socklen_t int
|
||||||
|
|
||||||
/* Define if you have the ANSI C header files. */
|
/* Define if you have the ANSI C header files. */
|
||||||
#define STDC_HEADERS 1
|
#define STDC_HEADERS 1
|
||||||
|
|
||||||
|
74
configure.in
74
configure.in
@@ -295,6 +295,14 @@ exit (rc != 0 ? 1 : 0); }],[
|
|||||||
[ac_cv_gethostbyname_args=0])],
|
[ac_cv_gethostbyname_args=0])],
|
||||||
[ac_cv_gethostbyname_args=0])])
|
[ac_cv_gethostbyname_args=0])])
|
||||||
|
|
||||||
|
if test "$ac_cv_func_gethostbyname_r" = "yes"; then
|
||||||
|
if test "$ac_cv_gethostbyname_args" = "0"; then
|
||||||
|
dnl there's a gethostbyname_r() function, but we don't know how
|
||||||
|
dnl many arguments it wants!
|
||||||
|
AC_MSG_ERROR([couldn't figure out how to use gethostbyname_r()])
|
||||||
|
fi
|
||||||
|
fi
|
||||||
|
|
||||||
])
|
])
|
||||||
|
|
||||||
dnl **********************************************************************
|
dnl **********************************************************************
|
||||||
@@ -391,6 +399,36 @@ AC_CHECK_FUNC(gethostname, , AC_CHECK_LIB(ucb, gethostname))
|
|||||||
dnl dl lib?
|
dnl dl lib?
|
||||||
AC_CHECK_FUNC(dlopen, , AC_CHECK_LIB(dl, dlopen))
|
AC_CHECK_FUNC(dlopen, , AC_CHECK_LIB(dl, dlopen))
|
||||||
|
|
||||||
|
dnl **********************************************************************
|
||||||
|
dnl Check for the random seed preferences
|
||||||
|
dnl **********************************************************************
|
||||||
|
|
||||||
|
AC_ARG_WITH(egd-socket,
|
||||||
|
[ --with-egd-socket=FILE Entropy Gathering Daemon socket pathname],
|
||||||
|
[ EGD_SOCKET="$withval" ]
|
||||||
|
)
|
||||||
|
if test -n "$EGD_SOCKET" ; then
|
||||||
|
AC_DEFINE_UNQUOTED(EGD_SOCKET, "$EGD_SOCKET")
|
||||||
|
fi
|
||||||
|
|
||||||
|
dnl Check for user-specified random device
|
||||||
|
AC_ARG_WITH(random,
|
||||||
|
[ --with-random=FILE read randomness from FILE (default=/dev/urandom)],
|
||||||
|
[ RANDOM_FILE="$withval" ],
|
||||||
|
[
|
||||||
|
dnl Check for random device
|
||||||
|
AC_CHECK_FILE("/dev/urandom",
|
||||||
|
[
|
||||||
|
RANDOM_FILE="/dev/urandom";
|
||||||
|
]
|
||||||
|
)
|
||||||
|
]
|
||||||
|
)
|
||||||
|
if test -n "$RANDOM_FILE" ; then
|
||||||
|
AC_SUBST(RANDOM_FILE)
|
||||||
|
AC_DEFINE_UNQUOTED(RANDOM_FILE, "$RANDOM_FILE")
|
||||||
|
fi
|
||||||
|
|
||||||
dnl **********************************************************************
|
dnl **********************************************************************
|
||||||
dnl Check for the presence of Kerberos4 libraries and headers
|
dnl Check for the presence of Kerberos4 libraries and headers
|
||||||
dnl **********************************************************************
|
dnl **********************************************************************
|
||||||
@@ -545,7 +583,8 @@ else
|
|||||||
dnl these can only exist if openssl exists
|
dnl these can only exist if openssl exists
|
||||||
|
|
||||||
AC_CHECK_FUNCS( RAND_status \
|
AC_CHECK_FUNCS( RAND_status \
|
||||||
RAND_screen )
|
RAND_screen \
|
||||||
|
RAND_egd )
|
||||||
|
|
||||||
fi
|
fi
|
||||||
|
|
||||||
@@ -642,6 +681,7 @@ AC_CHECK_HEADERS( \
|
|||||||
winsock.h \
|
winsock.h \
|
||||||
time.h \
|
time.h \
|
||||||
io.h \
|
io.h \
|
||||||
|
pwd.h
|
||||||
)
|
)
|
||||||
|
|
||||||
dnl Check for libz header
|
dnl Check for libz header
|
||||||
@@ -662,6 +702,28 @@ AC_CHECK_SIZEOF(long long, 4)
|
|||||||
# check for ssize_t
|
# check for ssize_t
|
||||||
AC_CHECK_TYPE(ssize_t, int)
|
AC_CHECK_TYPE(ssize_t, int)
|
||||||
|
|
||||||
|
dnl
|
||||||
|
dnl We can't just AC_CHECK_TYPE() for socklen_t since it doesn't appear
|
||||||
|
dnl in the standard headers. We egrep for it in the socket headers and
|
||||||
|
dnl if it is used there we assume we have the type defined, otherwise
|
||||||
|
dnl we search for it with AC_CHECK_TYPE() the "normal" way
|
||||||
|
dnl
|
||||||
|
|
||||||
|
if test "$ac_cv_header_sys_socket_h" = "yes"; then
|
||||||
|
AC_MSG_CHECKING(for socklen_t in sys/socket.h)
|
||||||
|
AC_EGREP_HEADER(socklen_t,
|
||||||
|
sys/socket.h,
|
||||||
|
socklen_t=yes
|
||||||
|
AC_MSG_RESULT(yes),
|
||||||
|
AC_MSG_RESULT(no))
|
||||||
|
fi
|
||||||
|
|
||||||
|
if test "$socklen_t" != "yes"; then
|
||||||
|
# check for socklen_t the standard way if it wasn't found before
|
||||||
|
AC_CHECK_TYPE(socklen_t, int)
|
||||||
|
fi
|
||||||
|
|
||||||
|
|
||||||
dnl Get system canonical name
|
dnl Get system canonical name
|
||||||
AC_CANONICAL_HOST
|
AC_CANONICAL_HOST
|
||||||
AC_DEFINE_UNQUOTED(OS, "${host}")
|
AC_DEFINE_UNQUOTED(OS, "${host}")
|
||||||
@@ -693,7 +755,9 @@ AC_CHECK_FUNCS( socket \
|
|||||||
sigaction \
|
sigaction \
|
||||||
signal \
|
signal \
|
||||||
getpass_r \
|
getpass_r \
|
||||||
strlcat
|
strlcat \
|
||||||
|
getpwuid \
|
||||||
|
geteuid
|
||||||
)
|
)
|
||||||
|
|
||||||
dnl removed 'getpass' check on October 26, 2000
|
dnl removed 'getpass' check on October 26, 2000
|
||||||
@@ -734,5 +798,9 @@ AC_OUTPUT( Makefile \
|
|||||||
packages/Linux/RPM/Makefile \
|
packages/Linux/RPM/Makefile \
|
||||||
packages/Linux/RPM/curl.spec \
|
packages/Linux/RPM/curl.spec \
|
||||||
packages/Linux/RPM/curl-ssl.spec \
|
packages/Linux/RPM/curl-ssl.spec \
|
||||||
tiny/Makefile )
|
perl/Makefile \
|
||||||
|
perl/Curl_easy/Makefile \
|
||||||
|
php/Makefile \
|
||||||
|
php/examples/Makefile
|
||||||
|
)
|
||||||
|
|
||||||
|
17
docs/BUGS
17
docs/BUGS
@@ -6,9 +6,9 @@
|
|||||||
|
|
||||||
BUGS
|
BUGS
|
||||||
|
|
||||||
Curl has grown substantially from that day, several years ago, when I
|
Curl and libcurl have grown substantially since the beginning. At the time
|
||||||
started fiddling with it. When I write this, there are 16500 lines of source
|
of writing (mid March 2001), there are 23000 lines of source code, and by
|
||||||
code, and by the time you read this it has probably grown even more.
|
the time you read this it has probably grown even more.
|
||||||
|
|
||||||
Of course there are lots of bugs left. And lots of misfeatures.
|
Of course there are lots of bugs left. And lots of misfeatures.
|
||||||
|
|
||||||
@@ -21,10 +21,11 @@ BUGS
|
|||||||
http://sourceforge.net/bugs/?group_id=976
|
http://sourceforge.net/bugs/?group_id=976
|
||||||
|
|
||||||
When reporting a bug, you should include information that will help us
|
When reporting a bug, you should include information that will help us
|
||||||
understand what's wrong, what's expected and how to repeat it. You therefore
|
understand what's wrong, what you expected to happen and how to repeat the
|
||||||
need to supply your operating system's name and version number (uname -a
|
bad behaviour. You therefore need to supply your operating system's name and
|
||||||
under a unix is fine), what version of curl you're using (curl -v is fine),
|
version number (uname -a under a unix is fine), what version of curl you're
|
||||||
what URL you were working with and anything else you think matters.
|
using (curl -V is fine), what URL you were working with and anything else
|
||||||
|
you think matters.
|
||||||
|
|
||||||
If curl crashed, causing a core dump (in unix), there is hardly any use to
|
If curl crashed, causing a core dump (in unix), there is hardly any use to
|
||||||
send that huge file to anyone of us. Unless we have an exact same system
|
send that huge file to anyone of us. Unless we have an exact same system
|
||||||
@@ -32,7 +33,7 @@ BUGS
|
|||||||
a stack trace and send that (much smaller) output to us instead!
|
a stack trace and send that (much smaller) output to us instead!
|
||||||
|
|
||||||
The address and how to subscribe to the mailing list is detailed in the
|
The address and how to subscribe to the mailing list is detailed in the
|
||||||
README.curl file.
|
MANUAL file.
|
||||||
|
|
||||||
HOW TO GET A STACK TRACE with a common unix debugger
|
HOW TO GET A STACK TRACE with a common unix debugger
|
||||||
====================================================
|
====================================================
|
||||||
|
@@ -13,7 +13,7 @@ To Think About When Contributing Source Code
|
|||||||
The License Issue
|
The License Issue
|
||||||
|
|
||||||
When contributing with code, you agree to put your changes and new code under
|
When contributing with code, you agree to put your changes and new code under
|
||||||
the same license curl and libcurl is already using.
|
the same license curl and libcurl is already using unless stated otherwise.
|
||||||
|
|
||||||
If you add a larger piece of code, you can opt to make that file or set of
|
If you add a larger piece of code, you can opt to make that file or set of
|
||||||
files to use a different license as long as they don't enfore any changes to
|
files to use a different license as long as they don't enfore any changes to
|
||||||
@@ -26,19 +26,19 @@ Naming
|
|||||||
Try using a non-confusing naming scheme for your new functions and variable
|
Try using a non-confusing naming scheme for your new functions and variable
|
||||||
names. It doesn't necessarily have to mean that you should use the same as in
|
names. It doesn't necessarily have to mean that you should use the same as in
|
||||||
other places of the code, just that the names should be logical,
|
other places of the code, just that the names should be logical,
|
||||||
understandable and be named according to what they're used for.
|
understandable and be named according to what they're used for. File-local
|
||||||
|
functions should be made static.
|
||||||
|
|
||||||
Indenting
|
Indenting
|
||||||
|
|
||||||
Please try using the same indenting levels and bracing method as all the
|
Please try using the same indenting levels and bracing method as all the
|
||||||
other code already does. It makes the source code a lot easier to follow if
|
other code already does. It makes the source code a lot easier to follow if
|
||||||
all of it is written using the same style. I don't ask you to like it, I just
|
all of it is written using the same style. We don't ask you to like it, we
|
||||||
ask you to follow the tradition! ;-)
|
just ask you to follow the tradition! ;-)
|
||||||
|
|
||||||
Commenting
|
Commenting
|
||||||
|
|
||||||
Comment your source code extensively. I don't see myself as a very good
|
Comment your source code extensively. Commented code is quality code and
|
||||||
source commenter, but I try to become one. Commented code is quality code and
|
|
||||||
enables future modifications much more. Uncommented code much more risk being
|
enables future modifications much more. Uncommented code much more risk being
|
||||||
completely replaced when someone wants to extend things, since other persons'
|
completely replaced when someone wants to extend things, since other persons'
|
||||||
source code can get quite hard to read.
|
source code can get quite hard to read.
|
||||||
@@ -71,9 +71,9 @@ Separate Patches Doing Different Things
|
|||||||
Patch Against Recent Sources
|
Patch Against Recent Sources
|
||||||
|
|
||||||
Please try to get the latest available sources to make your patches
|
Please try to get the latest available sources to make your patches
|
||||||
against. It makes my life so much easier. The very best is if you get the
|
against. It makes the life of the developers so much easier. The very best is
|
||||||
most up-to-date sources from the CVS repository, but the latest release
|
if you get the most up-to-date sources from the CVS repository, but the
|
||||||
archive is quite OK as well!
|
latest release archive is quite OK as well!
|
||||||
|
|
||||||
Document
|
Document
|
||||||
|
|
||||||
@@ -91,9 +91,9 @@ Write Access to CVS Repository
|
|||||||
|
|
||||||
Test Cases
|
Test Cases
|
||||||
|
|
||||||
Since the introduction of the test suite, we will get the possibility to
|
Since the introduction of the test suite, we can quickly verify that the main
|
||||||
quickly verify that the main features are working as supposed to. To maintain
|
features are working as they're supposed to. To maintain this situation and
|
||||||
this situation and improve it, all new features and functions that are added
|
improve it, all new features and functions that are added need to be tested
|
||||||
need tro be tested. Every feature that is added should get at least one valid
|
in the test suite. Every feature that is added should get at least one valid
|
||||||
test case that verifies that it works as documented. If every submitter also
|
test case that verifies that it works as documented. If every submitter also
|
||||||
post a few test cases, it won't end up as a heavy burden on a single person!
|
post a few test cases, it won't end up as a heavy burden on a single person!
|
||||||
|
153
docs/FAQ
153
docs/FAQ
@@ -1,4 +1,4 @@
|
|||||||
Updated: February 16, 2001 (http://curl.haxx.se/docs/faq.shtml)
|
Updated: March 23, 2001 (http://curl.haxx.se/docs/faq.shtml)
|
||||||
_ _ ____ _
|
_ _ ____ _
|
||||||
___| | | | _ \| |
|
___| | | | _ \| |
|
||||||
/ __| | | | |_) | |
|
/ __| | | | |_) | |
|
||||||
@@ -12,6 +12,8 @@ FAQ
|
|||||||
1.2 What is libcurl?
|
1.2 What is libcurl?
|
||||||
1.3 What is cURL not?
|
1.3 What is cURL not?
|
||||||
1.4 When will you make curl do XXXX ?
|
1.4 When will you make curl do XXXX ?
|
||||||
|
1.5 Who makes cURL?
|
||||||
|
1.6 What do you get for making cURL?
|
||||||
|
|
||||||
2. Install Related Problems
|
2. Install Related Problems
|
||||||
2.1 configure doesn't find OpenSSL even when it is installed
|
2.1 configure doesn't find OpenSSL even when it is installed
|
||||||
@@ -31,7 +33,7 @@ FAQ
|
|||||||
3.7 Can I use curl to delete/rename a file through FTP?
|
3.7 Can I use curl to delete/rename a file through FTP?
|
||||||
3.8 How do I tell curl to follow HTTP redirects?
|
3.8 How do I tell curl to follow HTTP redirects?
|
||||||
3.9 How do I use curl in PHP?
|
3.9 How do I use curl in PHP?
|
||||||
3.10 What about SOAP, WEBDAV, XML-RPC or similar protocols over HTTP?
|
3.10 What about SOAP, WebDAV, XML-RPC or similar protocols over HTTP?
|
||||||
|
|
||||||
4. Running Problems
|
4. Running Problems
|
||||||
4.1 Problems connecting to SSL servers.
|
4.1 Problems connecting to SSL servers.
|
||||||
@@ -79,14 +81,16 @@ FAQ
|
|||||||
Curl supports a range of common internet protocols, currently including
|
Curl supports a range of common internet protocols, currently including
|
||||||
HTTP, HTTPS, FTP, GOPHER, LDAP, DICT, TELNET and FILE.
|
HTTP, HTTPS, FTP, GOPHER, LDAP, DICT, TELNET and FILE.
|
||||||
|
|
||||||
We spell it cURL or just curl.
|
We spell it cURL or just curl. We pronounce it with an initial k sound:
|
||||||
|
[kurl].
|
||||||
|
|
||||||
1.2 What is libcurl?
|
1.2 What is libcurl?
|
||||||
|
|
||||||
libcurl is a reliable, higly portable multiprotocol file transfer library.
|
libcurl is a reliable and portable library which provides you with an easy
|
||||||
|
interface to a range of common internet protocols.
|
||||||
|
|
||||||
Any application is free to use libcurl, even commercial or closed-source
|
You can use libcurl for free in your application even if it is commercial
|
||||||
ones.
|
or closed-source.
|
||||||
|
|
||||||
1.3 What is cURL not?
|
1.3 What is cURL not?
|
||||||
|
|
||||||
@@ -112,33 +116,59 @@ FAQ
|
|||||||
|
|
||||||
1.4 When will you make curl do XXXX ?
|
1.4 When will you make curl do XXXX ?
|
||||||
|
|
||||||
I love suggestions of what to change in order to make curl and libcurl
|
We love suggestions of what to change in order to make curl and libcurl
|
||||||
better. I do however believe in a few rules when it comes to the future of
|
better. We do however believe in a few rules when it comes to the future of
|
||||||
curl:
|
curl:
|
||||||
|
|
||||||
* It is to remain a command line tool. If you want GUIs or fancy scripting
|
* Curl is to remain a command line tool. If you want GUIs or fancy scripting
|
||||||
capabilities, you're free to write another tool that uses libcurl and that
|
capabilities, you're free to write another tool that uses libcurl and that
|
||||||
offers this. There's no point in having one single tool that does every
|
offers this. There's no point in having a single tool that does every
|
||||||
imaginable thing. That's also one of the great advantages of having the
|
imaginable thing. That's also one of the great advantages of having the
|
||||||
core of curl as a library: libcurl.
|
core of curl as a library.
|
||||||
|
|
||||||
* I do not add things to curl that other small and available tools already
|
* We do not add things to curl that other small and available tools already
|
||||||
do very fine at the side. Curl's output is fine to pipe into another
|
do very fine at the side. Curl's output is fine to pipe into another
|
||||||
program or redirect to another file for the next program to interpret.
|
program or redirect to another file for the next program to interpret.
|
||||||
|
|
||||||
* I focus on protocol related issues and improvements. If you wanna do more
|
* We focus on protocol related issues and improvements. If you wanna do more
|
||||||
magic with the supported protocols than curl currently does, chances are
|
magic with the supported protocols than curl currently does, chances are
|
||||||
big I will agree. If you wanna add more protocols, I may very well
|
big I will agree. If you wanna add more protocols, I may very well
|
||||||
agree.
|
agree.
|
||||||
|
|
||||||
* If you want me to make all the work while you wait for me to implement it
|
* If you want someone else to make all the work while you wait for us to
|
||||||
for you, that is not a very friendly attitude. I spend a considerable time
|
implement it for you, that is not a very friendly attitude. We spend a
|
||||||
already on maintaining and developing curl. In order to get more out of
|
considerable time already on maintaining and developing curl. In order to
|
||||||
me, I trust you will offer some of your time and efforts in return.
|
get more out of us, you should consider trading in some of your time and
|
||||||
|
efforts in return.
|
||||||
|
|
||||||
* If you write the code, chances are bigger that it will get into curl
|
* If you write the code, chances are bigger that it will get into curl
|
||||||
faster.
|
faster.
|
||||||
|
|
||||||
|
1.5 Who makes cURL?
|
||||||
|
|
||||||
|
cURL and libcurl are not made by any single individual. Sure, Daniel
|
||||||
|
Stenberg writes the major parts, but various people's submissions are
|
||||||
|
important and crucial. Anyone can post their changes and improvements and
|
||||||
|
have them inserted in the main sources (of course on the condition that
|
||||||
|
developers agree on that the fixes are good).
|
||||||
|
|
||||||
|
The list of contributors in the bottom of the man page is only a small part
|
||||||
|
of all the people that every day provide us with bug reports, suggestions,
|
||||||
|
ideas and source code.
|
||||||
|
|
||||||
|
curl is developed by a community, with Daniel at the wheel.
|
||||||
|
|
||||||
|
1.6 What do you get for making cURL?
|
||||||
|
|
||||||
|
Project cURL is entirely free and open, without any commercial interests or
|
||||||
|
money involved. No person gets paid in any way for developing curl. We all
|
||||||
|
do this volountarily on our spare time.
|
||||||
|
|
||||||
|
We get some help from companies. Contactor Data hosts the curl web site and
|
||||||
|
the main mailing list, Haxx owns the curl web site's domain and
|
||||||
|
sourceforge.net hosts several project tools we take advantage from like the
|
||||||
|
bug tracker, mailing lists and more.
|
||||||
|
|
||||||
2. Install Related Problems
|
2. Install Related Problems
|
||||||
|
|
||||||
2.1. configure doesn't find OpenSSL even when it is installed
|
2.1. configure doesn't find OpenSSL even when it is installed
|
||||||
@@ -182,23 +212,24 @@ FAQ
|
|||||||
|
|
||||||
2.2. Does curl work/build with other SSL libraries?
|
2.2. Does curl work/build with other SSL libraries?
|
||||||
|
|
||||||
Curl has been written to use OpenSSL, although I doubt there would be much
|
Curl has been written to use OpenSSL, although there should not be much
|
||||||
problems using a different library. If anyone does "port" curl to use a
|
problems using a different library. If anyone does "port" curl to use a
|
||||||
different SSL library, I am of course very interested in getting the patch!
|
different SSL library, we are of course very interested in getting the
|
||||||
|
patch!
|
||||||
|
|
||||||
2.3. Where can I find a copy of LIBEAY32.DLL?
|
2.3. Where can I find a copy of LIBEAY32.DLL?
|
||||||
|
|
||||||
That is an OpenSSL binary built for Windows.
|
That is an OpenSSL binary built for Windows.
|
||||||
|
|
||||||
Curl uses OpenSSL to do the SSL stuff. The LIBEAY32.DLL is what curl needs
|
Curl uses OpenSSL to do the SSL stuff. The LIBEAY32.DLL is what curl needs
|
||||||
on a windows machine to do https://. Check out the curl web page to find
|
on a windows machine to do https://. Check out the curl web site to find
|
||||||
accurate and up-to-date pointers to recent OpenSSL DDLs and other binary
|
accurate and up-to-date pointers to recent OpenSSL DDLs and other binary
|
||||||
packages.
|
packages.
|
||||||
|
|
||||||
2.4. Does cURL support Socks (RFC 1928) ?
|
2.4. Does cURL support Socks (RFC 1928) ?
|
||||||
|
|
||||||
No. Nobody has wanted it that badly yet. I would appriciate patches that
|
No. Nobody has wanted it that badly yet. We appriciate patches that bring
|
||||||
brings this functionality.
|
this functionality.
|
||||||
|
|
||||||
|
|
||||||
3. Usage problems
|
3. Usage problems
|
||||||
@@ -220,7 +251,7 @@ FAQ
|
|||||||
|
|
||||||
3.2. How do I tell curl to resume a transfer?
|
3.2. How do I tell curl to resume a transfer?
|
||||||
|
|
||||||
Curl supports resume both ways on FTP, download ways on HTTP.
|
Curl supports resumed transfers both ways on both FTP and HTTP.
|
||||||
|
|
||||||
Try the -C option.
|
Try the -C option.
|
||||||
|
|
||||||
@@ -232,10 +263,10 @@ FAQ
|
|||||||
use the -F type. In all the most common cases, you should use -d which then
|
use the -F type. In all the most common cases, you should use -d which then
|
||||||
causes a posting with the type 'application/x-www-form-urlencoded'.
|
causes a posting with the type 'application/x-www-form-urlencoded'.
|
||||||
|
|
||||||
I have described this in some detail in the README.curl file, and if you
|
This is described in some detail in the README.curl file, and if you don't
|
||||||
don't understand it the first time, read it again before you post questions
|
understand it the first time, read it again before you post questions about
|
||||||
about this to the mailing list. I would also suggest that you read through
|
this to the mailing list. Also, try reading through the mailing list
|
||||||
the mailing list archives for old postings and questions regarding this.
|
archives for old postings and questions regarding this.
|
||||||
|
|
||||||
3.4. How do I tell curl to run custom FTP commands?
|
3.4. How do I tell curl to run custom FTP commands?
|
||||||
|
|
||||||
@@ -292,7 +323,7 @@ FAQ
|
|||||||
invoke the curl tool using a command line. This is the way to use curl if
|
invoke the curl tool using a command line. This is the way to use curl if
|
||||||
you're using PHP3 or PHP4 built without curl module support.
|
you're using PHP3 or PHP4 built without curl module support.
|
||||||
|
|
||||||
3.10 What about SOAP, WEBDAV, XML-RPC or similar protocols over HTTP?
|
3.10 What about SOAP, WebDAV, XML-RPC or similar protocols over HTTP?
|
||||||
|
|
||||||
Curl adheres to the HTTP spec, which basically means you can play with *any*
|
Curl adheres to the HTTP spec, which basically means you can play with *any*
|
||||||
protocol that is built ontop of HTTP. Protocols such as SOAP, WEBDAV and
|
protocol that is built ontop of HTTP. Protocols such as SOAP, WEBDAV and
|
||||||
@@ -306,9 +337,9 @@ FAQ
|
|||||||
|
|
||||||
4.1. Problems connecting to SSL servers.
|
4.1. Problems connecting to SSL servers.
|
||||||
|
|
||||||
It took a very long time before I could sort out why curl had problems
|
It took a very long time before we could sort out why curl had problems to
|
||||||
to connect to certain SSL servers when using SSLeay or OpenSSL v0.9+.
|
connect to certain SSL servers when using SSLeay or OpenSSL v0.9+. The
|
||||||
The error sometimes showed up similar to:
|
error sometimes showed up similar to:
|
||||||
|
|
||||||
16570:error:1407D071:SSL routines:SSL2_READ:bad mac decode:s2_pkt.c:233:
|
16570:error:1407D071:SSL routines:SSL2_READ:bad mac decode:s2_pkt.c:233:
|
||||||
|
|
||||||
@@ -316,12 +347,12 @@ FAQ
|
|||||||
requests properly. To correct this problem, tell curl to select SSLv2 from
|
requests properly. To correct this problem, tell curl to select SSLv2 from
|
||||||
the command line (-2/--sslv2).
|
the command line (-2/--sslv2).
|
||||||
|
|
||||||
I have also seen examples where the remote server didn't like the SSLv2
|
There has also been examples where the remote server didn't like the SSLv2
|
||||||
request and instead you had to force curl to use SSLv3 with -3/--sslv3.
|
request and instead you had to force curl to use SSLv3 with -3/--sslv3.
|
||||||
|
|
||||||
4.2. Why do I get problems when I use & or % in the URL?
|
4.2. Why do I get problems when I use & or % in the URL?
|
||||||
|
|
||||||
In general unix shells, the & letter is treated special and when used it
|
In general unix shells, the & letter is treated special and when used, it
|
||||||
runs the specified command in the background. To safely send the & as a part
|
runs the specified command in the background. To safely send the & as a part
|
||||||
of a URL, you should qoute the entire URL by using single (') or double (")
|
of a URL, you should qoute the entire URL by using single (') or double (")
|
||||||
quotes around it.
|
quotes around it.
|
||||||
@@ -346,8 +377,8 @@ FAQ
|
|||||||
curl '{curl,www}.haxx.se'
|
curl '{curl,www}.haxx.se'
|
||||||
|
|
||||||
To be able to use those letters as actual parts of the URL (without using
|
To be able to use those letters as actual parts of the URL (without using
|
||||||
them for the curl URL "globbing" system), use the -g/--globoff option
|
them for the curl URL "globbing" system), use the -g/--globoff option (curl
|
||||||
(included in curl 7.6 and later):
|
7.6 and later):
|
||||||
|
|
||||||
curl -g 'www.site.com/weirdname[].html'
|
curl -g 'www.site.com/weirdname[].html'
|
||||||
|
|
||||||
@@ -363,8 +394,8 @@ FAQ
|
|||||||
|
|
||||||
4.5 Why do I get return code XXX from a HTTP server?
|
4.5 Why do I get return code XXX from a HTTP server?
|
||||||
|
|
||||||
RFC2616 clearly explains the return codes. I'll make a short transcript
|
RFC2616 clearly explains the return codes. This is a short transcript. Go
|
||||||
here. Go read the RFC for exact details:
|
read the RFC for exact details:
|
||||||
|
|
||||||
4.5.1 "400 Bad Request"
|
4.5.1 "400 Bad Request"
|
||||||
|
|
||||||
@@ -400,7 +431,7 @@ FAQ
|
|||||||
|
|
||||||
4.7. How do I keep usernames and passwords secret in Curl command lines?
|
4.7. How do I keep usernames and passwords secret in Curl command lines?
|
||||||
|
|
||||||
I see this problem as two parts:
|
This problem has two sides:
|
||||||
|
|
||||||
The first part is to avoid having clear-text passwords in the command line
|
The first part is to avoid having clear-text passwords in the command line
|
||||||
so that they don't appear in 'ps' outputs and similar. That is easily
|
so that they don't appear in 'ps' outputs and similar. That is easily
|
||||||
@@ -447,9 +478,8 @@ FAQ
|
|||||||
programs. libcurl will use thread-safe functions instead of non-safe ones if
|
programs. libcurl will use thread-safe functions instead of non-safe ones if
|
||||||
your system has such.
|
your system has such.
|
||||||
|
|
||||||
I am very interested in once and for all getting some kind of report or
|
We would appriciate some kind of report or README file from those who have
|
||||||
README file from those who have used libcurl in a threaded environment,
|
used libcurl in a threaded environment.
|
||||||
since I haven't and I get this question more and more frequently!
|
|
||||||
|
|
||||||
5.2 How can I receive all data into a large memory chunk?
|
5.2 How can I receive all data into a large memory chunk?
|
||||||
|
|
||||||
@@ -486,11 +516,18 @@ FAQ
|
|||||||
|
|
||||||
5.3 How do I fetch multiple files with libcurl?
|
5.3 How do I fetch multiple files with libcurl?
|
||||||
|
|
||||||
The easy interface of libcurl does not support multiple requests using the
|
Starting with version 7.7, curl and libcurl will have excellent support for
|
||||||
same connection. The only available way to do multiple requests is to
|
transferring multiple files. You should just repeatedly set new URLs with
|
||||||
init/perform/cleanup for each request.
|
curl_easy_setopt() and then transfer it with curl_easy_perform(). The handle
|
||||||
|
you get from curl_easy_init() is not only reusable starting with libcurl
|
||||||
|
7.7, but also you're encouraged to reuse it if you can, as that will enable
|
||||||
|
libcurl to use persistant connections.
|
||||||
|
|
||||||
5.4 Does libcurl do Winsock initing on win32 systems?
|
For libcurl prior to 7.7, there was no multiple file support. The only
|
||||||
|
available way to do multiple requests was to init/perform/cleanup for each
|
||||||
|
transfer.
|
||||||
|
|
||||||
|
5.4 Does libcurl do Winsock initialization on win32 systems?
|
||||||
|
|
||||||
No.
|
No.
|
||||||
|
|
||||||
@@ -512,17 +549,16 @@ FAQ
|
|||||||
|
|
||||||
5.6 What about Keep-Alive or persistant connections?
|
5.6 What about Keep-Alive or persistant connections?
|
||||||
|
|
||||||
This is closely related to issue 5.3. Since libcurl has no real support
|
Starting with version 7.7, curl and libcurl will have excellent support for
|
||||||
for doing multiple file transfers, there's no support for Keep-Alive or
|
persistant connections when transferring several files from the same server.
|
||||||
persistant connections either.
|
Curl will attempt to reuse connections for all URLs specified on the same
|
||||||
|
command line/config file, and libcurl will reuse connections for all
|
||||||
|
transfers that are made using the same libcurl handle.
|
||||||
|
|
||||||
This is of course subject to change as soon as libcurl gets support for
|
Previous versions had no persistant connection support.
|
||||||
multiple files. Feel free to join in and make this change happen sooner!
|
|
||||||
|
|
||||||
6. License Issues
|
6. License Issues
|
||||||
|
|
||||||
NOTE: This section concerns curl 7.5.2 or later!
|
|
||||||
|
|
||||||
Curl and libcurl are released under a MIT/X derivate license *or* the MPL,
|
Curl and libcurl are released under a MIT/X derivate license *or* the MPL,
|
||||||
the Mozilla Public License. To get a really good answer to your license
|
the Mozilla Public License. To get a really good answer to your license
|
||||||
conflict questions, you should study the MPL and MIT/X licenses and the
|
conflict questions, you should study the MPL and MIT/X licenses and the
|
||||||
@@ -567,9 +603,10 @@ FAQ
|
|||||||
|
|
||||||
No.
|
No.
|
||||||
|
|
||||||
We carefully picked this license years ago and a large amount of people have
|
We have carefully picked this license after years of development and
|
||||||
contributed with source code knowing that this is the license we use. This
|
discussions and a large amount of people have contributed with source code
|
||||||
license puts the restrictions we want on curl/libcurl and it does not spread
|
knowing that this is the license we use. This license puts the restrictions
|
||||||
to other programs or libraries that use it. The recent dual license
|
we want on curl/libcurl and it does not spread to other programs or
|
||||||
modification should make it possible for everyone to use libcurl or curl in
|
libraries that use it. The recent dual license modification should make it
|
||||||
their projects, no matter what license they already have in use.
|
possible for everyone to use libcurl or curl in their projects, no matter
|
||||||
|
what license they already have in use.
|
||||||
|
@@ -17,12 +17,14 @@ Misc
|
|||||||
- progress bar/time specs while downloading
|
- progress bar/time specs while downloading
|
||||||
- "standard" proxy environment variables support
|
- "standard" proxy environment variables support
|
||||||
- config file support
|
- config file support
|
||||||
- compiles on win32
|
- compiles on win32 (reported built on 29 operating systems)
|
||||||
- redirectable stderr
|
- redirectable stderr
|
||||||
- use selected network interface for outgoing traffic
|
- use selected network interface for outgoing traffic
|
||||||
- IPv6 support
|
- IPv6 support
|
||||||
|
- persistant connections
|
||||||
|
|
||||||
HTTP
|
HTTP
|
||||||
|
- HTTP/1.1 compliant
|
||||||
- GET
|
- GET
|
||||||
- PUT
|
- PUT
|
||||||
- HEAD
|
- HEAD
|
||||||
@@ -72,6 +74,7 @@ FTP
|
|||||||
|
|
||||||
TELNET
|
TELNET
|
||||||
- connection negotiation
|
- connection negotiation
|
||||||
|
- custom telnet options
|
||||||
- stdin/stdout I/O
|
- stdin/stdout I/O
|
||||||
|
|
||||||
LDAP (*2)
|
LDAP (*2)
|
||||||
|
101
docs/INSTALL
101
docs/INSTALL
@@ -7,24 +7,35 @@
|
|||||||
How To Compile
|
How To Compile
|
||||||
|
|
||||||
Curl has been compiled and built on numerous different operating systems. The
|
Curl has been compiled and built on numerous different operating systems. The
|
||||||
way to proceed is mainly divided in two different ways: the unix way or the
|
way to proceed is mainly divided in two different ways: the unix way or the
|
||||||
windows way.
|
windows way.
|
||||||
|
|
||||||
If you're using Windows (95, 98, NT) or OS/2, you should continue reading from
|
If you're using Windows (95/98/NT/ME/2000 or whatever) or OS/2, you should
|
||||||
the Win32 or OS/2 headers further down. All other systems should be capable of
|
continue reading from the Win32 or OS/2 headers further down. All other
|
||||||
being installed as described below.
|
systems should be capable of being installed as described below.
|
||||||
|
|
||||||
UNIX
|
UNIX
|
||||||
====
|
====
|
||||||
|
|
||||||
The configure script *always* tries to find a working SSL library unless
|
A normal unix installation is made in three or four steps (after you've
|
||||||
explicitly told not to. If you have OpenSSL installed in the default
|
unpacked the source archive):
|
||||||
search path for your compiler/linker, you don't need to do anything
|
|
||||||
special:
|
|
||||||
|
|
||||||
./configure
|
./configure
|
||||||
|
make
|
||||||
|
make test (optional)
|
||||||
|
make install
|
||||||
|
|
||||||
If you have OpenSSL installed in /usr/local/ssl, you can run configure
|
You probably need to be root when doing the last command.
|
||||||
|
|
||||||
|
If you want to install curl in a different file hierarchy than /usr/local,
|
||||||
|
you need to specify that already when running configure:
|
||||||
|
|
||||||
|
./configure --prefix=/path/to/curl/tree
|
||||||
|
|
||||||
|
The configure script always tries to find a working SSL library unless
|
||||||
|
explicitly told not to. If you have OpenSSL installed in the default search
|
||||||
|
path for your compiler/linker, you don't need to do anything special. If
|
||||||
|
you have OpenSSL installed in e.g /usr/local/ssl, you can run configure
|
||||||
like:
|
like:
|
||||||
|
|
||||||
./configure --with-ssl
|
./configure --with-ssl
|
||||||
@@ -54,33 +65,11 @@ UNIX
|
|||||||
env CPPFLAGS="-I/path/to/ssl/include" LDFLAGS="-L/path/to/ssl/lib" \
|
env CPPFLAGS="-I/path/to/ssl/include" LDFLAGS="-L/path/to/ssl/lib" \
|
||||||
./configure
|
./configure
|
||||||
|
|
||||||
If your SSL library was compiled with rsaref (usually for use in
|
If your SSL library was compiled with rsaref (usually for use in the United
|
||||||
the United States), you may also need to set:
|
States), you may also need to set:
|
||||||
|
|
||||||
LIBS=-lRSAglue -lrsaref
|
LIBS=-lRSAglue -lrsaref
|
||||||
(from Doug Kaufman <dkaufman@rahul.net>)
|
(as suggested by Doug Kaufman)
|
||||||
|
|
||||||
Without SSL support, just run:
|
|
||||||
|
|
||||||
./configure
|
|
||||||
|
|
||||||
Then run:
|
|
||||||
|
|
||||||
make
|
|
||||||
|
|
||||||
Use the executable `curl` in src/ directory.
|
|
||||||
|
|
||||||
To install curl on your system, run
|
|
||||||
|
|
||||||
make install
|
|
||||||
|
|
||||||
This will copy curl to /usr/local/bin/ (or $prefix/bin if you used the
|
|
||||||
--prefix option to configure) and it copies the man pages, the lib and the
|
|
||||||
include files to suitable places.
|
|
||||||
|
|
||||||
To make sure everything runs as supposed, run the test suite:
|
|
||||||
|
|
||||||
make test
|
|
||||||
|
|
||||||
KNOWN PROBLEMS
|
KNOWN PROBLEMS
|
||||||
|
|
||||||
@@ -109,7 +98,7 @@ UNIX
|
|||||||
they're executable and set to appear in the path *BEFORE* the actual (but
|
they're executable and set to appear in the path *BEFORE* the actual (but
|
||||||
obsolete) autoconf and autoheader scripts.
|
obsolete) autoconf and autoheader scripts.
|
||||||
|
|
||||||
OPTIONS
|
MORE OPTIONS
|
||||||
|
|
||||||
Remember, to force configure to use the standard cc compiler if both
|
Remember, to force configure to use the standard cc compiler if both
|
||||||
cc and gcc are present, run configure like
|
cc and gcc are present, run configure like
|
||||||
@@ -156,29 +145,27 @@ Win32
|
|||||||
MingW32 (GCC-2.95) style
|
MingW32 (GCC-2.95) style
|
||||||
------------------------
|
------------------------
|
||||||
Run the 'mingw32.bat' file to get the proper environment variables
|
Run the 'mingw32.bat' file to get the proper environment variables
|
||||||
set, then run 'make -f Makefile.m32' in the lib/ dir and then
|
set, then run 'make mingw32' in the root dir.
|
||||||
'make -f Makefile.m32' in the src/ dir.
|
|
||||||
|
|
||||||
If you have any problems linking libraries or finding header files,
|
If you have any problems linking libraries or finding header files, be
|
||||||
be sure to look at the provided "Makefile.m32" files for the proper
|
sure to verify that the provided "Makefile.m32" files use the proper
|
||||||
paths, and adjust as necessary.
|
paths, and adjust as necessary.
|
||||||
|
|
||||||
Cygwin style
|
Cygwin style
|
||||||
------------
|
------------
|
||||||
Almost identical to the unix installation. Run the configure script
|
Almost identical to the unix installation. Run the configure script in
|
||||||
in the curl root with 'sh configure'. Make sure you have the sh
|
the curl root with 'sh configure'. Make sure you have the sh
|
||||||
executable in /bin/ or you'll see the configure fail towards the
|
executable in /bin/ or you'll see the configure fail towards the end.
|
||||||
end.
|
|
||||||
|
|
||||||
Run 'make'
|
Run 'make'
|
||||||
|
|
||||||
Microsoft command line style
|
Microsoft command line style
|
||||||
----------------------------
|
----------------------------
|
||||||
Run the 'vcvars32.bat' file to get the proper environment variables
|
Run the 'vcvars32.bat' file to get the proper environment variables
|
||||||
set, then run 'nmake -f Makefile.vc6' in the lib/ dir and then
|
set, then run 'nmake vc' in the root dir.
|
||||||
'nmake -f Makefile.vc6' in the src/ dir.
|
|
||||||
|
|
||||||
The vcvars32.bat file is part of the Microsoft development environment.
|
The vcvars32.bat file is part of the Microsoft development
|
||||||
|
environment.
|
||||||
|
|
||||||
IDE-style
|
IDE-style
|
||||||
-------------------------
|
-------------------------
|
||||||
@@ -196,9 +183,9 @@ Win32
|
|||||||
For VC++ 6, there's an included Makefile.vc6 that should be possible
|
For VC++ 6, there's an included Makefile.vc6 that should be possible
|
||||||
to use out-of-the-box.
|
to use out-of-the-box.
|
||||||
|
|
||||||
Microsoft note: add /Zm200 to the compiler options, as the hugehelp.c
|
Microsoft note: add /Zm200 to the compiler options to increase the
|
||||||
won't compile otherwise due to "too long puts string" or something
|
compiler's memory allocation limit, as the hugehelp.c won't compile
|
||||||
like that!
|
due to "too long puts string".
|
||||||
|
|
||||||
|
|
||||||
With SSL:
|
With SSL:
|
||||||
@@ -206,26 +193,24 @@ Win32
|
|||||||
MingW32 (GCC-2.95) style
|
MingW32 (GCC-2.95) style
|
||||||
------------------------
|
------------------------
|
||||||
Run the 'mingw32.bat' file to get the proper environment variables
|
Run the 'mingw32.bat' file to get the proper environment variables
|
||||||
set, then run 'make -f Makefile.m32 SSL=1' in the lib/ dir and then
|
set, then run 'make mingw32-ssl' in the root dir.
|
||||||
'make -f Makefile.m32 SSL=1' in the src/ dir.
|
|
||||||
|
|
||||||
If you have any problems linking libraries or finding header files,
|
If you have any problems linking libraries or finding header files, be
|
||||||
be sure to look at the provided "Makefile.m32" files for the proper
|
sure to look at the provided "Makefile.m32" files for the proper
|
||||||
paths, and adjust as necessary.
|
paths, and adjust as necessary.
|
||||||
|
|
||||||
Cygwin style
|
Cygwin style
|
||||||
------------
|
------------
|
||||||
|
|
||||||
Haven't done, nor got any reports on how to do. It should although be
|
Haven't done, nor got any reports on how to do. It should although be
|
||||||
identical to the unix setup for the same purpose. See above.
|
identical to the unix setup for the same purpose. See above.
|
||||||
|
|
||||||
Microsoft command line style
|
Microsoft command line style
|
||||||
----------------------------
|
----------------------------
|
||||||
Run the 'vcvars32.bat' file to get the proper environment variables
|
Run the 'vcvars32.bat' file to get the proper environment variables
|
||||||
set, then run 'nmake -f Makefile.vc6 release-ssl' in the lib/ dir and
|
set, then run 'nmake vc-ssl' in the root dir.
|
||||||
then 'nmake -f Makefile.vc6' in the src/ dir.
|
|
||||||
|
|
||||||
The vcvars32.bat file is part of the Microsoft development environment.
|
The vcvars32.bat file is part of the Microsoft development
|
||||||
|
environment.
|
||||||
|
|
||||||
Microsoft / Borland style
|
Microsoft / Borland style
|
||||||
-------------------------
|
-------------------------
|
||||||
@@ -266,7 +251,7 @@ IBM OS/2
|
|||||||
|
|
||||||
PORTS
|
PORTS
|
||||||
=====
|
=====
|
||||||
Just to show off, this is a probably incomplete list of known hardware and
|
This is a probably incomplete list of known hardware and
|
||||||
operating systems that curl has been compiled for:
|
operating systems that curl has been compiled for:
|
||||||
|
|
||||||
- Ultrix
|
- Ultrix
|
||||||
|
102
docs/INTERNALS
102
docs/INTERNALS
@@ -1,4 +1,4 @@
|
|||||||
Updated for curl 7.6 on January 26, 2001
|
Updated for curl 7.7 on March 13, 2001
|
||||||
_ _ ____ _
|
_ _ ____ _
|
||||||
___| | | | _ \| |
|
___| | | | _ \| |
|
||||||
/ __| | | | |_) | |
|
/ __| | | | |_) | |
|
||||||
@@ -7,11 +7,11 @@
|
|||||||
|
|
||||||
INTERNALS
|
INTERNALS
|
||||||
|
|
||||||
The project is kind of split in two. The library and the client. The client
|
The project is split in two. The library and the client. The client part uses
|
||||||
part uses the library, but the library is meant to be designed to allow other
|
the library, but the library is designed to allow other applications to use
|
||||||
applications to use it.
|
it.
|
||||||
|
|
||||||
Thus, the largest amount of code and complexity is in the library part.
|
The largest amount of code and complexity is in the library part.
|
||||||
|
|
||||||
CVS
|
CVS
|
||||||
===
|
===
|
||||||
@@ -35,13 +35,13 @@ Windows vs Unix
|
|||||||
the same at all places except for the header file that defines them. The
|
the same at all places except for the header file that defines them. The
|
||||||
macros in use are sclose(), sread() and swrite().
|
macros in use are sclose(), sread() and swrite().
|
||||||
|
|
||||||
2. Windows requires a couple of init calls for the socket stuff
|
2. Windows requires a couple of init calls for the socket stuff.
|
||||||
|
|
||||||
Those must be made by the application that uses libcurl, in curl that means
|
Those must be made by the application that uses libcurl, in curl that means
|
||||||
src/main.c has some code #ifdef'ed to do just that.
|
src/main.c has some code #ifdef'ed to do just that.
|
||||||
|
|
||||||
3. The file descriptors for network communication and file operations are
|
3. The file descriptors for network communication and file operations are
|
||||||
not easily interchangable as in unix
|
not easily interchangable as in unix.
|
||||||
|
|
||||||
We avoid this by not trying any funny tricks on file descriptors.
|
We avoid this by not trying any funny tricks on file descriptors.
|
||||||
|
|
||||||
@@ -51,10 +51,10 @@ Windows vs Unix
|
|||||||
|
|
||||||
We set stdout to binary under windows
|
We set stdout to binary under windows
|
||||||
|
|
||||||
Inside the source code, I do make an effort to avoid '#ifdef WIN32'. All
|
Inside the source code, We make an effort to avoid '#ifdef [Your OS]'. All
|
||||||
conditionals that deal with features *should* instead be in the format
|
conditionals that deal with features *should* instead be in the format
|
||||||
'#ifdef HAVE_THAT_WEIRD_FUNCTION'. Since Windows can't run configure scripts,
|
'#ifdef HAVE_THAT_WEIRD_FUNCTION'. Since Windows can't run configure scripts,
|
||||||
I maintain two config-win32.h files (one in / and one in src/) that are
|
we maintain two config-win32.h files (one in / and one in src/) that are
|
||||||
supposed to look exactly as a config.h file would have looked like on a
|
supposed to look exactly as a config.h file would have looked like on a
|
||||||
Windows machine!
|
Windows machine!
|
||||||
|
|
||||||
@@ -64,12 +64,6 @@ Windows vs Unix
|
|||||||
Library
|
Library
|
||||||
=======
|
=======
|
||||||
|
|
||||||
As described elsewhere, libcurl is meant to get two different "layers" of
|
|
||||||
interfaces. At the present point only the high-level, the "easy", interface
|
|
||||||
has been fully implemented and documented. We assume the easy-interface in
|
|
||||||
this description, the low-level interface will be documented when fully
|
|
||||||
implemented.
|
|
||||||
|
|
||||||
There are plenty of entry points to the library, namely each publicly defined
|
There are plenty of entry points to the library, namely each publicly defined
|
||||||
function that libcurl offers to applications. All of those functions are
|
function that libcurl offers to applications. All of those functions are
|
||||||
rather small and easy-to-follow. All the ones prefixed with 'curl_easy' are
|
rather small and easy-to-follow. All the ones prefixed with 'curl_easy' are
|
||||||
@@ -103,8 +97,9 @@ Library
|
|||||||
lib/sendf.c) function to send printf-style formatted data to the remote host
|
lib/sendf.c) function to send printf-style formatted data to the remote host
|
||||||
and when they're ready to make the actual file transfer they call the
|
and when they're ready to make the actual file transfer they call the
|
||||||
Curl_Transfer() function (in lib/transfer.c) to setup the transfer and
|
Curl_Transfer() function (in lib/transfer.c) to setup the transfer and
|
||||||
returns. curl_transfer() then calls _Tranfer() in lib/transfer.c that
|
returns. Curl_perform() then calls Transfer() in lib/transfer.c that performs
|
||||||
performs the entire file transfer.
|
the entire file transfer. Curl_perform() is what does the main "connect - do
|
||||||
|
- transfer - done" loop. It loops if there's a Location: to follow.
|
||||||
|
|
||||||
During transfer, the progress functions in lib/progress.c are called at a
|
During transfer, the progress functions in lib/progress.c are called at a
|
||||||
frequent interval (or at the user's choice, a specified callback might get
|
frequent interval (or at the user's choice, a specified callback might get
|
||||||
@@ -114,6 +109,22 @@ Library
|
|||||||
When completed, the curl_easy_cleanup() should be called to free up used
|
When completed, the curl_easy_cleanup() should be called to free up used
|
||||||
resources.
|
resources.
|
||||||
|
|
||||||
|
A quick roundup on internal function sequences (many of these call
|
||||||
|
protocol-specific function-pointers):
|
||||||
|
|
||||||
|
curl_connect - connects to a remote site and does initial connect fluff
|
||||||
|
This also checks for an existing connection to the requested site and uses
|
||||||
|
that one if it is possible.
|
||||||
|
|
||||||
|
curl_do - starts a transfer
|
||||||
|
curl_transfer() - transfers data
|
||||||
|
curl_done - ends a transfer
|
||||||
|
|
||||||
|
curl_disconnect - disconnects from a remote site. This is called when the
|
||||||
|
disconnect is really requested, which doesn't necessarily have to be
|
||||||
|
exactly after curl_done in case we want to keep the connection open for
|
||||||
|
a while.
|
||||||
|
|
||||||
HTTP(S)
|
HTTP(S)
|
||||||
|
|
||||||
HTTP offers a lot and is the protocol in curl that uses the most lines of
|
HTTP offers a lot and is the protocol in curl that uses the most lines of
|
||||||
@@ -129,6 +140,14 @@ Library
|
|||||||
the source by the use of curl_read() for reading and curl_write() for writing
|
the source by the use of curl_read() for reading and curl_write() for writing
|
||||||
data to the remote server.
|
data to the remote server.
|
||||||
|
|
||||||
|
http_chunks.c contains functions that understands HTTP 1.1 chunked transfer
|
||||||
|
encoding.
|
||||||
|
|
||||||
|
An interesting detail with the HTTP(S) request, is the add_buffer() series of
|
||||||
|
functions we use. They append data to one single buffer, and when the
|
||||||
|
building is done the entire request is sent off in one single write. This is
|
||||||
|
done this way to overcome problems with flawed firewalls and lame servers.
|
||||||
|
|
||||||
FTP
|
FTP
|
||||||
|
|
||||||
The Curl_if2ip() function can be used for getting the IP number of a
|
The Curl_if2ip() function can be used for getting the IP number of a
|
||||||
@@ -160,7 +179,7 @@ Library
|
|||||||
URL encoding and decoding, called escaping and unescaping in the source code,
|
URL encoding and decoding, called escaping and unescaping in the source code,
|
||||||
is found in lib/escape.c.
|
is found in lib/escape.c.
|
||||||
|
|
||||||
While transfering data in _Transfer() a few functions might get
|
While transfering data in Transfer() a few functions might get
|
||||||
used. curl_getdate() in lib/getdate.c is for HTTP date comparisons (and
|
used. curl_getdate() in lib/getdate.c is for HTTP date comparisons (and
|
||||||
more).
|
more).
|
||||||
|
|
||||||
@@ -182,6 +201,34 @@ Library
|
|||||||
exists in lib/getpass.c. libcurl offers a custom callback that can be used
|
exists in lib/getpass.c. libcurl offers a custom callback that can be used
|
||||||
instead of this, but it doesn't change much to us.
|
instead of this, but it doesn't change much to us.
|
||||||
|
|
||||||
|
Persistant Connections
|
||||||
|
======================
|
||||||
|
|
||||||
|
With curl 7.7, we added persistant connection support to libcurl which has
|
||||||
|
introduced a somewhat different treatmeant of things inside of libcurl.
|
||||||
|
|
||||||
|
o The 'UrlData' struct returned in the curl_easy_init() call must never
|
||||||
|
hold connection-oriented data. It is meant to hold the root data as well
|
||||||
|
as all the options etc that the library-user may choose.
|
||||||
|
o The 'UrlData' struct holds the cache array of pointers to 'connectdata'
|
||||||
|
structs. There's one connectdata struct for each connection that libcurl
|
||||||
|
knows about.
|
||||||
|
o This also enables the 'curl handle' to be reused on subsequent transfers,
|
||||||
|
something that was illegal in pre-7.7 versions.
|
||||||
|
o When we are about to perform a transfer with curl_easy_perform(), we first
|
||||||
|
check for an already existing connection in the cache that we can use,
|
||||||
|
otherwise we create a new one and add to the cache. If the cache is full
|
||||||
|
already when we add a new connection, we close one of the present ones. We
|
||||||
|
select which one to close dependent on the close policy that may have been
|
||||||
|
previously set.
|
||||||
|
o When the tranfer operation is complete, we try to leave the connection open.
|
||||||
|
Particular options may tell us not to, and protocols may signal closure on
|
||||||
|
connections and then we don't keep it open of course.
|
||||||
|
o When curl_easy_cleanup() is called, we close all still opened connections.
|
||||||
|
|
||||||
|
You do realize that the curl handle must be re-used in order for the
|
||||||
|
persistant connections to work.
|
||||||
|
|
||||||
Library Symbols
|
Library Symbols
|
||||||
===============
|
===============
|
||||||
|
|
||||||
@@ -236,12 +283,12 @@ Memory Debugging
|
|||||||
deal with resources that might give us problems if we "leak" them. The
|
deal with resources that might give us problems if we "leak" them. The
|
||||||
functions in the memdebug system do nothing fancy, they do their normal
|
functions in the memdebug system do nothing fancy, they do their normal
|
||||||
function and then log information about what they just did. The logged data
|
function and then log information about what they just did. The logged data
|
||||||
is then analyzed after a complete session,
|
can then be analyzed after a complete session,
|
||||||
|
|
||||||
memanalyze.pl is a perl script present only in CVS (not part of the release
|
memanalyze.pl is a perl script present only present in CVS (not part of the
|
||||||
archives) that analyzes a log file generated by the memdebug system. It
|
release archives) that analyzes a log file generated by the memdebug
|
||||||
detects if resources are allocated but never freed and other kinds of errors
|
system. It detects if resources are allocated but never freed and other kinds
|
||||||
related to resource management.
|
of errors related to resource management.
|
||||||
|
|
||||||
Use -DMALLOCDEBUG when compiling to enable memory debugging.
|
Use -DMALLOCDEBUG when compiling to enable memory debugging.
|
||||||
|
|
||||||
@@ -256,8 +303,8 @@ Test Suite
|
|||||||
httpserver.pl and ftpserver.pl before all the test cases are performed. The
|
httpserver.pl and ftpserver.pl before all the test cases are performed. The
|
||||||
test suite currently only runs on unix-like platforms.
|
test suite currently only runs on unix-like platforms.
|
||||||
|
|
||||||
You'll find a complete description of the test case data files in the README
|
You'll find a complete description of the test case data files in the
|
||||||
file in the test directory.
|
tests/README file.
|
||||||
|
|
||||||
The test suite automatically detects if curl was built with the memory
|
The test suite automatically detects if curl was built with the memory
|
||||||
debugging enabled, and if it was it will detect memory leaks too.
|
debugging enabled, and if it was it will detect memory leaks too.
|
||||||
@@ -269,6 +316,7 @@ Building Releases
|
|||||||
released, run the 'maketgz' script (using 'make distcheck' will give you a
|
released, run the 'maketgz' script (using 'make distcheck' will give you a
|
||||||
pretty good view on the status of the current sources). maketgz prompts for
|
pretty good view on the status of the current sources). maketgz prompts for
|
||||||
version number of the client and the library before it creates a release
|
version number of the client and the library before it creates a release
|
||||||
archive.
|
archive. maketgz uses 'make dist' for the actual archive building, why you
|
||||||
|
need to fill in the Makefile.am files properly for which files that should
|
||||||
|
be included in the release archives.
|
||||||
|
|
||||||
You must have autoconf installed to build release archives.
|
|
||||||
|
115
docs/LIBCURL
115
docs/LIBCURL
@@ -4,58 +4,91 @@
|
|||||||
| | | |_) | (__| |_| | | | |
|
| | | |_) | (__| |_| | | | |
|
||||||
|_|_|_.__/ \___|\__,_|_| |_|
|
|_|_|_.__/ \___|\__,_|_| |_|
|
||||||
|
|
||||||
|
How To Use Libcurl In Your C/C++ Program
|
||||||
|
|
||||||
How To Use Libcurl In Your Program
|
[ libcurl can be used directly from within your PHP or Perl programs as well,
|
||||||
|
look elsewhere for documentation on this ]
|
||||||
|
|
||||||
Interfaces
|
The interface is meant to be very simple for applictions/programmers, hence
|
||||||
|
the name "easy". We have therefore minimized the number of entries.
|
||||||
libcurl currently offers two different interfaces to the URL transfer
|
|
||||||
engine. They can be seen as one low-level and one high-level, in the sense
|
|
||||||
that the low-level one will allow you to deal with a lot more details but on
|
|
||||||
the other hand not offer as many fancy features (such as Location:
|
|
||||||
following). The high-level interface is supposed to be a built-in
|
|
||||||
implementation of the low-level interface. You will not be able to mix
|
|
||||||
function calls from the different layers.
|
|
||||||
|
|
||||||
As we currently ONLY support the high-level interface, the so called easy
|
|
||||||
interface, I will not attempt to describe any low-level functions at this
|
|
||||||
point.
|
|
||||||
|
|
||||||
Function descriptions
|
|
||||||
|
|
||||||
The interface is meant to be very simple for very simple
|
|
||||||
implementations. Thus, we have minimized the number of entries.
|
|
||||||
|
|
||||||
The Easy Interface
|
The Easy Interface
|
||||||
|
|
||||||
When using the easy interface, you init your easy-session and get a handle,
|
When using the easy interface, you init your session and get a handle, which
|
||||||
which you use as input to the following interface functions you use.
|
you use as input to the following interface functions you use. Use
|
||||||
|
curl_easy_init() to get the handle.
|
||||||
|
|
||||||
You continue by setting all the options you want in the upcoming transfer,
|
You continue by setting all the options you want in the upcoming transfer,
|
||||||
most important among them is the URL itself. You might want to set some
|
most important among them is the URL itself (you can't transfer anything
|
||||||
callbacks as well that will be called from the library when data is available
|
without a specified URL as you may have figured out yourself). You might want
|
||||||
etc.
|
to set some callbacks as well that will be called from the library when data
|
||||||
|
is available etc. curl_easy_setopt() is there for this.
|
||||||
|
|
||||||
When all is setup, you tell libcurl to perform the transfer. It will then do
|
When all is setup, you tell libcurl to perform the transfer using
|
||||||
the entire operation and won't return until it is done or failed.
|
curl_easy_perform(). It will then do the entire operation and won't return
|
||||||
|
until it is done or failed.
|
||||||
|
|
||||||
After the transfer has been made, you cleanup the easy-session's handle and
|
After the transfer has been made, you cleanup the session with
|
||||||
libcurl is entirely off the hook!
|
curl_easy_cleanup() and libcurl is entirely off the hook! If you want
|
||||||
|
persistant connections, you don't cleanup immediately, but instead run ahead
|
||||||
|
and perform other transfers. See the chapter below for Persistant
|
||||||
|
Connections.
|
||||||
|
|
||||||
curl_easy_init()
|
While the above mentioned four functions are the main functions to use in the
|
||||||
curl_easy_setopt()
|
easy interface, there is a series of other helpful functions to use. They
|
||||||
curl_easy_perform()
|
are:
|
||||||
curl_easy_cleanup()
|
|
||||||
|
|
||||||
While the above four functions are the main functions to use in the easy
|
curl_version() - displays the libcurl version
|
||||||
interface, there is a series of helpful functions to use. They are:
|
curl_getdate() - converts a date string to time_t
|
||||||
|
curl_getenv() - portable environment variable reader
|
||||||
|
curl_easy_getinfo() - get information about a performed transfer
|
||||||
|
curl_formparse() - helps building a HTTP form POST
|
||||||
|
curl_formfree() - free a list built with curl_formparse()
|
||||||
|
curl_slist_append() - builds a linked list
|
||||||
|
curl_slist_free_all() - frees a whole curl_slist
|
||||||
|
|
||||||
curl_version() - displays the libcurl version
|
For details on these, read the separate man pages.
|
||||||
curl_getdate() - converts a date string to time_t
|
|
||||||
curl_getenv() - portable environment variable reader
|
|
||||||
curl_formparse() - helps building a HTTP form POST
|
|
||||||
curl_slist_append() - builds a linked list
|
|
||||||
curl_slist_free_all() - frees a whole curl_slist
|
|
||||||
|
|
||||||
Read the separate man pages for these functions for details!
|
Portability
|
||||||
|
|
||||||
|
libcurl works *exactly* the same, on any of the platforms it compiles and
|
||||||
|
builds on.
|
||||||
|
|
||||||
|
There's only one caution, and that is the win32 platform that may(*) require
|
||||||
|
you to init the winsock stuff before you use the libcurl functions. Details
|
||||||
|
on this are noted on the curl_easy_init() man page.
|
||||||
|
|
||||||
|
(*) = it appears as if users of the cygwin environment get this done
|
||||||
|
automatically.
|
||||||
|
|
||||||
|
Threads
|
||||||
|
|
||||||
|
Never *ever* call curl-functions simultaneously using the same handle from
|
||||||
|
several threads. libcurl is thread-safe and can be used in any number of
|
||||||
|
threads, but you must use separate curl handles if you want to use libcurl in
|
||||||
|
more than one thread simultaneously.
|
||||||
|
|
||||||
|
Persistant Connections
|
||||||
|
|
||||||
|
With libcurl 7.7, persistant connections were added. Persistant connections
|
||||||
|
means that libcurl can re-use the same connection for several transfers, if
|
||||||
|
the conditions are right.
|
||||||
|
|
||||||
|
libcurl will *always* attempt to use persistant connections. Whenever you use
|
||||||
|
curl_easy_perform(), libcurl will attempt to use an existing connection to do
|
||||||
|
the transfer, and if none exists it'll open a new one that will be subject
|
||||||
|
for re-use on a possible following call to curl_easy_perform().
|
||||||
|
|
||||||
|
To allow libcurl to take full advantage of persistant connections, you should
|
||||||
|
do as many of your file transfers as possible using the same curl
|
||||||
|
handle. When you call curl_easy_cleanup(), all the possibly open connections
|
||||||
|
held by libcurl will be closed and forgotten.
|
||||||
|
|
||||||
|
Note that the options set with curl_easy_setopt() will be used in on every
|
||||||
|
repeat curl_easy_perform() call
|
||||||
|
|
||||||
|
Compatibility with older libcurls
|
||||||
|
|
||||||
|
Repeated curl_easy_perform() calls on the same handle were not supported in
|
||||||
|
pre-7.7 versions, and caused confusion and defined behaviour.
|
||||||
|
|
||||||
|
105
docs/MANUAL
105
docs/MANUAL
@@ -15,22 +15,26 @@ SIMPLE USAGE
|
|||||||
|
|
||||||
curl ftp://ftp.funet.fi/README
|
curl ftp://ftp.funet.fi/README
|
||||||
|
|
||||||
Get a gopher document from funet's gopher server:
|
|
||||||
|
|
||||||
curl gopher://gopher.funet.fi
|
|
||||||
|
|
||||||
Get a web page from a server using port 8000:
|
Get a web page from a server using port 8000:
|
||||||
|
|
||||||
curl http://www.weirdserver.com:8000/
|
curl http://www.weirdserver.com:8000/
|
||||||
|
|
||||||
Get a list of the root directory of an FTP site:
|
Get a list of the root directory of an FTP site:
|
||||||
|
|
||||||
curl ftp://ftp.fts.frontec.se/
|
curl ftp://cool.haxx.se/
|
||||||
|
|
||||||
|
Get a gopher document from funet's gopher server:
|
||||||
|
|
||||||
|
curl gopher://gopher.funet.fi
|
||||||
|
|
||||||
Get the definition of curl from a dictionary:
|
Get the definition of curl from a dictionary:
|
||||||
|
|
||||||
curl dict://dict.org/m:curl
|
curl dict://dict.org/m:curl
|
||||||
|
|
||||||
|
Fetch two documents at once:
|
||||||
|
|
||||||
|
curl ftp://cool.haxx.se/ http://www.weirdserver.com:8000/
|
||||||
|
|
||||||
DOWNLOAD TO A FILE
|
DOWNLOAD TO A FILE
|
||||||
|
|
||||||
Get a web page and store in a local file:
|
Get a web page and store in a local file:
|
||||||
@@ -43,6 +47,10 @@ DOWNLOAD TO A FILE
|
|||||||
|
|
||||||
curl -O http://www.netscape.com/index.html
|
curl -O http://www.netscape.com/index.html
|
||||||
|
|
||||||
|
Fetch two files and store them with their remote names:
|
||||||
|
|
||||||
|
curl -O www.haxx.se/index.html -O curl.haxx.se/download.html
|
||||||
|
|
||||||
USING PASSWORDS
|
USING PASSWORDS
|
||||||
|
|
||||||
FTP
|
FTP
|
||||||
@@ -178,7 +186,7 @@ DETAILED INFORMATION
|
|||||||
-D/--dump-header option when getting files from both FTP and HTTP, and it
|
-D/--dump-header option when getting files from both FTP and HTTP, and it
|
||||||
will then store the headers in the specified file.
|
will then store the headers in the specified file.
|
||||||
|
|
||||||
Store the HTTP headers in a separate file:
|
Store the HTTP headers in a separate file (headers.txt in the example):
|
||||||
|
|
||||||
curl --dump-header headers.txt curl.haxx.se
|
curl --dump-header headers.txt curl.haxx.se
|
||||||
|
|
||||||
@@ -237,32 +245,32 @@ POST (HTTP)
|
|||||||
|
|
||||||
-F accepts parameters like -F "name=contents". If you want the contents to
|
-F accepts parameters like -F "name=contents". If you want the contents to
|
||||||
be read from a file, use <@filename> as contents. When specifying a file,
|
be read from a file, use <@filename> as contents. When specifying a file,
|
||||||
you can also specify which content type the file is, by appending
|
you can also specify the file content type by appending ';type=<mime type>'
|
||||||
';type=<mime type>' to the file name. You can also post contents of several
|
to the file name. You can also post the contents of several files in one field.
|
||||||
files in one field. So that the field name 'coolfiles' can be sent three
|
For example, the field name 'coolfiles' is used to send three files, with
|
||||||
files with different content types in a manner similar to:
|
different content types using the following syntax:
|
||||||
|
|
||||||
curl -F "coolfiles=@fil1.gif;type=image/gif,fil2.txt,fil3.html" \
|
curl -F "coolfiles=@fil1.gif;type=image/gif,fil2.txt,fil3.html" \
|
||||||
http://www.post.com/postit.cgi
|
http://www.post.com/postit.cgi
|
||||||
|
|
||||||
If content-type is not specified, curl will try to guess from the extension
|
If the content-type is not specified, curl will try to guess from the file
|
||||||
(it only knows a few), or use the previously specified type (from an earlier
|
extension (it only knows a few), or use the previously specified type
|
||||||
file if several files are specified in a list) or finally using the default
|
(from an earlier file if several files are specified in a list) or else it
|
||||||
type 'text/plain'.
|
will using the default type 'text/plain'.
|
||||||
|
|
||||||
Emulate a fill-in form with -F. Let's say you fill in three fields in a
|
Emulate a fill-in form with -F. Let's say you fill in three fields in a
|
||||||
form. One field is a file name which to post, one field is your name and one
|
form. One field is a file name which to post, one field is your name and one
|
||||||
field is a file description. We want to post the file we have written named
|
field is a file description. We want to post the file we have written named
|
||||||
"cooltext.txt". To let curl do the posting of this data instead of your
|
"cooltext.txt". To let curl do the posting of this data instead of your
|
||||||
favourite browser, you have to check out the HTML of the form page to get to
|
favourite browser, you have to read the HTML source of the form page and find
|
||||||
know the names of the input fields. In our example, the input field names are
|
the names of the input fields. In our example, the input field names are
|
||||||
'file', 'yourname' and 'filedescription'.
|
'file', 'yourname' and 'filedescription'.
|
||||||
|
|
||||||
curl -F "file=@cooltext.txt" -F "yourname=Daniel" \
|
curl -F "file=@cooltext.txt" -F "yourname=Daniel" \
|
||||||
-F "filedescription=Cool text file with cool text inside" \
|
-F "filedescription=Cool text file with cool text inside" \
|
||||||
http://www.post.com/postit.cgi
|
http://www.post.com/postit.cgi
|
||||||
|
|
||||||
So, to send two files in one post you can do it in two ways:
|
To send two files in one post you can do it in two ways:
|
||||||
|
|
||||||
1. Send multiple files in a single "field" with a single field name:
|
1. Send multiple files in a single "field" with a single field name:
|
||||||
|
|
||||||
@@ -272,11 +280,11 @@ POST (HTTP)
|
|||||||
|
|
||||||
curl -F "docpicture=@dog.gif" -F "catpicture=@cat.gif"
|
curl -F "docpicture=@dog.gif" -F "catpicture=@cat.gif"
|
||||||
|
|
||||||
REFERER
|
REFERRER
|
||||||
|
|
||||||
A HTTP request has the option to include information about which address
|
A HTTP request has the option to include information about which address
|
||||||
that referred to actual page, and curl allows the user to specify that
|
that referred to actual page. Curl allows you to specify the
|
||||||
referrer to get specified on the command line. It is especially useful to
|
referrer to be used on the command line. It is especially useful to
|
||||||
fool or trick stupid servers or CGI scripts that rely on that information
|
fool or trick stupid servers or CGI scripts that rely on that information
|
||||||
being available or contain certain data.
|
being available or contain certain data.
|
||||||
|
|
||||||
@@ -345,13 +353,17 @@ COOKIES
|
|||||||
Note that by specifying -b you enable the "cookie awareness" and with -L
|
Note that by specifying -b you enable the "cookie awareness" and with -L
|
||||||
you can make curl follow a location: (which often is used in combination
|
you can make curl follow a location: (which often is used in combination
|
||||||
with cookies). So that if a site sends cookies and a location, you can
|
with cookies). So that if a site sends cookies and a location, you can
|
||||||
use a non-existing file to trig the cookie awareness like:
|
use a non-existing file to trigger the cookie awareness like:
|
||||||
|
|
||||||
curl -L -b empty-file www.example.com
|
curl -L -b empty.txt www.example.com
|
||||||
|
|
||||||
The file to read cookies from must be formatted using plain HTTP headers OR
|
The file to read cookies from must be formatted using plain HTTP headers OR
|
||||||
as netscape's cookie file. Curl will determine what kind it is based on the
|
as netscape's cookie file. Curl will determine what kind it is based on the
|
||||||
file contents.
|
file contents. In the above command, curl will parse the header and store
|
||||||
|
the cookies received from www.example.com. curl will send to the server the
|
||||||
|
stored cookies which match the request as it follows the location. The
|
||||||
|
file "empty.txt" may be a non-existant file.
|
||||||
|
|
||||||
|
|
||||||
PROGRESS METER
|
PROGRESS METER
|
||||||
|
|
||||||
@@ -384,12 +396,12 @@ PROGRESS METER
|
|||||||
|
|
||||||
SPEED LIMIT
|
SPEED LIMIT
|
||||||
|
|
||||||
Curl offers the user to set conditions regarding transfer speed that must
|
Curl allows the user to set the transfer speed conditions that must be met
|
||||||
be met to let the transfer keep going. By using the switch -y and -Y you
|
to let the transfer keep going. By using the switch -y and -Y you
|
||||||
can make curl abort transfers if the transfer speed doesn't exceed your
|
can make curl abort transfers if the transfer speed is below the specified
|
||||||
given lowest limit for a specified time.
|
lowest limit for a specified time.
|
||||||
|
|
||||||
To let curl abandon downloading this page if its slower than 3000 bytes per
|
To have curl abort the download if the speed is slower than 3000 bytes per
|
||||||
second for 1 minute, run:
|
second for 1 minute, run:
|
||||||
|
|
||||||
curl -y 3000 -Y 60 www.far-away-site.com
|
curl -y 3000 -Y 60 www.far-away-site.com
|
||||||
@@ -455,9 +467,13 @@ EXTRA HEADERS
|
|||||||
|
|
||||||
curl -H "X-you-and-me: yes" www.love.com
|
curl -H "X-you-and-me: yes" www.love.com
|
||||||
|
|
||||||
This can also be useful in case you want curl to send a different text in
|
This can also be useful in case you want curl to send a different text in a
|
||||||
a header than it normally does. The -H header you specify then replaces the
|
header than it normally does. The -H header you specify then replaces the
|
||||||
header curl would normally send.
|
header curl would normally send. If you replace an internal header with an
|
||||||
|
empty one, you prevent that header from being sent. To prevent the Host:
|
||||||
|
header from being used:
|
||||||
|
|
||||||
|
curl -H "Host:" www.server.com
|
||||||
|
|
||||||
FTP and PATH NAMES
|
FTP and PATH NAMES
|
||||||
|
|
||||||
@@ -598,7 +614,7 @@ RESUMING FILE TRANSFERS
|
|||||||
(*1) = This requires that the ftp server supports the non-standard command
|
(*1) = This requires that the ftp server supports the non-standard command
|
||||||
SIZE. If it doesn't, curl will say so.
|
SIZE. If it doesn't, curl will say so.
|
||||||
|
|
||||||
(*2) = This requires that the wb server supports at least HTTP/1.1. If it
|
(*2) = This requires that the web server supports at least HTTP/1.1. If it
|
||||||
doesn't, curl will say so.
|
doesn't, curl will say so.
|
||||||
|
|
||||||
TIME CONDITIONS
|
TIME CONDITIONS
|
||||||
@@ -745,6 +761,25 @@ TELNET
|
|||||||
to track when the login prompt is received and send the username and
|
to track when the login prompt is received and send the username and
|
||||||
password accordingly.
|
password accordingly.
|
||||||
|
|
||||||
|
PERSISTANT CONNECTIONS
|
||||||
|
|
||||||
|
Specifying multiple files on a single command line will make curl transfer
|
||||||
|
all of them, one after the other in the specified order.
|
||||||
|
|
||||||
|
libcurl will attempt to use persistant connections for the transfers so that
|
||||||
|
the second transfer to the same host can use the same connection that was
|
||||||
|
already initiated and was left open in the previous transfer. This greatly
|
||||||
|
decreases connection time for all but the first transfer and it makes a far
|
||||||
|
better use of the network.
|
||||||
|
|
||||||
|
Note that curl cannot use persistant connections for transfers that are used
|
||||||
|
in subsequence curl invokes. Try to stuff as many URLs as possible on the
|
||||||
|
same command line if they are using the same host, as that'll make the
|
||||||
|
transfers faster. If you use a http proxy for file transfers, practicly
|
||||||
|
all transfers will be persistant.
|
||||||
|
|
||||||
|
Persistant connections were introduced in curl 7.7.
|
||||||
|
|
||||||
MAILING LISTS
|
MAILING LISTS
|
||||||
|
|
||||||
For your convenience, we have several open mailing lists to discuss curl,
|
For your convenience, we have several open mailing lists to discuss curl,
|
||||||
@@ -753,10 +788,10 @@ MAILING LISTS
|
|||||||
To subscribe to the main curl list, mail curl-request@contactor.se with
|
To subscribe to the main curl list, mail curl-request@contactor.se with
|
||||||
"subscribe <fill in your email address>" in the body.
|
"subscribe <fill in your email address>" in the body.
|
||||||
|
|
||||||
To subscribe to the libcurl users list, follow the instructions at
|
To subscribe to the curl-library users/deverlopers list, follow the
|
||||||
http://curl.haxx.se/mail/
|
instructions at http://curl.haxx.se/mail/
|
||||||
|
|
||||||
To subscribe to the curl announce list, to only get information about new
|
To subscribe to the curl-announce list, to only get information about new
|
||||||
releases, follow the instructions at http://curl.haxx.se/mail/
|
releases, follow the instructions at http://curl.haxx.se/mail/
|
||||||
|
|
||||||
To subscribe to the curl-and-PHP list in which curl using with PHP is
|
To subscribe to the curl-and-PHP list in which curl using with PHP is
|
||||||
|
@@ -17,7 +17,9 @@ man_MANS = \
|
|||||||
curl_getenv.3 \
|
curl_getenv.3 \
|
||||||
curl_slist_append.3 \
|
curl_slist_append.3 \
|
||||||
curl_slist_free_all.3 \
|
curl_slist_free_all.3 \
|
||||||
curl_version.3
|
curl_version.3 \
|
||||||
|
curl_escape.3 \
|
||||||
|
curl_unescape.3
|
||||||
|
|
||||||
EXTRA_DIST = $(man_MANS) \
|
EXTRA_DIST = $(man_MANS) \
|
||||||
MANUAL BUGS CONTRIBUTE FAQ FEATURES INTERNALS \
|
MANUAL BUGS CONTRIBUTE FAQ FEATURES INTERNALS \
|
||||||
|
86
docs/TODO
86
docs/TODO
@@ -6,41 +6,49 @@
|
|||||||
|
|
||||||
TODO
|
TODO
|
||||||
|
|
||||||
For the future
|
Things to do in project cURL. Please tell me what you think, contribute and
|
||||||
|
send me patches that improve things!
|
||||||
|
|
||||||
Ok, this is what I wanna do with Curl. Please tell me what you think, and
|
To do for the next release:
|
||||||
please don't hesitate to contribute and send me patches that improve this
|
|
||||||
product! (Yes, you may add things not mentioned here, these are just a
|
|
||||||
few teasers...)
|
|
||||||
|
|
||||||
* Make SSL session ids get used if multiple HTTPS documents from the same
|
* Make SSL session ids get used if multiple HTTPS documents from the same
|
||||||
host is requested.
|
host is requested.
|
||||||
|
|
||||||
* Make the curl tool support URLs that start with @ that would then mean that
|
* Document the undocumented libcurl functions: the printf clones (like
|
||||||
the following is a plain list with URLs to download. Thus @filename.txt
|
curl_msprintf, curl_mfprintf, curl_msnprintf, curl_maprintf and
|
||||||
reads a list of URLs from a local file. A fancy option would then be to
|
curl_mvfprintf) and the string compare functions (curl_strequal
|
||||||
support @http://whatever.com that would first load a list and then get the
|
and curl_strnequal).
|
||||||
URLs mentioned in the list. I figure -O or something would have to be
|
|
||||||
implied by such an action.
|
To do in a future release (random order):
|
||||||
|
|
||||||
|
* Rewrite parts of the test suite. Make a (XML?) format to store all
|
||||||
|
test-data in a single for a single test case. The current system makes far
|
||||||
|
too many separate files. We also need to have the test suite support
|
||||||
|
different behaviors, like when libcurl is compiled for IPv6 support and
|
||||||
|
thus performs a different set of FTP commands.
|
||||||
|
|
||||||
|
* Add configure options that disables certain protocols in libcurl to
|
||||||
|
decrease footprint. '--disable-[protocol]' where protocol is http, ftp,
|
||||||
|
telnet, ldap, dict or file.
|
||||||
|
|
||||||
|
* Extend the test suite to include telnet and https. The telnet could just do
|
||||||
|
ftp or http operations (for which we have test servers) and the https would
|
||||||
|
probably work against/with some of the openssl tools.
|
||||||
|
|
||||||
* Add a command line option that allows the output file to get the same time
|
* Add a command line option that allows the output file to get the same time
|
||||||
stamp as the remote file. libcurl already is capable of fetching the remote
|
stamp as the remote file. libcurl already is capable of fetching the remote
|
||||||
file's date.
|
file's date.
|
||||||
|
|
||||||
* Make the SSL layer option capable of using the Mozilla Security Services as
|
* Make curl's SSL layer option capable of using other free SSL libraries.
|
||||||
an alternative to OpenSSL:
|
Such as the Mozilla Security Services
|
||||||
http://www.mozilla.org/projects/security/pki/nss/
|
(http://www.mozilla.org/projects/security/pki/nss/) and GNUTLS
|
||||||
|
(http://gnutls.hellug.gr/)
|
||||||
* Make sure the low-level interface works. highlevel.c should basically be
|
|
||||||
possible to write using that interface. Document the low-level interface
|
|
||||||
|
|
||||||
* Make the easy-interface support multiple file transfers. If they're done
|
|
||||||
to the same host, they should use persistant connections or similar.
|
|
||||||
Figure out a nice design for this.
|
|
||||||
|
|
||||||
* Add asynchronous name resolving, as this enables full timeout support for
|
* Add asynchronous name resolving, as this enables full timeout support for
|
||||||
fork() systems.
|
fork() systems.
|
||||||
|
|
||||||
|
* Non-blocking connect(), also to make timeouts work on windows.
|
||||||
|
|
||||||
* Move non-URL related functions that are used by both the lib and the curl
|
* Move non-URL related functions that are used by both the lib and the curl
|
||||||
application to a separate "portability lib".
|
application to a separate "portability lib".
|
||||||
|
|
||||||
@@ -48,14 +56,13 @@ For the future
|
|||||||
something being worked on in this area) and perl (we have seen the first
|
something being worked on in this area) and perl (we have seen the first
|
||||||
versions of this!) comes to mind. Python anyone?
|
versions of this!) comes to mind. Python anyone?
|
||||||
|
|
||||||
* "Content-Encoding: compress/gzip/zlib"
|
* "Content-Encoding: compress/gzip/zlib" HTTP 1.1 clearly defines how to get
|
||||||
|
and decode compressed documents. There is the zlib that is pretty good at
|
||||||
|
decompressing stuff. This work was started in October 1999 but halted again
|
||||||
|
since it proved more work than we thought. It is still a good idea to
|
||||||
|
implement though.
|
||||||
|
|
||||||
HTTP 1.1 clearly defines how to get and decode compressed documents. There
|
* Authentication: NTLM. Support for that MS crap called NTLM
|
||||||
is the zlib that is pretty good at decompressing stuff. This work was
|
|
||||||
started in October 1999 but halted again since it proved more work than we
|
|
||||||
thought. It is still a good idea to implement though.
|
|
||||||
|
|
||||||
* Authentication: NTLM. It would be to support that MS crap called NTLM
|
|
||||||
authentication. MS proxies and servers sometime require that. Since that
|
authentication. MS proxies and servers sometime require that. Since that
|
||||||
protocol is a proprietary one, it involves reverse engineering and network
|
protocol is a proprietary one, it involves reverse engineering and network
|
||||||
sniffing. This should however be a library-based functionality. There are a
|
sniffing. This should however be a library-based functionality. There are a
|
||||||
@@ -67,33 +74,22 @@ For the future
|
|||||||
|
|
||||||
* RFC2617 compliance, "Digest Access Authentication"
|
* RFC2617 compliance, "Digest Access Authentication"
|
||||||
A valid test page seem to exist at:
|
A valid test page seem to exist at:
|
||||||
http://hopf.math.nwu.edu/testpage/digest/
|
http://hopf.math.nwu.edu/testpage/digest/
|
||||||
And some friendly person's server source code is available at
|
And some friendly person's server source code is available at
|
||||||
http://hopf.math.nwu.edu/digestauth/index.html
|
http://hopf.math.nwu.edu/digestauth/index.html
|
||||||
|
|
||||||
Then there's the Apache mod_digest source code too of course. It seems as
|
Then there's the Apache mod_digest source code too of course. It seems as
|
||||||
if Netscape doesn't support this, and not many servers do. Although this is
|
if Netscape doesn't support this, and not many servers do. Although this is
|
||||||
a lot better authentication method than the more common "Basic". Basic
|
a lot better authentication method than the more common "Basic". Basic
|
||||||
sends the password in cleartext over the network, this "Digest" method uses
|
sends the password in cleartext over the network, this "Digest" method uses
|
||||||
a challange-response protocol which increases security quite a lot.
|
a challange-response protocol which increases security quite a lot.
|
||||||
|
|
||||||
* Multiple Proxies?
|
|
||||||
Is there anyone that actually uses serial-proxies? I mean, send CONNECT to
|
|
||||||
the first proxy to connect to the second proxy to which you send CONNECT to
|
|
||||||
connect to the remote host (or even more iterations). Is there anyone
|
|
||||||
wanting curl to support it? (Not that it would be hard, just confusing...)
|
|
||||||
|
|
||||||
* Other proxies
|
* Other proxies
|
||||||
Ftp-kind proxy, Socks5, whatever kind of proxies are there?
|
Ftp-kind proxy, Socks5, whatever kind of proxies are there?
|
||||||
|
|
||||||
* IPv6 Awareness and support
|
* IPv6 Awareness and support. (This is partly done.) RFC 2428 "FTP
|
||||||
Where ever it would fit. configure search for v6-versions of a few
|
Extensions for IPv6 and NATs" is interesting. PORT should be replaced with
|
||||||
functions and then use them instead is of course the first thing to do...
|
EPRT for IPv6 (done), and EPSV instead of PASV. HTTP proxies are left to
|
||||||
RFC 2428 "FTP Extensions for IPv6 and NATs" will be interesting. PORT
|
add support for.
|
||||||
should be replaced with EPRT for IPv6, and EPSV instead of PASV.
|
|
||||||
|
|
||||||
* SSL for more protocols, like SSL-FTP...
|
* SSL for more protocols, like SSL-FTP...
|
||||||
(http://search.ietf.org/internet-drafts/draft-murray-auth-ftp-ssl-05.txt)
|
(http://search.ietf.org/internet-drafts/draft-murray-auth-ftp-ssl-05.txt)
|
||||||
|
|
||||||
* HTTP POST resume using Range:
|
|
||||||
|
|
||||||
|
53
docs/curl.1
53
docs/curl.1
@@ -2,7 +2,7 @@
|
|||||||
.\" nroff -man curl.1
|
.\" nroff -man curl.1
|
||||||
.\" Written by Daniel Stenberg
|
.\" Written by Daniel Stenberg
|
||||||
.\"
|
.\"
|
||||||
.TH curl 1 "19 January 2001" "Curl 7.6" "Curl Manual"
|
.TH curl 1 "24 March 2001" "Curl 7.7" "Curl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl \- get a URL with FTP, TELNET, LDAP, GOPHER, DICT, FILE, HTTP or
|
curl \- get a URL with FTP, TELNET, LDAP, GOPHER, DICT, FILE, HTTP or
|
||||||
HTTPS syntax.
|
HTTPS syntax.
|
||||||
@@ -41,6 +41,12 @@ supported at the moment:
|
|||||||
|
|
||||||
Starting with curl 7.6, you can specify any amount of URLs on the command
|
Starting with curl 7.6, you can specify any amount of URLs on the command
|
||||||
line. They will be fetched in a sequential manner in the specified order.
|
line. They will be fetched in a sequential manner in the specified order.
|
||||||
|
|
||||||
|
Starting with curl 7.7, curl will attempt to re-use connections for multiple
|
||||||
|
file transfers, so that getting many files from the same server will not do
|
||||||
|
multiple connects/handshakes. This improves speed. Of course this is only done
|
||||||
|
on files specified on a single command line and cannot be used between
|
||||||
|
separate curl invokes.
|
||||||
.SH OPTIONS
|
.SH OPTIONS
|
||||||
.IP "-a/--append"
|
.IP "-a/--append"
|
||||||
(FTP)
|
(FTP)
|
||||||
@@ -85,6 +91,14 @@ also be enforced by using an URL that ends with ";type=A". This option causes
|
|||||||
data sent to stdout to be in text mode for win32 systems.
|
data sent to stdout to be in text mode for win32 systems.
|
||||||
|
|
||||||
If this option is used twice, the second one will disable ASCII usage.
|
If this option is used twice, the second one will disable ASCII usage.
|
||||||
|
.IP "--connect-timeout <seconds>"
|
||||||
|
Maximum time in seconds that you allow the connection to the server to take.
|
||||||
|
This only limits the connection phase, once curl has connected this option is
|
||||||
|
of no more use. This option doesn't work in win32 systems. See also the
|
||||||
|
.I "--max-time"
|
||||||
|
option.
|
||||||
|
|
||||||
|
If this option is used serveral times, the last one will be used.
|
||||||
.IP "-c/--continue"
|
.IP "-c/--continue"
|
||||||
.B Deprecated. Use '-C -' instead.
|
.B Deprecated. Use '-C -' instead.
|
||||||
Continue/Resume a previous file transfer. This instructs curl to
|
Continue/Resume a previous file transfer. This instructs curl to
|
||||||
@@ -105,15 +119,16 @@ HTTP resume is only possible with HTTP/1.1 or later servers.
|
|||||||
|
|
||||||
If this option is used serveral times, the last one will be used.
|
If this option is used serveral times, the last one will be used.
|
||||||
.IP "-d/--data <data>"
|
.IP "-d/--data <data>"
|
||||||
(HTTP) Sends the specified data in a POST request to the HTTP server. Note
|
(HTTP) Sends the specified data in a POST request to the HTTP server, in a way
|
||||||
that the data is sent exactly as specified with no extra processing (with all
|
that can emulate as if a user has filled in a HTML form and pressed the submit
|
||||||
newlines cut off). The data is expected to be "url-encoded". This will cause
|
button. Note that the data is sent exactly as specified with no extra
|
||||||
curl to pass the data to the server using the content-type
|
processing (with all newlines cut off). The data is expected to be
|
||||||
application/x-www-form-urlencoded. Compare to -F. If more than one -d/--data
|
"url-encoded". This will cause curl to pass the data to the server using the
|
||||||
option is used on the same command line, the data pieces specified will be
|
content-type application/x-www-form-urlencoded. Compare to -F. If more than
|
||||||
merged together with a separating &-letter. Thus, using '-d name=daniel -d
|
one -d/--data option is used on the same command line, the data pieces
|
||||||
skill=lousy' would generate a post chunk that looks like
|
specified will be merged together with a separating &-letter. Thus, using '-d
|
||||||
'name=daniel&skill=lousy'.
|
name=daniel -d skill=lousy' would generate a post chunk that looks like
|
||||||
|
\&'name=daniel&skill=lousy'.
|
||||||
|
|
||||||
If you start the data with the letter @, the rest should be a file name to
|
If you start the data with the letter @, the rest should be a file name to
|
||||||
read the data from, or - if you want curl to read the data from stdin. The
|
read the data from, or - if you want curl to read the data from stdin. The
|
||||||
@@ -160,6 +175,11 @@ previous URL when it follows a Location: header. The ";auto" string can be
|
|||||||
used alone, even if you don't set an initial referer.
|
used alone, even if you don't set an initial referer.
|
||||||
|
|
||||||
If this option is used serveral times, the last one will be used.
|
If this option is used serveral times, the last one will be used.
|
||||||
|
.IP "--egd-file <file>"
|
||||||
|
(HTTPS) Specify the path name to the Entropy Gathering Daemon socket. The
|
||||||
|
socket is used to seed the random engine for SSL connections. See also the
|
||||||
|
.I "--random-file"
|
||||||
|
option.
|
||||||
.IP "-E/--cert <certificate[:password]>"
|
.IP "-E/--cert <certificate[:password]>"
|
||||||
(HTTPS)
|
(HTTPS)
|
||||||
Tells curl to use the specified certificate file when getting a file
|
Tells curl to use the specified certificate file when getting a file
|
||||||
@@ -195,7 +215,7 @@ get attached in the post as a file upload, while the < makes a text field and
|
|||||||
just get the contents for that text field from a file.
|
just get the contents for that text field from a file.
|
||||||
|
|
||||||
Example, to send your password file to the server, where
|
Example, to send your password file to the server, where
|
||||||
'password' is the name of the form-field to which /etc/passwd will be the
|
\&'password' is the name of the form-field to which /etc/passwd will be the
|
||||||
input:
|
input:
|
||||||
|
|
||||||
.B curl
|
.B curl
|
||||||
@@ -283,6 +303,9 @@ If this option is used twice, the second will again disable location following.
|
|||||||
Maximum time in seconds that you allow the whole operation to take. This is
|
Maximum time in seconds that you allow the whole operation to take. This is
|
||||||
useful for preventing your batch jobs from hanging for hours due to slow
|
useful for preventing your batch jobs from hanging for hours due to slow
|
||||||
networks or links going down. This doesn't work fully in win32 systems.
|
networks or links going down. This doesn't work fully in win32 systems.
|
||||||
|
See also the
|
||||||
|
.I "--connect-timeout"
|
||||||
|
option.
|
||||||
|
|
||||||
If this option is used serveral times, the last one will be used.
|
If this option is used serveral times, the last one will be used.
|
||||||
.IP "-M/--manual"
|
.IP "-M/--manual"
|
||||||
@@ -302,7 +325,7 @@ directory.
|
|||||||
A quick and very simple example of how to setup a
|
A quick and very simple example of how to setup a
|
||||||
.I .netrc
|
.I .netrc
|
||||||
to allow curl to ftp to the machine host.domain.com with user name
|
to allow curl to ftp to the machine host.domain.com with user name
|
||||||
'myself' and password 'secret' should look similar to:
|
\&'myself' and password 'secret' should look similar to:
|
||||||
|
|
||||||
.B "machine host.domain.com login myself password secret"
|
.B "machine host.domain.com login myself password secret"
|
||||||
|
|
||||||
@@ -377,6 +400,12 @@ to be run before and after the transfer. If the server returns failure for one
|
|||||||
of the commands, the entire operation will be aborted.
|
of the commands, the entire operation will be aborted.
|
||||||
|
|
||||||
This option can be used multiple times.
|
This option can be used multiple times.
|
||||||
|
.IP "--random-file <file>"
|
||||||
|
(HTTPS) Specify the path name to file containing what will be considered as
|
||||||
|
random data. The data is used to seed the random engine for SSL connections.
|
||||||
|
See also the
|
||||||
|
.I "--edg-file"
|
||||||
|
option.
|
||||||
.IP "-r/--range <range>"
|
.IP "-r/--range <range>"
|
||||||
(HTTP/FTP)
|
(HTTP/FTP)
|
||||||
Retrieve a byte range (i.e a partial document) from a HTTP/1.1 or FTP
|
Retrieve a byte range (i.e a partial document) from a HTTP/1.1 or FTP
|
||||||
|
@@ -2,13 +2,13 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_easy_cleanup 3 "22 May 2000" "Curl 7.0" "libcurl Manual"
|
.TH curl_easy_cleanup 3 "5 March 2001" "libcurl 7.7" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_easy_cleanup - End a libcurl "easy" session
|
curl_easy_cleanup - End a libcurl session
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/easy.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "curl_easy_cleanup(CURL *" handle ");
|
.BI "curl_easy_cleanup(CURL *" handle ");"
|
||||||
.ad
|
.ad
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
This function must be the last function to call for a curl session. It is the
|
This function must be the last function to call for a curl session. It is the
|
||||||
@@ -17,6 +17,10 @@ opposite of the
|
|||||||
function and must be called with the same
|
function and must be called with the same
|
||||||
.I handle
|
.I handle
|
||||||
as input as the curl_easy_init call returned.
|
as input as the curl_easy_init call returned.
|
||||||
|
|
||||||
|
This will effectively close all connections libcurl has been used and possibly
|
||||||
|
has kept open until now. Don't call this function if you intend to transfer
|
||||||
|
more files (libcurl 7.7 or later).
|
||||||
.SH RETURN VALUE
|
.SH RETURN VALUE
|
||||||
None
|
None
|
||||||
.SH "SEE ALSO"
|
.SH "SEE ALSO"
|
||||||
|
@@ -2,11 +2,11 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_easy_init 3 "4 February 2001" "Curl 7.6.1" "libcurl Manual"
|
.TH curl_easy_init 3 "5 March 2001" "libcurl 7.6.1" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_easy_getinfo - Extract information from a curl session (added in 7.4)
|
curl_easy_getinfo - Extract information from a curl session (added in 7.4)
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/easy.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "CURLcode curl_easy_getinfo(CURL *curl, CURLINFO info, ... );"
|
.BI "CURLcode curl_easy_getinfo(CURL *curl, CURLINFO info, ... );"
|
||||||
.ad
|
.ad
|
||||||
@@ -83,8 +83,8 @@ verification that was requested (using the CURLOPT_SSL_VERIFYPEER option to
|
|||||||
curl_easy_setopt). (Added in 7.4.2)
|
curl_easy_setopt). (Added in 7.4.2)
|
||||||
.TP
|
.TP
|
||||||
.B CURLINFO_CONTENT_LENGTH_DOWNLOAD
|
.B CURLINFO_CONTENT_LENGTH_DOWNLOAD
|
||||||
Pass a pointer to a double to receive the content-length of the download.
|
Pass a pointer to a double to receive the content-length of the download. This
|
||||||
(Added in 7.6.1)
|
is the value read from the Content-Length: field. (Added in 7.6.1)
|
||||||
.TP
|
.TP
|
||||||
.B CURLINFO_CONTENT_LENGTH_UPLOAD
|
.B CURLINFO_CONTENT_LENGTH_UPLOAD
|
||||||
Pass a pointer to a double to receive the specified size of the upload.
|
Pass a pointer to a double to receive the specified size of the upload.
|
||||||
|
@@ -2,11 +2,11 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_easy_init 3 "26 September 2000" "Curl 7.0" "libcurl Manual"
|
.TH curl_easy_init 3 "5 March 2001" "libcurl 7.7" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_easy_init - Start a libcurl "easy" session
|
curl_easy_init - Start a libcurl session
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/easy.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "CURL *curl_easy_init( );"
|
.BI "CURL *curl_easy_init( );"
|
||||||
.ad
|
.ad
|
||||||
@@ -19,6 +19,10 @@ when the operation is complete.
|
|||||||
|
|
||||||
On win32 systems, you need to init the winsock stuff manually, libcurl will
|
On win32 systems, you need to init the winsock stuff manually, libcurl will
|
||||||
not do that for you. WSAStartup() and WSACleanup() should be used accordingly.
|
not do that for you. WSAStartup() and WSACleanup() should be used accordingly.
|
||||||
|
|
||||||
|
Using libcurl 7.7 and later, you should perform all your sequential file
|
||||||
|
transfers using the same curl handle. This enables libcurl to use persistant
|
||||||
|
connections where possible.
|
||||||
.SH RETURN VALUE
|
.SH RETURN VALUE
|
||||||
If this function returns NULL, something went wrong and you cannot use the
|
If this function returns NULL, something went wrong and you cannot use the
|
||||||
other curl functions.
|
other curl functions.
|
||||||
|
@@ -2,11 +2,11 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_easy_perform 3 "1 Mar 2001" "Curl 7.0" "libcurl Manual"
|
.TH curl_easy_perform 3 "5 Mar 2001" "libcurl 7.7" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_easy_perform - Do the actual transfer in a "easy" session
|
curl_easy_perform - Perform a file transfer
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/easy.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "CURLcode curl_easy_perform(CURL *" handle ");
|
.BI "CURLcode curl_easy_perform(CURL *" handle ");
|
||||||
.ad
|
.ad
|
||||||
@@ -17,9 +17,22 @@ It must be called with the same
|
|||||||
.I handle
|
.I handle
|
||||||
as input as the curl_easy_init call returned.
|
as input as the curl_easy_init call returned.
|
||||||
|
|
||||||
You are only allowed to call this function once using the same handle. If you
|
libcurl version 7.7 or later (for older versions see below): You can do any
|
||||||
want to do repeated calls, you must call curl_easy_cleanup and curl_easy_init
|
amount of calls to curl_easy_perform() while using the same handle. If you
|
||||||
again first.
|
intend to transfer more than one file, you are even encouraged to do
|
||||||
|
so. libcurl will then attempt to re-use the same connection for the following
|
||||||
|
transfers, thus making the operations faster, less CPU intense and using less
|
||||||
|
network resources. Just note that you will have to use
|
||||||
|
.I curl_easy_setopt
|
||||||
|
between the invokes to set options for the following curl_easy_perform.
|
||||||
|
|
||||||
|
You must never call this function simultaneously from two places using the
|
||||||
|
same handle. Let the function return first before invoking it another time. If
|
||||||
|
you want parallel transfers, you must use several curl handles.
|
||||||
|
|
||||||
|
Before libcurl version 7.7: You are only allowed to call this function once
|
||||||
|
using the same handle. If you want to do repeated calls, you must call
|
||||||
|
curl_easy_cleanup and curl_easy_init again first.
|
||||||
.SH RETURN VALUE
|
.SH RETURN VALUE
|
||||||
0 means everything was ok, non-zero means an error occurred as
|
0 means everything was ok, non-zero means an error occurred as
|
||||||
.I <curl/curl.h>
|
.I <curl/curl.h>
|
||||||
|
@@ -2,13 +2,13 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_easy_setopt 3 "2 February 2001" "Curl 7.5" "libcurl Manual"
|
.TH curl_easy_setopt 3 "30 March 2001" "libcurl 7.7" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_easy_setopt - Set curl easy-session options
|
curl_easy_setopt - Set curl easy-session options
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/easy.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "CURLcode curl_easy_setopt(CURL *" handle ", CURLoption "option ", ...);
|
.BI "CURLcode curl_easy_setopt(CURL *" handle ", CURLoption "option ", ...);"
|
||||||
.ad
|
.ad
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
curl_easy_setopt() is called to tell libcurl how to behave in a number of
|
curl_easy_setopt() is called to tell libcurl how to behave in a number of
|
||||||
@@ -20,7 +20,18 @@ followed by a parameter. That parameter can be a long, a function pointer or
|
|||||||
an object pointer, all depending on what the option in question expects. Read
|
an object pointer, all depending on what the option in question expects. Read
|
||||||
this manual carefully as bad input values may cause libcurl to behave badly!
|
this manual carefully as bad input values may cause libcurl to behave badly!
|
||||||
You can only set one option in each function call. A typical application uses
|
You can only set one option in each function call. A typical application uses
|
||||||
many calls in the setup phase.
|
many curl_easy_setopt() calls in the setup phase.
|
||||||
|
|
||||||
|
NOTE: strings passed to libcurl as 'char *' arguments, will not be copied by
|
||||||
|
the library. Instead you should keep them available until libcurl no longer
|
||||||
|
needs them. Failing to do so will cause very odd behaviour or even crashes.
|
||||||
|
|
||||||
|
More note: the options set with this function call are valid for the
|
||||||
|
forthcoming data transfers that are performed when you invoke
|
||||||
|
.I curl_easy_perform .
|
||||||
|
The options are not in any way reset between transfers, so if you want
|
||||||
|
subsequent transfers with different options, you must change them between the
|
||||||
|
transfers.
|
||||||
|
|
||||||
The
|
The
|
||||||
.I "handle"
|
.I "handle"
|
||||||
@@ -86,14 +97,16 @@ libcurl what the expected size of the infile is.
|
|||||||
.TP
|
.TP
|
||||||
.B CURLOPT_URL
|
.B CURLOPT_URL
|
||||||
The actual URL to deal with. The parameter should be a char * to a zero
|
The actual URL to deal with. The parameter should be a char * to a zero
|
||||||
terminated string. NOTE: this option is currently required!
|
terminated string. The string must remain present until curl no longer needs
|
||||||
|
it, as it doesn't copy the string. NOTE: this option is required to be set
|
||||||
|
before curl_easy_perform() is called.
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_PROXY
|
.B CURLOPT_PROXY
|
||||||
If you need libcurl to use a http proxy to access the outside world, set the
|
If you need libcurl to use a http proxy to access the outside world, set the
|
||||||
proxy string with this option. The parameter should be a char * to a zero
|
proxy string with this option. The parameter should be a char * to a zero
|
||||||
terminated string. To specify port number in this string, append":[port]" to
|
terminated string. To specify port number in this string, append :[port] to
|
||||||
the end of the host name. The proxy string may be prefixed with
|
the end of the host name. The proxy string may be prefixed with
|
||||||
"[protocol]://" since any such prefix will be ignored.
|
[protocol]:// since any such prefix will be ignored.
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_PROXYPORT
|
.B CURLOPT_PROXYPORT
|
||||||
Set this long with this option to set the proxy port to use unless it is
|
Set this long with this option to set the proxy port to use unless it is
|
||||||
@@ -189,9 +202,11 @@ prompted for it.
|
|||||||
.TP
|
.TP
|
||||||
.B CURLOPT_RANGE
|
.B CURLOPT_RANGE
|
||||||
Pass a char * as parameter, which should contain the specified range you
|
Pass a char * as parameter, which should contain the specified range you
|
||||||
want. It should be in the format "X-Y", where X or Y may be left out. The HTTP
|
want. It should be in the format "X-Y", where X or Y may be left out. HTTP
|
||||||
transfers also support several intervals, separated with commas as in
|
transfers also support several intervals, separated with commas as in
|
||||||
.I "X-Y,N-M".
|
.I "X-Y,N-M"
|
||||||
|
. Using this kind of multiple intervals will cause the HTTP server to send the
|
||||||
|
response document in pieces.
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_ERRORBUFFER
|
.B CURLOPT_ERRORBUFFER
|
||||||
Pass a char * to a buffer that the libcurl may store human readable error
|
Pass a char * to a buffer that the libcurl may store human readable error
|
||||||
@@ -202,7 +217,8 @@ library. The buffer must be at least CURL_ERROR_SIZE big.
|
|||||||
Pass a long as parameter containing the maximum time in seconds that you allow
|
Pass a long as parameter containing the maximum time in seconds that you allow
|
||||||
the libcurl transfer operation to take. Do note that normally, name lookups
|
the libcurl transfer operation to take. Do note that normally, name lookups
|
||||||
maky take a considerable time and that limiting the operation to less than a
|
maky take a considerable time and that limiting the operation to less than a
|
||||||
few minutes risk aborting perfectly normal operations.
|
few minutes risk aborting perfectly normal operations. This option will cause
|
||||||
|
curl to use the SIGALRM to enable timeouting system calls.
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_POSTFIELDS
|
.B CURLOPT_POSTFIELDS
|
||||||
Pass a char * as parameter, which should be the full data to post in a HTTP
|
Pass a char * as parameter, which should be the full data to post in a HTTP
|
||||||
@@ -258,7 +274,11 @@ curl_slist' structs properly filled in. Use
|
|||||||
.I curl_slist_append(3)
|
.I curl_slist_append(3)
|
||||||
to create the list and
|
to create the list and
|
||||||
.I curl_slist_free_all(3)
|
.I curl_slist_free_all(3)
|
||||||
to clean up an entire list.
|
to clean up an entire list. If you add a header that is otherwise generated
|
||||||
|
and used by libcurl internally, your added one will be used instead. If you
|
||||||
|
add a header with no contents as in 'Accept:', the internally used header will
|
||||||
|
just get disabled. Thus, using this option you can add new headers, replace
|
||||||
|
internal headers and remove internal headers.
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_HTTPPOST
|
.B CURLOPT_HTTPPOST
|
||||||
Tells libcurl you want a multipart/formdata HTTP POST to be made and you
|
Tells libcurl you want a multipart/formdata HTTP POST to be made and you
|
||||||
@@ -297,7 +317,9 @@ struct curl_slist structs properly filled in as described for
|
|||||||
.I "CURLOPT_QUOTE"
|
.I "CURLOPT_QUOTE"
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_WRITEHEADER
|
.B CURLOPT_WRITEHEADER
|
||||||
Pass a FILE * to be used to write the header part of the received data to.
|
Pass a FILE * to be used to write the header part of the received data to. The
|
||||||
|
headers are guaranteed to be written one-by-one to this file handle and only
|
||||||
|
complete lines are written. Parsing headers should be easy enough using this.
|
||||||
.TP
|
.TP
|
||||||
.B CURLOPT_COOKIEFILE
|
.B CURLOPT_COOKIEFILE
|
||||||
Pass a pointer to a zero terminated string as parameter. It should contain the
|
Pass a pointer to a zero terminated string as parameter. It should contain the
|
||||||
@@ -337,7 +359,7 @@ name. (Added in libcurl 7.3)
|
|||||||
.B CURLOPT_KRB4LEVEL
|
.B CURLOPT_KRB4LEVEL
|
||||||
Pass a char * as parameter. Set the krb4 security level, this also enables
|
Pass a char * as parameter. Set the krb4 security level, this also enables
|
||||||
krb4 awareness. This is a string, 'clear', 'safe', 'confidential' or
|
krb4 awareness. This is a string, 'clear', 'safe', 'confidential' or
|
||||||
'private'. If the string is set but doesn't match one of these, 'private'
|
\&'private'. If the string is set but doesn't match one of these, 'private'
|
||||||
will be used. Set the string to NULL to disable kerberos4. The kerberos
|
will be used. Set the string to NULL to disable kerberos4. The kerberos
|
||||||
support only works for FTP. (Added in libcurl 7.3)
|
support only works for FTP. (Added in libcurl 7.3)
|
||||||
.TP
|
.TP
|
||||||
@@ -410,6 +432,59 @@ Pass a long. The set number will be the redirection limit. If that many
|
|||||||
redirections have been followed, the next redirect will cause an error. This
|
redirections have been followed, the next redirect will cause an error. This
|
||||||
option only makes sense if the CURLOPT_FOLLOWLOCATION is used at the same
|
option only makes sense if the CURLOPT_FOLLOWLOCATION is used at the same
|
||||||
time. (Added in 7.5)
|
time. (Added in 7.5)
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_MAXCONNECTS
|
||||||
|
Pass a long. The set number will be the persistant connection cache size. The
|
||||||
|
set amount will be the maximum amount of simultaneous connections that libcurl
|
||||||
|
may cache between file transfers. Default is 5, and there isn't much point in
|
||||||
|
changing this value unless you are perfectly aware of how this work and
|
||||||
|
changes libcurl's behaviour. Note: if you have already performed transfers
|
||||||
|
with this curl handle, setting a smaller MAXCONNECTS than before may cause
|
||||||
|
open connections to unnecessarily get closed. (Added in 7.7)
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_CLOSEPOLICY
|
||||||
|
Pass a long. This option sets what policy libcurl should use when the
|
||||||
|
connection cache is filled and one of the open connections has to be closed to
|
||||||
|
make room for a new connection. This must be one of the CURLCLOSEPOLICY_*
|
||||||
|
defines. Use CURLCLOSEPOLICY_LEAST_RECENTLY_USED to make libcurl close the
|
||||||
|
connection that was least recently used, that connection is also least likely
|
||||||
|
to be capable of re-use. Use CURLCLOSEPOLICY_OLDEST to make libcurl close the
|
||||||
|
oldest connection, the one that was created first among the ones in the
|
||||||
|
connection cache. The other close policies are not support yet. (Added in 7.7)
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_FRESH_CONNECT
|
||||||
|
Pass a long. Set to non-zero to make the next transfer use a new connection by
|
||||||
|
force. If the connection cache is full before this connection, one of the
|
||||||
|
existinf connections will be closed as according to the set policy. This
|
||||||
|
option should be used with caution and only if you understand what it
|
||||||
|
does. Set to 0 to have libcurl attempt re-use of an existing connection.
|
||||||
|
(Added in 7.7)
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_FORBID_REUSE
|
||||||
|
Pass a long. Set to non-zero to make the next transfer explicitly close the
|
||||||
|
connection when done. Normally, libcurl keep all connections alive when done
|
||||||
|
with one transfer in case there comes a succeeding one that can re-use them.
|
||||||
|
This option should be used with caution and only if you understand what it
|
||||||
|
does. Set to 0 to have libcurl keep the connection open for possibly later
|
||||||
|
re-use. (Added in 7.7)
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_RANDOM_FILE
|
||||||
|
Pass a char * to a zero terminated file name. The file will be used to read
|
||||||
|
from to seed the random engine for SSL. The more random the specified file is,
|
||||||
|
the more secure will the SSL connection become.
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_FORBID_REUSE
|
||||||
|
Pass a char * to the zero terminated path name to the Entropy Gathering Daemon
|
||||||
|
socket. It will be used to seed the random engine for SSL.
|
||||||
|
.TP
|
||||||
|
.B CURLOPT_CONNECTTIMEOUT
|
||||||
|
Pass a long. It should contain the maximum time in seconds that you allow the
|
||||||
|
connection to the server to take. This only limits the connection phase, once
|
||||||
|
it has connected, this option is of no more use. Set to zero to disable
|
||||||
|
connection timeout (it will then only timeout on the system's internal
|
||||||
|
timeouts). This option doesn't work in win32 systems. See also the
|
||||||
|
.I CURLOPT_TIMEOUT
|
||||||
|
option.
|
||||||
.PP
|
.PP
|
||||||
.SH RETURN VALUE
|
.SH RETURN VALUE
|
||||||
0 means the option was set properly, non-zero means an error as
|
0 means the option was set properly, non-zero means an error as
|
||||||
|
28
docs/curl_escape.3
Normal file
28
docs/curl_escape.3
Normal file
@@ -0,0 +1,28 @@
|
|||||||
|
.\" You can view this file with:
|
||||||
|
.\" nroff -man [file]
|
||||||
|
.\" Written by daniel@haxx.se
|
||||||
|
.\"
|
||||||
|
.TH curl_escape 3 "22 March 2001" "libcurl 7.7" "libcurl Manual"
|
||||||
|
.SH NAME
|
||||||
|
curl_escape - URL encodes the given string
|
||||||
|
.SH SYNOPSIS
|
||||||
|
.B #include <curl/curl.h>
|
||||||
|
.sp
|
||||||
|
.BI "char *curl_escape( char *" url ", int "length " );"
|
||||||
|
.ad
|
||||||
|
.SH DESCRIPTION
|
||||||
|
This function will convert the given input string to an URL encoded string and
|
||||||
|
return that as a new allocated string. All input characters that are not a-z,
|
||||||
|
A-Z or 0-9 will be converted to their "URL escaped" version. If a sequence of
|
||||||
|
%NN (where NN is a two-digit hexadecimal number) is found in the string to
|
||||||
|
encode, that 3-letter combination will be copied to the output unmodifed,
|
||||||
|
assuming that it is an already encoded piece of data.
|
||||||
|
|
||||||
|
If the 'length' argument is set to 0, curl_escape() will use strlen() on the
|
||||||
|
input 'url' string to find out the size.
|
||||||
|
|
||||||
|
You must free() the returned string when you're done with it.
|
||||||
|
.SH RETURN VALUE
|
||||||
|
A pointer to a zero terminated string or NULL if it failed.
|
||||||
|
.SH "SEE ALSO"
|
||||||
|
.I curl_unescape(), RFC 2396
|
@@ -2,7 +2,7 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_formfree 3 "17 November 2000" "Curl 7.5" "libcurl Manual"
|
.TH curl_formfree 3 "5 March 2001" "libcurl 7.5" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_formfree - free a previously build multipart/formdata HTTP POST chain
|
curl_formfree - free a previously build multipart/formdata HTTP POST chain
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_formparse 3 "22 February 2001" "Curl 7.0" "libcurl Manual"
|
.TH curl_formparse 3 "5 March 2001" "libcurl 7.0" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_formparse - add a section to a multipart/formdata HTTP POST
|
curl_formparse - add a section to a multipart/formdata HTTP POST
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_getdate 3 "2 June 2000" "Curl 7.0" "libcurl Manual"
|
.TH curl_getdate 3 "5 March 2001" "libcurl 7.0" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_getdate - Convert an date in a ASCII string to number of seconds since
|
curl_getdate - Convert an date in a ASCII string to number of seconds since
|
||||||
January 1, 1970
|
January 1, 1970
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_getenv 3 "2 June 2000" "Curl 7.0" "libcurl Manual"
|
.TH curl_getenv 3 "5 March 2001" "libcurl 7.0" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_getenv - return value for environment name
|
curl_getenv - return value for environment name
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
|
@@ -2,7 +2,7 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_slist_append 3 "1 Mar 2001" "Curl 7.7" "libcurl Manual"
|
.TH curl_slist_append 3 "5 March 2001" "libcurl 7.0" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_slist_append - add a string to an slist
|
curl_slist_append - add a string to an slist
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
|
@@ -2,13 +2,13 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_slist_free_all 3 "2 June 2000" "Curl 7.0" "libcurl Manual"
|
.TH curl_slist_free_all 3 "5 March 2001" "libcurl 7.0" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_slist_free_all - free an entire curl_slist list
|
curl_slist_free_all - free an entire curl_slist list
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/curl.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "void curl_slist_free_all(struct curl_slit *" list);
|
.BI "void curl_slist_free_all(struct curl_slist *" list);
|
||||||
.ad
|
.ad
|
||||||
.SH DESCRIPTION
|
.SH DESCRIPTION
|
||||||
curl_slist_free_all() removes all traces of a previously built curl_slist
|
curl_slist_free_all() removes all traces of a previously built curl_slist
|
||||||
|
27
docs/curl_unescape.3
Normal file
27
docs/curl_unescape.3
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
.\" You can view this file with:
|
||||||
|
.\" nroff -man [file]
|
||||||
|
.\" Written by daniel@haxx.se
|
||||||
|
.\"
|
||||||
|
.TH curl_unescape 3 "22 March 2001" "libcurl 7.7" "libcurl Manual"
|
||||||
|
.SH NAME
|
||||||
|
curl_unescape - URL decodes the given string
|
||||||
|
.SH SYNOPSIS
|
||||||
|
.B #include <curl/curl.h>
|
||||||
|
.sp
|
||||||
|
.BI "char *curl_unescape( char *" url ", int "length " );"
|
||||||
|
.ad
|
||||||
|
.SH DESCRIPTION
|
||||||
|
This function will convert the given URL encoded input string to a "plain
|
||||||
|
string" and return that as a new allocated string. All input characters that
|
||||||
|
are URL encoded (%XX where XX is a two-digit hexadecimal number, or +) will be
|
||||||
|
converted to their plain text versions (up to a ? letter, no letters to the
|
||||||
|
right of a ? letter will be converted).
|
||||||
|
|
||||||
|
If the 'length' argument is set to 0, curl_unescape() will use strlen() on the
|
||||||
|
input 'url' string to find out the size.
|
||||||
|
|
||||||
|
You must free() the returned string when you're done with it.
|
||||||
|
.SH RETURN VALUE
|
||||||
|
A pointer to a zero terminated string or NULL if it failed.
|
||||||
|
.SH "SEE ALSO"
|
||||||
|
.I curl_escape(), RFC 2396
|
@@ -2,11 +2,11 @@
|
|||||||
.\" nroff -man [file]
|
.\" nroff -man [file]
|
||||||
.\" Written by daniel@haxx.se
|
.\" Written by daniel@haxx.se
|
||||||
.\"
|
.\"
|
||||||
.TH curl_version 3 "2 June 2000" "Curl 7.0" "libcurl Manual"
|
.TH curl_version 3 "5 March 2001" "libcurl 7.0" "libcurl Manual"
|
||||||
.SH NAME
|
.SH NAME
|
||||||
curl_version - returns the libcurl version string
|
curl_version - returns the libcurl version string
|
||||||
.SH SYNOPSIS
|
.SH SYNOPSIS
|
||||||
.B #include <curl/easy.h>
|
.B #include <curl/curl.h>
|
||||||
.sp
|
.sp
|
||||||
.BI "char *curl_version( );"
|
.BI "char *curl_version( );"
|
||||||
.ad
|
.ad
|
||||||
@@ -14,9 +14,9 @@ curl_version - returns the libcurl version string
|
|||||||
Returns a human readable string with the version number of libcurl and some of
|
Returns a human readable string with the version number of libcurl and some of
|
||||||
its important components (like OpenSSL version).
|
its important components (like OpenSSL version).
|
||||||
|
|
||||||
Do note that this returns the actual running lib's version, you might have
|
Note: this returns the actual running lib's version, you might have installed
|
||||||
installed a newer lib's include files in your system which may turn your
|
a newer lib's include files in your system which may turn your LIBCURL_VERSION
|
||||||
LIBCURL_VERSION #define value to differ from this result.
|
#define value to differ from this result.
|
||||||
.SH RETURN VALUE
|
.SH RETURN VALUE
|
||||||
A pointer to a zero terminated string.
|
A pointer to a zero terminated string.
|
||||||
.SH "SEE ALSO"
|
.SH "SEE ALSO"
|
||||||
|
@@ -6,8 +6,7 @@ AUTOMAKE_OPTIONS = foreign no-dependencies
|
|||||||
|
|
||||||
EXTRA_DIST =
|
EXTRA_DIST =
|
||||||
README curlgtk.c sepheaders.c simple.c postit.c \
|
README curlgtk.c sepheaders.c simple.c postit.c \
|
||||||
win32sockets.c \
|
win32sockets.c persistant.c ftpget.c Makefile.example
|
||||||
getpageinvar.php simpleget.php simplepost.php
|
|
||||||
|
|
||||||
all:
|
all:
|
||||||
@echo "done"
|
@echo "done"
|
||||||
|
41
docs/examples/Makefile.example
Normal file
41
docs/examples/Makefile.example
Normal file
@@ -0,0 +1,41 @@
|
|||||||
|
#############################################################################
|
||||||
|
# _ _ ____ _
|
||||||
|
# Project ___| | | | _ \| |
|
||||||
|
# / __| | | | |_) | |
|
||||||
|
# | (__| |_| | _ <| |___
|
||||||
|
# \___|\___/|_| \_\_____|
|
||||||
|
#
|
||||||
|
# $Id$
|
||||||
|
#
|
||||||
|
|
||||||
|
# What to call the final executable
|
||||||
|
TARGET = example
|
||||||
|
|
||||||
|
# Which object files that the executable consists of
|
||||||
|
OBJS= ftpget.o
|
||||||
|
|
||||||
|
# What compiler to use
|
||||||
|
CC = gcc
|
||||||
|
|
||||||
|
# Compiler flags, -g for debug, -c to make an object file
|
||||||
|
CFLAGS = -c -g
|
||||||
|
|
||||||
|
# This should point to a directory that holds libcurl, if it isn't
|
||||||
|
# in the system's standard lib dir
|
||||||
|
# We also set a -L to include the directory where we have the openssl
|
||||||
|
# libraries
|
||||||
|
LDFLAGS = -L/home/dast/lib -L/usr/local/ssl/lib
|
||||||
|
|
||||||
|
# We need -lcurl for the curl stuff
|
||||||
|
# We need -lsocket and -lnsl when on Solaris
|
||||||
|
# We need -lssl and -lcrypto when using libcurl with SSL support
|
||||||
|
# We need -ldl for dlopen() if that is in libdl
|
||||||
|
LIBS = -lcurl -lsocket -lnsl -lssl -lcrypto -dl
|
||||||
|
|
||||||
|
# Link the target with all objects and libraries
|
||||||
|
$(TARGET) : $(OBJS)
|
||||||
|
$(CC) $(LDFLAGS) $(LIBS) -o $(TARGET) $(OBJS)
|
||||||
|
|
||||||
|
# Compile the source files into object files
|
||||||
|
ftpget.o : ftpget.c
|
||||||
|
$(CC) $(CFLAGS) $<
|
@@ -7,5 +7,8 @@ advantage of libcurl.
|
|||||||
If you end up with other small but still useful example sources, please mail
|
If you end up with other small but still useful example sources, please mail
|
||||||
them for submission in future packages and on the web site.
|
them for submission in future packages and on the web site.
|
||||||
|
|
||||||
There are examples for different languages and environments. Browse around to
|
The Makefile.example is an example makefile that could be used to build these
|
||||||
find those that fit you.
|
examples. Just edit the file according to your system and requirements first.
|
||||||
|
|
||||||
|
Try the php/examples/ directory for PHP programming snippets!
|
||||||
|
|
||||||
|
44
docs/examples/ftpget.c
Normal file
44
docs/examples/ftpget.c
Normal file
@@ -0,0 +1,44 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* _ _ ____ _
|
||||||
|
* Project ___| | | | _ \| |
|
||||||
|
* / __| | | | |_) | |
|
||||||
|
* | (__| |_| | _ <| |___
|
||||||
|
* \___|\___/|_| \_\_____|
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
|
||||||
|
#include <curl/curl.h>
|
||||||
|
#include <curl/types.h>
|
||||||
|
#include <curl/easy.h>
|
||||||
|
|
||||||
|
/* to make this work under windows, use the win32-functions from the
|
||||||
|
win32socket.c file as well */
|
||||||
|
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
CURL *curl;
|
||||||
|
CURLcode res;
|
||||||
|
FILE *ftpfile;
|
||||||
|
|
||||||
|
/* local file name to store the file as */
|
||||||
|
ftpfile = fopen("curl.tar.gz", "wb"); /* b is binary for win */
|
||||||
|
|
||||||
|
curl = curl_easy_init();
|
||||||
|
if(curl) {
|
||||||
|
/* Get curl 7.7 from sunet.se's FTP site: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_URL,
|
||||||
|
"ftp://ftp.sunet.se/pub/www/utilities/curl/curl-7.7.tar.gz");
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FILE, ftpfile);
|
||||||
|
res = curl_easy_perform(curl);
|
||||||
|
|
||||||
|
/* always cleanup */
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
}
|
||||||
|
|
||||||
|
fclose(ftpfile); /* close the local file */
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
53
docs/examples/persistant.c
Normal file
53
docs/examples/persistant.c
Normal file
@@ -0,0 +1,53 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* _ _ ____ _
|
||||||
|
* Project ___| | | | _ \| |
|
||||||
|
* / __| | | | |_) | |
|
||||||
|
* | (__| |_| | _ <| |___
|
||||||
|
* \___|\___/|_| \_\_____|
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
*/
|
||||||
|
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <unistd.h>
|
||||||
|
|
||||||
|
#include <curl/curl.h>
|
||||||
|
|
||||||
|
/* to make this work under windows, use the win32-functions from the
|
||||||
|
docs/examples/win32socket.c file as well */
|
||||||
|
|
||||||
|
/* This example REQUIRES libcurl 7.7 or later */
|
||||||
|
#if (LIBCURL_VERSION_NUM < 0x070700)
|
||||||
|
#error Too old libcurl version, upgrade or stay away.
|
||||||
|
#endif
|
||||||
|
|
||||||
|
int main(int argc, char **argv)
|
||||||
|
{
|
||||||
|
CURL *curl;
|
||||||
|
CURLcode res;
|
||||||
|
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
/* this sends all memory debug messages to a specified logfile */
|
||||||
|
curl_memdebug("memdump");
|
||||||
|
#endif
|
||||||
|
|
||||||
|
curl = curl_easy_init();
|
||||||
|
if(curl) {
|
||||||
|
curl_easy_setopt(curl, CURLOPT_VERBOSE, 1);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_HEADER, 1);
|
||||||
|
|
||||||
|
/* get the first document */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_URL, "http://curl.haxx.se/");
|
||||||
|
res = curl_easy_perform(curl);
|
||||||
|
|
||||||
|
/* get another document from the same server using the same
|
||||||
|
connection */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_URL, "http://curl.haxx.se/docs/");
|
||||||
|
res = curl_easy_perform(curl);
|
||||||
|
|
||||||
|
/* always cleanup */
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
}
|
||||||
|
|
||||||
|
return 0;
|
||||||
|
}
|
@@ -97,71 +97,57 @@ typedef int (*curl_passwd_callback)(void *clientp,
|
|||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
CURLE_OK = 0,
|
CURLE_OK = 0,
|
||||||
CURLE_UNSUPPORTED_PROTOCOL,
|
CURLE_UNSUPPORTED_PROTOCOL, /* 1 */
|
||||||
CURLE_FAILED_INIT,
|
CURLE_FAILED_INIT, /* 2 */
|
||||||
CURLE_URL_MALFORMAT,
|
CURLE_URL_MALFORMAT, /* 3 */
|
||||||
CURLE_URL_MALFORMAT_USER,
|
CURLE_URL_MALFORMAT_USER, /* 4 */
|
||||||
CURLE_COULDNT_RESOLVE_PROXY,
|
CURLE_COULDNT_RESOLVE_PROXY, /* 5 */
|
||||||
CURLE_COULDNT_RESOLVE_HOST,
|
CURLE_COULDNT_RESOLVE_HOST, /* 6 */
|
||||||
CURLE_COULDNT_CONNECT,
|
CURLE_COULDNT_CONNECT, /* 7 */
|
||||||
CURLE_FTP_WEIRD_SERVER_REPLY,
|
CURLE_FTP_WEIRD_SERVER_REPLY, /* 8 */
|
||||||
CURLE_FTP_ACCESS_DENIED,
|
CURLE_FTP_ACCESS_DENIED, /* 9 */
|
||||||
CURLE_FTP_USER_PASSWORD_INCORRECT,
|
CURLE_FTP_USER_PASSWORD_INCORRECT, /* 10 */
|
||||||
CURLE_FTP_WEIRD_PASS_REPLY,
|
CURLE_FTP_WEIRD_PASS_REPLY, /* 11 */
|
||||||
CURLE_FTP_WEIRD_USER_REPLY,
|
CURLE_FTP_WEIRD_USER_REPLY, /* 12 */
|
||||||
CURLE_FTP_WEIRD_PASV_REPLY,
|
CURLE_FTP_WEIRD_PASV_REPLY, /* 13 */
|
||||||
CURLE_FTP_WEIRD_227_FORMAT,
|
CURLE_FTP_WEIRD_227_FORMAT, /* 14 */
|
||||||
CURLE_FTP_CANT_GET_HOST,
|
CURLE_FTP_CANT_GET_HOST, /* 15 */
|
||||||
CURLE_FTP_CANT_RECONNECT,
|
CURLE_FTP_CANT_RECONNECT, /* 16 */
|
||||||
CURLE_FTP_COULDNT_SET_BINARY,
|
CURLE_FTP_COULDNT_SET_BINARY, /* 17 */
|
||||||
CURLE_PARTIAL_FILE,
|
CURLE_PARTIAL_FILE, /* 18 */
|
||||||
CURLE_FTP_COULDNT_RETR_FILE,
|
CURLE_FTP_COULDNT_RETR_FILE, /* 19 */
|
||||||
CURLE_FTP_WRITE_ERROR,
|
CURLE_FTP_WRITE_ERROR, /* 20 */
|
||||||
CURLE_FTP_QUOTE_ERROR,
|
CURLE_FTP_QUOTE_ERROR, /* 21 */
|
||||||
CURLE_HTTP_NOT_FOUND,
|
CURLE_HTTP_NOT_FOUND, /* 22 */
|
||||||
CURLE_WRITE_ERROR,
|
CURLE_WRITE_ERROR, /* 23 */
|
||||||
|
CURLE_MALFORMAT_USER, /* 24 - user name is illegally specified */
|
||||||
|
CURLE_FTP_COULDNT_STOR_FILE, /* 25 - failed FTP upload */
|
||||||
|
CURLE_READ_ERROR, /* 26 - could open/read from file */
|
||||||
|
CURLE_OUT_OF_MEMORY, /* 27 */
|
||||||
|
CURLE_OPERATION_TIMEOUTED, /* 28 - the timeout time was reached */
|
||||||
|
CURLE_FTP_COULDNT_SET_ASCII, /* 29 - TYPE A failed */
|
||||||
|
CURLE_FTP_PORT_FAILED, /* 30 - FTP PORT operation failed */
|
||||||
|
CURLE_FTP_COULDNT_USE_REST, /* 31 - the REST command failed */
|
||||||
|
CURLE_FTP_COULDNT_GET_SIZE, /* 32 - the SIZE command failed */
|
||||||
|
CURLE_HTTP_RANGE_ERROR, /* 33 - RANGE "command" didn't work */
|
||||||
|
CURLE_HTTP_POST_ERROR, /* 34 */
|
||||||
|
CURLE_SSL_CONNECT_ERROR, /* 35 - wrong when connecting with SSL */
|
||||||
|
CURLE_FTP_BAD_DOWNLOAD_RESUME, /* 36 - couldn't resume download */
|
||||||
|
CURLE_FILE_COULDNT_READ_FILE, /* 37 */
|
||||||
|
CURLE_LDAP_CANNOT_BIND, /* 38 */
|
||||||
|
CURLE_LDAP_SEARCH_FAILED, /* 39 */
|
||||||
|
CURLE_LIBRARY_NOT_FOUND, /* 40 */
|
||||||
|
CURLE_FUNCTION_NOT_FOUND, /* 41 */
|
||||||
|
CURLE_ABORTED_BY_CALLBACK, /* 42 */
|
||||||
|
CURLE_BAD_FUNCTION_ARGUMENT, /* 43 */
|
||||||
|
CURLE_BAD_CALLING_ORDER, /* 44 */
|
||||||
|
CURLE_HTTP_PORT_FAILED, /* 45 - HTTP Interface operation failed */
|
||||||
|
CURLE_BAD_PASSWORD_ENTERED, /* 46 - my_getpass() returns fail */
|
||||||
|
CURLE_TOO_MANY_REDIRECTS , /* 47 - catch endless re-direct loops */
|
||||||
|
CURLE_UNKNOWN_TELNET_OPTION, /* 48 - User specified an unknown option */
|
||||||
|
CURLE_TELNET_OPTION_SYNTAX , /* 49 - Malformed telnet option */
|
||||||
|
|
||||||
CURLE_MALFORMAT_USER, /* the user name is illegally specified */
|
CURL_LAST /* never use! */
|
||||||
CURLE_FTP_COULDNT_STOR_FILE, /* failed FTP upload */
|
|
||||||
CURLE_READ_ERROR, /* could open/read from file */
|
|
||||||
|
|
||||||
CURLE_OUT_OF_MEMORY,
|
|
||||||
CURLE_OPERATION_TIMEOUTED, /* the timeout time was reached */
|
|
||||||
CURLE_FTP_COULDNT_SET_ASCII, /* TYPE A failed */
|
|
||||||
|
|
||||||
CURLE_FTP_PORT_FAILED, /* FTP PORT operation failed */
|
|
||||||
|
|
||||||
CURLE_FTP_COULDNT_USE_REST, /* the REST command failed */
|
|
||||||
CURLE_FTP_COULDNT_GET_SIZE, /* the SIZE command failed */
|
|
||||||
|
|
||||||
CURLE_HTTP_RANGE_ERROR, /* The RANGE "command" didn't seem to work */
|
|
||||||
|
|
||||||
CURLE_HTTP_POST_ERROR,
|
|
||||||
|
|
||||||
CURLE_SSL_CONNECT_ERROR, /* something was wrong when connecting with SSL */
|
|
||||||
|
|
||||||
CURLE_FTP_BAD_DOWNLOAD_RESUME, /* couldn't resume download */
|
|
||||||
|
|
||||||
CURLE_FILE_COULDNT_READ_FILE,
|
|
||||||
|
|
||||||
CURLE_LDAP_CANNOT_BIND,
|
|
||||||
CURLE_LDAP_SEARCH_FAILED,
|
|
||||||
CURLE_LIBRARY_NOT_FOUND,
|
|
||||||
CURLE_FUNCTION_NOT_FOUND,
|
|
||||||
|
|
||||||
CURLE_ABORTED_BY_CALLBACK,
|
|
||||||
CURLE_BAD_FUNCTION_ARGUMENT,
|
|
||||||
CURLE_BAD_CALLING_ORDER,
|
|
||||||
|
|
||||||
CURLE_HTTP_PORT_FAILED, /* HTTP Interface operation failed */
|
|
||||||
|
|
||||||
CURLE_BAD_PASSWORD_ENTERED, /* when the my_getpass() returns fail */
|
|
||||||
CURLE_TOO_MANY_REDIRECTS , /* catch endless re-direct loops */
|
|
||||||
|
|
||||||
CURLE_UNKNOWN_TELNET_OPTION , /* User specified an unknown option */
|
|
||||||
CURLE_TELNET_OPTION_SYNTAX , /* Malformed telnet option */
|
|
||||||
|
|
||||||
CURL_LAST
|
|
||||||
} CURLcode;
|
} CURLcode;
|
||||||
|
|
||||||
/* This is just to make older programs not break: */
|
/* This is just to make older programs not break: */
|
||||||
@@ -412,6 +398,38 @@ typedef enum {
|
|||||||
/* This points to a linked list of telnet options */
|
/* This points to a linked list of telnet options */
|
||||||
CINIT(TELNETOPTIONS, OBJECTPOINT, 70),
|
CINIT(TELNETOPTIONS, OBJECTPOINT, 70),
|
||||||
|
|
||||||
|
/* Max amount of cached alive connections */
|
||||||
|
CINIT(MAXCONNECTS, LONG, 71),
|
||||||
|
|
||||||
|
/* What policy to use when closing connections when the cache is filled
|
||||||
|
up */
|
||||||
|
CINIT(CLOSEPOLICY, LONG, 72),
|
||||||
|
|
||||||
|
/* Callback to use when CURLCLOSEPOLICY_CALLBACK is set */
|
||||||
|
CINIT(CLOSEFUNCTION, FUNCTIONPOINT, 73),
|
||||||
|
|
||||||
|
/* Set to explicitly use a new connection for the upcoming transfer.
|
||||||
|
Do not use this unless you're absolutely sure of this, as it makes the
|
||||||
|
operation slower and is less friendly for the network. */
|
||||||
|
CINIT(FRESH_CONNECT, LONG, 74),
|
||||||
|
|
||||||
|
/* Set to explicitly forbid the upcoming transfer's connection to be re-used
|
||||||
|
when done. Do not use this unless you're absolutely sure of this, as it
|
||||||
|
makes the operation slower and is less friendly for the network. */
|
||||||
|
CINIT(FORBID_REUSE, LONG, 75),
|
||||||
|
|
||||||
|
/* Set to a file name that contains random data for libcurl to use to
|
||||||
|
seed the random engine when doing SSL connects. */
|
||||||
|
CINIT(RANDOM_FILE, OBJECTPOINT, 76),
|
||||||
|
|
||||||
|
/* Set to the Entropy Gathering Daemon socket pathname */
|
||||||
|
CINIT(EGDSOCKET, OBJECTPOINT, 77),
|
||||||
|
|
||||||
|
/* Time-out connect operations after this amount of seconds, if connects
|
||||||
|
are OK within this time, then fine... This only aborts the connect
|
||||||
|
phase. [Only works on unix-style/SIGALRM operating systems] */
|
||||||
|
CINIT(CONNECTTIMEOUT, LONG, 78),
|
||||||
|
|
||||||
CURLOPT_LASTENTRY /* the last unusued */
|
CURLOPT_LASTENTRY /* the last unusued */
|
||||||
} CURLoption;
|
} CURLoption;
|
||||||
|
|
||||||
@@ -437,10 +455,10 @@ typedef enum {
|
|||||||
|
|
||||||
NOTE: they return TRUE if the strings match *case insensitively*.
|
NOTE: they return TRUE if the strings match *case insensitively*.
|
||||||
*/
|
*/
|
||||||
extern int (Curl_strequal)(const char *s1, const char *s2);
|
extern int (curl_strequal)(const char *s1, const char *s2);
|
||||||
extern int (Curl_strnequal)(const char *s1, const char *s2, size_t n);
|
extern int (curl_strnequal)(const char *s1, const char *s2, size_t n);
|
||||||
#define strequal(a,b) Curl_strequal(a,b)
|
#define strequal(a,b) curl_strequal(a,b)
|
||||||
#define strnequal(a,b,c) Curl_strnequal(a,b,c)
|
#define strnequal(a,b,c) curl_strnequal(a,b,c)
|
||||||
|
|
||||||
/* external form function */
|
/* external form function */
|
||||||
int curl_formparse(char *string,
|
int curl_formparse(char *string,
|
||||||
@@ -457,9 +475,14 @@ char *curl_getenv(char *variable);
|
|||||||
/* Returns a static ascii string of the libcurl version. */
|
/* Returns a static ascii string of the libcurl version. */
|
||||||
char *curl_version(void);
|
char *curl_version(void);
|
||||||
|
|
||||||
|
/* Escape and unescape URL encoding in strings. The functions return a new
|
||||||
|
* allocated string or NULL if an error occurred. */
|
||||||
|
char *curl_escape(char *string, int length);
|
||||||
|
char *curl_unescape(char *string, int length);
|
||||||
|
|
||||||
/* This is the version number */
|
/* This is the version number */
|
||||||
#define LIBCURL_VERSION "7.7-alpha2"
|
#define LIBCURL_VERSION "7.7.1"
|
||||||
#define LIBCURL_VERSION_NUM 0x070000
|
#define LIBCURL_VERSION_NUM 0x070701
|
||||||
|
|
||||||
/* linked-list structure for the CURLOPT_QUOTE option (and other) */
|
/* linked-list structure for the CURLOPT_QUOTE option (and other) */
|
||||||
struct curl_slist {
|
struct curl_slist {
|
||||||
@@ -516,21 +539,6 @@ typedef enum {
|
|||||||
before it can be included! */
|
before it can be included! */
|
||||||
#include <curl/easy.h> /* nothing in curl is fun without the easy stuff */
|
#include <curl/easy.h> /* nothing in curl is fun without the easy stuff */
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_getinfo()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Request internal information from the curl session with this function.
|
|
||||||
* The third argument MUST be a pointer to a long or a pointer to a char *.
|
|
||||||
* The data pointed to will be filled in accordingly and can be relied upon
|
|
||||||
* only if the function returns CURLE_OK.
|
|
||||||
* This function is intended to get used *AFTER* a performed transfer, all
|
|
||||||
* results are undefined before the transfer is completed.
|
|
||||||
*/
|
|
||||||
CURLcode curl_getinfo(CURL *curl, CURLINFO info, ...);
|
|
||||||
|
|
||||||
|
|
||||||
typedef enum {
|
typedef enum {
|
||||||
CURLCLOSEPOLICY_NONE, /* first, never use this */
|
CURLCLOSEPOLICY_NONE, /* first, never use this */
|
||||||
|
|
||||||
|
@@ -16,7 +16,7 @@ lib_LTLIBRARIES = libcurl.la
|
|||||||
INCLUDES = -I$(top_srcdir)/include
|
INCLUDES = -I$(top_srcdir)/include
|
||||||
|
|
||||||
|
|
||||||
libcurl_la_LDFLAGS = -version-info 1:0:0
|
libcurl_la_LDFLAGS = -version-info 2:0:0
|
||||||
# This flag accepts an argument of the form current[:revision[:age]]. So,
|
# This flag accepts an argument of the form current[:revision[:age]]. So,
|
||||||
# passing -version-info 3:12:1 sets current to 3, revision to 12, and age to
|
# passing -version-info 3:12:1 sets current to 3, revision to 12, and age to
|
||||||
# 1.
|
# 1.
|
||||||
@@ -58,7 +58,8 @@ getenv.c ldap.h ssluse.h \
|
|||||||
escape.c mprintf.c telnet.c \
|
escape.c mprintf.c telnet.c \
|
||||||
escape.h getpass.c netrc.c telnet.h \
|
escape.h getpass.c netrc.c telnet.h \
|
||||||
getinfo.c transfer.c strequal.c strequal.h easy.c \
|
getinfo.c transfer.c strequal.c strequal.h easy.c \
|
||||||
security.h security.c krb4.c krb4.h memdebug.c memdebug.h inet_ntoa_r.h
|
security.h security.c krb4.c krb4.h memdebug.c memdebug.h inet_ntoa_r.h \
|
||||||
|
http_chunks.c http_chunks.h
|
||||||
|
|
||||||
noinst_HEADERS = setup.h transfer.h
|
noinst_HEADERS = setup.h transfer.h
|
||||||
|
|
||||||
|
@@ -33,13 +33,13 @@ libcurl_a_SOURCES = arpa_telnet.h file.c getpass.h netrc.h timeval.c base64.c \
|
|||||||
urldata.h transfer.c getdate.h ldap.c ssluse.c version.c transfer.h getenv.c \
|
urldata.h transfer.c getdate.h ldap.c ssluse.c version.c transfer.h getenv.c \
|
||||||
ldap.h ssluse.h escape.c getenv.h mprintf.c telnet.c escape.h getpass.c netrc.c \
|
ldap.h ssluse.h escape.c getenv.h mprintf.c telnet.c escape.h getpass.c netrc.c \
|
||||||
telnet.h getinfo.c strequal.c strequal.h easy.c security.h \
|
telnet.h getinfo.c strequal.c strequal.h easy.c security.h \
|
||||||
security.c krb4.c
|
security.c krb4.h krb4.c memdebug.h memdebug.c inet_ntoa_r.h http_chunks.h http_chunks.c
|
||||||
|
|
||||||
libcurl_a_OBJECTS = file.o timeval.o base64.o hostip.o progress.o \
|
libcurl_a_OBJECTS = file.o timeval.o base64.o hostip.o progress.o \
|
||||||
formdata.o cookie.o http.o sendf.o ftp.o url.o dict.o if2ip.o \
|
formdata.o cookie.o http.o sendf.o ftp.o url.o dict.o if2ip.o \
|
||||||
speedcheck.o getdate.o transfer.o ldap.o ssluse.o version.o \
|
speedcheck.o getdate.o transfer.o ldap.o ssluse.o version.o \
|
||||||
getenv.o escape.o mprintf.o telnet.o getpass.o netrc.o getinfo.o \
|
getenv.o escape.o mprintf.o telnet.o getpass.o netrc.o getinfo.o \
|
||||||
strequal.o easy.o security.o krb4.o
|
strequal.o easy.o security.o krb4.o memdebug.o http_chunks.o
|
||||||
|
|
||||||
LIBRARIES = $(libcurl_a_LIBRARIES)
|
LIBRARIES = $(libcurl_a_LIBRARIES)
|
||||||
SOURCES = $(libcurl_a_SOURCES)
|
SOURCES = $(libcurl_a_SOURCES)
|
||||||
|
@@ -38,6 +38,7 @@ RELEASE_OBJS= \
|
|||||||
formdatar.obj \
|
formdatar.obj \
|
||||||
ftpr.obj \
|
ftpr.obj \
|
||||||
httpr.obj \
|
httpr.obj \
|
||||||
|
http_chunksr.obj \
|
||||||
ldapr.obj \
|
ldapr.obj \
|
||||||
dictr.obj \
|
dictr.obj \
|
||||||
telnetr.obj \
|
telnetr.obj \
|
||||||
@@ -68,6 +69,7 @@ DEBUG_OBJS= \
|
|||||||
formdatad.obj \
|
formdatad.obj \
|
||||||
ftpd.obj \
|
ftpd.obj \
|
||||||
httpd.obj \
|
httpd.obj \
|
||||||
|
http_chunksd.obj \
|
||||||
ldapd.obj \
|
ldapd.obj \
|
||||||
dictd.obj \
|
dictd.obj \
|
||||||
telnetd.obj \
|
telnetd.obj \
|
||||||
@@ -98,6 +100,7 @@ RELEASE_SSL_OBJS= \
|
|||||||
formdatars.obj \
|
formdatars.obj \
|
||||||
ftprs.obj \
|
ftprs.obj \
|
||||||
httprs.obj \
|
httprs.obj \
|
||||||
|
http_chunksrs.obj \
|
||||||
ldaprs.obj \
|
ldaprs.obj \
|
||||||
dictrs.obj \
|
dictrs.obj \
|
||||||
telnetrs.obj \
|
telnetrs.obj \
|
||||||
@@ -128,6 +131,7 @@ LINK_OBJS= \
|
|||||||
formdata.obj \
|
formdata.obj \
|
||||||
ftp.obj \
|
ftp.obj \
|
||||||
http.obj \
|
http.obj \
|
||||||
|
http_chunks.obj \
|
||||||
ldap.obj \
|
ldap.obj \
|
||||||
dict.obj \
|
dict.obj \
|
||||||
telnet.obj \
|
telnet.obj \
|
||||||
@@ -176,6 +180,8 @@ ftpr.obj: ftp.c
|
|||||||
$(CCR) $(CFLAGS) ftp.c
|
$(CCR) $(CFLAGS) ftp.c
|
||||||
httpr.obj: http.c
|
httpr.obj: http.c
|
||||||
$(CCR) $(CFLAGS) http.c
|
$(CCR) $(CFLAGS) http.c
|
||||||
|
http_chunksr.obj: http_chunks.c
|
||||||
|
$(CCR) $(CFLAGS) http_chunks.c
|
||||||
ldapr.obj: ldap.c
|
ldapr.obj: ldap.c
|
||||||
$(CCR) $(CFLAGS) ldap.c
|
$(CCR) $(CFLAGS) ldap.c
|
||||||
dictr.obj: dict.c
|
dictr.obj: dict.c
|
||||||
@@ -234,6 +240,8 @@ ftpd.obj: ftp.c
|
|||||||
$(CCD) $(CFLAGS) ftp.c
|
$(CCD) $(CFLAGS) ftp.c
|
||||||
httpd.obj: http.c
|
httpd.obj: http.c
|
||||||
$(CCD) $(CFLAGS) http.c
|
$(CCD) $(CFLAGS) http.c
|
||||||
|
http_chunksd.obj: http_chunks.c
|
||||||
|
$(CCD) $(CFLAGS) http_chunks.c
|
||||||
ldapd.obj: ldap.c
|
ldapd.obj: ldap.c
|
||||||
$(CCD) $(CFLAGS) ldap.c
|
$(CCD) $(CFLAGS) ldap.c
|
||||||
dictd.obj: dict.c
|
dictd.obj: dict.c
|
||||||
@@ -291,8 +299,8 @@ formdatars.obj: formdata.c
|
|||||||
$(CCRS) $(CFLAGS) formdata.c
|
$(CCRS) $(CFLAGS) formdata.c
|
||||||
ftprs.obj: ftp.c
|
ftprs.obj: ftp.c
|
||||||
$(CCRS) $(CFLAGS) ftp.c
|
$(CCRS) $(CFLAGS) ftp.c
|
||||||
httprs.obj: http.c
|
http_chunksrs.obj: http_chunks.c
|
||||||
$(CCRS) $(CFLAGS) http.c
|
$(CCRS) $(CFLAGS) http_chunks.c
|
||||||
ldaprs.obj: ldap.c
|
ldaprs.obj: ldap.c
|
||||||
$(CCRS) $(CFLAGS) ldap.c
|
$(CCRS) $(CFLAGS) ldap.c
|
||||||
dictrs.obj: dict.c
|
dictrs.obj: dict.c
|
||||||
|
@@ -100,7 +100,7 @@ CURLcode Curl_dict(struct connectdata *conn)
|
|||||||
char *path = conn->path;
|
char *path = conn->path;
|
||||||
long *bytecount = &conn->bytecount;
|
long *bytecount = &conn->bytecount;
|
||||||
|
|
||||||
if(data->bits.user_passwd) {
|
if(conn->bits.user_passwd) {
|
||||||
/* AUTH is missing */
|
/* AUTH is missing */
|
||||||
}
|
}
|
||||||
|
|
||||||
|
19
lib/easy.c
19
lib/easy.c
@@ -83,15 +83,11 @@ CURL *curl_easy_init(void)
|
|||||||
CURLcode res;
|
CURLcode res;
|
||||||
struct UrlData *data;
|
struct UrlData *data;
|
||||||
|
|
||||||
if(curl_init())
|
|
||||||
return NULL;
|
|
||||||
|
|
||||||
/* We use curl_open() with undefined URL so far */
|
/* We use curl_open() with undefined URL so far */
|
||||||
res = curl_open((CURL **)&data, NULL);
|
res = Curl_open((CURL **)&data, NULL);
|
||||||
if(res != CURLE_OK)
|
if(res != CURLE_OK)
|
||||||
return NULL;
|
return NULL;
|
||||||
|
|
||||||
data->interf = CURLI_EASY; /* mark it as an easy one */
|
|
||||||
/* SAC */
|
/* SAC */
|
||||||
data->device = NULL;
|
data->device = NULL;
|
||||||
|
|
||||||
@@ -119,16 +115,16 @@ CURLcode curl_easy_setopt(CURL *curl, CURLoption tag, ...)
|
|||||||
if(tag < CURLOPTTYPE_OBJECTPOINT) {
|
if(tag < CURLOPTTYPE_OBJECTPOINT) {
|
||||||
/* This is a LONG type */
|
/* This is a LONG type */
|
||||||
param_long = va_arg(arg, long);
|
param_long = va_arg(arg, long);
|
||||||
curl_setopt(data, tag, param_long);
|
Curl_setopt(data, tag, param_long);
|
||||||
}
|
}
|
||||||
else if(tag < CURLOPTTYPE_FUNCTIONPOINT) {
|
else if(tag < CURLOPTTYPE_FUNCTIONPOINT) {
|
||||||
/* This is a object pointer type */
|
/* This is a object pointer type */
|
||||||
param_obj = va_arg(arg, void *);
|
param_obj = va_arg(arg, void *);
|
||||||
curl_setopt(data, tag, param_obj);
|
Curl_setopt(data, tag, param_obj);
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
param_func = va_arg(arg, func_T );
|
param_func = va_arg(arg, func_T );
|
||||||
curl_setopt(data, tag, param_func);
|
Curl_setopt(data, tag, param_func);
|
||||||
}
|
}
|
||||||
|
|
||||||
va_end(arg);
|
va_end(arg);
|
||||||
@@ -137,13 +133,12 @@ CURLcode curl_easy_setopt(CURL *curl, CURLoption tag, ...)
|
|||||||
|
|
||||||
CURLcode curl_easy_perform(CURL *curl)
|
CURLcode curl_easy_perform(CURL *curl)
|
||||||
{
|
{
|
||||||
return curl_transfer(curl);
|
return Curl_perform(curl);
|
||||||
}
|
}
|
||||||
|
|
||||||
void curl_easy_cleanup(CURL *curl)
|
void curl_easy_cleanup(CURL *curl)
|
||||||
{
|
{
|
||||||
curl_close(curl);
|
Curl_close(curl);
|
||||||
curl_free();
|
|
||||||
}
|
}
|
||||||
|
|
||||||
CURLcode curl_easy_getinfo(CURL *curl, CURLINFO info, ...)
|
CURLcode curl_easy_getinfo(CURL *curl, CURLINFO info, ...)
|
||||||
@@ -153,5 +148,5 @@ CURLcode curl_easy_getinfo(CURL *curl, CURLINFO info, ...)
|
|||||||
va_start(arg, info);
|
va_start(arg, info);
|
||||||
paramp = va_arg(arg, void *);
|
paramp = va_arg(arg, void *);
|
||||||
|
|
||||||
return curl_getinfo(curl, info, paramp);
|
return Curl_getinfo(curl, info, paramp);
|
||||||
}
|
}
|
||||||
|
131
lib/escape.c
131
lib/escape.c
@@ -25,6 +25,7 @@
|
|||||||
* allocated string or NULL if an error occurred. */
|
* allocated string or NULL if an error occurred. */
|
||||||
|
|
||||||
#include "setup.h"
|
#include "setup.h"
|
||||||
|
#include <ctype.h>
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
@@ -36,74 +37,88 @@
|
|||||||
#include "memdebug.h"
|
#include "memdebug.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
char *curl_escape(char *string)
|
char *curl_escape(char *string, int length)
|
||||||
{
|
{
|
||||||
int alloc=strlen(string)+1;
|
int alloc = (length?length:strlen(string))+1;
|
||||||
char *ns = malloc(alloc);
|
char *ns = malloc(alloc);
|
||||||
unsigned char in;
|
unsigned char in;
|
||||||
int newlen = alloc;
|
int newlen = alloc;
|
||||||
int index=0;
|
int index=0;
|
||||||
|
|
||||||
while(*string) {
|
while(length--) {
|
||||||
in = *string;
|
in = *string;
|
||||||
if(' ' == in)
|
if(' ' == in)
|
||||||
ns[index++] = '+';
|
ns[index++] = '+';
|
||||||
else if(!(in >= 'a' && in <= 'z') &&
|
else if(!(in >= 'a' && in <= 'z') &&
|
||||||
!(in >= 'A' && in <= 'Z') &&
|
!(in >= 'A' && in <= 'Z') &&
|
||||||
!(in >= '0' && in <= '9')) {
|
!(in >= '0' && in <= '9')) {
|
||||||
/* encode it */
|
/* encode it */
|
||||||
newlen += 2; /* the size grows with two, since this'll become a %XX */
|
if(('%' == in) &&
|
||||||
if(newlen > alloc) {
|
(length>=2) &&
|
||||||
alloc *= 2;
|
isxdigit((int)string[1]) &&
|
||||||
ns = realloc(ns, alloc);
|
isxdigit((int)string[2]) ) {
|
||||||
if(!ns)
|
/*
|
||||||
return NULL;
|
* This is an already encoded letter, leave it!
|
||||||
}
|
*/
|
||||||
sprintf(&ns[index], "%%%02X", in);
|
memcpy(&ns[index], string, 3);
|
||||||
index+=3;
|
string+=2;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
/* just copy this */
|
/* encode this now */
|
||||||
ns[index++]=in;
|
|
||||||
|
newlen += 2; /* the size grows with two, since this'll become a %XX */
|
||||||
|
if(newlen > alloc) {
|
||||||
|
alloc *= 2;
|
||||||
|
ns = realloc(ns, alloc);
|
||||||
|
if(!ns)
|
||||||
|
return NULL;
|
||||||
|
}
|
||||||
|
sprintf(&ns[index], "%%%02X", in);
|
||||||
}
|
}
|
||||||
string++;
|
index+=3;
|
||||||
}
|
}
|
||||||
ns[index]=0; /* terminate it */
|
else {
|
||||||
return ns;
|
/* just copy this */
|
||||||
|
ns[index++]=in;
|
||||||
|
}
|
||||||
|
string++;
|
||||||
|
}
|
||||||
|
ns[index]=0; /* terminate it */
|
||||||
|
return ns;
|
||||||
}
|
}
|
||||||
|
|
||||||
char *curl_unescape(char *string, int length)
|
char *curl_unescape(char *string, int length)
|
||||||
{
|
{
|
||||||
int alloc = (length?length:strlen(string))+1;
|
int alloc = (length?length:strlen(string))+1;
|
||||||
char *ns = malloc(alloc);
|
char *ns = malloc(alloc);
|
||||||
unsigned char in;
|
unsigned char in;
|
||||||
int index=0;
|
int index=0;
|
||||||
int hex;
|
unsigned int hex;
|
||||||
char querypart=FALSE; /* everything to the right of a '?' letter is
|
char querypart=FALSE; /* everything to the right of a '?' letter is
|
||||||
the "query part" where '+' should become ' '.
|
the "query part" where '+' should become ' '.
|
||||||
RFC 2316, section 3.10 */
|
RFC 2316, section 3.10 */
|
||||||
|
|
||||||
while(--alloc > 0) {
|
while(--alloc > 0) {
|
||||||
in = *string;
|
in = *string;
|
||||||
if(querypart && ('+' == in))
|
if(querypart && ('+' == in))
|
||||||
in = ' ';
|
in = ' ';
|
||||||
else if(!querypart && ('?' == in)) {
|
else if(!querypart && ('?' == in)) {
|
||||||
/* we have "walked in" to the query part */
|
/* we have "walked in" to the query part */
|
||||||
querypart=TRUE;
|
querypart=TRUE;
|
||||||
|
}
|
||||||
|
else if('%' == in) {
|
||||||
|
/* encoded part */
|
||||||
|
if(sscanf(string+1, "%02X", &hex)) {
|
||||||
|
in = hex;
|
||||||
|
string+=2;
|
||||||
|
alloc-=2;
|
||||||
}
|
}
|
||||||
else if('%' == in) {
|
}
|
||||||
/* encoded part */
|
|
||||||
if(sscanf(string+1, "%02X", &hex)) {
|
ns[index++] = in;
|
||||||
in = hex;
|
string++;
|
||||||
string+=2;
|
}
|
||||||
alloc-=2;
|
ns[index]=0; /* terminate it */
|
||||||
}
|
return ns;
|
||||||
}
|
|
||||||
|
|
||||||
ns[index++] = in;
|
|
||||||
string++;
|
|
||||||
}
|
|
||||||
ns[index]=0; /* terminate it */
|
|
||||||
return ns;
|
|
||||||
|
|
||||||
}
|
}
|
||||||
|
@@ -26,7 +26,7 @@
|
|||||||
/* Escape and unescape URL encoding in strings. The functions return a new
|
/* Escape and unescape URL encoding in strings. The functions return a new
|
||||||
* allocated string or NULL if an error occurred. */
|
* allocated string or NULL if an error occurred. */
|
||||||
|
|
||||||
char *curl_escape(char *string);
|
char *curl_escape(char *string, int length);
|
||||||
char *curl_unescape(char *string, int length);
|
char *curl_unescape(char *string, int length);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
61
lib/file.c
61
lib/file.c
@@ -91,29 +91,24 @@
|
|||||||
#include "memdebug.h"
|
#include "memdebug.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
CURLcode file(struct connectdata *conn)
|
/* Emulate a connect-then-transfer protocol. We connect to the file here */
|
||||||
|
CURLcode Curl_file_connect(struct connectdata *conn)
|
||||||
{
|
{
|
||||||
/* This implementation ignores the host name in conformance with
|
char *actual_path = curl_unescape(conn->path, 0);
|
||||||
RFC 1738. Only local files (reachable via the standard file system)
|
struct FILE *file;
|
||||||
are supported. This means that files on remotely mounted directories
|
|
||||||
(via NFS, Samba, NT sharing) can be accessed through a file:// URL
|
|
||||||
*/
|
|
||||||
CURLcode res = CURLE_OK;
|
|
||||||
char *path = conn->path;
|
|
||||||
struct stat statbuf;
|
|
||||||
size_t expected_size=-1;
|
|
||||||
size_t nread;
|
|
||||||
struct UrlData *data = conn->data;
|
|
||||||
char *buf = data->buffer;
|
|
||||||
int bytecount = 0;
|
|
||||||
struct timeval start = Curl_tvnow();
|
|
||||||
struct timeval now = start;
|
|
||||||
int fd;
|
int fd;
|
||||||
char *actual_path = curl_unescape(path, 0);
|
|
||||||
|
|
||||||
#if defined(WIN32) || defined(__EMX__)
|
#if defined(WIN32) || defined(__EMX__)
|
||||||
int i;
|
int i;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
file = (struct FILE *)malloc(sizeof(struct FILE));
|
||||||
|
if(!file)
|
||||||
|
return CURLE_OUT_OF_MEMORY;
|
||||||
|
|
||||||
|
memset(file, 0, sizeof(struct FILE));
|
||||||
|
conn->proto.file = file;
|
||||||
|
|
||||||
|
#if defined(WIN32) || defined(__EMX__)
|
||||||
/* change path separators from '/' to '\\' for Windows and OS/2 */
|
/* change path separators from '/' to '\\' for Windows and OS/2 */
|
||||||
for (i=0; actual_path[i] != '\0'; ++i)
|
for (i=0; actual_path[i] != '\0'; ++i)
|
||||||
if (actual_path[i] == '/')
|
if (actual_path[i] == '/')
|
||||||
@@ -126,9 +121,37 @@ CURLcode file(struct connectdata *conn)
|
|||||||
free(actual_path);
|
free(actual_path);
|
||||||
|
|
||||||
if(fd == -1) {
|
if(fd == -1) {
|
||||||
failf(data, "Couldn't open file %s", path);
|
failf(conn->data, "Couldn't open file %s", conn->path);
|
||||||
return CURLE_FILE_COULDNT_READ_FILE;
|
return CURLE_FILE_COULDNT_READ_FILE;
|
||||||
}
|
}
|
||||||
|
file->fd = fd;
|
||||||
|
|
||||||
|
return CURLE_OK;
|
||||||
|
}
|
||||||
|
|
||||||
|
/* This is the do-phase, separated from the connect-phase above */
|
||||||
|
|
||||||
|
CURLcode Curl_file(struct connectdata *conn)
|
||||||
|
{
|
||||||
|
/* This implementation ignores the host name in conformance with
|
||||||
|
RFC 1738. Only local files (reachable via the standard file system)
|
||||||
|
are supported. This means that files on remotely mounted directories
|
||||||
|
(via NFS, Samba, NT sharing) can be accessed through a file:// URL
|
||||||
|
*/
|
||||||
|
CURLcode res = CURLE_OK;
|
||||||
|
struct stat statbuf;
|
||||||
|
size_t expected_size=-1;
|
||||||
|
size_t nread;
|
||||||
|
struct UrlData *data = conn->data;
|
||||||
|
char *buf = data->buffer;
|
||||||
|
int bytecount = 0;
|
||||||
|
struct timeval start = Curl_tvnow();
|
||||||
|
struct timeval now = start;
|
||||||
|
int fd;
|
||||||
|
|
||||||
|
/* get the fd from the connection phase */
|
||||||
|
fd = conn->proto.file->fd;
|
||||||
|
|
||||||
if( -1 != fstat(fd, &statbuf)) {
|
if( -1 != fstat(fd, &statbuf)) {
|
||||||
/* we could stat it, then read out the size */
|
/* we could stat it, then read out the size */
|
||||||
expected_size = statbuf.st_size;
|
expected_size = statbuf.st_size;
|
||||||
|
@@ -23,6 +23,6 @@
|
|||||||
*
|
*
|
||||||
* $Id$
|
* $Id$
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
CURLcode file(struct connectdata *conn);
|
CURLcode Curl_file(struct connectdata *conn);
|
||||||
|
CURLcode Curl_file_connect(struct connectdata *conn);
|
||||||
#endif
|
#endif
|
||||||
|
77
lib/ftp.c
77
lib/ftp.c
@@ -77,6 +77,9 @@
|
|||||||
#include "krb4.h"
|
#include "krb4.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
#include "strequal.h"
|
||||||
|
#include "ssluse.h"
|
||||||
|
|
||||||
#define _MPRINTF_REPLACE /* use our functions only */
|
#define _MPRINTF_REPLACE /* use our functions only */
|
||||||
#include <curl/mprintf.h>
|
#include <curl/mprintf.h>
|
||||||
|
|
||||||
@@ -119,8 +122,8 @@ static CURLcode AllowServerConnect(struct UrlData *data,
|
|||||||
size_t size = sizeof(struct sockaddr_in);
|
size_t size = sizeof(struct sockaddr_in);
|
||||||
struct sockaddr_in add;
|
struct sockaddr_in add;
|
||||||
|
|
||||||
getsockname(sock, (struct sockaddr *) &add, (int *)&size);
|
getsockname(sock, (struct sockaddr *) &add, (socklen_t *)&size);
|
||||||
s=accept(sock, (struct sockaddr *) &add, (int *)&size);
|
s=accept(sock, (struct sockaddr *) &add, (socklen_t *)&size);
|
||||||
|
|
||||||
sclose(sock); /* close the first socket */
|
sclose(sock); /* close the first socket */
|
||||||
|
|
||||||
@@ -296,6 +299,9 @@ CURLcode Curl_ftp_connect(struct connectdata *conn)
|
|||||||
memset(ftp, 0, sizeof(struct FTP));
|
memset(ftp, 0, sizeof(struct FTP));
|
||||||
conn->proto.ftp = ftp;
|
conn->proto.ftp = ftp;
|
||||||
|
|
||||||
|
/* We always support persistant connections on ftp */
|
||||||
|
conn->bits.close = FALSE;
|
||||||
|
|
||||||
/* get some initial data into the ftp struct */
|
/* get some initial data into the ftp struct */
|
||||||
ftp->bytecountp = &conn->bytecount;
|
ftp->bytecountp = &conn->bytecount;
|
||||||
|
|
||||||
@@ -311,6 +317,14 @@ CURLcode Curl_ftp_connect(struct connectdata *conn)
|
|||||||
return result;
|
return result;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if(conn->protocol & PROT_FTPS) {
|
||||||
|
/* FTPS is simply ftp with SSL for the control channel */
|
||||||
|
/* now, perform the SSL initialization for this socket */
|
||||||
|
if(Curl_SSLConnect(conn))
|
||||||
|
return CURLE_SSL_CONNECT_ERROR;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
/* The first thing we do is wait for the "220*" line: */
|
/* The first thing we do is wait for the "220*" line: */
|
||||||
nread = Curl_GetFTPResponse(conn->firstsocket, buf, conn, &ftpcode);
|
nread = Curl_GetFTPResponse(conn->firstsocket, buf, conn, &ftpcode);
|
||||||
if(nread < 0)
|
if(nread < 0)
|
||||||
@@ -332,8 +346,6 @@ CURLcode Curl_ftp_connect(struct connectdata *conn)
|
|||||||
set a valid level */
|
set a valid level */
|
||||||
sec_request_prot(conn, data->krb4_level);
|
sec_request_prot(conn, data->krb4_level);
|
||||||
|
|
||||||
data->cmdchannel = fdopen(conn->firstsocket, "w");
|
|
||||||
|
|
||||||
if(sec_login(conn) != 0)
|
if(sec_login(conn) != 0)
|
||||||
infof(data, "Logging in with password in cleartext!\n");
|
infof(data, "Logging in with password in cleartext!\n");
|
||||||
else
|
else
|
||||||
@@ -546,13 +558,14 @@ CURLcode _ftp(struct connectdata *conn)
|
|||||||
char *buf = data->buffer; /* this is our buffer */
|
char *buf = data->buffer; /* this is our buffer */
|
||||||
/* for the ftp PORT mode */
|
/* for the ftp PORT mode */
|
||||||
int portsock=-1;
|
int portsock=-1;
|
||||||
struct sockaddr_in serv_addr;
|
|
||||||
char hostent_buf[8192];
|
|
||||||
#if defined (HAVE_INET_NTOA_R)
|
#if defined (HAVE_INET_NTOA_R)
|
||||||
char ntoa_buf[64];
|
char ntoa_buf[64];
|
||||||
#endif
|
#endif
|
||||||
#ifdef ENABLE_IPV6
|
#ifdef ENABLE_IPV6
|
||||||
struct addrinfo *ai;
|
struct addrinfo *ai;
|
||||||
|
#else
|
||||||
|
struct sockaddr_in serv_addr;
|
||||||
|
char hostent_buf[8192];
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct curl_slist *qitem; /* QUOTE item */
|
struct curl_slist *qitem; /* QUOTE item */
|
||||||
@@ -712,20 +725,20 @@ CURLcode _ftp(struct connectdata *conn)
|
|||||||
#ifdef ENABLE_IPV6
|
#ifdef ENABLE_IPV6
|
||||||
struct addrinfo hints, *res, *ai;
|
struct addrinfo hints, *res, *ai;
|
||||||
struct sockaddr_storage ss;
|
struct sockaddr_storage ss;
|
||||||
int sslen;
|
socklen_t sslen;
|
||||||
char hbuf[NI_MAXHOST];
|
char hbuf[NI_MAXHOST];
|
||||||
char *localaddr;
|
|
||||||
struct sockaddr *sa=(struct sockaddr *)&ss;
|
struct sockaddr *sa=(struct sockaddr *)&ss;
|
||||||
#ifdef NI_WITHSCOPEID
|
#ifdef NI_WITHSCOPEID
|
||||||
const int niflags = NI_NUMERICHOST | NI_NUMERICSERV | NI_WITHSCOPEID;
|
const int niflags = NI_NUMERICHOST | NI_NUMERICSERV | NI_WITHSCOPEID;
|
||||||
#else
|
#else
|
||||||
const int niflags = NI_NUMERICHOST | NI_NUMERICSERV;
|
const int niflags = NI_NUMERICHOST | NI_NUMERICSERV;
|
||||||
#endif
|
#endif
|
||||||
unsigned char *ap;
|
char *ap;
|
||||||
unsigned char *pp;
|
char *pp;
|
||||||
int alen, plen;
|
int alen, plen;
|
||||||
char portmsgbuf[4096], tmp[4096];
|
char portmsgbuf[4096], tmp[4096];
|
||||||
char *p;
|
|
||||||
char *mode[] = { "EPRT", "LPRT", "PORT", NULL };
|
char *mode[] = { "EPRT", "LPRT", "PORT", NULL };
|
||||||
char **modep;
|
char **modep;
|
||||||
|
|
||||||
@@ -758,13 +771,13 @@ CURLcode _ftp(struct connectdata *conn)
|
|||||||
continue;
|
continue;
|
||||||
|
|
||||||
if (bind(portsock, ai->ai_addr, ai->ai_addrlen) < 0) {
|
if (bind(portsock, ai->ai_addr, ai->ai_addrlen) < 0) {
|
||||||
close(portsock);
|
sclose(portsock);
|
||||||
portsock = -1;
|
portsock = -1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (listen(portsock, 1) < 0) {
|
if (listen(portsock, 1) < 0) {
|
||||||
close(portsock);
|
sclose(portsock);
|
||||||
portsock = -1;
|
portsock = -1;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
@@ -875,7 +888,7 @@ again:;
|
|||||||
}
|
}
|
||||||
|
|
||||||
if (!*modep) {
|
if (!*modep) {
|
||||||
close(portsock);
|
sclose(portsock);
|
||||||
freeaddrinfo(res);
|
freeaddrinfo(res);
|
||||||
return CURLE_FTP_PORT_FAILED;
|
return CURLE_FTP_PORT_FAILED;
|
||||||
}
|
}
|
||||||
@@ -929,7 +942,7 @@ again:;
|
|||||||
size = sizeof(add);
|
size = sizeof(add);
|
||||||
|
|
||||||
if(getsockname(portsock, (struct sockaddr *) &add,
|
if(getsockname(portsock, (struct sockaddr *) &add,
|
||||||
(int *)&size)<0) {
|
(socklen_t *)&size)<0) {
|
||||||
failf(data, "getsockname() failed");
|
failf(data, "getsockname() failed");
|
||||||
return CURLE_FTP_PORT_FAILED;
|
return CURLE_FTP_PORT_FAILED;
|
||||||
}
|
}
|
||||||
@@ -1024,9 +1037,10 @@ again:;
|
|||||||
struct addrinfo *res;
|
struct addrinfo *res;
|
||||||
#else
|
#else
|
||||||
struct hostent *he;
|
struct hostent *he;
|
||||||
#endif
|
|
||||||
char *str=buf,*ip_addr;
|
|
||||||
char *hostdataptr=NULL;
|
char *hostdataptr=NULL;
|
||||||
|
char *ip_addr;
|
||||||
|
#endif
|
||||||
|
char *str=buf;
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* New 227-parser June 3rd 1999.
|
* New 227-parser June 3rd 1999.
|
||||||
@@ -1695,16 +1709,9 @@ size_t Curl_ftpsendf(int fd, struct connectdata *conn, char *fmt, ...)
|
|||||||
|
|
||||||
strcat(s, "\r\n"); /* append a trailing CRLF */
|
strcat(s, "\r\n"); /* append a trailing CRLF */
|
||||||
|
|
||||||
#ifdef KRB4
|
bytes_written=0;
|
||||||
if(conn->sec_complete && conn->data->cmdchannel) {
|
Curl_write(conn, fd, s, strlen(s), &bytes_written);
|
||||||
bytes_written = sec_fprintf(conn, conn->data->cmdchannel, s);
|
|
||||||
fflush(conn->data->cmdchannel);
|
|
||||||
}
|
|
||||||
else
|
|
||||||
#endif /* KRB4 */
|
|
||||||
{
|
|
||||||
bytes_written = swrite(fd, s, strlen(s));
|
|
||||||
}
|
|
||||||
return(bytes_written);
|
return(bytes_written);
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1713,12 +1720,14 @@ CURLcode Curl_ftp_disconnect(struct connectdata *conn)
|
|||||||
{
|
{
|
||||||
struct FTP *ftp= conn->proto.ftp;
|
struct FTP *ftp= conn->proto.ftp;
|
||||||
|
|
||||||
if(ftp->user)
|
/* The FTP session may or may not have been allocated/setup at this point! */
|
||||||
free(ftp->user);
|
if(ftp) {
|
||||||
if(ftp->passwd)
|
if(ftp->user)
|
||||||
free(ftp->passwd);
|
free(ftp->user);
|
||||||
if(ftp->entrypath)
|
if(ftp->passwd)
|
||||||
free(ftp->entrypath);
|
free(ftp->passwd);
|
||||||
|
if(ftp->entrypath)
|
||||||
|
free(ftp->entrypath);
|
||||||
|
}
|
||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
187
lib/getdate.c
187
lib/getdate.c
@@ -104,6 +104,11 @@
|
|||||||
# include <string.h>
|
# include <string.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#if __GNUC__ < 2 || (__GNUC__ == 2 && __GNUC_MINOR__ < 7)
|
#if __GNUC__ < 2 || (__GNUC__ == 2 && __GNUC_MINOR__ < 7)
|
||||||
# define __attribute__(x)
|
# define __attribute__(x)
|
||||||
#endif
|
#endif
|
||||||
@@ -127,44 +132,44 @@
|
|||||||
then those parser generators need to be fixed instead of adding those
|
then those parser generators need to be fixed instead of adding those
|
||||||
names to this list. */
|
names to this list. */
|
||||||
|
|
||||||
#define yymaxdepth gd_maxdepth
|
#define yymaxdepth Curl_gd_maxdepth
|
||||||
#define yyparse gd_parse
|
#define yyparse Curl_gd_parse
|
||||||
#define yylex gd_lex
|
#define yylex Curl_gd_lex
|
||||||
#define yyerror gd_error
|
#define yyerror Curl_gd_error
|
||||||
#define yylval gd_lval
|
#define yylval Curl_gd_lval
|
||||||
#define yychar gd_char
|
#define yychar Curl_gd_char
|
||||||
#define yydebug gd_debug
|
#define yydebug Curl_gd_debug
|
||||||
#define yypact gd_pact
|
#define yypact Curl_gd_pact
|
||||||
#define yyr1 gd_r1
|
#define yyr1 Curl_gd_r1
|
||||||
#define yyr2 gd_r2
|
#define yyr2 Curl_gd_r2
|
||||||
#define yydef gd_def
|
#define yydef Curl_gd_def
|
||||||
#define yychk gd_chk
|
#define yychk Curl_gd_chk
|
||||||
#define yypgo gd_pgo
|
#define yypgo Curl_gd_pgo
|
||||||
#define yyact gd_act
|
#define yyact Curl_gd_act
|
||||||
#define yyexca gd_exca
|
#define yyexca Curl_gd_exca
|
||||||
#define yyerrflag gd_errflag
|
#define yyerrflag Curl_gd_errflag
|
||||||
#define yynerrs gd_nerrs
|
#define yynerrs Curl_gd_nerrs
|
||||||
#define yyps gd_ps
|
#define yyps Curl_gd_ps
|
||||||
#define yypv gd_pv
|
#define yypv Curl_gd_pv
|
||||||
#define yys gd_s
|
#define yys Curl_gd_s
|
||||||
#define yy_yys gd_yys
|
#define yy_yys Curl_gd_yys
|
||||||
#define yystate gd_state
|
#define yystate Curl_gd_state
|
||||||
#define yytmp gd_tmp
|
#define yytmp Curl_gd_tmp
|
||||||
#define yyv gd_v
|
#define yyv Curl_gd_v
|
||||||
#define yy_yyv gd_yyv
|
#define yy_yyv Curl_gd_yyv
|
||||||
#define yyval gd_val
|
#define yyval Curl_gd_val
|
||||||
#define yylloc gd_lloc
|
#define yylloc Curl_gd_lloc
|
||||||
#define yyreds gd_reds /* With YYDEBUG defined */
|
#define yyreds Curl_gd_reds /* With YYDEBUG defined */
|
||||||
#define yytoks gd_toks /* With YYDEBUG defined */
|
#define yytoks Curl_gd_toks /* With YYDEBUG defined */
|
||||||
#define yylhs gd_yylhs
|
#define yylhs Curl_gd_yylhs
|
||||||
#define yylen gd_yylen
|
#define yylen Curl_gd_yylen
|
||||||
#define yydefred gd_yydefred
|
#define yydefred Curl_gd_yydefred
|
||||||
#define yydgoto gd_yydgoto
|
#define yydgoto Curl_gd_yydgoto
|
||||||
#define yysindex gd_yysindex
|
#define yysindex Curl_gd_yysindex
|
||||||
#define yyrindex gd_yyrindex
|
#define yyrindex Curl_gd_yyrindex
|
||||||
#define yygindex gd_yygindex
|
#define yygindex Curl_gd_yygindex
|
||||||
#define yytable gd_yytable
|
#define yytable Curl_gd_yytable
|
||||||
#define yycheck gd_yycheck
|
#define yycheck Curl_gd_yycheck
|
||||||
|
|
||||||
static int yylex ();
|
static int yylex ();
|
||||||
static int yyerror ();
|
static int yyerror ();
|
||||||
@@ -222,7 +227,7 @@ static int yyRelSeconds;
|
|||||||
static int yyRelYear;
|
static int yyRelYear;
|
||||||
|
|
||||||
|
|
||||||
#line 205 "getdate.y"
|
#line 210 "getdate.y"
|
||||||
typedef union {
|
typedef union {
|
||||||
int Number;
|
int Number;
|
||||||
enum _MERIDIAN Meridian;
|
enum _MERIDIAN Meridian;
|
||||||
@@ -305,11 +310,11 @@ static const short yyrhs[] = { -1,
|
|||||||
|
|
||||||
#if YYDEBUG != 0
|
#if YYDEBUG != 0
|
||||||
static const short yyrline[] = { 0,
|
static const short yyrline[] = { 0,
|
||||||
221, 222, 225, 228, 231, 234, 237, 240, 243, 249,
|
226, 227, 230, 233, 236, 239, 242, 245, 248, 254,
|
||||||
255, 264, 270, 282, 285, 288, 294, 298, 302, 308,
|
260, 269, 275, 287, 290, 293, 299, 303, 307, 313,
|
||||||
312, 330, 336, 342, 346, 351, 355, 362, 370, 373,
|
317, 335, 341, 347, 351, 356, 360, 367, 375, 378,
|
||||||
376, 379, 382, 385, 388, 391, 394, 397, 400, 403,
|
381, 384, 387, 390, 393, 396, 399, 402, 405, 408,
|
||||||
406, 409, 412, 415, 418, 421, 424, 429, 462, 466
|
411, 414, 417, 420, 423, 426, 429, 434, 467, 471
|
||||||
};
|
};
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
@@ -933,37 +938,37 @@ yyreduce:
|
|||||||
switch (yyn) {
|
switch (yyn) {
|
||||||
|
|
||||||
case 3:
|
case 3:
|
||||||
#line 225 "getdate.y"
|
#line 230 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHaveTime++;
|
yyHaveTime++;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 4:
|
case 4:
|
||||||
#line 228 "getdate.y"
|
#line 233 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHaveZone++;
|
yyHaveZone++;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 5:
|
case 5:
|
||||||
#line 231 "getdate.y"
|
#line 236 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHaveDate++;
|
yyHaveDate++;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 6:
|
case 6:
|
||||||
#line 234 "getdate.y"
|
#line 239 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHaveDay++;
|
yyHaveDay++;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 7:
|
case 7:
|
||||||
#line 237 "getdate.y"
|
#line 242 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHaveRel++;
|
yyHaveRel++;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 9:
|
case 9:
|
||||||
#line 243 "getdate.y"
|
#line 248 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHour = yyvsp[-1].Number;
|
yyHour = yyvsp[-1].Number;
|
||||||
yyMinutes = 0;
|
yyMinutes = 0;
|
||||||
@@ -972,7 +977,7 @@ case 9:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 10:
|
case 10:
|
||||||
#line 249 "getdate.y"
|
#line 254 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHour = yyvsp[-3].Number;
|
yyHour = yyvsp[-3].Number;
|
||||||
yyMinutes = yyvsp[-1].Number;
|
yyMinutes = yyvsp[-1].Number;
|
||||||
@@ -981,7 +986,7 @@ case 10:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 11:
|
case 11:
|
||||||
#line 255 "getdate.y"
|
#line 260 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHour = yyvsp[-3].Number;
|
yyHour = yyvsp[-3].Number;
|
||||||
yyMinutes = yyvsp[-1].Number;
|
yyMinutes = yyvsp[-1].Number;
|
||||||
@@ -993,7 +998,7 @@ case 11:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 12:
|
case 12:
|
||||||
#line 264 "getdate.y"
|
#line 269 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHour = yyvsp[-5].Number;
|
yyHour = yyvsp[-5].Number;
|
||||||
yyMinutes = yyvsp[-3].Number;
|
yyMinutes = yyvsp[-3].Number;
|
||||||
@@ -1002,7 +1007,7 @@ case 12:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 13:
|
case 13:
|
||||||
#line 270 "getdate.y"
|
#line 275 "getdate.y"
|
||||||
{
|
{
|
||||||
yyHour = yyvsp[-5].Number;
|
yyHour = yyvsp[-5].Number;
|
||||||
yyMinutes = yyvsp[-3].Number;
|
yyMinutes = yyvsp[-3].Number;
|
||||||
@@ -1015,53 +1020,53 @@ case 13:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 14:
|
case 14:
|
||||||
#line 282 "getdate.y"
|
#line 287 "getdate.y"
|
||||||
{
|
{
|
||||||
yyTimezone = yyvsp[0].Number;
|
yyTimezone = yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 15:
|
case 15:
|
||||||
#line 285 "getdate.y"
|
#line 290 "getdate.y"
|
||||||
{
|
{
|
||||||
yyTimezone = yyvsp[0].Number - 60;
|
yyTimezone = yyvsp[0].Number - 60;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 16:
|
case 16:
|
||||||
#line 289 "getdate.y"
|
#line 294 "getdate.y"
|
||||||
{
|
{
|
||||||
yyTimezone = yyvsp[-1].Number - 60;
|
yyTimezone = yyvsp[-1].Number - 60;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 17:
|
case 17:
|
||||||
#line 294 "getdate.y"
|
#line 299 "getdate.y"
|
||||||
{
|
{
|
||||||
yyDayOrdinal = 1;
|
yyDayOrdinal = 1;
|
||||||
yyDayNumber = yyvsp[0].Number;
|
yyDayNumber = yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 18:
|
case 18:
|
||||||
#line 298 "getdate.y"
|
#line 303 "getdate.y"
|
||||||
{
|
{
|
||||||
yyDayOrdinal = 1;
|
yyDayOrdinal = 1;
|
||||||
yyDayNumber = yyvsp[-1].Number;
|
yyDayNumber = yyvsp[-1].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 19:
|
case 19:
|
||||||
#line 302 "getdate.y"
|
#line 307 "getdate.y"
|
||||||
{
|
{
|
||||||
yyDayOrdinal = yyvsp[-1].Number;
|
yyDayOrdinal = yyvsp[-1].Number;
|
||||||
yyDayNumber = yyvsp[0].Number;
|
yyDayNumber = yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 20:
|
case 20:
|
||||||
#line 308 "getdate.y"
|
#line 313 "getdate.y"
|
||||||
{
|
{
|
||||||
yyMonth = yyvsp[-2].Number;
|
yyMonth = yyvsp[-2].Number;
|
||||||
yyDay = yyvsp[0].Number;
|
yyDay = yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 21:
|
case 21:
|
||||||
#line 312 "getdate.y"
|
#line 317 "getdate.y"
|
||||||
{
|
{
|
||||||
/* Interpret as YYYY/MM/DD if $1 >= 1000, otherwise as MM/DD/YY.
|
/* Interpret as YYYY/MM/DD if $1 >= 1000, otherwise as MM/DD/YY.
|
||||||
The goal in recognizing YYYY/MM/DD is solely to support legacy
|
The goal in recognizing YYYY/MM/DD is solely to support legacy
|
||||||
@@ -1082,7 +1087,7 @@ case 21:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 22:
|
case 22:
|
||||||
#line 330 "getdate.y"
|
#line 335 "getdate.y"
|
||||||
{
|
{
|
||||||
/* ISO 8601 format. yyyy-mm-dd. */
|
/* ISO 8601 format. yyyy-mm-dd. */
|
||||||
yyYear = yyvsp[-2].Number;
|
yyYear = yyvsp[-2].Number;
|
||||||
@@ -1091,7 +1096,7 @@ case 22:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 23:
|
case 23:
|
||||||
#line 336 "getdate.y"
|
#line 341 "getdate.y"
|
||||||
{
|
{
|
||||||
/* e.g. 17-JUN-1992. */
|
/* e.g. 17-JUN-1992. */
|
||||||
yyDay = yyvsp[-2].Number;
|
yyDay = yyvsp[-2].Number;
|
||||||
@@ -1100,14 +1105,14 @@ case 23:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 24:
|
case 24:
|
||||||
#line 342 "getdate.y"
|
#line 347 "getdate.y"
|
||||||
{
|
{
|
||||||
yyMonth = yyvsp[-1].Number;
|
yyMonth = yyvsp[-1].Number;
|
||||||
yyDay = yyvsp[0].Number;
|
yyDay = yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 25:
|
case 25:
|
||||||
#line 346 "getdate.y"
|
#line 351 "getdate.y"
|
||||||
{
|
{
|
||||||
yyMonth = yyvsp[-3].Number;
|
yyMonth = yyvsp[-3].Number;
|
||||||
yyDay = yyvsp[-2].Number;
|
yyDay = yyvsp[-2].Number;
|
||||||
@@ -1115,14 +1120,14 @@ case 25:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 26:
|
case 26:
|
||||||
#line 351 "getdate.y"
|
#line 356 "getdate.y"
|
||||||
{
|
{
|
||||||
yyMonth = yyvsp[0].Number;
|
yyMonth = yyvsp[0].Number;
|
||||||
yyDay = yyvsp[-1].Number;
|
yyDay = yyvsp[-1].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 27:
|
case 27:
|
||||||
#line 355 "getdate.y"
|
#line 360 "getdate.y"
|
||||||
{
|
{
|
||||||
yyMonth = yyvsp[-1].Number;
|
yyMonth = yyvsp[-1].Number;
|
||||||
yyDay = yyvsp[-2].Number;
|
yyDay = yyvsp[-2].Number;
|
||||||
@@ -1130,7 +1135,7 @@ case 27:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 28:
|
case 28:
|
||||||
#line 362 "getdate.y"
|
#line 367 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelSeconds = -yyRelSeconds;
|
yyRelSeconds = -yyRelSeconds;
|
||||||
yyRelMinutes = -yyRelMinutes;
|
yyRelMinutes = -yyRelMinutes;
|
||||||
@@ -1141,115 +1146,115 @@ case 28:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 30:
|
case 30:
|
||||||
#line 373 "getdate.y"
|
#line 378 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelYear += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelYear += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 31:
|
case 31:
|
||||||
#line 376 "getdate.y"
|
#line 381 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelYear += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelYear += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 32:
|
case 32:
|
||||||
#line 379 "getdate.y"
|
#line 384 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelYear += yyvsp[0].Number;
|
yyRelYear += yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 33:
|
case 33:
|
||||||
#line 382 "getdate.y"
|
#line 387 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelMonth += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelMonth += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 34:
|
case 34:
|
||||||
#line 385 "getdate.y"
|
#line 390 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelMonth += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelMonth += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 35:
|
case 35:
|
||||||
#line 388 "getdate.y"
|
#line 393 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelMonth += yyvsp[0].Number;
|
yyRelMonth += yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 36:
|
case 36:
|
||||||
#line 391 "getdate.y"
|
#line 396 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelDay += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelDay += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 37:
|
case 37:
|
||||||
#line 394 "getdate.y"
|
#line 399 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelDay += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelDay += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 38:
|
case 38:
|
||||||
#line 397 "getdate.y"
|
#line 402 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelDay += yyvsp[0].Number;
|
yyRelDay += yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 39:
|
case 39:
|
||||||
#line 400 "getdate.y"
|
#line 405 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelHour += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelHour += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 40:
|
case 40:
|
||||||
#line 403 "getdate.y"
|
#line 408 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelHour += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelHour += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 41:
|
case 41:
|
||||||
#line 406 "getdate.y"
|
#line 411 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelHour += yyvsp[0].Number;
|
yyRelHour += yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 42:
|
case 42:
|
||||||
#line 409 "getdate.y"
|
#line 414 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelMinutes += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelMinutes += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 43:
|
case 43:
|
||||||
#line 412 "getdate.y"
|
#line 417 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelMinutes += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelMinutes += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 44:
|
case 44:
|
||||||
#line 415 "getdate.y"
|
#line 420 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelMinutes += yyvsp[0].Number;
|
yyRelMinutes += yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 45:
|
case 45:
|
||||||
#line 418 "getdate.y"
|
#line 423 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelSeconds += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelSeconds += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 46:
|
case 46:
|
||||||
#line 421 "getdate.y"
|
#line 426 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelSeconds += yyvsp[-1].Number * yyvsp[0].Number;
|
yyRelSeconds += yyvsp[-1].Number * yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 47:
|
case 47:
|
||||||
#line 424 "getdate.y"
|
#line 429 "getdate.y"
|
||||||
{
|
{
|
||||||
yyRelSeconds += yyvsp[0].Number;
|
yyRelSeconds += yyvsp[0].Number;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 48:
|
case 48:
|
||||||
#line 430 "getdate.y"
|
#line 435 "getdate.y"
|
||||||
{
|
{
|
||||||
if (yyHaveTime && yyHaveDate && !yyHaveRel)
|
if (yyHaveTime && yyHaveDate && !yyHaveRel)
|
||||||
yyYear = yyvsp[0].Number;
|
yyYear = yyvsp[0].Number;
|
||||||
@@ -1282,13 +1287,13 @@ case 48:
|
|||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 49:
|
case 49:
|
||||||
#line 463 "getdate.y"
|
#line 468 "getdate.y"
|
||||||
{
|
{
|
||||||
yyval.Meridian = MER24;
|
yyval.Meridian = MER24;
|
||||||
;
|
;
|
||||||
break;}
|
break;}
|
||||||
case 50:
|
case 50:
|
||||||
#line 467 "getdate.y"
|
#line 472 "getdate.y"
|
||||||
{
|
{
|
||||||
yyval.Meridian = yyvsp[0].Meridian;
|
yyval.Meridian = yyvsp[0].Meridian;
|
||||||
;
|
;
|
||||||
@@ -1515,7 +1520,7 @@ yyerrhandle:
|
|||||||
}
|
}
|
||||||
return 1;
|
return 1;
|
||||||
}
|
}
|
||||||
#line 472 "getdate.y"
|
#line 477 "getdate.y"
|
||||||
|
|
||||||
|
|
||||||
/* Include this file down here because bison inserts code above which
|
/* Include this file down here because bison inserts code above which
|
||||||
|
@@ -80,6 +80,11 @@
|
|||||||
# include <string.h>
|
# include <string.h>
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#if __GNUC__ < 2 || (__GNUC__ == 2 && __GNUC_MINOR__ < 7)
|
#if __GNUC__ < 2 || (__GNUC__ == 2 && __GNUC_MINOR__ < 7)
|
||||||
# define __attribute__(x)
|
# define __attribute__(x)
|
||||||
#endif
|
#endif
|
||||||
@@ -103,44 +108,44 @@
|
|||||||
then those parser generators need to be fixed instead of adding those
|
then those parser generators need to be fixed instead of adding those
|
||||||
names to this list. */
|
names to this list. */
|
||||||
|
|
||||||
#define yymaxdepth gd_maxdepth
|
#define yymaxdepth Curl_gd_maxdepth
|
||||||
#define yyparse gd_parse
|
#define yyparse Curl_gd_parse
|
||||||
#define yylex gd_lex
|
#define yylex Curl_gd_lex
|
||||||
#define yyerror gd_error
|
#define yyerror Curl_gd_error
|
||||||
#define yylval gd_lval
|
#define yylval Curl_gd_lval
|
||||||
#define yychar gd_char
|
#define yychar Curl_gd_char
|
||||||
#define yydebug gd_debug
|
#define yydebug Curl_gd_debug
|
||||||
#define yypact gd_pact
|
#define yypact Curl_gd_pact
|
||||||
#define yyr1 gd_r1
|
#define yyr1 Curl_gd_r1
|
||||||
#define yyr2 gd_r2
|
#define yyr2 Curl_gd_r2
|
||||||
#define yydef gd_def
|
#define yydef Curl_gd_def
|
||||||
#define yychk gd_chk
|
#define yychk Curl_gd_chk
|
||||||
#define yypgo gd_pgo
|
#define yypgo Curl_gd_pgo
|
||||||
#define yyact gd_act
|
#define yyact Curl_gd_act
|
||||||
#define yyexca gd_exca
|
#define yyexca Curl_gd_exca
|
||||||
#define yyerrflag gd_errflag
|
#define yyerrflag Curl_gd_errflag
|
||||||
#define yynerrs gd_nerrs
|
#define yynerrs Curl_gd_nerrs
|
||||||
#define yyps gd_ps
|
#define yyps Curl_gd_ps
|
||||||
#define yypv gd_pv
|
#define yypv Curl_gd_pv
|
||||||
#define yys gd_s
|
#define yys Curl_gd_s
|
||||||
#define yy_yys gd_yys
|
#define yy_yys Curl_gd_yys
|
||||||
#define yystate gd_state
|
#define yystate Curl_gd_state
|
||||||
#define yytmp gd_tmp
|
#define yytmp Curl_gd_tmp
|
||||||
#define yyv gd_v
|
#define yyv Curl_gd_v
|
||||||
#define yy_yyv gd_yyv
|
#define yy_yyv Curl_gd_yyv
|
||||||
#define yyval gd_val
|
#define yyval Curl_gd_val
|
||||||
#define yylloc gd_lloc
|
#define yylloc Curl_gd_lloc
|
||||||
#define yyreds gd_reds /* With YYDEBUG defined */
|
#define yyreds Curl_gd_reds /* With YYDEBUG defined */
|
||||||
#define yytoks gd_toks /* With YYDEBUG defined */
|
#define yytoks Curl_gd_toks /* With YYDEBUG defined */
|
||||||
#define yylhs gd_yylhs
|
#define yylhs Curl_gd_yylhs
|
||||||
#define yylen gd_yylen
|
#define yylen Curl_gd_yylen
|
||||||
#define yydefred gd_yydefred
|
#define yydefred Curl_gd_yydefred
|
||||||
#define yydgoto gd_yydgoto
|
#define yydgoto Curl_gd_yydgoto
|
||||||
#define yysindex gd_yysindex
|
#define yysindex Curl_gd_yysindex
|
||||||
#define yyrindex gd_yyrindex
|
#define yyrindex Curl_gd_yyrindex
|
||||||
#define yygindex gd_yygindex
|
#define yygindex Curl_gd_yygindex
|
||||||
#define yytable gd_yytable
|
#define yytable Curl_gd_yytable
|
||||||
#define yycheck gd_yycheck
|
#define yycheck Curl_gd_yycheck
|
||||||
|
|
||||||
static int yylex ();
|
static int yylex ();
|
||||||
static int yyerror ();
|
static int yyerror ();
|
||||||
|
@@ -31,7 +31,7 @@
|
|||||||
#include <string.h>
|
#include <string.h>
|
||||||
#include <stdarg.h>
|
#include <stdarg.h>
|
||||||
|
|
||||||
CURLcode curl_getinfo(CURL *curl, CURLINFO info, ...)
|
CURLcode Curl_getinfo(CURL *curl, CURLINFO info, ...)
|
||||||
{
|
{
|
||||||
va_list arg;
|
va_list arg;
|
||||||
long *param_longp;
|
long *param_longp;
|
||||||
|
@@ -66,6 +66,11 @@
|
|||||||
# endif
|
# endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
/* no perror? make an fprintf! */
|
/* no perror? make an fprintf! */
|
||||||
#ifndef HAVE_PERROR
|
#ifndef HAVE_PERROR
|
||||||
# define perror(x) fprintf(stderr, "Error in: %s\n", x)
|
# define perror(x) fprintf(stderr, "Error in: %s\n", x)
|
||||||
|
39
lib/http.c
39
lib/http.c
@@ -104,6 +104,7 @@
|
|||||||
#include "memdebug.h"
|
#include "memdebug.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* ------------------------------------------------------------------------- */
|
||||||
/*
|
/*
|
||||||
* The add_buffer series of functions are used to build one large memory chunk
|
* The add_buffer series of functions are used to build one large memory chunk
|
||||||
* from repeated function invokes. Used so that the entire HTTP request can
|
* from repeated function invokes. Used so that the entire HTTP request can
|
||||||
@@ -205,7 +206,7 @@ CURLcode add_buffer(send_buffer *in, void *inptr, size_t size)
|
|||||||
}
|
}
|
||||||
|
|
||||||
/* end of the add_buffer functions */
|
/* end of the add_buffer functions */
|
||||||
/*****************************************************************************/
|
/* ------------------------------------------------------------------------- */
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Read everything until a newline.
|
* Read everything until a newline.
|
||||||
@@ -282,7 +283,7 @@ CURLcode Curl_ConnectHTTPProxyTunnel(struct connectdata *conn,
|
|||||||
"%s"
|
"%s"
|
||||||
"\r\n",
|
"\r\n",
|
||||||
hostname, remote_port,
|
hostname, remote_port,
|
||||||
(data->bits.proxy_user_passwd)?conn->allocptr.proxyuserpwd:"",
|
(conn->bits.proxy_user_passwd)?conn->allocptr.proxyuserpwd:"",
|
||||||
(data->useragent?conn->allocptr.uagent:"")
|
(data->useragent?conn->allocptr.uagent:"")
|
||||||
);
|
);
|
||||||
|
|
||||||
@@ -309,6 +310,9 @@ CURLcode Curl_ConnectHTTPProxyTunnel(struct connectdata *conn,
|
|||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* HTTP stuff to do at connect-time.
|
||||||
|
*/
|
||||||
CURLcode Curl_http_connect(struct connectdata *conn)
|
CURLcode Curl_http_connect(struct connectdata *conn)
|
||||||
{
|
{
|
||||||
struct UrlData *data;
|
struct UrlData *data;
|
||||||
@@ -336,7 +340,7 @@ CURLcode Curl_http_connect(struct connectdata *conn)
|
|||||||
return CURLE_SSL_CONNECT_ERROR;
|
return CURLE_SSL_CONNECT_ERROR;
|
||||||
}
|
}
|
||||||
|
|
||||||
if(data->bits.user_passwd && !data->bits.this_is_a_follow) {
|
if(conn->bits.user_passwd && !data->bits.this_is_a_follow) {
|
||||||
/* Authorization: is requested, this is not a followed location, get the
|
/* Authorization: is requested, this is not a followed location, get the
|
||||||
original host name */
|
original host name */
|
||||||
data->auth_host = strdup(conn->hostname);
|
data->auth_host = strdup(conn->hostname);
|
||||||
@@ -402,6 +406,9 @@ CURLcode Curl_http(struct connectdata *conn)
|
|||||||
else
|
else
|
||||||
http = conn->proto.http;
|
http = conn->proto.http;
|
||||||
|
|
||||||
|
/* We default to persistant connections */
|
||||||
|
conn->bits.close = FALSE;
|
||||||
|
|
||||||
if ( (conn->protocol&(PROT_HTTP|PROT_FTP)) &&
|
if ( (conn->protocol&(PROT_HTTP|PROT_FTP)) &&
|
||||||
data->bits.upload) {
|
data->bits.upload) {
|
||||||
data->bits.http_put=1;
|
data->bits.http_put=1;
|
||||||
@@ -416,7 +423,7 @@ CURLcode Curl_http(struct connectdata *conn)
|
|||||||
conn->allocptr.uagent=NULL;
|
conn->allocptr.uagent=NULL;
|
||||||
}
|
}
|
||||||
|
|
||||||
if((data->bits.user_passwd) && !checkheaders(data, "Authorization:")) {
|
if((conn->bits.user_passwd) && !checkheaders(data, "Authorization:")) {
|
||||||
char *authorization;
|
char *authorization;
|
||||||
|
|
||||||
/* To prevent the user+password to get sent to other than the original
|
/* To prevent the user+password to get sent to other than the original
|
||||||
@@ -462,10 +469,14 @@ CURLcode Curl_http(struct connectdata *conn)
|
|||||||
http->sendit = Curl_getFormData(data->httppost, &http->postsize);
|
http->sendit = Curl_getFormData(data->httppost, &http->postsize);
|
||||||
}
|
}
|
||||||
|
|
||||||
if(!checkheaders(data, "Host:") &&
|
if(!checkheaders(data, "Host:")) {
|
||||||
!conn->allocptr.host) {
|
/* if ptr_host is already set, it is almost OK since we only re-use
|
||||||
/* if ptr_host is already set, it is OK since we only re-use connections
|
connections to the very same host and port, but when we use a HTTP
|
||||||
to the very same host and port */
|
proxy we have a persistant connect and yet we must change the Host:
|
||||||
|
header! */
|
||||||
|
|
||||||
|
if(conn->allocptr.host)
|
||||||
|
free(conn->allocptr.host);
|
||||||
|
|
||||||
if(((conn->protocol&PROT_HTTPS) && (conn->remote_port == PORT_HTTPS)) ||
|
if(((conn->protocol&PROT_HTTPS) && (conn->remote_port == PORT_HTTPS)) ||
|
||||||
(!(conn->protocol&PROT_HTTPS) && (conn->remote_port == PORT_HTTP)) )
|
(!(conn->protocol&PROT_HTTPS) && (conn->remote_port == PORT_HTTP)) )
|
||||||
@@ -595,10 +606,14 @@ CURLcode Curl_http(struct connectdata *conn)
|
|||||||
(data->bits.http_post || data->bits.http_formpost)?"POST":
|
(data->bits.http_post || data->bits.http_formpost)?"POST":
|
||||||
(data->bits.http_put)?"PUT":"GET"),
|
(data->bits.http_put)?"PUT":"GET"),
|
||||||
ppath,
|
ppath,
|
||||||
(data->bits.proxy_user_passwd && conn->allocptr.proxyuserpwd)?conn->allocptr.proxyuserpwd:"",
|
(conn->bits.proxy_user_passwd &&
|
||||||
(data->bits.user_passwd && conn->allocptr.userpwd)?conn->allocptr.userpwd:"",
|
conn->allocptr.proxyuserpwd)?conn->allocptr.proxyuserpwd:"",
|
||||||
(data->bits.set_range && conn->allocptr.rangeline)?conn->allocptr.rangeline:"",
|
(conn->bits.user_passwd && conn->allocptr.userpwd)?
|
||||||
(data->useragent && *data->useragent && conn->allocptr.uagent)?conn->allocptr.uagent:"",
|
conn->allocptr.userpwd:"",
|
||||||
|
(data->bits.set_range && conn->allocptr.rangeline)?
|
||||||
|
conn->allocptr.rangeline:"",
|
||||||
|
(data->useragent && *data->useragent && conn->allocptr.uagent)?
|
||||||
|
conn->allocptr.uagent:"",
|
||||||
(conn->allocptr.cookie?conn->allocptr.cookie:""), /* Cookie: <data> */
|
(conn->allocptr.cookie?conn->allocptr.cookie:""), /* Cookie: <data> */
|
||||||
(conn->allocptr.host?conn->allocptr.host:""), /* Host: host */
|
(conn->allocptr.host?conn->allocptr.host:""), /* Host: host */
|
||||||
http->p_pragma?http->p_pragma:"",
|
http->p_pragma?http->p_pragma:"",
|
||||||
|
@@ -35,4 +35,9 @@ CURLcode Curl_http_done(struct connectdata *conn);
|
|||||||
CURLcode Curl_http_connect(struct connectdata *conn);
|
CURLcode Curl_http_connect(struct connectdata *conn);
|
||||||
CURLcode Curl_http_close(struct connectdata *conn);
|
CURLcode Curl_http_close(struct connectdata *conn);
|
||||||
|
|
||||||
|
/* The following functions are defined in http_chunks.c */
|
||||||
|
void Curl_httpchunk_init(struct connectdata *conn);
|
||||||
|
CHUNKcode Curl_httpchunk_read(struct connectdata *conn, char *datap,
|
||||||
|
ssize_t length, ssize_t *wrote);
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
222
lib/http_chunks.c
Normal file
222
lib/http_chunks.c
Normal file
@@ -0,0 +1,222 @@
|
|||||||
|
/*****************************************************************************
|
||||||
|
* _ _ ____ _
|
||||||
|
* Project ___| | | | _ \| |
|
||||||
|
* / __| | | | |_) | |
|
||||||
|
* | (__| |_| | _ <| |___
|
||||||
|
* \___|\___/|_| \_\_____|
|
||||||
|
*
|
||||||
|
* Copyright (C) 2001, Daniel Stenberg, <daniel@haxx.se>, et al.
|
||||||
|
*
|
||||||
|
* In order to be useful for every potential user, curl and libcurl are
|
||||||
|
* dual-licensed under the MPL and the MIT/X-derivate licenses.
|
||||||
|
*
|
||||||
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
||||||
|
* copies of the Software, and permit persons to whom the Software is
|
||||||
|
* furnished to do so, under the terms of the MPL or the MIT/X-derivate
|
||||||
|
* licenses. You may pick one of these licenses.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
*****************************************************************************/
|
||||||
|
#include "setup.h"
|
||||||
|
|
||||||
|
/* -- WIN32 approved -- */
|
||||||
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
|
#include <stdarg.h>
|
||||||
|
#include <stdlib.h>
|
||||||
|
#include <ctype.h>
|
||||||
|
|
||||||
|
#include "urldata.h" /* it includes http_chunks.h */
|
||||||
|
#include "sendf.h" /* for the client write stuff */
|
||||||
|
|
||||||
|
#define _MPRINTF_REPLACE /* use our functions only */
|
||||||
|
#include <curl/mprintf.h>
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Chunk format (simplified):
|
||||||
|
*
|
||||||
|
* <HEX SIZE>[ chunk extension ] CRLF
|
||||||
|
* <DATA>
|
||||||
|
*
|
||||||
|
* Highlights from RFC2616 section 3.6 say:
|
||||||
|
|
||||||
|
The chunked encoding modifies the body of a message in order to
|
||||||
|
transfer it as a series of chunks, each with its own size indicator,
|
||||||
|
followed by an OPTIONAL trailer containing entity-header fields. This
|
||||||
|
allows dynamically produced content to be transferred along with the
|
||||||
|
information necessary for the recipient to verify that it has
|
||||||
|
received the full message.
|
||||||
|
|
||||||
|
Chunked-Body = *chunk
|
||||||
|
last-chunk
|
||||||
|
trailer
|
||||||
|
CRLF
|
||||||
|
|
||||||
|
chunk = chunk-size [ chunk-extension ] CRLF
|
||||||
|
chunk-data CRLF
|
||||||
|
chunk-size = 1*HEX
|
||||||
|
last-chunk = 1*("0") [ chunk-extension ] CRLF
|
||||||
|
|
||||||
|
chunk-extension= *( ";" chunk-ext-name [ "=" chunk-ext-val ] )
|
||||||
|
chunk-ext-name = token
|
||||||
|
chunk-ext-val = token | quoted-string
|
||||||
|
chunk-data = chunk-size(OCTET)
|
||||||
|
trailer = *(entity-header CRLF)
|
||||||
|
|
||||||
|
The chunk-size field is a string of hex digits indicating the size of
|
||||||
|
the chunk. The chunked encoding is ended by any chunk whose size is
|
||||||
|
zero, followed by the trailer, which is terminated by an empty line.
|
||||||
|
|
||||||
|
*/
|
||||||
|
|
||||||
|
|
||||||
|
void Curl_httpchunk_init(struct connectdata *conn)
|
||||||
|
{
|
||||||
|
struct Curl_chunker *chunk = &conn->proto.http->chunk;
|
||||||
|
chunk->hexindex=0; /* start at 0 */
|
||||||
|
chunk->dataleft=0; /* no data left yet! */
|
||||||
|
chunk->state = CHUNK_HEX; /* we get hex first! */
|
||||||
|
}
|
||||||
|
|
||||||
|
/*
|
||||||
|
* chunk_read() returns a OK for normal operations, or a positive return code
|
||||||
|
* for errors. STOP means this sequence of chunks is complete. The 'wrote'
|
||||||
|
* argument is set to tell the caller how many bytes we actually passed to the
|
||||||
|
* client (for byte-counting and whatever).
|
||||||
|
*
|
||||||
|
* The states and the state-machine is further explained in the header file.
|
||||||
|
*/
|
||||||
|
CHUNKcode Curl_httpchunk_read(struct connectdata *conn,
|
||||||
|
char *datap,
|
||||||
|
ssize_t length,
|
||||||
|
ssize_t *wrote)
|
||||||
|
{
|
||||||
|
CURLcode result;
|
||||||
|
struct Curl_chunker *ch = &conn->proto.http->chunk;
|
||||||
|
int piece;
|
||||||
|
*wrote = 0; /* nothing yet */
|
||||||
|
|
||||||
|
while(length) {
|
||||||
|
switch(ch->state) {
|
||||||
|
case CHUNK_HEX:
|
||||||
|
if(isxdigit((int)*datap)) {
|
||||||
|
if(ch->hexindex < MAXNUM_SIZE) {
|
||||||
|
ch->hexbuffer[ch->hexindex] = *datap;
|
||||||
|
datap++;
|
||||||
|
length--;
|
||||||
|
ch->hexindex++;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
return CHUNKE_TOO_LONG_HEX; /* longer hex than we support */
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
if(0 == ch->hexindex) {
|
||||||
|
/* This is illegal data, we received junk where we expected
|
||||||
|
a hexadecimal digit. */
|
||||||
|
return CHUNKE_ILLEGAL_HEX;
|
||||||
|
}
|
||||||
|
/* length and datap are unmodified */
|
||||||
|
ch->hexbuffer[ch->hexindex]=0;
|
||||||
|
ch->datasize=strtoul(ch->hexbuffer, NULL, 16);
|
||||||
|
ch->state = CHUNK_POSTHEX;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CHUNK_POSTHEX:
|
||||||
|
/* In this state, we're waiting for CRLF to arrive. We support
|
||||||
|
this to allow so called chunk-extensions to show up here
|
||||||
|
before the CRLF comes. */
|
||||||
|
if(*datap == '\r')
|
||||||
|
ch->state = CHUNK_CR;
|
||||||
|
length--;
|
||||||
|
datap++;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CHUNK_CR:
|
||||||
|
/* waiting for the LF */
|
||||||
|
if(*datap == '\n') {
|
||||||
|
/* we're now expecting data to come, unless size was zero! */
|
||||||
|
if(0 == ch->datasize) {
|
||||||
|
ch->state = CHUNK_STOP; /* stop reading! */
|
||||||
|
if(1 == length) {
|
||||||
|
/* This was the final byte, return right now */
|
||||||
|
return CHUNKE_STOP;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
else
|
||||||
|
ch->state = CHUNK_DATA;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
/* previously we got a fake CR, go back to CR waiting! */
|
||||||
|
ch->state = CHUNK_CR;
|
||||||
|
datap++;
|
||||||
|
length--;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CHUNK_DATA:
|
||||||
|
/* we get pure and fine data
|
||||||
|
|
||||||
|
We expect another 'datasize' of data. We have 'length' right now,
|
||||||
|
it can be more or less than 'datasize'. Get the smallest piece.
|
||||||
|
*/
|
||||||
|
piece = (ch->datasize >= length)?length:ch->datasize;
|
||||||
|
|
||||||
|
/* Write the data portion available */
|
||||||
|
result = Curl_client_write(conn->data, CLIENTWRITE_BODY, datap, piece);
|
||||||
|
if(result)
|
||||||
|
return CHUNKE_WRITE_ERROR;
|
||||||
|
*wrote += piece;
|
||||||
|
|
||||||
|
ch->datasize -= piece; /* decrease amount left to expect */
|
||||||
|
datap += piece; /* move read pointer forward */
|
||||||
|
length -= piece; /* decrease space left in this round */
|
||||||
|
|
||||||
|
if(0 == ch->datasize)
|
||||||
|
/* end of data this round, we now expect a trailing CRLF */
|
||||||
|
ch->state = CHUNK_POSTCR;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CHUNK_POSTCR:
|
||||||
|
if(*datap == '\r') {
|
||||||
|
ch->state = CHUNK_POSTLF;
|
||||||
|
datap++;
|
||||||
|
length--;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return CHUNKE_BAD_CHUNK;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CHUNK_POSTLF:
|
||||||
|
if(*datap == '\n') {
|
||||||
|
/*
|
||||||
|
* The last one before we go back to hex state and start all
|
||||||
|
* over.
|
||||||
|
*/
|
||||||
|
Curl_httpchunk_init(conn);
|
||||||
|
datap++;
|
||||||
|
length--;
|
||||||
|
}
|
||||||
|
else
|
||||||
|
return CHUNKE_BAD_CHUNK;
|
||||||
|
break;
|
||||||
|
|
||||||
|
case CHUNK_STOP:
|
||||||
|
/* If we arrive here, there is data left in the end of the buffer
|
||||||
|
even if there's no more chunks to read */
|
||||||
|
ch->dataleft = length;
|
||||||
|
return CHUNKE_STOP; /* return stop */
|
||||||
|
default:
|
||||||
|
return CHUNKE_STATE_ERROR;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return CHUNKE_OK;
|
||||||
|
}
|
87
lib/http_chunks.h
Normal file
87
lib/http_chunks.h
Normal file
@@ -0,0 +1,87 @@
|
|||||||
|
#ifndef __HTTP_CHUNKS_H
|
||||||
|
#define __HTTP_CHUNKS_H
|
||||||
|
/*****************************************************************************
|
||||||
|
* _ _ ____ _
|
||||||
|
* Project ___| | | | _ \| |
|
||||||
|
* / __| | | | |_) | |
|
||||||
|
* | (__| |_| | _ <| |___
|
||||||
|
* \___|\___/|_| \_\_____|
|
||||||
|
*
|
||||||
|
* Copyright (C) 2001, Daniel Stenberg, <daniel@haxx.se>, et al.
|
||||||
|
*
|
||||||
|
* In order to be useful for every potential user, curl and libcurl are
|
||||||
|
* dual-licensed under the MPL and the MIT/X-derivate licenses.
|
||||||
|
*
|
||||||
|
* You may opt to use, copy, modify, merge, publish, distribute and/or sell
|
||||||
|
* copies of the Software, and permit persons to whom the Software is
|
||||||
|
* furnished to do so, under the terms of the MPL or the MIT/X-derivate
|
||||||
|
* licenses. You may pick one of these licenses.
|
||||||
|
*
|
||||||
|
* This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
|
||||||
|
* KIND, either express or implied.
|
||||||
|
*
|
||||||
|
* $Id$
|
||||||
|
*****************************************************************************/
|
||||||
|
/*
|
||||||
|
* The longest possible hexadecimal number we support in a chunked transfer.
|
||||||
|
* Weird enough, RFC2616 doesn't set a maximum size! Since we use strtoul()
|
||||||
|
* to convert it, we "only" support 2^32 bytes chunk data.
|
||||||
|
*/
|
||||||
|
#define MAXNUM_SIZE 16
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CHUNK_FIRST, /* never use */
|
||||||
|
|
||||||
|
/* In this we await and buffer all hexadecimal digits until we get one
|
||||||
|
that isn't a hexadecimal digit. When done, we go POSTHEX */
|
||||||
|
CHUNK_HEX,
|
||||||
|
|
||||||
|
/* We have received the hexadecimal digit and we eat all characters until
|
||||||
|
we get a CRLF pair. When we see a CR we go to the CR state. */
|
||||||
|
CHUNK_POSTHEX,
|
||||||
|
|
||||||
|
/* A single CR has been found and we should get a LF right away in this
|
||||||
|
state or we go back to POSTHEX. When LF is received, we go to DATA.
|
||||||
|
If the size given was zero, we set state to STOP and return. */
|
||||||
|
CHUNK_CR,
|
||||||
|
|
||||||
|
/* We eat the amount of data specified. When done, we move on to the
|
||||||
|
POST_CR state. */
|
||||||
|
CHUNK_DATA,
|
||||||
|
|
||||||
|
/* POSTCR should get a CR and nothing else, then move to POSTLF */
|
||||||
|
CHUNK_POSTCR,
|
||||||
|
|
||||||
|
/* POSTLF should get a LF and nothing else, then move back to HEX as
|
||||||
|
the CRLF combination marks the end of a chunk */
|
||||||
|
CHUNK_POSTLF,
|
||||||
|
|
||||||
|
/* This is mainly used to really mark that we're out of the game.
|
||||||
|
NOTE: that there's a 'dataleft' field in the struct that will tell how
|
||||||
|
many bytes that were not passed to the client in the end of the last
|
||||||
|
buffer! */
|
||||||
|
CHUNK_STOP,
|
||||||
|
|
||||||
|
CHUNK_LAST /* never use */
|
||||||
|
} ChunkyState;
|
||||||
|
|
||||||
|
typedef enum {
|
||||||
|
CHUNKE_STOP = -1,
|
||||||
|
CHUNKE_OK = 0,
|
||||||
|
CHUNKE_TOO_LONG_HEX = 1,
|
||||||
|
CHUNKE_ILLEGAL_HEX,
|
||||||
|
CHUNKE_BAD_CHUNK,
|
||||||
|
CHUNKE_WRITE_ERROR,
|
||||||
|
CHUNKE_STATE_ERROR,
|
||||||
|
CHUNKE_LAST
|
||||||
|
} CHUNKcode;
|
||||||
|
|
||||||
|
struct Curl_chunker {
|
||||||
|
char hexbuffer[ MAXNUM_SIZE + 1];
|
||||||
|
int hexindex;
|
||||||
|
ChunkyState state;
|
||||||
|
size_t datasize;
|
||||||
|
size_t dataleft; /* untouched data amount at the end of the last buffer */
|
||||||
|
};
|
||||||
|
|
||||||
|
#endif
|
@@ -70,6 +70,11 @@
|
|||||||
#include "inet_ntoa_r.h"
|
#include "inet_ntoa_r.h"
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#define SYS_ERROR -1
|
#define SYS_ERROR -1
|
||||||
|
|
||||||
char *Curl_if2ip(char *interface, char *buf, int buf_size)
|
char *Curl_if2ip(char *interface, char *buf, int buf_size)
|
||||||
@@ -90,6 +95,7 @@ char *Curl_if2ip(char *interface, char *buf, int buf_size)
|
|||||||
strcpy(req.ifr_name, interface);
|
strcpy(req.ifr_name, interface);
|
||||||
req.ifr_addr.sa_family = AF_INET;
|
req.ifr_addr.sa_family = AF_INET;
|
||||||
if (SYS_ERROR == ioctl(dummy, SIOCGIFADDR, &req, sizeof(req))) {
|
if (SYS_ERROR == ioctl(dummy, SIOCGIFADDR, &req, sizeof(req))) {
|
||||||
|
sclose(dummy);
|
||||||
return NULL;
|
return NULL;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -104,7 +110,7 @@ char *Curl_if2ip(char *interface, char *buf, int buf_size)
|
|||||||
ip[buf_size - 1] = 0;
|
ip[buf_size - 1] = 0;
|
||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
close(dummy);
|
sclose(dummy);
|
||||||
}
|
}
|
||||||
return ip;
|
return ip;
|
||||||
}
|
}
|
||||||
|
@@ -2,41 +2,31 @@
|
|||||||
; Definition file for the DLL version of the LIBCURL library from curl
|
; Definition file for the DLL version of the LIBCURL library from curl
|
||||||
;
|
;
|
||||||
|
|
||||||
LIBRARY CURL
|
LIBRARY LIBCURL
|
||||||
|
|
||||||
DESCRIPTION 'curl libcurl - http://curl.haxx.se'
|
DESCRIPTION 'curl libcurl - http://curl.haxx.se'
|
||||||
|
|
||||||
EXPORTS
|
EXPORTS
|
||||||
curl_close @ 1 ;
|
curl_easy_cleanup @ 1 ;
|
||||||
curl_connect @ 2 ;
|
curl_easy_getinfo @ 2 ;
|
||||||
curl_disconnect @ 3 ;
|
curl_easy_init @ 3 ;
|
||||||
curl_do @ 4 ;
|
curl_easy_perform @ 4 ;
|
||||||
curl_done @ 5 ;
|
curl_easy_setopt @ 5 ;
|
||||||
curl_easy_cleanup @ 6 ;
|
curl_escape @ 6 ;
|
||||||
curl_easy_getinfo @ 7 ;
|
curl_formparse @ 7 ;
|
||||||
curl_easy_init @ 8 ;
|
curl_formfree @ 8 ;
|
||||||
curl_easy_perform @ 9 ;
|
curl_getdate @ 9 ;
|
||||||
curl_easy_setopt @ 10 ;
|
curl_getenv @ 10 ;
|
||||||
curl_escape @ 11 ;
|
curl_slist_append @ 11 ;
|
||||||
curl_formparse @ 12 ;
|
curl_slist_free_all @ 12 ;
|
||||||
curl_free @ 13 ;
|
curl_unescape @ 13 ;
|
||||||
curl_getdate @ 14 ;
|
curl_version @ 14 ;
|
||||||
curl_getenv @ 15 ;
|
curl_maprintf @ 15 ;
|
||||||
curl_init @ 16 ;
|
curl_mfprintf @ 16 ;
|
||||||
curl_open @ 17 ;
|
curl_mprintf @ 17 ;
|
||||||
curl_read @ 18 ;
|
curl_msprintf @ 18 ;
|
||||||
curl_setopt @ 19 ;
|
curl_msnprintf @ 19 ;
|
||||||
curl_slist_append @ 20 ;
|
curl_mvfprintf @ 20 ;
|
||||||
curl_slist_free_all @ 21 ;
|
curl_strequal @ 21 ;
|
||||||
curl_transfer @ 22 ;
|
curl_strnequal @ 22 ;
|
||||||
curl_unescape @ 23 ;
|
|
||||||
curl_version @ 24 ;
|
|
||||||
curl_write @ 25 ;
|
|
||||||
curl_maprintf @ 26 ;
|
|
||||||
curl_mfprintf @ 27 ;
|
|
||||||
curl_mprintf @ 28 ;
|
|
||||||
curl_msprintf @ 29 ;
|
|
||||||
curl_msnprintf @ 30 ;
|
|
||||||
curl_mvfprintf @ 31 ;
|
|
||||||
Curl_strequal @ 32 ;
|
|
||||||
Curl_strnequal @ 33 ;
|
|
||||||
|
@@ -120,7 +120,7 @@ int curl_socket(int domain, int type, int protocol, int line, char *source)
|
|||||||
return sockfd;
|
return sockfd;
|
||||||
}
|
}
|
||||||
|
|
||||||
int curl_accept(int s, struct sockaddr *addr, int *addrlen,
|
int curl_accept(int s, struct sockaddr *addr, socklen_t *addrlen,
|
||||||
int line, char *source)
|
int line, char *source)
|
||||||
{
|
{
|
||||||
int sockfd=(accept)(s, addr, addrlen);
|
int sockfd=(accept)(s, addr, addrlen);
|
||||||
|
@@ -13,7 +13,7 @@ void curl_memdebug(char *logname);
|
|||||||
/* file descriptor manipulators */
|
/* file descriptor manipulators */
|
||||||
int curl_socket(int domain, int type, int protocol, int, char *);
|
int curl_socket(int domain, int type, int protocol, int, char *);
|
||||||
int curl_sclose(int sockfd, int, char *);
|
int curl_sclose(int sockfd, int, char *);
|
||||||
int curl_accept(int s, struct sockaddr *addr, int *addrlen,
|
int curl_accept(int s, struct sockaddr *addr, socklen_t *addrlen,
|
||||||
int line, char *source);
|
int line, char *source);
|
||||||
|
|
||||||
/* FILE functions */
|
/* FILE functions */
|
||||||
|
41
lib/netrc.c
41
lib/netrc.c
@@ -27,10 +27,26 @@
|
|||||||
#include <stdlib.h>
|
#include <stdlib.h>
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
|
#ifdef HAVE_SYS_TYPES_H
|
||||||
|
#include <sys/types.h>
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_UNISTD_H
|
||||||
|
#include <unistd.h>
|
||||||
|
#endif
|
||||||
|
#ifdef HAVE_PWD_H
|
||||||
|
#include <pwd.h>
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
|
|
||||||
#include "strequal.h"
|
#include "strequal.h"
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
/* Debug this single source file with:
|
/* Debug this single source file with:
|
||||||
'make netrc' then run './netrc'!
|
'make netrc' then run './netrc'!
|
||||||
|
|
||||||
@@ -60,7 +76,7 @@ int Curl_parsenetrc(char *host,
|
|||||||
char netrcbuffer[256];
|
char netrcbuffer[256];
|
||||||
int retcode=1;
|
int retcode=1;
|
||||||
|
|
||||||
char *home = curl_getenv("HOME"); /* portable environment reader */
|
char *home = NULL;
|
||||||
int state=NOTHING;
|
int state=NOTHING;
|
||||||
|
|
||||||
char state_login=0;
|
char state_login=0;
|
||||||
@@ -68,11 +84,25 @@ int Curl_parsenetrc(char *host,
|
|||||||
|
|
||||||
#define NETRC DOT_CHAR "netrc"
|
#define NETRC DOT_CHAR "netrc"
|
||||||
|
|
||||||
if(!home)
|
#if defined(HAVE_GETPWUID) && defined(HAVE_GETEUID)
|
||||||
return -1;
|
struct passwd *pw;
|
||||||
|
pw= getpwuid(geteuid());
|
||||||
|
if (pw)
|
||||||
|
home = pw->pw_dir;
|
||||||
|
#else
|
||||||
|
void *pw=NULL;
|
||||||
|
#endif
|
||||||
|
|
||||||
|
if(NULL == pw) {
|
||||||
|
home = curl_getenv("HOME"); /* portable environment reader */
|
||||||
|
if(!home) {
|
||||||
|
return -1;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
if(strlen(home)>(sizeof(netrcbuffer)-strlen(NETRC))) {
|
if(strlen(home)>(sizeof(netrcbuffer)-strlen(NETRC))) {
|
||||||
free(home);
|
if(NULL==pw)
|
||||||
|
free(home);
|
||||||
return -1;
|
return -1;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -140,7 +170,8 @@ int Curl_parsenetrc(char *host,
|
|||||||
fclose(file);
|
fclose(file);
|
||||||
}
|
}
|
||||||
|
|
||||||
free(home);
|
if(NULL==pw)
|
||||||
|
free(home);
|
||||||
|
|
||||||
return retcode;
|
return retcode;
|
||||||
}
|
}
|
||||||
|
39
lib/sendf.c
39
lib/sendf.c
@@ -120,7 +120,7 @@ void curl_slist_free_all(struct curl_slist *list)
|
|||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
/* infof() is for info message along the way */
|
/* Curl_infof() is for info message along the way */
|
||||||
|
|
||||||
void Curl_infof(struct UrlData *data, char *fmt, ...)
|
void Curl_infof(struct UrlData *data, char *fmt, ...)
|
||||||
{
|
{
|
||||||
@@ -133,7 +133,7 @@ void Curl_infof(struct UrlData *data, char *fmt, ...)
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
/* failf() is for messages stating why we failed, the LAST one will be
|
/* Curl_failf() is for messages stating why we failed, the LAST one will be
|
||||||
returned for the user (if requested) */
|
returned for the user (if requested) */
|
||||||
|
|
||||||
void Curl_failf(struct UrlData *data, char *fmt, ...)
|
void Curl_failf(struct UrlData *data, char *fmt, ...)
|
||||||
@@ -142,7 +142,7 @@ void Curl_failf(struct UrlData *data, char *fmt, ...)
|
|||||||
va_start(ap, fmt);
|
va_start(ap, fmt);
|
||||||
if(data->errorbuffer)
|
if(data->errorbuffer)
|
||||||
vsnprintf(data->errorbuffer, CURL_ERROR_SIZE, fmt, ap);
|
vsnprintf(data->errorbuffer, CURL_ERROR_SIZE, fmt, ap);
|
||||||
else {
|
else if(!data->bits.mute) {
|
||||||
/* no errorbuffer receives this, write to data->err instead */
|
/* no errorbuffer receives this, write to data->err instead */
|
||||||
vfprintf(data->err, fmt, ap);
|
vfprintf(data->err, fmt, ap);
|
||||||
fprintf(data->err, "\n");
|
fprintf(data->err, "\n");
|
||||||
@@ -213,23 +213,6 @@ CURLcode Curl_write(struct connectdata *conn, int sockfd,
|
|||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* External write-function, writes to the data-socket.
|
|
||||||
* Takes care of plain sockets, SSL or kerberos transparently.
|
|
||||||
*/
|
|
||||||
CURLcode curl_write(CURLconnect *c_conn, char *buf, size_t amount,
|
|
||||||
size_t *n)
|
|
||||||
{
|
|
||||||
struct connectdata *conn = (struct connectdata *)c_conn;
|
|
||||||
|
|
||||||
if(!n || !conn || (conn->handle != STRUCT_CONNECT))
|
|
||||||
return CURLE_FAILED_INIT;
|
|
||||||
|
|
||||||
return Curl_write(conn, conn->sockfd, buf, amount, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
/* client_write() sends data to the write callback(s)
|
/* client_write() sends data to the write callback(s)
|
||||||
|
|
||||||
The bit pattern defines to what "streams" to write to. Body and/or header.
|
The bit pattern defines to what "streams" to write to. Body and/or header.
|
||||||
@@ -299,19 +282,3 @@ CURLcode Curl_read(struct connectdata *conn, int sockfd,
|
|||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
/*
|
|
||||||
* The public read function reads from the 'sockfd' file descriptor only.
|
|
||||||
* Use the Curl_read() internally when you want to specify fd.
|
|
||||||
*/
|
|
||||||
|
|
||||||
CURLcode curl_read(CURLconnect *c_conn, char *buf, size_t buffersize,
|
|
||||||
ssize_t *n)
|
|
||||||
{
|
|
||||||
struct connectdata *conn = (struct connectdata *)c_conn;
|
|
||||||
|
|
||||||
if(!n || !conn || (conn->handle != STRUCT_CONNECT))
|
|
||||||
return CURLE_FAILED_INIT;
|
|
||||||
|
|
||||||
return Curl_read(conn, conn->sockfd, buf, buffersize, n);
|
|
||||||
}
|
|
||||||
|
|
||||||
|
@@ -24,6 +24,7 @@
|
|||||||
#include "setup.h"
|
#include "setup.h"
|
||||||
|
|
||||||
#include <stdio.h>
|
#include <stdio.h>
|
||||||
|
#include <string.h>
|
||||||
#if defined(__MINGW32__)
|
#if defined(__MINGW32__)
|
||||||
#include <winsock.h>
|
#include <winsock.h>
|
||||||
#endif
|
#endif
|
||||||
|
134
lib/ssluse.c
134
lib/ssluse.c
@@ -35,6 +35,7 @@
|
|||||||
#include "formdata.h" /* for the boundary function */
|
#include "formdata.h" /* for the boundary function */
|
||||||
|
|
||||||
#ifdef USE_SSLEAY
|
#ifdef USE_SSLEAY
|
||||||
|
#include <openssl/rand.h>
|
||||||
|
|
||||||
static char global_passwd[64];
|
static char global_passwd[64];
|
||||||
|
|
||||||
@@ -58,16 +59,108 @@ static int passwd_callback(char *buf, int num, int verify
|
|||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* This function is *highly* inspired by (and parts are directly stolen
|
static
|
||||||
* from) source from the SSLeay package written by Eric Young
|
bool seed_enough(struct connectdata *conn, /* unused for now */
|
||||||
* (eay@cryptsoft.com). */
|
int nread)
|
||||||
|
{
|
||||||
|
#ifdef HAVE_RAND_STATUS
|
||||||
|
/* only available in OpenSSL 0.9.5a and later */
|
||||||
|
if(RAND_status())
|
||||||
|
return TRUE;
|
||||||
|
#else
|
||||||
|
if(nread > 500)
|
||||||
|
/* this is a very silly decision to make */
|
||||||
|
return TRUE;
|
||||||
|
#endif
|
||||||
|
return FALSE; /* not enough */
|
||||||
|
}
|
||||||
|
|
||||||
static
|
static
|
||||||
int cert_stuff(struct UrlData *data,
|
int random_the_seed(struct connectdata *conn)
|
||||||
struct connectdata *conn,
|
{
|
||||||
|
char *buf = conn->data->buffer; /* point to the big buffer */
|
||||||
|
int nread=0;
|
||||||
|
struct UrlData *data=conn->data;
|
||||||
|
|
||||||
|
/* Q: should we add support for a random file name as a libcurl option?
|
||||||
|
A: Yes, it is here */
|
||||||
|
|
||||||
|
#ifndef RANDOM_FILE
|
||||||
|
/* if RANDOM_FILE isn't defined, we only perform this if an option tells
|
||||||
|
us to! */
|
||||||
|
if(data->ssl.random_file)
|
||||||
|
#define RANDOM_FILE "" /* doesn't matter won't be used */
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
/* let the option override the define */
|
||||||
|
nread += RAND_load_file((data->ssl.random_file?
|
||||||
|
data->ssl.random_file:RANDOM_FILE),
|
||||||
|
16384);
|
||||||
|
if(seed_enough(conn, nread))
|
||||||
|
return nread;
|
||||||
|
}
|
||||||
|
|
||||||
|
#if defined(HAVE_RAND_EGD)
|
||||||
|
/* only available in OpenSSL 0.9.5 and later */
|
||||||
|
/* EGD_SOCKET is set at configure time or not at all */
|
||||||
|
#ifndef EGD_SOCKET
|
||||||
|
/* If we don't have the define set, we only do this if the egd-option
|
||||||
|
is set */
|
||||||
|
if(data->ssl.egdsocket)
|
||||||
|
#define EGD_SOCKET "" /* doesn't matter won't be used */
|
||||||
|
#endif
|
||||||
|
{
|
||||||
|
/* If there's an option and a define, the option overrides the
|
||||||
|
define */
|
||||||
|
int ret = RAND_egd(data->ssl.egdsocket?data->ssl.egdsocket:EGD_SOCKET);
|
||||||
|
if(-1 != ret) {
|
||||||
|
nread += ret;
|
||||||
|
if(seed_enough(conn, nread))
|
||||||
|
return nread;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* If we get here, it means we need to seed the PRNG using a "silly"
|
||||||
|
approach! */
|
||||||
|
#ifdef HAVE_RAND_SCREEN
|
||||||
|
/* This one gets a random value by reading the currently shown screen */
|
||||||
|
RAND_screen();
|
||||||
|
nread = 100; /* just a value */
|
||||||
|
#else
|
||||||
|
{
|
||||||
|
int len;
|
||||||
|
char *area = Curl_FormBoundary();
|
||||||
|
if(!area)
|
||||||
|
return 3; /* out of memory */
|
||||||
|
|
||||||
|
len = strlen(area);
|
||||||
|
RAND_seed(area, len);
|
||||||
|
|
||||||
|
free(area); /* now remove the random junk */
|
||||||
|
}
|
||||||
|
#endif
|
||||||
|
|
||||||
|
/* generates a default path for the random seed file */
|
||||||
|
buf[0]=0; /* blank it first */
|
||||||
|
RAND_file_name(buf, BUFSIZE);
|
||||||
|
if ( buf[0] ) {
|
||||||
|
/* we got a file name to try */
|
||||||
|
nread += RAND_load_file(buf, 16384);
|
||||||
|
if(seed_enough(conn, nread))
|
||||||
|
return nread;
|
||||||
|
}
|
||||||
|
|
||||||
|
infof(conn->data, "Your connection is using a weak random seed!\n");
|
||||||
|
return nread;
|
||||||
|
}
|
||||||
|
|
||||||
|
static
|
||||||
|
int cert_stuff(struct connectdata *conn,
|
||||||
char *cert_file,
|
char *cert_file,
|
||||||
char *key_file)
|
char *key_file)
|
||||||
{
|
{
|
||||||
|
struct UrlData *data = conn->data;
|
||||||
if (cert_file != NULL) {
|
if (cert_file != NULL) {
|
||||||
SSL *ssl;
|
SSL *ssl;
|
||||||
X509 *x509;
|
X509 *x509;
|
||||||
@@ -123,9 +216,6 @@ int cert_stuff(struct UrlData *data,
|
|||||||
return(1);
|
return(1);
|
||||||
}
|
}
|
||||||
|
|
||||||
#endif
|
|
||||||
|
|
||||||
#ifdef USE_SSLEAY
|
|
||||||
static
|
static
|
||||||
int cert_verify_callback(int ok, X509_STORE_CTX *ctx)
|
int cert_verify_callback(int ok, X509_STORE_CTX *ctx)
|
||||||
{
|
{
|
||||||
@@ -156,28 +246,8 @@ Curl_SSLConnect(struct connectdata *conn)
|
|||||||
/* Lets get nice error messages */
|
/* Lets get nice error messages */
|
||||||
SSL_load_error_strings();
|
SSL_load_error_strings();
|
||||||
|
|
||||||
#ifdef HAVE_RAND_STATUS
|
/* Make funny stuff to get random input */
|
||||||
/* RAND_status() was introduced in OpenSSL 0.9.5 */
|
random_the_seed(conn);
|
||||||
if(0 == RAND_status())
|
|
||||||
#endif
|
|
||||||
{
|
|
||||||
/* We need to seed the PRNG properly! */
|
|
||||||
#ifdef HAVE_RAND_SCREEN
|
|
||||||
/* This one gets a random value by reading the currently shown screen */
|
|
||||||
RAND_screen();
|
|
||||||
#else
|
|
||||||
int len;
|
|
||||||
char *area = Curl_FormBoundary();
|
|
||||||
if(!area)
|
|
||||||
return 3; /* out of memory */
|
|
||||||
|
|
||||||
len = strlen(area);
|
|
||||||
|
|
||||||
RAND_seed(area, len);
|
|
||||||
|
|
||||||
free(area); /* now remove the random junk */
|
|
||||||
#endif
|
|
||||||
}
|
|
||||||
|
|
||||||
/* Setup all the global SSL stuff */
|
/* Setup all the global SSL stuff */
|
||||||
SSLeay_add_ssl_algorithms();
|
SSLeay_add_ssl_algorithms();
|
||||||
@@ -202,7 +272,7 @@ Curl_SSLConnect(struct connectdata *conn)
|
|||||||
}
|
}
|
||||||
|
|
||||||
if(data->cert) {
|
if(data->cert) {
|
||||||
if (!cert_stuff(data, conn, data->cert, data->cert)) {
|
if (!cert_stuff(conn, data->cert, data->cert)) {
|
||||||
failf(data, "couldn't use certificate!\n");
|
failf(data, "couldn't use certificate!\n");
|
||||||
return 2;
|
return 2;
|
||||||
}
|
}
|
||||||
@@ -288,7 +358,7 @@ Curl_SSLConnect(struct connectdata *conn)
|
|||||||
X509_free(conn->ssl.server_cert);
|
X509_free(conn->ssl.server_cert);
|
||||||
#else /* USE_SSLEAY */
|
#else /* USE_SSLEAY */
|
||||||
/* this is for "-ansi -Wall -pedantic" to stop complaining! (rabe) */
|
/* this is for "-ansi -Wall -pedantic" to stop complaining! (rabe) */
|
||||||
(void) data;
|
(void) conn;
|
||||||
#endif
|
#endif
|
||||||
return 0;
|
return 0;
|
||||||
}
|
}
|
||||||
|
@@ -25,7 +25,7 @@
|
|||||||
|
|
||||||
#include <string.h>
|
#include <string.h>
|
||||||
|
|
||||||
int Curl_strequal(const char *first, const char *second)
|
int curl_strequal(const char *first, const char *second)
|
||||||
{
|
{
|
||||||
#if defined(HAVE_STRCASECMP)
|
#if defined(HAVE_STRCASECMP)
|
||||||
return !strcasecmp(first, second);
|
return !strcasecmp(first, second);
|
||||||
@@ -45,7 +45,7 @@ int Curl_strequal(const char *first, const char *second)
|
|||||||
#endif
|
#endif
|
||||||
}
|
}
|
||||||
|
|
||||||
int Curl_strnequal(const char *first, const char *second, size_t max)
|
int curl_strnequal(const char *first, const char *second, size_t max)
|
||||||
{
|
{
|
||||||
#if defined(HAVE_STRCASECMP)
|
#if defined(HAVE_STRCASECMP)
|
||||||
return !strncasecmp(first, second, max);
|
return !strncasecmp(first, second, max);
|
||||||
|
@@ -22,10 +22,14 @@
|
|||||||
*
|
*
|
||||||
* $Id$
|
* $Id$
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
int Curl_strequal(const char *first, const char *second);
|
|
||||||
int Curl_strnequal(const char *first, const char *second, size_t max);
|
|
||||||
|
|
||||||
#define strequal(a,b) Curl_strequal(a,b)
|
/*
|
||||||
#define strnequal(a,b,c) Curl_strnequal(a,b,c)
|
* These two actually are public functions.
|
||||||
|
*/
|
||||||
|
int curl_strequal(const char *first, const char *second);
|
||||||
|
int curl_strnequal(const char *first, const char *second, size_t max);
|
||||||
|
|
||||||
|
#define strequal(a,b) curl_strequal(a,b)
|
||||||
|
#define strnequal(a,b,c) curl_strnequal(a,b,c)
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
381
lib/telnet.c
381
lib/telnet.c
@@ -82,18 +82,23 @@
|
|||||||
|
|
||||||
#include "arpa_telnet.h"
|
#include "arpa_telnet.h"
|
||||||
|
|
||||||
|
/* The last #include file should be: */
|
||||||
|
#ifdef MALLOCDEBUG
|
||||||
|
#include "memdebug.h"
|
||||||
|
#endif
|
||||||
|
|
||||||
#define SUBBUFSIZE 512
|
#define SUBBUFSIZE 512
|
||||||
|
|
||||||
#define SB_CLEAR() subpointer = subbuffer;
|
#define SB_CLEAR(x) x->subpointer = x->subbuffer;
|
||||||
#define SB_TERM() { subend = subpointer; SB_CLEAR(); }
|
#define SB_TERM(x) { x->subend = x->subpointer; SB_CLEAR(x); }
|
||||||
#define SB_ACCUM(c) if (subpointer < (subbuffer+sizeof subbuffer)) { \
|
#define SB_ACCUM(x,c) if (x->subpointer < (x->subbuffer+sizeof x->subbuffer)) { \
|
||||||
*subpointer++ = (c); \
|
*x->subpointer++ = (c); \
|
||||||
}
|
}
|
||||||
|
|
||||||
#define SB_GET() ((*subpointer++)&0xff)
|
#define SB_GET(x) ((*x->subpointer++)&0xff)
|
||||||
#define SB_PEEK() ((*subpointer)&0xff)
|
#define SB_PEEK(x) ((*x->subpointer)&0xff)
|
||||||
#define SB_EOF() (subpointer >= subend)
|
#define SB_EOF(x) (x->subpointer >= x->subend)
|
||||||
#define SB_LEN() (subend - subpointer)
|
#define SB_LEN(x) (x->subend - x->subpointer)
|
||||||
|
|
||||||
static
|
static
|
||||||
void telrcv(struct connectdata *,
|
void telrcv(struct connectdata *,
|
||||||
@@ -113,14 +118,19 @@ static void printsub(struct UrlData *data,
|
|||||||
int direction, unsigned char *pointer, int length);
|
int direction, unsigned char *pointer, int length);
|
||||||
static void suboption(struct connectdata *);
|
static void suboption(struct connectdata *);
|
||||||
|
|
||||||
/* suboptions */
|
/* For negotiation compliant to RFC 1143 */
|
||||||
static char subbuffer[SUBBUFSIZE];
|
#define NO 0
|
||||||
static char *subpointer, *subend; /* buffer for sub-options */
|
#define YES 1
|
||||||
|
#define WANTYES 2
|
||||||
|
#define WANTNO 3
|
||||||
|
|
||||||
|
#define EMPTY 0
|
||||||
|
#define OPPOSITE 1
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Telnet receiver states for fsm
|
* Telnet receiver states for fsm
|
||||||
*/
|
*/
|
||||||
static enum
|
typedef enum
|
||||||
{
|
{
|
||||||
TS_DATA = 0,
|
TS_DATA = 0,
|
||||||
TS_IAC,
|
TS_IAC,
|
||||||
@@ -131,63 +141,76 @@ static enum
|
|||||||
TS_CR,
|
TS_CR,
|
||||||
TS_SB, /* sub-option collection */
|
TS_SB, /* sub-option collection */
|
||||||
TS_SE /* looking for sub-option end */
|
TS_SE /* looking for sub-option end */
|
||||||
} telrcv_state;
|
} TelnetReceive;
|
||||||
|
|
||||||
/* For negotiation compliant to RFC 1143 */
|
struct TELNET {
|
||||||
#define NO 0
|
int please_negotiate;
|
||||||
#define YES 1
|
int already_negotiated;
|
||||||
#define WANTYES 2
|
int us[256];
|
||||||
#define WANTNO 3
|
int usq[256];
|
||||||
|
int us_preferred[256];
|
||||||
|
int him[256];
|
||||||
|
int himq[256];
|
||||||
|
int him_preferred[256];
|
||||||
|
char *subopt_ttype; /* Set with suboption TTYPE */
|
||||||
|
char *subopt_xdisploc; /* Set with suboption XDISPLOC */
|
||||||
|
struct curl_slist *telnet_vars; /* Environment variables */
|
||||||
|
|
||||||
#define EMPTY 0
|
/* suboptions */
|
||||||
#define OPPOSITE 1
|
char subbuffer[SUBBUFSIZE];
|
||||||
|
char *subpointer, *subend; /* buffer for sub-options */
|
||||||
static int please_negotiate = 0;
|
|
||||||
static int already_negotiated = 0;
|
TelnetReceive telrcv_state;
|
||||||
static int us[256];
|
};
|
||||||
static int usq[256];
|
|
||||||
static int us_preferred[256];
|
|
||||||
static int him[256];
|
|
||||||
static int himq[256];
|
|
||||||
static int him_preferred[256];
|
|
||||||
|
|
||||||
static char *subopt_ttype = NULL; /* Set with suboption TTYPE */
|
|
||||||
static char *subopt_xdisploc = NULL; /* Set with suboption XDISPLOC */
|
|
||||||
static struct curl_slist *telnet_vars = NULL; /* Environment variables */
|
|
||||||
|
|
||||||
static
|
static
|
||||||
void init_telnet(struct connectdata *conn)
|
CURLcode init_telnet(struct connectdata *conn)
|
||||||
{
|
{
|
||||||
telrcv_state = TS_DATA;
|
struct TELNET *tn;
|
||||||
|
|
||||||
|
tn = (struct TELNET *)malloc(sizeof(struct TELNET));
|
||||||
|
if(!tn)
|
||||||
|
return CURLE_OUT_OF_MEMORY;
|
||||||
|
|
||||||
|
conn->proto.telnet = (void *)tn; /* make us known */
|
||||||
|
|
||||||
|
memset(tn, 0, sizeof(struct TELNET));
|
||||||
|
|
||||||
|
tn->telrcv_state = TS_DATA;
|
||||||
|
|
||||||
/* Init suboptions */
|
/* Init suboptions */
|
||||||
SB_CLEAR();
|
SB_CLEAR(tn);
|
||||||
|
|
||||||
/* Set all options to NO */
|
/* Set all options to NO */
|
||||||
memset(us, NO, 256);
|
#if 0
|
||||||
memset(usq, NO, 256);
|
/* NO is zero => default fill pattern */
|
||||||
memset(us_preferred, NO, 256);
|
memset(tn->us, NO, 256);
|
||||||
memset(him, NO, 256);
|
memset(tn->usq, NO, 256);
|
||||||
memset(himq, NO, 256);
|
memset(tn->us_preferred, NO, 256);
|
||||||
memset(him_preferred, NO, 256);
|
memset(tn->him, NO, 256);
|
||||||
|
memset(tn->himq, NO, 256);
|
||||||
|
memset(tn->him_preferred, NO, 256);
|
||||||
|
#endif
|
||||||
/* Set the options we want by default */
|
/* Set the options we want by default */
|
||||||
us_preferred[TELOPT_BINARY] = YES;
|
tn->us_preferred[TELOPT_BINARY] = YES;
|
||||||
us_preferred[TELOPT_SGA] = YES;
|
tn->us_preferred[TELOPT_SGA] = YES;
|
||||||
him_preferred[TELOPT_BINARY] = YES;
|
tn->him_preferred[TELOPT_BINARY] = YES;
|
||||||
him_preferred[TELOPT_SGA] = YES;
|
tn->him_preferred[TELOPT_SGA] = YES;
|
||||||
|
|
||||||
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
static void negotiate(struct connectdata *conn)
|
static void negotiate(struct connectdata *conn)
|
||||||
{
|
{
|
||||||
int i;
|
int i;
|
||||||
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
|
||||||
for(i = 0;i < NTELOPTS;i++)
|
for(i = 0;i < NTELOPTS;i++)
|
||||||
{
|
{
|
||||||
if(us_preferred[i] == YES)
|
if(tn->us_preferred[i] == YES)
|
||||||
set_local_option(conn, i, YES);
|
set_local_option(conn, i, YES);
|
||||||
|
|
||||||
if(him_preferred[i] == YES)
|
if(tn->him_preferred[i] == YES)
|
||||||
set_remote_option(conn, i, YES);
|
set_remote_option(conn, i, YES);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -247,12 +270,13 @@ static void send_negotiation(struct connectdata *conn, int cmd, int option)
|
|||||||
static
|
static
|
||||||
void set_remote_option(struct connectdata *conn, int option, int newstate)
|
void set_remote_option(struct connectdata *conn, int option, int newstate)
|
||||||
{
|
{
|
||||||
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
if(newstate == YES)
|
if(newstate == YES)
|
||||||
{
|
{
|
||||||
switch(him[option])
|
switch(tn->him[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
him[option] = WANTYES;
|
tn->him[option] = WANTYES;
|
||||||
send_negotiation(conn, DO, option);
|
send_negotiation(conn, DO, option);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -261,11 +285,11 @@ void set_remote_option(struct connectdata *conn, int option, int newstate)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Already negotiating for YES, queue the request */
|
/* Already negotiating for YES, queue the request */
|
||||||
himq[option] = OPPOSITE;
|
tn->himq[option] = OPPOSITE;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
/* Error: already queued an enable request */
|
/* Error: already queued an enable request */
|
||||||
@@ -274,13 +298,13 @@ void set_remote_option(struct connectdata *conn, int option, int newstate)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Error: already negotiating for enable */
|
/* Error: already negotiating for enable */
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -288,34 +312,34 @@ void set_remote_option(struct connectdata *conn, int option, int newstate)
|
|||||||
}
|
}
|
||||||
else /* NO */
|
else /* NO */
|
||||||
{
|
{
|
||||||
switch(him[option])
|
switch(tn->him[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
/* Already disabled */
|
/* Already disabled */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case YES:
|
case YES:
|
||||||
him[option] = WANTNO;
|
tn->him[option] = WANTNO;
|
||||||
send_negotiation(conn, DONT, option);
|
send_negotiation(conn, DONT, option);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Already negotiating for NO */
|
/* Already negotiating for NO */
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
himq[option] = OPPOSITE;
|
tn->himq[option] = OPPOSITE;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
break;
|
break;
|
||||||
@@ -328,12 +352,13 @@ void set_remote_option(struct connectdata *conn, int option, int newstate)
|
|||||||
static
|
static
|
||||||
void rec_will(struct connectdata *conn, int option)
|
void rec_will(struct connectdata *conn, int option)
|
||||||
{
|
{
|
||||||
switch(him[option])
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
switch(tn->him[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
if(him_preferred[option] == YES)
|
if(tn->him_preferred[option] == YES)
|
||||||
{
|
{
|
||||||
him[option] = YES;
|
tn->him[option] = YES;
|
||||||
send_negotiation(conn, DO, option);
|
send_negotiation(conn, DO, option);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -347,29 +372,29 @@ void rec_will(struct connectdata *conn, int option)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Error: DONT answered by WILL */
|
/* Error: DONT answered by WILL */
|
||||||
him[option] = NO;
|
tn->him[option] = NO;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
/* Error: DONT answered by WILL */
|
/* Error: DONT answered by WILL */
|
||||||
him[option] = YES;
|
tn->him[option] = YES;
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
him[option] = YES;
|
tn->him[option] = YES;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
him[option] = WANTNO;
|
tn->him[option] = WANTNO;
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
send_negotiation(conn, DONT, option);
|
send_negotiation(conn, DONT, option);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -380,41 +405,42 @@ void rec_will(struct connectdata *conn, int option)
|
|||||||
static
|
static
|
||||||
void rec_wont(struct connectdata *conn, int option)
|
void rec_wont(struct connectdata *conn, int option)
|
||||||
{
|
{
|
||||||
switch(him[option])
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
switch(tn->him[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
/* Already disabled */
|
/* Already disabled */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case YES:
|
case YES:
|
||||||
him[option] = NO;
|
tn->him[option] = NO;
|
||||||
send_negotiation(conn, DONT, option);
|
send_negotiation(conn, DONT, option);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
him[option] = NO;
|
tn->him[option] = NO;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
him[option] = WANTYES;
|
tn->him[option] = WANTYES;
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
send_negotiation(conn, DO, option);
|
send_negotiation(conn, DO, option);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(himq[option])
|
switch(tn->himq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
him[option] = NO;
|
tn->him[option] = NO;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
him[option] = NO;
|
tn->him[option] = NO;
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -423,12 +449,13 @@ void rec_wont(struct connectdata *conn, int option)
|
|||||||
|
|
||||||
void set_local_option(struct connectdata *conn, int option, int newstate)
|
void set_local_option(struct connectdata *conn, int option, int newstate)
|
||||||
{
|
{
|
||||||
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
if(newstate == YES)
|
if(newstate == YES)
|
||||||
{
|
{
|
||||||
switch(us[option])
|
switch(tn->us[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
us[option] = WANTYES;
|
tn->us[option] = WANTYES;
|
||||||
send_negotiation(conn, WILL, option);
|
send_negotiation(conn, WILL, option);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
@@ -437,11 +464,11 @@ void set_local_option(struct connectdata *conn, int option, int newstate)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Already negotiating for YES, queue the request */
|
/* Already negotiating for YES, queue the request */
|
||||||
usq[option] = OPPOSITE;
|
tn->usq[option] = OPPOSITE;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
/* Error: already queued an enable request */
|
/* Error: already queued an enable request */
|
||||||
@@ -450,13 +477,13 @@ void set_local_option(struct connectdata *conn, int option, int newstate)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Error: already negotiating for enable */
|
/* Error: already negotiating for enable */
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
usq[option] = EMPTY;
|
tn->usq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -464,34 +491,34 @@ void set_local_option(struct connectdata *conn, int option, int newstate)
|
|||||||
}
|
}
|
||||||
else /* NO */
|
else /* NO */
|
||||||
{
|
{
|
||||||
switch(us[option])
|
switch(tn->us[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
/* Already disabled */
|
/* Already disabled */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case YES:
|
case YES:
|
||||||
us[option] = WANTNO;
|
tn->us[option] = WANTNO;
|
||||||
send_negotiation(conn, WONT, option);
|
send_negotiation(conn, WONT, option);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Already negotiating for NO */
|
/* Already negotiating for NO */
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
usq[option] = EMPTY;
|
tn->usq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
usq[option] = OPPOSITE;
|
tn->usq[option] = OPPOSITE;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
break;
|
break;
|
||||||
@@ -504,12 +531,13 @@ void set_local_option(struct connectdata *conn, int option, int newstate)
|
|||||||
static
|
static
|
||||||
void rec_do(struct connectdata *conn, int option)
|
void rec_do(struct connectdata *conn, int option)
|
||||||
{
|
{
|
||||||
switch(us[option])
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
switch(tn->us[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
if(us_preferred[option] == YES)
|
if(tn->us_preferred[option] == YES)
|
||||||
{
|
{
|
||||||
us[option] = YES;
|
tn->us[option] = YES;
|
||||||
send_negotiation(conn, WILL, option);
|
send_negotiation(conn, WILL, option);
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
@@ -523,29 +551,29 @@ void rec_do(struct connectdata *conn, int option)
|
|||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
/* Error: DONT answered by WILL */
|
/* Error: DONT answered by WILL */
|
||||||
us[option] = NO;
|
tn->us[option] = NO;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
/* Error: DONT answered by WILL */
|
/* Error: DONT answered by WILL */
|
||||||
us[option] = YES;
|
tn->us[option] = YES;
|
||||||
usq[option] = EMPTY;
|
tn->usq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
us[option] = YES;
|
tn->us[option] = YES;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
us[option] = WANTNO;
|
tn->us[option] = WANTNO;
|
||||||
himq[option] = EMPTY;
|
tn->himq[option] = EMPTY;
|
||||||
send_negotiation(conn, WONT, option);
|
send_negotiation(conn, WONT, option);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -556,41 +584,42 @@ void rec_do(struct connectdata *conn, int option)
|
|||||||
static
|
static
|
||||||
void rec_dont(struct connectdata *conn, int option)
|
void rec_dont(struct connectdata *conn, int option)
|
||||||
{
|
{
|
||||||
switch(us[option])
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
switch(tn->us[option])
|
||||||
{
|
{
|
||||||
case NO:
|
case NO:
|
||||||
/* Already disabled */
|
/* Already disabled */
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case YES:
|
case YES:
|
||||||
us[option] = NO;
|
tn->us[option] = NO;
|
||||||
send_negotiation(conn, WONT, option);
|
send_negotiation(conn, WONT, option);
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTNO:
|
case WANTNO:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
us[option] = NO;
|
tn->us[option] = NO;
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
us[option] = WANTYES;
|
tn->us[option] = WANTYES;
|
||||||
usq[option] = EMPTY;
|
tn->usq[option] = EMPTY;
|
||||||
send_negotiation(conn, WILL, option);
|
send_negotiation(conn, WILL, option);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
|
|
||||||
case WANTYES:
|
case WANTYES:
|
||||||
switch(usq[option])
|
switch(tn->usq[option])
|
||||||
{
|
{
|
||||||
case EMPTY:
|
case EMPTY:
|
||||||
us[option] = NO;
|
tn->us[option] = NO;
|
||||||
break;
|
break;
|
||||||
case OPPOSITE:
|
case OPPOSITE:
|
||||||
us[option] = NO;
|
tn->us[option] = NO;
|
||||||
usq[option] = EMPTY;
|
tn->usq[option] = EMPTY;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -717,16 +746,17 @@ static int check_telnet_options(struct connectdata *conn)
|
|||||||
char option_arg[256];
|
char option_arg[256];
|
||||||
char *buf;
|
char *buf;
|
||||||
struct UrlData *data = conn->data;
|
struct UrlData *data = conn->data;
|
||||||
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
|
||||||
/* Add the user name as an environment variable if it
|
/* Add the user name as an environment variable if it
|
||||||
was given on the command line */
|
was given on the command line */
|
||||||
if(data->bits.user_passwd)
|
if(conn->bits.user_passwd)
|
||||||
{
|
{
|
||||||
char *buf = malloc(256);
|
char *buf = malloc(256);
|
||||||
sprintf(buf, "USER,%s", data->user);
|
sprintf(buf, "USER,%s", data->user);
|
||||||
telnet_vars = curl_slist_append(telnet_vars, buf);
|
tn->telnet_vars = curl_slist_append(tn->telnet_vars, buf);
|
||||||
|
|
||||||
us_preferred[TELOPT_NEW_ENVIRON] = YES;
|
tn->us_preferred[TELOPT_NEW_ENVIRON] = YES;
|
||||||
}
|
}
|
||||||
|
|
||||||
for(head = data->telnet_options; head; head=head->next) {
|
for(head = data->telnet_options; head; head=head->next) {
|
||||||
@@ -735,15 +765,15 @@ static int check_telnet_options(struct connectdata *conn)
|
|||||||
|
|
||||||
/* Terminal type */
|
/* Terminal type */
|
||||||
if(strequal(option_keyword, "TTYPE")) {
|
if(strequal(option_keyword, "TTYPE")) {
|
||||||
subopt_ttype = option_arg;
|
tn->subopt_ttype = option_arg;
|
||||||
us_preferred[TELOPT_TTYPE] = YES;
|
tn->us_preferred[TELOPT_TTYPE] = YES;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
/* Display variable */
|
/* Display variable */
|
||||||
if(strequal(option_keyword, "XDISPLOC")) {
|
if(strequal(option_keyword, "XDISPLOC")) {
|
||||||
subopt_xdisploc = option_arg;
|
tn->subopt_xdisploc = option_arg;
|
||||||
us_preferred[TELOPT_XDISPLOC] = YES;
|
tn->us_preferred[TELOPT_XDISPLOC] = YES;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -752,8 +782,8 @@ static int check_telnet_options(struct connectdata *conn)
|
|||||||
buf = strdup(option_arg);
|
buf = strdup(option_arg);
|
||||||
if(!buf)
|
if(!buf)
|
||||||
return CURLE_OUT_OF_MEMORY;
|
return CURLE_OUT_OF_MEMORY;
|
||||||
telnet_vars = curl_slist_append(telnet_vars, buf);
|
tn->telnet_vars = curl_slist_append(tn->telnet_vars, buf);
|
||||||
us_preferred[TELOPT_NEW_ENVIRON] = YES;
|
tn->us_preferred[TELOPT_NEW_ENVIRON] = YES;
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -773,7 +803,6 @@ static int check_telnet_options(struct connectdata *conn)
|
|||||||
*
|
*
|
||||||
* Look at the sub-option buffer, and try to be helpful to the other
|
* Look at the sub-option buffer, and try to be helpful to the other
|
||||||
* side.
|
* side.
|
||||||
* No suboptions are supported yet.
|
|
||||||
*/
|
*/
|
||||||
|
|
||||||
static void suboption(struct connectdata *conn)
|
static void suboption(struct connectdata *conn)
|
||||||
@@ -786,22 +815,23 @@ static void suboption(struct connectdata *conn)
|
|||||||
char varname[128];
|
char varname[128];
|
||||||
char varval[128];
|
char varval[128];
|
||||||
struct UrlData *data = conn->data;
|
struct UrlData *data = conn->data;
|
||||||
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
|
||||||
printsub(data, '<', (unsigned char *)subbuffer, SB_LEN()+2);
|
printsub(data, '<', (unsigned char *)tn->subbuffer, SB_LEN(tn)+2);
|
||||||
switch (subchar = SB_GET()) {
|
switch (subchar = SB_GET(tn)) {
|
||||||
case TELOPT_TTYPE:
|
case TELOPT_TTYPE:
|
||||||
len = strlen(subopt_ttype) + 4 + 2;
|
len = strlen(tn->subopt_ttype) + 4 + 2;
|
||||||
snprintf((char *)temp, sizeof(temp),
|
snprintf((char *)temp, sizeof(temp),
|
||||||
"%c%c%c%c%s%c%c", IAC, SB, TELOPT_TTYPE,
|
"%c%c%c%c%s%c%c", IAC, SB, TELOPT_TTYPE,
|
||||||
TELQUAL_IS, subopt_ttype, IAC, SE);
|
TELQUAL_IS, tn->subopt_ttype, IAC, SE);
|
||||||
swrite(conn->firstsocket, temp, len);
|
swrite(conn->firstsocket, temp, len);
|
||||||
printsub(data, '>', &temp[2], len-2);
|
printsub(data, '>', &temp[2], len-2);
|
||||||
break;
|
break;
|
||||||
case TELOPT_XDISPLOC:
|
case TELOPT_XDISPLOC:
|
||||||
len = strlen(subopt_xdisploc) + 4 + 2;
|
len = strlen(tn->subopt_xdisploc) + 4 + 2;
|
||||||
snprintf((char *)temp, sizeof(temp),
|
snprintf((char *)temp, sizeof(temp),
|
||||||
"%c%c%c%c%s%c%c", IAC, SB, TELOPT_XDISPLOC,
|
"%c%c%c%c%s%c%c", IAC, SB, TELOPT_XDISPLOC,
|
||||||
TELQUAL_IS, subopt_xdisploc, IAC, SE);
|
TELQUAL_IS, tn->subopt_xdisploc, IAC, SE);
|
||||||
swrite(conn->firstsocket, temp, len);
|
swrite(conn->firstsocket, temp, len);
|
||||||
printsub(data, '>', &temp[2], len-2);
|
printsub(data, '>', &temp[2], len-2);
|
||||||
break;
|
break;
|
||||||
@@ -810,7 +840,7 @@ static void suboption(struct connectdata *conn)
|
|||||||
"%c%c%c%c", IAC, SB, TELOPT_NEW_ENVIRON, TELQUAL_IS);
|
"%c%c%c%c", IAC, SB, TELOPT_NEW_ENVIRON, TELQUAL_IS);
|
||||||
len = 4;
|
len = 4;
|
||||||
|
|
||||||
for(v = telnet_vars;v;v = v->next) {
|
for(v = tn->telnet_vars;v;v = v->next) {
|
||||||
tmplen = (strlen(v->data) + 1);
|
tmplen = (strlen(v->data) + 1);
|
||||||
/* Add the variable only if it fits */
|
/* Add the variable only if it fits */
|
||||||
if(len + tmplen < sizeof(temp)-6) {
|
if(len + tmplen < sizeof(temp)-6) {
|
||||||
@@ -839,15 +869,16 @@ void telrcv(struct connectdata *conn,
|
|||||||
unsigned char c;
|
unsigned char c;
|
||||||
int index = 0;
|
int index = 0;
|
||||||
struct UrlData *data = conn->data;
|
struct UrlData *data = conn->data;
|
||||||
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
|
||||||
while(count--)
|
while(count--)
|
||||||
{
|
{
|
||||||
c = inbuf[index++];
|
c = inbuf[index++];
|
||||||
|
|
||||||
switch (telrcv_state)
|
switch (tn->telrcv_state)
|
||||||
{
|
{
|
||||||
case TS_CR:
|
case TS_CR:
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
if (c == '\0')
|
if (c == '\0')
|
||||||
{
|
{
|
||||||
break; /* Ignore \0 after CR */
|
break; /* Ignore \0 after CR */
|
||||||
@@ -859,12 +890,12 @@ void telrcv(struct connectdata *conn,
|
|||||||
case TS_DATA:
|
case TS_DATA:
|
||||||
if (c == IAC)
|
if (c == IAC)
|
||||||
{
|
{
|
||||||
telrcv_state = TS_IAC;
|
tn->telrcv_state = TS_IAC;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
else if(c == '\r')
|
else if(c == '\r')
|
||||||
{
|
{
|
||||||
telrcv_state = TS_CR;
|
tn->telrcv_state = TS_CR;
|
||||||
}
|
}
|
||||||
|
|
||||||
Curl_client_write(data, CLIENTWRITE_BODY, (char *)&c, 1);
|
Curl_client_write(data, CLIENTWRITE_BODY, (char *)&c, 1);
|
||||||
@@ -875,20 +906,20 @@ void telrcv(struct connectdata *conn,
|
|||||||
switch (c)
|
switch (c)
|
||||||
{
|
{
|
||||||
case WILL:
|
case WILL:
|
||||||
telrcv_state = TS_WILL;
|
tn->telrcv_state = TS_WILL;
|
||||||
continue;
|
continue;
|
||||||
case WONT:
|
case WONT:
|
||||||
telrcv_state = TS_WONT;
|
tn->telrcv_state = TS_WONT;
|
||||||
continue;
|
continue;
|
||||||
case DO:
|
case DO:
|
||||||
telrcv_state = TS_DO;
|
tn->telrcv_state = TS_DO;
|
||||||
continue;
|
continue;
|
||||||
case DONT:
|
case DONT:
|
||||||
telrcv_state = TS_DONT;
|
tn->telrcv_state = TS_DONT;
|
||||||
continue;
|
continue;
|
||||||
case SB:
|
case SB:
|
||||||
SB_CLEAR();
|
SB_CLEAR(tn);
|
||||||
telrcv_state = TS_SB;
|
tn->telrcv_state = TS_SB;
|
||||||
continue;
|
continue;
|
||||||
case IAC:
|
case IAC:
|
||||||
Curl_client_write(data, CLIENTWRITE_BODY, (char *)&c, 1);
|
Curl_client_write(data, CLIENTWRITE_BODY, (char *)&c, 1);
|
||||||
@@ -900,45 +931,45 @@ void telrcv(struct connectdata *conn,
|
|||||||
printoption(data, "RCVD", IAC, c);
|
printoption(data, "RCVD", IAC, c);
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
case TS_WILL:
|
case TS_WILL:
|
||||||
printoption(data, "RCVD", WILL, c);
|
printoption(data, "RCVD", WILL, c);
|
||||||
please_negotiate = 1;
|
tn->please_negotiate = 1;
|
||||||
rec_will(conn, c);
|
rec_will(conn, c);
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
case TS_WONT:
|
case TS_WONT:
|
||||||
printoption(data, "RCVD", WONT, c);
|
printoption(data, "RCVD", WONT, c);
|
||||||
please_negotiate = 1;
|
tn->please_negotiate = 1;
|
||||||
rec_wont(conn, c);
|
rec_wont(conn, c);
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
case TS_DO:
|
case TS_DO:
|
||||||
printoption(data, "RCVD", DO, c);
|
printoption(data, "RCVD", DO, c);
|
||||||
please_negotiate = 1;
|
tn->please_negotiate = 1;
|
||||||
rec_do(conn, c);
|
rec_do(conn, c);
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
case TS_DONT:
|
case TS_DONT:
|
||||||
printoption(data, "RCVD", DONT, c);
|
printoption(data, "RCVD", DONT, c);
|
||||||
please_negotiate = 1;
|
tn->please_negotiate = 1;
|
||||||
rec_dont(conn, c);
|
rec_dont(conn, c);
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
case TS_SB:
|
case TS_SB:
|
||||||
if (c == IAC)
|
if (c == IAC)
|
||||||
{
|
{
|
||||||
telrcv_state = TS_SE;
|
tn->telrcv_state = TS_SE;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
SB_ACCUM(c);
|
SB_ACCUM(tn,c);
|
||||||
}
|
}
|
||||||
continue;
|
continue;
|
||||||
|
|
||||||
@@ -959,27 +990,27 @@ void telrcv(struct connectdata *conn,
|
|||||||
* we terminate the suboption, and process the
|
* we terminate the suboption, and process the
|
||||||
* partial suboption if we can.
|
* partial suboption if we can.
|
||||||
*/
|
*/
|
||||||
SB_ACCUM((unsigned char)IAC);
|
SB_ACCUM(tn, (unsigned char)IAC);
|
||||||
SB_ACCUM(c);
|
SB_ACCUM(tn, c);
|
||||||
subpointer -= 2;
|
tn->subpointer -= 2;
|
||||||
SB_TERM();
|
SB_TERM(tn);
|
||||||
|
|
||||||
printoption(data, "In SUBOPTION processing, RCVD", IAC, c);
|
printoption(data, "In SUBOPTION processing, RCVD", IAC, c);
|
||||||
suboption(conn); /* handle sub-option */
|
suboption(conn); /* handle sub-option */
|
||||||
telrcv_state = TS_IAC;
|
tn->telrcv_state = TS_IAC;
|
||||||
goto process_iac;
|
goto process_iac;
|
||||||
}
|
}
|
||||||
SB_ACCUM(c);
|
SB_ACCUM(tn,c);
|
||||||
telrcv_state = TS_SB;
|
tn->telrcv_state = TS_SB;
|
||||||
}
|
}
|
||||||
else
|
else
|
||||||
{
|
{
|
||||||
SB_ACCUM((unsigned char)IAC);
|
SB_ACCUM(tn, (unsigned char)IAC);
|
||||||
SB_ACCUM((unsigned char)SE);
|
SB_ACCUM(tn, (unsigned char)SE);
|
||||||
subpointer -= 2;
|
tn->subpointer -= 2;
|
||||||
SB_TERM();
|
SB_TERM(tn);
|
||||||
suboption(conn); /* handle sub-option */
|
suboption(conn); /* handle sub-option */
|
||||||
telrcv_state = TS_DATA;
|
tn->telrcv_state = TS_DATA;
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -988,7 +1019,12 @@ void telrcv(struct connectdata *conn,
|
|||||||
|
|
||||||
CURLcode Curl_telnet_done(struct connectdata *conn)
|
CURLcode Curl_telnet_done(struct connectdata *conn)
|
||||||
{
|
{
|
||||||
curl_slist_free_all(telnet_vars);
|
struct TELNET *tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
curl_slist_free_all(tn->telnet_vars);
|
||||||
|
|
||||||
|
free(conn->proto.telnet);
|
||||||
|
conn->proto.telnet = NULL;
|
||||||
|
|
||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -1003,9 +1039,14 @@ CURLcode Curl_telnet(struct connectdata *conn)
|
|||||||
bool keepon = TRUE;
|
bool keepon = TRUE;
|
||||||
char *buf = data->buffer;
|
char *buf = data->buffer;
|
||||||
ssize_t nread;
|
ssize_t nread;
|
||||||
|
struct TELNET *tn;
|
||||||
|
|
||||||
|
code = init_telnet(conn);
|
||||||
|
if(code)
|
||||||
|
return code;
|
||||||
|
|
||||||
|
tn = (struct TELNET *)conn->proto.telnet;
|
||||||
|
|
||||||
init_telnet(conn);
|
|
||||||
|
|
||||||
code = check_telnet_options(conn);
|
code = check_telnet_options(conn);
|
||||||
if(code)
|
if(code)
|
||||||
return code;
|
return code;
|
||||||
@@ -1060,9 +1101,9 @@ CURLcode Curl_telnet(struct connectdata *conn)
|
|||||||
/* Negotiate if the peer has started negotiating,
|
/* Negotiate if the peer has started negotiating,
|
||||||
otherwise don't. We don't want to speak telnet with
|
otherwise don't. We don't want to speak telnet with
|
||||||
non-telnet servers, like POP or SMTP. */
|
non-telnet servers, like POP or SMTP. */
|
||||||
if(please_negotiate && !already_negotiated) {
|
if(tn->please_negotiate && !tn->already_negotiated) {
|
||||||
negotiate(conn);
|
negotiate(conn);
|
||||||
already_negotiated = 1;
|
tn->already_negotiated = 1;
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@@ -53,7 +53,7 @@ gettimeofday (struct timeval *tp, void *nothing)
|
|||||||
#endif
|
#endif
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
struct timeval Curl_tvnow ()
|
struct timeval Curl_tvnow (void)
|
||||||
{
|
{
|
||||||
struct timeval now;
|
struct timeval now;
|
||||||
#ifdef HAVE_GETTIMEOFDAY
|
#ifdef HAVE_GETTIMEOFDAY
|
||||||
|
276
lib/transfer.c
276
lib/transfer.c
@@ -89,6 +89,7 @@
|
|||||||
#include "getpass.h"
|
#include "getpass.h"
|
||||||
#include "progress.h"
|
#include "progress.h"
|
||||||
#include "getdate.h"
|
#include "getdate.h"
|
||||||
|
#include "http.h"
|
||||||
|
|
||||||
#define _MPRINTF_REPLACE /* use our functions only */
|
#define _MPRINTF_REPLACE /* use our functions only */
|
||||||
#include <curl/mprintf.h>
|
#include <curl/mprintf.h>
|
||||||
@@ -106,7 +107,7 @@
|
|||||||
<butlerm@xmission.com>. */
|
<butlerm@xmission.com>. */
|
||||||
|
|
||||||
CURLcode static
|
CURLcode static
|
||||||
_Transfer(struct connectdata *c_conn)
|
Transfer(struct connectdata *c_conn)
|
||||||
{
|
{
|
||||||
ssize_t nread; /* number of bytes read */
|
ssize_t nread; /* number of bytes read */
|
||||||
int bytecount = 0; /* total number of bytes read */
|
int bytecount = 0; /* total number of bytes read */
|
||||||
@@ -126,7 +127,7 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
bool content_range = FALSE; /* set TRUE if Content-Range: was found */
|
bool content_range = FALSE; /* set TRUE if Content-Range: was found */
|
||||||
int offset = 0; /* possible resume offset read from the
|
int offset = 0; /* possible resume offset read from the
|
||||||
Content-Range: header */
|
Content-Range: header */
|
||||||
int code = 0; /* error code from the 'HTTP/1.? XXX' line */
|
int httpcode = 0; /* error code from the 'HTTP/1.? XXX' line */
|
||||||
int httpversion = -1; /* the last digit in the HTTP/1.1 string */
|
int httpversion = -1; /* the last digit in the HTTP/1.1 string */
|
||||||
|
|
||||||
/* for the low speed checks: */
|
/* for the low speed checks: */
|
||||||
@@ -141,9 +142,6 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
char *buf;
|
char *buf;
|
||||||
int maxfd;
|
int maxfd;
|
||||||
|
|
||||||
if(!conn || (conn->handle != STRUCT_CONNECT))
|
|
||||||
return CURLE_BAD_FUNCTION_ARGUMENT;
|
|
||||||
|
|
||||||
data = conn->data; /* there's the root struct */
|
data = conn->data; /* there's the root struct */
|
||||||
buf = data->buffer;
|
buf = data->buffer;
|
||||||
maxfd = (conn->sockfd>conn->writesockfd?conn->sockfd:conn->writesockfd)+1;
|
maxfd = (conn->sockfd>conn->writesockfd?conn->sockfd:conn->writesockfd)+1;
|
||||||
@@ -183,7 +181,7 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
int keepon=0;
|
int keepon=0;
|
||||||
|
|
||||||
/* timeout every X second
|
/* timeout every X second
|
||||||
- makes a better progressmeter (i.e even when no data is read, the
|
- makes a better progress meter (i.e even when no data is read, the
|
||||||
meter can be updated and reflect reality)
|
meter can be updated and reflect reality)
|
||||||
- allows removal of the alarm() crap
|
- allows removal of the alarm() crap
|
||||||
- variable timeout is easier
|
- variable timeout is easier
|
||||||
@@ -312,8 +310,11 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
/* we now have a full line that p points to */
|
/* we now have a full line that p points to */
|
||||||
if (('\n' == *p) || ('\r' == *p)) {
|
if (('\n' == *p) || ('\r' == *p)) {
|
||||||
/* Zero-length line means end of header! */
|
/* Zero-length line means end of header! */
|
||||||
|
#if 0
|
||||||
if (-1 != conn->size) /* if known */
|
if (-1 != conn->size) /* if known */
|
||||||
conn->size += bytecount; /* we append the already read size */
|
conn->size += bytecount; /* we append the already read
|
||||||
|
size */
|
||||||
|
#endif
|
||||||
|
|
||||||
|
|
||||||
if ('\r' == *p)
|
if ('\r' == *p)
|
||||||
@@ -323,7 +324,20 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
#if 0 /* headers are not included in the size */
|
#if 0 /* headers are not included in the size */
|
||||||
Curl_pgrsSetDownloadSize(data, conn->size);
|
Curl_pgrsSetDownloadSize(data, conn->size);
|
||||||
#endif
|
#endif
|
||||||
header = FALSE; /* no more header to parse! */
|
|
||||||
|
if(100 == httpcode) {
|
||||||
|
/*
|
||||||
|
* we have made a HTTP PUT or POST and this is 1.1-lingo
|
||||||
|
* that tells us that the server is OK with this and ready
|
||||||
|
* to receive our stuff.
|
||||||
|
* However, we'll get more headers now so we must get
|
||||||
|
* back into the header-parsing state!
|
||||||
|
*/
|
||||||
|
header = TRUE;
|
||||||
|
headerline = 0; /* we restart the header line counter */
|
||||||
|
}
|
||||||
|
else
|
||||||
|
header = FALSE; /* no more header to parse! */
|
||||||
|
|
||||||
/* now, only output this if the header AND body are requested:
|
/* now, only output this if the header AND body are requested:
|
||||||
*/
|
*/
|
||||||
@@ -337,18 +351,38 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
return urg;
|
return urg;
|
||||||
|
|
||||||
data->header_size += p - data->headerbuff;
|
data->header_size += p - data->headerbuff;
|
||||||
break; /* exit header line loop */
|
|
||||||
|
if(!header) {
|
||||||
|
/*
|
||||||
|
* end-of-headers.
|
||||||
|
*
|
||||||
|
* If we requested a "no body", this is a good time to get
|
||||||
|
* out and return home.
|
||||||
|
*/
|
||||||
|
if(data->bits.no_body)
|
||||||
|
return CURLE_OK;
|
||||||
|
break; /* exit header line loop */
|
||||||
|
}
|
||||||
|
|
||||||
|
/* We continue reading headers, so reset the line-based
|
||||||
|
header parsing variables hbufp && hbuflen */
|
||||||
|
hbufp = data->headerbuff;
|
||||||
|
hbuflen = 0;
|
||||||
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
if (!headerline++) {
|
if (!headerline++) {
|
||||||
/* This is the first header, it MUST be the error code line
|
/* This is the first header, it MUST be the error code line
|
||||||
or else we consiser this to be the body right away! */
|
or else we consiser this to be the body right away! */
|
||||||
if (2 == sscanf (p, " HTTP/1.%d %3d", &httpversion, &code)) {
|
if (2 == sscanf (p, " HTTP/1.%d %3d", &httpversion,
|
||||||
|
&httpcode)) {
|
||||||
/* 404 -> URL not found! */
|
/* 404 -> URL not found! */
|
||||||
if (
|
if (
|
||||||
( ((data->bits.http_follow_location) && (code >= 400))
|
( ((data->bits.http_follow_location) &&
|
||||||
|
(httpcode >= 400))
|
||||||
||
|
||
|
||||||
(!data->bits.http_follow_location && (code >= 300)))
|
(!data->bits.http_follow_location &&
|
||||||
|
(httpcode >= 300)))
|
||||||
&& (data->bits.http_fail_on_error)) {
|
&& (data->bits.http_fail_on_error)) {
|
||||||
/* If we have been told to fail hard on HTTP-errors,
|
/* If we have been told to fail hard on HTTP-errors,
|
||||||
here is the check for that: */
|
here is the check for that: */
|
||||||
@@ -356,7 +390,13 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
failf (data, "The requested file was not found");
|
failf (data, "The requested file was not found");
|
||||||
return CURLE_HTTP_NOT_FOUND;
|
return CURLE_HTTP_NOT_FOUND;
|
||||||
}
|
}
|
||||||
data->progress.httpcode = code;
|
data->progress.httpcode = httpcode;
|
||||||
|
data->progress.httpversion = httpversion;
|
||||||
|
if(httpversion == 0)
|
||||||
|
/* Default action for HTTP/1.0 must be to close, unless
|
||||||
|
we get one of those fancy headers that tell us the
|
||||||
|
server keeps it open for us! */
|
||||||
|
conn->bits.close = TRUE;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
header = FALSE; /* this is not a header line */
|
header = FALSE; /* this is not a header line */
|
||||||
@@ -365,8 +405,35 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
}
|
}
|
||||||
/* check for Content-Length: header lines to get size */
|
/* check for Content-Length: header lines to get size */
|
||||||
if (strnequal("Content-Length", p, 14) &&
|
if (strnequal("Content-Length", p, 14) &&
|
||||||
sscanf (p+14, ": %ld", &contentlength))
|
sscanf (p+14, ": %ld", &contentlength)) {
|
||||||
conn->size = contentlength;
|
conn->size = contentlength;
|
||||||
|
Curl_pgrsSetDownloadSize(data, contentlength);
|
||||||
|
}
|
||||||
|
else if((httpversion == 0) &&
|
||||||
|
conn->bits.httpproxy &&
|
||||||
|
strnequal("Proxy-Connection: keep-alive", p,
|
||||||
|
strlen("Proxy-Connection: keep-alive"))) {
|
||||||
|
/*
|
||||||
|
* When a HTTP/1.0 reply comes when using a proxy, the
|
||||||
|
* 'Proxy-Connection: keep-alive' line tells us the
|
||||||
|
* connection will be kept alive for our pleasure.
|
||||||
|
* Default action for 1.0 is to close.
|
||||||
|
*/
|
||||||
|
conn->bits.close = FALSE; /* don't close when done */
|
||||||
|
infof(data, "HTTP/1.0 proxy connection set to keep alive!\n");
|
||||||
|
}
|
||||||
|
else if((httpversion == 0) &&
|
||||||
|
strnequal("Connection: keep-alive", p,
|
||||||
|
strlen("Connection: keep-alive"))) {
|
||||||
|
/*
|
||||||
|
* A HTTP/1.0 reply with the 'Connection: keep-alive' line
|
||||||
|
* tells us the connection will be kept alive for our
|
||||||
|
* pleasure. Default action for 1.0 is to close.
|
||||||
|
*
|
||||||
|
* [RFC2068, section 19.7.1] */
|
||||||
|
conn->bits.close = FALSE; /* don't close when done */
|
||||||
|
infof(data, "HTTP/1.0 connection set to keep alive!\n");
|
||||||
|
}
|
||||||
else if (strnequal("Connection: close", p,
|
else if (strnequal("Connection: close", p,
|
||||||
strlen("Connection: close"))) {
|
strlen("Connection: close"))) {
|
||||||
/*
|
/*
|
||||||
@@ -377,10 +444,25 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
*/
|
*/
|
||||||
conn->bits.close = TRUE; /* close when done */
|
conn->bits.close = TRUE; /* close when done */
|
||||||
}
|
}
|
||||||
|
else if (strnequal("Transfer-Encoding: chunked", p,
|
||||||
|
strlen("Transfer-Encoding: chunked"))) {
|
||||||
|
/*
|
||||||
|
* [RFC 2616, section 3.6.1] A 'chunked' transfer encoding
|
||||||
|
* means that the server will send a series of "chunks". Each
|
||||||
|
* chunk starts with line with info (including size of the
|
||||||
|
* coming block) (terminated with CRLF), then a block of data
|
||||||
|
* with the previously mentioned size. There can be any amount
|
||||||
|
* of chunks, and a chunk-data set to zero signals the
|
||||||
|
* end-of-chunks. */
|
||||||
|
conn->bits.chunk = TRUE; /* chunks coming our way */
|
||||||
|
|
||||||
|
/* init our chunky engine */
|
||||||
|
Curl_httpchunk_init(conn);
|
||||||
|
}
|
||||||
else if (strnequal("Content-Range", p, 13)) {
|
else if (strnequal("Content-Range", p, 13)) {
|
||||||
if (sscanf (p+13, ": bytes %d-", &offset) ||
|
if (sscanf (p+13, ": bytes %d-", &offset) ||
|
||||||
sscanf (p+13, ": bytes: %d-", &offset)) {
|
sscanf (p+13, ": bytes: %d-", &offset)) {
|
||||||
/* This second format was added August 1st by Igor
|
/* This second format was added August 1st 2000 by Igor
|
||||||
Khristophorov since Sun's webserver JavaWebServer/1.1.1
|
Khristophorov since Sun's webserver JavaWebServer/1.1.1
|
||||||
obviously sends the header this way! :-( */
|
obviously sends the header this way! :-( */
|
||||||
if (data->resume_from == offset) {
|
if (data->resume_from == offset) {
|
||||||
@@ -401,7 +483,7 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
if(data->bits.get_filetime)
|
if(data->bits.get_filetime)
|
||||||
data->progress.filetime = timeofdoc;
|
data->progress.filetime = timeofdoc;
|
||||||
}
|
}
|
||||||
else if ((code >= 300 && code < 400) &&
|
else if ((httpcode >= 300 && httpcode < 400) &&
|
||||||
(data->bits.http_follow_location) &&
|
(data->bits.http_follow_location) &&
|
||||||
strnequal("Location: ", p, 10)) {
|
strnequal("Location: ", p, 10)) {
|
||||||
/* this is the URL that the server advices us to get instead */
|
/* this is the URL that the server advices us to get instead */
|
||||||
@@ -416,7 +498,7 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
ptr++;
|
ptr++;
|
||||||
backup = *ptr; /* store the ending letter */
|
backup = *ptr; /* store the ending letter */
|
||||||
*ptr = '\0'; /* zero terminate */
|
*ptr = '\0'; /* zero terminate */
|
||||||
data->newurl = strdup(start); /* clone string */
|
conn->newurl = strdup(start); /* clone string */
|
||||||
*ptr = backup; /* restore ending letter */
|
*ptr = backup; /* restore ending letter */
|
||||||
}
|
}
|
||||||
|
|
||||||
@@ -457,12 +539,12 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
|
|
||||||
if(0 == bodywrites) {
|
if(0 == bodywrites) {
|
||||||
/* These checks are only made the first time we are about to
|
/* These checks are only made the first time we are about to
|
||||||
write a chunk of the body */
|
write a piece of the body */
|
||||||
if(conn->protocol&PROT_HTTP) {
|
if(conn->protocol&PROT_HTTP) {
|
||||||
/* HTTP-only checks */
|
/* HTTP-only checks */
|
||||||
if (data->newurl) {
|
if (conn->newurl) {
|
||||||
/* abort after the headers if "follow Location" is set */
|
/* abort after the headers if "follow Location" is set */
|
||||||
infof (data, "Follow to new URL: %s\n", data->newurl);
|
infof (data, "Follow to new URL: %s\n", conn->newurl);
|
||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
else if (data->resume_from &&
|
else if (data->resume_from &&
|
||||||
@@ -500,7 +582,8 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
} /* switch */
|
} /* switch */
|
||||||
} /* two valid time strings */
|
} /* two valid time strings */
|
||||||
} /* we have a time condition */
|
} /* we have a time condition */
|
||||||
if(!conn->bits.close && (httpversion == 1)) {
|
|
||||||
|
if(!conn->bits.close) {
|
||||||
/* If this is not the last request before a close, we must
|
/* If this is not the last request before a close, we must
|
||||||
set the maximum download size to the size of the expected
|
set the maximum download size to the size of the expected
|
||||||
document or else, we won't know when to stop reading! */
|
document or else, we won't know when to stop reading! */
|
||||||
@@ -514,6 +597,31 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
} /* this is the first time we write a body part */
|
} /* this is the first time we write a body part */
|
||||||
bodywrites++;
|
bodywrites++;
|
||||||
|
|
||||||
|
if(conn->bits.chunk) {
|
||||||
|
/*
|
||||||
|
* Bless me father for I have sinned. Here comes a chunked
|
||||||
|
* transfer flying and we need to decode this properly. While
|
||||||
|
* the name says read, this function both reads and writes away
|
||||||
|
* the data. The returned 'nread' holds the number of actual
|
||||||
|
* data it wrote to the client. */
|
||||||
|
CHUNKcode res =
|
||||||
|
Curl_httpchunk_read(conn, str, nread, &nread);
|
||||||
|
|
||||||
|
if(CHUNKE_OK < res) {
|
||||||
|
failf(data, "Receeived problem in the chunky parser");
|
||||||
|
return CURLE_READ_ERROR;
|
||||||
|
}
|
||||||
|
else if(CHUNKE_STOP == res) {
|
||||||
|
/* we're done reading chunks! */
|
||||||
|
keepon &= ~KEEP_READ; /* read no more */
|
||||||
|
|
||||||
|
/* There are now possibly N number of bytes at the end of the
|
||||||
|
str buffer that weren't written to the client, but we don't
|
||||||
|
care about them right now. */
|
||||||
|
}
|
||||||
|
/* If it returned OK, we just keep going */
|
||||||
|
}
|
||||||
|
|
||||||
if(conn->maxdownload &&
|
if(conn->maxdownload &&
|
||||||
(bytecount + nread >= conn->maxdownload)) {
|
(bytecount + nread >= conn->maxdownload)) {
|
||||||
nread = conn->maxdownload - bytecount;
|
nread = conn->maxdownload - bytecount;
|
||||||
@@ -526,9 +634,12 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
|
|
||||||
Curl_pgrsSetDownloadCounter(data, (double)bytecount);
|
Curl_pgrsSetDownloadCounter(data, (double)bytecount);
|
||||||
|
|
||||||
urg = Curl_client_write(data, CLIENTWRITE_BODY, str, nread);
|
if(! conn->bits.chunk) {
|
||||||
if(urg)
|
/* If this is chunky transfer, it was already written */
|
||||||
return urg;
|
urg = Curl_client_write(data, CLIENTWRITE_BODY, str, nread);
|
||||||
|
if(urg)
|
||||||
|
return urg;
|
||||||
|
}
|
||||||
|
|
||||||
} /* if (! header and data to read ) */
|
} /* if (! header and data to read ) */
|
||||||
} /* if( read from socket ) */
|
} /* if( read from socket ) */
|
||||||
@@ -614,6 +725,11 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
contentlength-bytecount);
|
contentlength-bytecount);
|
||||||
return CURLE_PARTIAL_FILE;
|
return CURLE_PARTIAL_FILE;
|
||||||
}
|
}
|
||||||
|
else if(conn->bits.chunk && conn->proto.http->chunk.datasize) {
|
||||||
|
failf(data, "transfer closed with at least %d bytes remaining",
|
||||||
|
conn->proto.http->chunk.datasize);
|
||||||
|
return CURLE_PARTIAL_FILE;
|
||||||
|
}
|
||||||
if(Curl_pgrsUpdate(data))
|
if(Curl_pgrsUpdate(data))
|
||||||
return CURLE_ABORTED_BY_CALLBACK;
|
return CURLE_ABORTED_BY_CALLBACK;
|
||||||
|
|
||||||
@@ -625,29 +741,43 @@ _Transfer(struct connectdata *c_conn)
|
|||||||
return CURLE_OK;
|
return CURLE_OK;
|
||||||
}
|
}
|
||||||
|
|
||||||
typedef int (*func_T)(void);
|
CURLcode Curl_perform(CURL *curl)
|
||||||
|
|
||||||
CURLcode curl_transfer(CURL *curl)
|
|
||||||
{
|
{
|
||||||
CURLcode res;
|
CURLcode res;
|
||||||
struct UrlData *data = curl;
|
struct UrlData *data = (struct UrlData *)curl;
|
||||||
struct connectdata *c_connect=NULL;
|
struct connectdata *conn=NULL;
|
||||||
bool port=TRUE; /* allow data->use_port to set port to use */
|
bool port=TRUE; /* allow data->use_port to set port to use */
|
||||||
|
char *newurl = NULL; /* possibly a new URL to follow to! */
|
||||||
|
|
||||||
|
data->followlocation=0; /* reset the location-follow counter */
|
||||||
|
data->bits.this_is_a_follow = FALSE; /* reset this */
|
||||||
|
|
||||||
Curl_pgrsStartNow(data);
|
Curl_pgrsStartNow(data);
|
||||||
|
|
||||||
do {
|
do {
|
||||||
Curl_pgrsTime(data, TIMER_STARTSINGLE);
|
Curl_pgrsTime(data, TIMER_STARTSINGLE);
|
||||||
res = curl_connect(curl, (CURLconnect **)&c_connect, port);
|
res = Curl_connect(data, &conn, port);
|
||||||
if(res == CURLE_OK) {
|
if(res == CURLE_OK) {
|
||||||
res = curl_do(c_connect);
|
res = Curl_do(conn);
|
||||||
if(res == CURLE_OK) {
|
if(res == CURLE_OK) {
|
||||||
res = _Transfer(c_connect); /* now fetch that URL please */
|
res = Transfer(conn); /* now fetch that URL please */
|
||||||
if(res == CURLE_OK)
|
if(res == CURLE_OK) {
|
||||||
res = curl_done(c_connect);
|
/*
|
||||||
|
* We must duplicate the new URL here as the connection data
|
||||||
|
* may be free()ed in the Curl_done() function.
|
||||||
|
*/
|
||||||
|
newurl = conn->newurl?strdup(conn->newurl):NULL;
|
||||||
|
|
||||||
|
res = Curl_done(conn);
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
if((res == CURLE_OK) && data->newurl) {
|
/*
|
||||||
|
* Important: 'conn' cannot be used here, since it may have been closed
|
||||||
|
* in 'Curl_done' or other functions.
|
||||||
|
*/
|
||||||
|
|
||||||
|
if((res == CURLE_OK) && newurl) {
|
||||||
/* Location: redirect
|
/* Location: redirect
|
||||||
|
|
||||||
This is assumed to happen for HTTP(S) only!
|
This is assumed to happen for HTTP(S) only!
|
||||||
@@ -660,9 +790,6 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
|
|
||||||
if (data->maxredirs && (data->followlocation >= data->maxredirs)) {
|
if (data->maxredirs && (data->followlocation >= data->maxredirs)) {
|
||||||
failf(data,"Maximum (%d) redirects followed", data->maxredirs);
|
failf(data,"Maximum (%d) redirects followed", data->maxredirs);
|
||||||
#ifdef USE_OLD_DISCONNECT
|
|
||||||
curl_disconnect(c_connect);
|
|
||||||
#endif
|
|
||||||
res=CURLE_TOO_MANY_REDIRECTS;
|
res=CURLE_TOO_MANY_REDIRECTS;
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
@@ -687,7 +814,7 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
data->bits.http_set_referer = TRUE; /* might have been false */
|
data->bits.http_set_referer = TRUE; /* might have been false */
|
||||||
}
|
}
|
||||||
|
|
||||||
if(2 != sscanf(data->newurl, "%15[^:]://%c", prot, &letter)) {
|
if(2 != sscanf(newurl, "%15[^:]://%c", prot, &letter)) {
|
||||||
/***
|
/***
|
||||||
*DANG* this is an RFC 2068 violation. The URL is supposed
|
*DANG* this is an RFC 2068 violation. The URL is supposed
|
||||||
to be absolute and this doesn't seem to be that!
|
to be absolute and this doesn't seem to be that!
|
||||||
@@ -700,19 +827,21 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
char *pathsep;
|
char *pathsep;
|
||||||
char *newest;
|
char *newest;
|
||||||
|
|
||||||
|
/* we must make our own copy of the URL to play with, as it may
|
||||||
|
point to read-only data */
|
||||||
|
char *url_clone=strdup(data->url);
|
||||||
|
|
||||||
|
if(!url_clone)
|
||||||
|
return CURLE_OUT_OF_MEMORY;
|
||||||
|
|
||||||
/* protsep points to the start of the host name */
|
/* protsep points to the start of the host name */
|
||||||
protsep=strstr(data->url, "//");
|
protsep=strstr(url_clone, "//");
|
||||||
if(!protsep)
|
if(!protsep)
|
||||||
protsep=data->url;
|
protsep=url_clone;
|
||||||
else {
|
else
|
||||||
port=FALSE; /* we got a full URL and thus we should not obey the
|
|
||||||
port number that might have been set by the user
|
|
||||||
in data->use_port */
|
|
||||||
|
|
||||||
protsep+=2; /* pass the slashes */
|
protsep+=2; /* pass the slashes */
|
||||||
}
|
|
||||||
|
|
||||||
if('/' != data->newurl[0]) {
|
if('/' != newurl[0]) {
|
||||||
/* First we need to find out if there's a ?-letter in the URL,
|
/* First we need to find out if there's a ?-letter in the URL,
|
||||||
and cut it and the right-side of that off */
|
and cut it and the right-side of that off */
|
||||||
pathsep = strrchr(protsep, '?');
|
pathsep = strrchr(protsep, '?');
|
||||||
@@ -733,16 +862,17 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
*pathsep=0;
|
*pathsep=0;
|
||||||
}
|
}
|
||||||
|
|
||||||
newest=(char *)malloc( strlen(data->url) +
|
newest=(char *)malloc( strlen(url_clone) +
|
||||||
1 + /* possible slash */
|
1 + /* possible slash */
|
||||||
strlen(data->newurl) + 1/* zero byte */);
|
strlen(newurl) + 1/* zero byte */);
|
||||||
|
|
||||||
if(!newest)
|
if(!newest)
|
||||||
return CURLE_OUT_OF_MEMORY;
|
return CURLE_OUT_OF_MEMORY;
|
||||||
sprintf(newest, "%s%s%s", data->url, ('/' == data->newurl[0])?"":"/",
|
sprintf(newest, "%s%s%s", url_clone, ('/' == newurl[0])?"":"/",
|
||||||
data->newurl);
|
newurl);
|
||||||
free(data->newurl);
|
free(newurl);
|
||||||
data->newurl = newest;
|
free(url_clone);
|
||||||
|
newurl = newest;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
/* This is an absolute URL, don't use the custom port number */
|
/* This is an absolute URL, don't use the custom port number */
|
||||||
@@ -753,8 +883,8 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
free(data->url);
|
free(data->url);
|
||||||
|
|
||||||
/* TBD: set the URL with curl_setopt() */
|
/* TBD: set the URL with curl_setopt() */
|
||||||
data->url = data->newurl;
|
data->url = newurl;
|
||||||
data->newurl = NULL; /* don't show! */
|
|
||||||
data->bits.urlstringalloc = TRUE; /* the URL is allocated */
|
data->bits.urlstringalloc = TRUE; /* the URL is allocated */
|
||||||
|
|
||||||
infof(data, "Follows Location: to new URL: '%s'\n", data->url);
|
infof(data, "Follows Location: to new URL: '%s'\n", data->url);
|
||||||
@@ -768,7 +898,6 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
switch(data->progress.httpcode) {
|
switch(data->progress.httpcode) {
|
||||||
case 300: /* Multiple Choices */
|
case 300: /* Multiple Choices */
|
||||||
case 301: /* Moved Permanently */
|
case 301: /* Moved Permanently */
|
||||||
case 302: /* Found */
|
|
||||||
case 306: /* Not used */
|
case 306: /* Not used */
|
||||||
case 307: /* Temporary Redirect */
|
case 307: /* Temporary Redirect */
|
||||||
default: /* for all unknown ones */
|
default: /* for all unknown ones */
|
||||||
@@ -776,6 +905,24 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
* seem to be OK to POST to.
|
* seem to be OK to POST to.
|
||||||
*/
|
*/
|
||||||
break;
|
break;
|
||||||
|
case 302: /* Found */
|
||||||
|
/* (From 10.3.3)
|
||||||
|
|
||||||
|
Note: RFC 1945 and RFC 2068 specify that the client is not allowed
|
||||||
|
to change the method on the redirected request. However, most
|
||||||
|
existing user agent implementations treat 302 as if it were a 303
|
||||||
|
response, performing a GET on the Location field-value regardless
|
||||||
|
of the original request method. The status codes 303 and 307 have
|
||||||
|
been added for servers that wish to make unambiguously clear which
|
||||||
|
kind of reaction is expected of the client.
|
||||||
|
|
||||||
|
(From 10.3.4)
|
||||||
|
|
||||||
|
Note: Many pre-HTTP/1.1 user agents do not understand the 303
|
||||||
|
status. When interoperability with such clients is a concern, the
|
||||||
|
302 status code may be used instead, since most user agents react
|
||||||
|
to a 302 response as described here for 303.
|
||||||
|
*/
|
||||||
case 303: /* See Other */
|
case 303: /* See Other */
|
||||||
/* Disable both types of POSTs, since doing a second POST when
|
/* Disable both types of POSTs, since doing a second POST when
|
||||||
* following isn't what anyone would want! */
|
* following isn't what anyone would want! */
|
||||||
@@ -799,22 +946,19 @@ CURLcode curl_transfer(CURL *curl)
|
|||||||
*/
|
*/
|
||||||
break;
|
break;
|
||||||
}
|
}
|
||||||
#ifdef USE_OLD_DISCONNECT
|
|
||||||
curl_disconnect(c_connect);
|
|
||||||
#endif
|
|
||||||
continue;
|
continue;
|
||||||
}
|
}
|
||||||
|
|
||||||
#ifdef USE_OLD_DISCONNECT
|
|
||||||
curl_disconnect(c_connect);
|
|
||||||
#endif
|
|
||||||
}
|
}
|
||||||
break; /* it only reaches here when this shouldn't loop */
|
break; /* it only reaches here when this shouldn't loop */
|
||||||
|
|
||||||
} while(1); /* loop if Location: */
|
} while(1); /* loop if Location: */
|
||||||
|
|
||||||
if(data->newurl)
|
if(newurl)
|
||||||
free(data->newurl);
|
free(newurl);
|
||||||
|
|
||||||
|
/* make sure the alarm is switched off! */
|
||||||
|
if(data->timeout || data->connecttimeout)
|
||||||
|
myalarm(0);
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
@@ -22,8 +22,9 @@
|
|||||||
*
|
*
|
||||||
* $Id$
|
* $Id$
|
||||||
*****************************************************************************/
|
*****************************************************************************/
|
||||||
CURLcode curl_transfer(CURL *curl);
|
CURLcode Curl_perform(CURL *curl);
|
||||||
|
|
||||||
|
/* This sets up a forthcoming transfer */
|
||||||
CURLcode
|
CURLcode
|
||||||
Curl_Transfer (struct connectdata *data,
|
Curl_Transfer (struct connectdata *data,
|
||||||
int sockfd, /* socket to read from or -1 */
|
int sockfd, /* socket to read from or -1 */
|
||||||
|
282
lib/urldata.h
282
lib/urldata.h
@@ -79,6 +79,8 @@
|
|||||||
|
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
|
|
||||||
|
#include "http_chunks.h" /* for the structs and enum stuff */
|
||||||
|
|
||||||
/* Download buffer size, keep it fairly big for speed reasons */
|
/* Download buffer size, keep it fairly big for speed reasons */
|
||||||
#define BUFSIZE (1024*50)
|
#define BUFSIZE (1024*50)
|
||||||
|
|
||||||
@@ -96,27 +98,6 @@
|
|||||||
#define MAX(x,y) ((x)>(y)?(x):(y))
|
#define MAX(x,y) ((x)>(y)?(x):(y))
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
/* Type of handle. All publicly returned 'handles' in the curl interface
|
|
||||||
have a handle first in the struct that describes what kind of handle it
|
|
||||||
is. Used to detect bad handle usage. */
|
|
||||||
typedef enum {
|
|
||||||
STRUCT_NONE,
|
|
||||||
STRUCT_OPEN,
|
|
||||||
STRUCT_CONNECT,
|
|
||||||
STRUCT_LAST
|
|
||||||
} Handle;
|
|
||||||
|
|
||||||
/* Connecting to a remote server using the curl interface is moving through
|
|
||||||
a state machine, this type is used to store the current state */
|
|
||||||
typedef enum {
|
|
||||||
CONN_NONE, /* illegal state */
|
|
||||||
CONN_INIT, /* curl_connect() has been called */
|
|
||||||
CONN_DO, /* curl_do() has been called successfully */
|
|
||||||
CONN_DONE, /* curl_done() has been called successfully */
|
|
||||||
CONN_ERROR, /* and error has occurred */
|
|
||||||
CONN_LAST /* illegal state */
|
|
||||||
} ConnState;
|
|
||||||
|
|
||||||
#ifdef KRB4
|
#ifdef KRB4
|
||||||
/* Types needed for krb4-ftp connections */
|
/* Types needed for krb4-ftp connections */
|
||||||
struct krb4buffer {
|
struct krb4buffer {
|
||||||
@@ -150,6 +131,8 @@ struct ssl_config_data {
|
|||||||
long verifypeer; /* set TRUE if this is desired */
|
long verifypeer; /* set TRUE if this is desired */
|
||||||
char *CApath; /* DOES NOT WORK ON WINDOWS */
|
char *CApath; /* DOES NOT WORK ON WINDOWS */
|
||||||
char *CAfile; /* cerficate to verify peer against */
|
char *CAfile; /* cerficate to verify peer against */
|
||||||
|
char *random_file; /* path to file containing "random" data */
|
||||||
|
char *egdsocket; /* path to file containing the EGD daemon socket */
|
||||||
};
|
};
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@@ -167,6 +150,8 @@ struct HTTP {
|
|||||||
struct Form form;
|
struct Form form;
|
||||||
size_t (*storefread)(char *, size_t , size_t , FILE *);
|
size_t (*storefread)(char *, size_t , size_t , FILE *);
|
||||||
FILE *in;
|
FILE *in;
|
||||||
|
|
||||||
|
struct Curl_chunker chunk;
|
||||||
};
|
};
|
||||||
|
|
||||||
/****************************************************************************
|
/****************************************************************************
|
||||||
@@ -183,12 +168,23 @@ struct FTP {
|
|||||||
char *entrypath; /* the PWD reply when we logged on */
|
char *entrypath; /* the PWD reply when we logged on */
|
||||||
};
|
};
|
||||||
|
|
||||||
|
/****************************************************************************
|
||||||
|
* FILE unique setup
|
||||||
|
***************************************************************************/
|
||||||
|
struct FILE {
|
||||||
|
int fd; /* open file descriptor to read from! */
|
||||||
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Boolean values that concerns this connection.
|
* Boolean values that concerns this connection.
|
||||||
*/
|
*/
|
||||||
struct ConnectBits {
|
struct ConnectBits {
|
||||||
bool close; /* if set, we close the connection after this request */
|
bool close; /* if set, we close the connection after this request */
|
||||||
bool reuse; /* if set, this is a re-used connection */
|
bool reuse; /* if set, this is a re-used connection */
|
||||||
|
bool chunk; /* if set, this is a chunked transfer-encoding */
|
||||||
|
bool httpproxy; /* if set, this transfer is done through a http proxy */
|
||||||
|
bool user_passwd; /* do we use user+password for this connection? */
|
||||||
|
bool proxy_user_passwd; /* user+password for the proxy? */
|
||||||
};
|
};
|
||||||
|
|
||||||
/*
|
/*
|
||||||
@@ -197,18 +193,10 @@ struct ConnectBits {
|
|||||||
*/
|
*/
|
||||||
struct connectdata {
|
struct connectdata {
|
||||||
/**** Fields set when inited and not modified again */
|
/**** Fields set when inited and not modified again */
|
||||||
|
|
||||||
/* To better see what kind of struct that is passed as input, *ALL* publicly
|
|
||||||
returned handles MUST have this initial 'Handle'. */
|
|
||||||
Handle handle; /* struct identifier */
|
|
||||||
struct UrlData *data; /* link to the root CURL struct */
|
struct UrlData *data; /* link to the root CURL struct */
|
||||||
|
|
||||||
int connectindex; /* what index in the connects index this particular
|
int connectindex; /* what index in the connects index this particular
|
||||||
struct has */
|
struct has */
|
||||||
|
|
||||||
/**** curl_connect() phase fields */
|
|
||||||
ConnState state; /* for state dependent actions */
|
|
||||||
|
|
||||||
long protocol; /* PROT_* flags concerning the protocol set */
|
long protocol; /* PROT_* flags concerning the protocol set */
|
||||||
#define PROT_MISSING (1<<0)
|
#define PROT_MISSING (1<<0)
|
||||||
#define PROT_GOPHER (1<<1)
|
#define PROT_GOPHER (1<<1)
|
||||||
@@ -219,6 +207,7 @@ struct connectdata {
|
|||||||
#define PROT_DICT (1<<6)
|
#define PROT_DICT (1<<6)
|
||||||
#define PROT_LDAP (1<<7)
|
#define PROT_LDAP (1<<7)
|
||||||
#define PROT_FILE (1<<8)
|
#define PROT_FILE (1<<8)
|
||||||
|
#define PROT_FTPS (1<<9)
|
||||||
|
|
||||||
#ifdef ENABLE_IPV6
|
#ifdef ENABLE_IPV6
|
||||||
struct addrinfo *hp; /* host info pointer list */
|
struct addrinfo *hp; /* host info pointer list */
|
||||||
@@ -238,7 +227,11 @@ struct connectdata {
|
|||||||
not the proxy port! */
|
not the proxy port! */
|
||||||
char *ppath;
|
char *ppath;
|
||||||
long bytecount;
|
long bytecount;
|
||||||
struct timeval now; /* current time */
|
|
||||||
|
char *proxyhost; /* name of the http proxy host */
|
||||||
|
|
||||||
|
struct timeval now; /* "current" time */
|
||||||
|
struct timeval created; /* creation time */
|
||||||
int firstsocket; /* the main socket to use */
|
int firstsocket; /* the main socket to use */
|
||||||
int secondarysocket; /* for i.e ftp transfers */
|
int secondarysocket; /* for i.e ftp transfers */
|
||||||
|
|
||||||
@@ -297,8 +290,10 @@ struct connectdata {
|
|||||||
char *host; /* free later if not NULL */
|
char *host; /* free later if not NULL */
|
||||||
} allocptr;
|
} allocptr;
|
||||||
|
|
||||||
#ifdef KRB4
|
char *newurl; /* This can only be set if a Location: was in the
|
||||||
|
document headers */
|
||||||
|
|
||||||
|
#ifdef KRB4
|
||||||
enum protection_level command_prot;
|
enum protection_level command_prot;
|
||||||
enum protection_level data_prot;
|
enum protection_level data_prot;
|
||||||
enum protection_level request_data_prot;
|
enum protection_level request_data_prot;
|
||||||
@@ -318,9 +313,9 @@ struct connectdata {
|
|||||||
struct HTTP *gopher; /* alias, just for the sake of being more readable */
|
struct HTTP *gopher; /* alias, just for the sake of being more readable */
|
||||||
struct HTTP *https; /* alias, just for the sake of being more readable */
|
struct HTTP *https; /* alias, just for the sake of being more readable */
|
||||||
struct FTP *ftp;
|
struct FTP *ftp;
|
||||||
#if 0 /* no need for special ones for these: */
|
|
||||||
struct TELNET *telnet;
|
|
||||||
struct FILE *file;
|
struct FILE *file;
|
||||||
|
void *telnet; /* private for telnet.c-eyes only */
|
||||||
|
#if 0 /* no need for special ones for these: */
|
||||||
struct LDAP *ldap;
|
struct LDAP *ldap;
|
||||||
struct DICT *dict;
|
struct DICT *dict;
|
||||||
#endif
|
#endif
|
||||||
@@ -354,6 +349,7 @@ struct Progress {
|
|||||||
double t_connect;
|
double t_connect;
|
||||||
double t_pretransfer;
|
double t_pretransfer;
|
||||||
int httpcode;
|
int httpcode;
|
||||||
|
int httpversion;
|
||||||
time_t filetime; /* If requested, this is might get set. It may be 0 if
|
time_t filetime; /* If requested, this is might get set. It may be 0 if
|
||||||
the time was unretrievable */
|
the time was unretrievable */
|
||||||
|
|
||||||
@@ -397,28 +393,22 @@ struct Configbits {
|
|||||||
bool httpproxy;
|
bool httpproxy;
|
||||||
bool mute;
|
bool mute;
|
||||||
bool no_body;
|
bool no_body;
|
||||||
bool proxy_user_passwd;
|
|
||||||
bool set_port;
|
bool set_port;
|
||||||
bool set_range;
|
bool set_range;
|
||||||
bool upload;
|
bool upload;
|
||||||
bool use_netrc;
|
bool use_netrc;
|
||||||
bool user_passwd;
|
|
||||||
bool verbose;
|
bool verbose;
|
||||||
bool this_is_a_follow; /* this is a followed Location: request */
|
bool this_is_a_follow; /* this is a followed Location: request */
|
||||||
bool krb4; /* kerberos4 connection requested */
|
bool krb4; /* kerberos4 connection requested */
|
||||||
bool proxystringalloc; /* the http proxy string is malloc()'ed */
|
bool proxystringalloc; /* the http proxy string is malloc()'ed */
|
||||||
bool rangestringalloc; /* the range string is malloc()'ed */
|
bool rangestringalloc; /* the range string is malloc()'ed */
|
||||||
bool urlstringalloc; /* the URL string is malloc()'ed */
|
bool urlstringalloc; /* the URL string is malloc()'ed */
|
||||||
|
bool reuse_forbid; /* if this is forbidden to be reused, close
|
||||||
|
after use */
|
||||||
|
bool reuse_fresh; /* do not re-use an existing connection for this
|
||||||
|
transfer */
|
||||||
};
|
};
|
||||||
|
|
||||||
/* What type of interface that intiated this struct */
|
|
||||||
typedef enum {
|
|
||||||
CURLI_NONE,
|
|
||||||
CURLI_EASY,
|
|
||||||
CURLI_NORMAL,
|
|
||||||
CURLI_LAST
|
|
||||||
} CurlInterface;
|
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* As of April 11, 2000 we're now trying to split up the urldata struct in
|
* As of April 11, 2000 we're now trying to split up the urldata struct in
|
||||||
* three different parts:
|
* three different parts:
|
||||||
@@ -445,9 +435,6 @@ typedef enum {
|
|||||||
*/
|
*/
|
||||||
|
|
||||||
struct UrlData {
|
struct UrlData {
|
||||||
Handle handle; /* struct identifier */
|
|
||||||
CurlInterface interf; /* created by WHAT interface? */
|
|
||||||
|
|
||||||
/*************** Global - specific items ************/
|
/*************** Global - specific items ************/
|
||||||
FILE *err; /* the stderr writes goes here */
|
FILE *err; /* the stderr writes goes here */
|
||||||
char *errorbuffer; /* store failure messages in here */
|
char *errorbuffer; /* store failure messages in here */
|
||||||
@@ -509,8 +496,9 @@ struct UrlData {
|
|||||||
curl_passwd_callback fpasswd;
|
curl_passwd_callback fpasswd;
|
||||||
void *passwd_client; /* pointer to pass to the passwd callback */
|
void *passwd_client; /* pointer to pass to the passwd callback */
|
||||||
|
|
||||||
long timeout; /* in seconds, 0 means no timeout */
|
long timeout; /* in seconds, 0 means no timeout */
|
||||||
long infilesize; /* size of file to upload, -1 means unknown */
|
long connecttimeout; /* in seconds, 0 means no timeout */
|
||||||
|
long infilesize; /* size of file to upload, -1 means unknown */
|
||||||
|
|
||||||
char buffer[BUFSIZE+1]; /* buffer with size BUFSIZE */
|
char buffer[BUFSIZE+1]; /* buffer with size BUFSIZE */
|
||||||
|
|
||||||
@@ -523,9 +511,6 @@ struct UrlData {
|
|||||||
|
|
||||||
char *cookie; /* HTTP cookie string to send */
|
char *cookie; /* HTTP cookie string to send */
|
||||||
|
|
||||||
char *newurl; /* This can only be set if a Location: was in the
|
|
||||||
document headers */
|
|
||||||
|
|
||||||
struct curl_slist *headers; /* linked list of extra headers */
|
struct curl_slist *headers; /* linked list of extra headers */
|
||||||
struct HttpPost *httppost; /* linked list of POST data */
|
struct HttpPost *httppost; /* linked list of POST data */
|
||||||
|
|
||||||
@@ -568,10 +553,6 @@ struct UrlData {
|
|||||||
char proxypasswd[MAX_CURL_PASSWORD_LENGTH];
|
char proxypasswd[MAX_CURL_PASSWORD_LENGTH];
|
||||||
|
|
||||||
char *krb4_level; /* what security level */
|
char *krb4_level; /* what security level */
|
||||||
#ifdef KRB4
|
|
||||||
FILE *cmdchannel;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
struct timeval keeps_speed; /* this should be request-specific */
|
struct timeval keeps_speed; /* this should be request-specific */
|
||||||
|
|
||||||
/* 'connects' will be an allocated array with pointers. If the pointer is
|
/* 'connects' will be an allocated array with pointers. If the pointer is
|
||||||
@@ -585,191 +566,26 @@ struct UrlData {
|
|||||||
#define LIBCURL_NAME "libcurl"
|
#define LIBCURL_NAME "libcurl"
|
||||||
#define LIBCURL_ID LIBCURL_NAME " " LIBCURL_VERSION " " SSL_ID
|
#define LIBCURL_ID LIBCURL_NAME " " LIBCURL_VERSION " " SSL_ID
|
||||||
|
|
||||||
|
CURLcode Curl_getinfo(CURL *curl, CURLINFO info, ...);
|
||||||
|
|
||||||
/*
|
/*
|
||||||
* Here follows function prototypes from what we used to plan to call
|
* Here follows function prototypes from what we used to plan to call
|
||||||
* the "low level" interface. It is no longer prioritized and it is not likely
|
* the "low level" interface. It is no longer prioritized and it is not likely
|
||||||
* to ever be supported to external users.
|
* to ever be supported to external users.
|
||||||
|
*
|
||||||
|
* I removed all the comments to them as well, as they were no longer accurate
|
||||||
|
* and they're not meant for "public use" anymore.
|
||||||
*/
|
*/
|
||||||
|
|
||||||
/*
|
CURLcode Curl_open(CURL **curl, char *url);
|
||||||
* NAME curl_init()
|
CURLcode Curl_setopt(CURL *handle, CURLoption option, ...);
|
||||||
*
|
CURLcode Curl_close(CURL *curl); /* the opposite of curl_open() */
|
||||||
* DESCRIPTION
|
CURLcode Curl_connect(struct UrlData *,
|
||||||
*
|
struct connectdata **,
|
||||||
* Inits libcurl globally. This must be used before any libcurl calls can
|
|
||||||
* be used. This may install global plug-ins or whatever. (This does not
|
|
||||||
* do winsock inits in Windows.)
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* curl_init();
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
CURLcode curl_init(void);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_init()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Frees libcurl globally. This must be used after all libcurl calls have
|
|
||||||
* been used. This may remove global plug-ins or whatever. (This does not
|
|
||||||
* do winsock cleanups in Windows.)
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* curl_free(curl);
|
|
||||||
*
|
|
||||||
*/
|
|
||||||
void curl_free(void);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_open()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Opens a general curl session. It does not try to connect or do anything
|
|
||||||
* on the network because of this call. The specified URL is only required
|
|
||||||
* to enable curl to figure out what protocol to "activate".
|
|
||||||
*
|
|
||||||
* A session should be looked upon as a series of requests to a single host. A
|
|
||||||
* session interacts with one host only, using one single protocol.
|
|
||||||
*
|
|
||||||
* The URL is not required. If set to "" or NULL, it can still be set later
|
|
||||||
* using the curl_setopt() function. If the curl_connect() function is called
|
|
||||||
* without the URL being known, it will return error.
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURLcode result;
|
|
||||||
* CURL *curl;
|
|
||||||
* result = curl_open(&curl, "http://curl.haxx.nu/libcurl/");
|
|
||||||
* if(result != CURL_OK) {
|
|
||||||
* return result;
|
|
||||||
* }
|
|
||||||
* */
|
|
||||||
CURLcode curl_open(CURL **curl, char *url);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_setopt()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Sets a particular option to the specified value.
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURL curl;
|
|
||||||
* curl_setopt(curl, CURL_HTTP_FOLLOW_LOCATION, TRUE);
|
|
||||||
*/
|
|
||||||
CURLcode curl_setopt(CURL *handle, CURLoption option, ...);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_close()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Closes a session previously opened with curl_open()
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURL *curl;
|
|
||||||
* CURLcode result;
|
|
||||||
*
|
|
||||||
* result = curl_close(curl);
|
|
||||||
*/
|
|
||||||
CURLcode curl_close(CURL *curl); /* the opposite of curl_open() */
|
|
||||||
|
|
||||||
CURLcode curl_read(CURLconnect *c_conn, char *buf, size_t buffersize,
|
|
||||||
ssize_t *n);
|
|
||||||
CURLcode curl_write(CURLconnect *c_conn, char *buf, size_t amount,
|
|
||||||
size_t *n);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_connect()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Connects to the peer server and performs the initial setup. This function
|
|
||||||
* writes a connect handle to its second argument that is a unique handle for
|
|
||||||
* this connect. This allows multiple connects from the same handle returned
|
|
||||||
* by curl_open().
|
|
||||||
*
|
|
||||||
* By setting 'allow_port' to FALSE, the data->use_port will *NOT* be
|
|
||||||
* respected.
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURLCode result;
|
|
||||||
* CURL curl;
|
|
||||||
* CURLconnect connect;
|
|
||||||
* result = curl_connect(curl, &connect); */
|
|
||||||
|
|
||||||
CURLcode curl_connect(CURL *curl,
|
|
||||||
CURLconnect **in_connect,
|
|
||||||
bool allow_port);
|
bool allow_port);
|
||||||
|
CURLcode Curl_do(struct connectdata *);
|
||||||
/*
|
CURLcode Curl_done(struct connectdata *);
|
||||||
* NAME curl_do()
|
CURLcode Curl_disconnect(struct connectdata *);
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* (Note: May 3rd 2000: this function does not currently allow you to
|
|
||||||
* specify a document, it will use the one set previously)
|
|
||||||
*
|
|
||||||
* This function asks for the particular document, file or resource that
|
|
||||||
* resides on the server we have connected to. You may specify a full URL,
|
|
||||||
* just an absolute path or even a relative path. That means, if you're just
|
|
||||||
* getting one file from the remote site, you can use the same URL as input
|
|
||||||
* for both curl_open() as well as for this function.
|
|
||||||
*
|
|
||||||
* In the even there is a host name, port number, user name or password parts
|
|
||||||
* in the URL, you can use the 'flags' argument to ignore them completely, or
|
|
||||||
* at your choice, make the function fail if you're trying to get a URL from
|
|
||||||
* different host than you connected to with curl_connect().
|
|
||||||
*
|
|
||||||
* You can only get one document at a time using the same connection. When one
|
|
||||||
* document has been received you can although request again.
|
|
||||||
*
|
|
||||||
* When the transfer is done, curl_done() MUST be called.
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURLCode result;
|
|
||||||
* char *url;
|
|
||||||
* CURLconnect *connect;
|
|
||||||
* result = curl_do(connect, url, CURL_DO_NONE); */
|
|
||||||
CURLcode curl_do(CURLconnect *in_conn);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_done()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* When the transfer following a curl_do() call is done, this function should
|
|
||||||
* get called.
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURLCode result;
|
|
||||||
* char *url;
|
|
||||||
* CURLconnect *connect;
|
|
||||||
* result = curl_done(connect); */
|
|
||||||
CURLcode curl_done(CURLconnect *connect);
|
|
||||||
|
|
||||||
/*
|
|
||||||
* NAME curl_disconnect()
|
|
||||||
*
|
|
||||||
* DESCRIPTION
|
|
||||||
*
|
|
||||||
* Disconnects from the peer server and performs connection cleanup.
|
|
||||||
*
|
|
||||||
* EXAMPLE
|
|
||||||
*
|
|
||||||
* CURLcode result;
|
|
||||||
* CURLconnect *connect;
|
|
||||||
* result = curl_disconnect(connect); */
|
|
||||||
CURLcode curl_disconnect(CURLconnect *connect);
|
|
||||||
|
|
||||||
|
|
||||||
#endif
|
#endif
|
||||||
|
2
maketgz
2
maketgz
@@ -11,7 +11,7 @@ libversion="$version"
|
|||||||
# Now we have a section to get the major, minor and patch number from the
|
# Now we have a section to get the major, minor and patch number from the
|
||||||
# full version string. We create a single hexadecimal number from it '0xMMmmpp'
|
# full version string. We create a single hexadecimal number from it '0xMMmmpp'
|
||||||
#
|
#
|
||||||
perl='$a=<STDIN>;@p=split("\\.",$a);for(0..2){printf STDOUT ("%02x",$p[0+$_]);}';
|
perl='$a=<STDIN>;@p=split("[\\.-]",$a);for(0..2){printf STDOUT ("%02x",$p[0+$_]);}';
|
||||||
|
|
||||||
numeric=`echo $libversion | perl -e "$perl"`
|
numeric=`echo $libversion | perl -e "$perl"`
|
||||||
|
|
||||||
|
@@ -1 +1,3 @@
|
|||||||
SUBDIRS = Win32 Linux
|
SUBDIRS = Win32 Linux
|
||||||
|
|
||||||
|
EXTRA_DIST = README
|
||||||
|
35
perl/Curl_easy/Changes
Normal file
35
perl/Curl_easy/Changes
Normal file
@@ -0,0 +1,35 @@
|
|||||||
|
Revision history for Perl extension Curl::easy.
|
||||||
|
Check out the file README for more info.
|
||||||
|
|
||||||
|
1.0.2 Tue Oct 10 2000:
|
||||||
|
- runs with libcurl 7.4
|
||||||
|
- modified curl_easy_getinfo(). It now calls curl_getinfo() that has
|
||||||
|
been added to libcurl in version 7.4.
|
||||||
|
|
||||||
|
1.0.1 Tue Oct 10 2000:
|
||||||
|
- Added some missing features of curl_easy_setopt():
|
||||||
|
- CURLOPT_ERRORBUFFER now works by passing the name of a perl
|
||||||
|
variable that shall be crated and the errormessage (if any)
|
||||||
|
be stored to.
|
||||||
|
- Passing filehandles (Options FILE, INFILE and WRITEHEADER) now works.
|
||||||
|
Have a look at test.pl to see how it works...
|
||||||
|
|
||||||
|
- Added a new function, curl_easy_getinfo(), that for now always
|
||||||
|
returns the number of bytes that where written to disk during the last
|
||||||
|
download. If the curl_easy_getinfo() function is included in libcurl,
|
||||||
|
(as promised by Daniel ;-)) i will turn this into just a call to this
|
||||||
|
function.
|
||||||
|
|
||||||
|
1.0 Thu Oct 5 2000:
|
||||||
|
- first released version
|
||||||
|
- runs with libcurl 7.3
|
||||||
|
- some features of curl_easy_setopt() are still missing:
|
||||||
|
- passing function pointers doesn't work (options WRITEFUNCTION,
|
||||||
|
READFUNCTION and PROGRESSFUNCTION).
|
||||||
|
- passing FILE * pointers doesn't work (options FILE, INFILE and
|
||||||
|
WRITEHEADER).
|
||||||
|
- passing linked lists doesn't work (options HTTPHEADER and
|
||||||
|
HTTPPOST).
|
||||||
|
- setting the buffer where to store error messages in doesn't work
|
||||||
|
(option ERRORBUFFER).
|
||||||
|
|
6
perl/Curl_easy/MANIFEST
Normal file
6
perl/Curl_easy/MANIFEST
Normal file
@@ -0,0 +1,6 @@
|
|||||||
|
Changes
|
||||||
|
MANIFEST
|
||||||
|
Makefile.PL
|
||||||
|
easy.pm
|
||||||
|
easy.xs
|
||||||
|
test.pl
|
14
perl/Curl_easy/Makefile.PL
Normal file
14
perl/Curl_easy/Makefile.PL
Normal file
@@ -0,0 +1,14 @@
|
|||||||
|
# Makefile.PL for Perl extension Curl::easy.
|
||||||
|
# Check out the file README for more info.
|
||||||
|
|
||||||
|
use ExtUtils::MakeMaker;
|
||||||
|
# See lib/ExtUtils/MakeMaker.pm for details of how to influence
|
||||||
|
# the contents of the Makefile that is written.
|
||||||
|
WriteMakefile(
|
||||||
|
'NAME' => 'Curl::easy',
|
||||||
|
'VERSION_FROM' => 'easy.pm', # finds $VERSION
|
||||||
|
'LIBS' => ['-lcurl '], # e.g., '-lm'
|
||||||
|
'DEFINE' => '-Wall', # e.g., '-DHAVE_SOMETHING'
|
||||||
|
'INC' => '', # e.g., '-I/usr/include/other'
|
||||||
|
'clean' => {FILES => "head.out body.out"}
|
||||||
|
);
|
1
perl/Curl_easy/Makefile.am
Normal file
1
perl/Curl_easy/Makefile.am
Normal file
@@ -0,0 +1 @@
|
|||||||
|
EXTRA_DIST = Changes easy.pm easy.xs Makefile.PL MANIFEST README test.pl
|
27
perl/Curl_easy/README
Normal file
27
perl/Curl_easy/README
Normal file
@@ -0,0 +1,27 @@
|
|||||||
|
README for Perl extension Curl::easy.
|
||||||
|
|
||||||
|
The perl module Curl::easy provides an interface to the cURL library "libcurl".
|
||||||
|
See http://curl.haxx.se/ for more information on cURL and libcurl.
|
||||||
|
|
||||||
|
This module requires libcurl and the corresponding headerfiles to be
|
||||||
|
installed. You then may install this module via the usual way:
|
||||||
|
|
||||||
|
perl Makefile.PL
|
||||||
|
make
|
||||||
|
make test
|
||||||
|
make install
|
||||||
|
|
||||||
|
The module provides the same functionality as libcurl provides to C programs,
|
||||||
|
please refer to the documentation of libcurl.
|
||||||
|
|
||||||
|
A short example how to use the module may be found in test.pl.
|
||||||
|
|
||||||
|
This Software is distributed AS IS, WITHOUT WARRANTY OF ANY KIND,
|
||||||
|
either express or implied. Send praise, patches, money, beer and
|
||||||
|
pizza to the author. Send complaints to /dev/null. ;-)
|
||||||
|
|
||||||
|
The author of this module is Georg Horn <horn@koblenz-net.de>
|
||||||
|
|
||||||
|
The latest version of this module can be dowloaded from
|
||||||
|
http://koblenz-net.de/~horn/export/
|
||||||
|
|
139
perl/Curl_easy/easy.pm
Normal file
139
perl/Curl_easy/easy.pm
Normal file
@@ -0,0 +1,139 @@
|
|||||||
|
# Perl interface for libcurl. Check out the file README for more info.
|
||||||
|
|
||||||
|
package Curl::easy;
|
||||||
|
|
||||||
|
use strict;
|
||||||
|
use Carp;
|
||||||
|
use vars qw($VERSION @ISA @EXPORT @EXPORT_OK $AUTOLOAD);
|
||||||
|
|
||||||
|
require Exporter;
|
||||||
|
require DynaLoader;
|
||||||
|
require AutoLoader;
|
||||||
|
|
||||||
|
@ISA = qw(Exporter DynaLoader);
|
||||||
|
# Items to export into callers namespace by default. Note: do not export
|
||||||
|
# names by default without a very good reason. Use EXPORT_OK instead.
|
||||||
|
# Do not simply export all your public functions/methods/constants.
|
||||||
|
@EXPORT = qw(
|
||||||
|
CURLOPT_AUTOREFERER
|
||||||
|
CURLOPT_COOKIE
|
||||||
|
CURLOPT_COOKIEFILE
|
||||||
|
CURLOPT_CRLF
|
||||||
|
CURLOPT_CUSTOMREQUEST
|
||||||
|
CURLOPT_ERRORBUFFER
|
||||||
|
CURLOPT_FAILONERROR
|
||||||
|
CURLOPT_FILE
|
||||||
|
CURLOPT_FOLLOWLOCATION
|
||||||
|
CURLOPT_FTPAPPEND
|
||||||
|
CURLOPT_FTPASCII
|
||||||
|
CURLOPT_FTPLISTONLY
|
||||||
|
CURLOPT_FTPPORT
|
||||||
|
CURLOPT_HEADER
|
||||||
|
CURLOPT_HTTPHEADER
|
||||||
|
CURLOPT_HTTPPOST
|
||||||
|
CURLOPT_HTTPPROXYTUNNEL
|
||||||
|
CURLOPT_HTTPREQUEST
|
||||||
|
CURLOPT_INFILE
|
||||||
|
CURLOPT_INFILESIZE
|
||||||
|
CURLOPT_INTERFACE
|
||||||
|
CURLOPT_KRB4LEVEL
|
||||||
|
CURLOPT_LOW_SPEED_LIMIT
|
||||||
|
CURLOPT_LOW_SPEED_TIME
|
||||||
|
CURLOPT_MUTE
|
||||||
|
CURLOPT_NETRC
|
||||||
|
CURLOPT_NOBODY
|
||||||
|
CURLOPT_NOPROGRESS
|
||||||
|
CURLOPT_NOTHING
|
||||||
|
CURLOPT_PORT
|
||||||
|
CURLOPT_POST
|
||||||
|
CURLOPT_POSTFIELDS
|
||||||
|
CURLOPT_POSTFIELDSIZE
|
||||||
|
CURLOPT_POSTQUOTE
|
||||||
|
CURLOPT_PROGRESSDATA
|
||||||
|
CURLOPT_PROGRESSFUNCTION
|
||||||
|
CURLOPT_PROXY
|
||||||
|
CURLOPT_PROXYPORT
|
||||||
|
CURLOPT_PROXYUSERPWD
|
||||||
|
CURLOPT_PUT
|
||||||
|
CURLOPT_QUOTE
|
||||||
|
CURLOPT_RANGE
|
||||||
|
CURLOPT_READFUNCTION
|
||||||
|
CURLOPT_REFERER
|
||||||
|
CURLOPT_RESUME_FROM
|
||||||
|
CURLOPT_SSLCERT
|
||||||
|
CURLOPT_SSLCERTPASSWD
|
||||||
|
CURLOPT_SSLVERSION
|
||||||
|
CURLOPT_STDERR
|
||||||
|
CURLOPT_TIMECONDITION
|
||||||
|
CURLOPT_TIMEOUT
|
||||||
|
CURLOPT_TIMEVALUE
|
||||||
|
CURLOPT_TRANSFERTEXT
|
||||||
|
CURLOPT_UPLOAD
|
||||||
|
CURLOPT_URL
|
||||||
|
CURLOPT_USERAGENT
|
||||||
|
CURLOPT_USERPWD
|
||||||
|
CURLOPT_VERBOSE
|
||||||
|
CURLOPT_WRITEFUNCTION
|
||||||
|
CURLOPT_WRITEHEADER
|
||||||
|
|
||||||
|
CURLINFO_EFFECTIVE_URL
|
||||||
|
CURLINFO_HTTP_CODE
|
||||||
|
CURLINFO_TOTAL_TIME
|
||||||
|
CURLINFO_NAMELOOKUP_TIME
|
||||||
|
CURLINFO_CONNECT_TIME
|
||||||
|
CURLINFO_PRETRANSFER_TIME
|
||||||
|
CURLINFO_SIZE_UPLOAD
|
||||||
|
CURLINFO_SIZE_DOWNLOAD
|
||||||
|
CURLINFO_SPEED_DOWNLOAD
|
||||||
|
CURLINFO_SPEED_UPLOAD
|
||||||
|
CURLINFO_HEADER_SIZE
|
||||||
|
CURLINFO_REQUEST_SIZE
|
||||||
|
);
|
||||||
|
$VERSION = '1.0.1';
|
||||||
|
|
||||||
|
sub AUTOLOAD {
|
||||||
|
# This AUTOLOAD is used to 'autoload' constants from the constant()
|
||||||
|
# XS function.
|
||||||
|
|
||||||
|
(my $constname = $AUTOLOAD) =~ s/.*:://;
|
||||||
|
return constant($constname, 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
bootstrap Curl::easy $VERSION;
|
||||||
|
|
||||||
|
# Preloaded methods go here.
|
||||||
|
|
||||||
|
# Autoload methods go after =cut, and are processed by the autosplit program.
|
||||||
|
|
||||||
|
1;
|
||||||
|
__END__
|
||||||
|
# Below is the stub of documentation for your module. You better edit it!
|
||||||
|
|
||||||
|
=head1 NAME
|
||||||
|
|
||||||
|
Curl::easy - Perl extension for libcurl
|
||||||
|
|
||||||
|
=head1 SYNOPSIS
|
||||||
|
|
||||||
|
use Curl::easy;
|
||||||
|
|
||||||
|
$CURL = curl_easy_init();
|
||||||
|
$CURLcode = curl_easy_setopt($CURL, CURLoption, Value);
|
||||||
|
$CURLcode = curl_easy_perform($CURL);
|
||||||
|
curl_easy_cleanup($CURL);
|
||||||
|
|
||||||
|
|
||||||
|
=head1 DESCRIPTION
|
||||||
|
|
||||||
|
This perl module provides an interface to the libcurl C library. See
|
||||||
|
http://curl.haxx.se/ for more information on cURL and libcurl.
|
||||||
|
|
||||||
|
=head1 AUTHOR
|
||||||
|
|
||||||
|
Georg Horn <horn@koblenz-net.de>
|
||||||
|
|
||||||
|
=head1 SEE ALSO
|
||||||
|
|
||||||
|
http://curl.haxx.se/
|
||||||
|
|
||||||
|
=cut
|
290
perl/Curl_easy/easy.xs
Normal file
290
perl/Curl_easy/easy.xs
Normal file
@@ -0,0 +1,290 @@
|
|||||||
|
/* Perl interface for libcurl. Check out the file README for more info. */
|
||||||
|
|
||||||
|
#include "EXTERN.h"
|
||||||
|
#include "perl.h"
|
||||||
|
#include "XSUB.h"
|
||||||
|
|
||||||
|
#include <curl/curl.h>
|
||||||
|
#include <curl/easy.h>
|
||||||
|
|
||||||
|
|
||||||
|
/* Buffer and varname for option CURLOPT_ERRORBUFFER */
|
||||||
|
|
||||||
|
static char errbuf[CURL_ERROR_SIZE];
|
||||||
|
static char *errbufvarname = NULL;
|
||||||
|
|
||||||
|
|
||||||
|
static int
|
||||||
|
constant(char *name, int arg)
|
||||||
|
{
|
||||||
|
errno = 0;
|
||||||
|
if (strncmp(name, "CURLINFO_", 9) == 0) {
|
||||||
|
name += 9;
|
||||||
|
switch (*name) {
|
||||||
|
case 'A':
|
||||||
|
case 'B':
|
||||||
|
case 'C':
|
||||||
|
case 'D':
|
||||||
|
if (strEQ(name, "CONNECT_TIME")) return CURLINFO_CONNECT_TIME;
|
||||||
|
break;
|
||||||
|
case 'E':
|
||||||
|
case 'F':
|
||||||
|
if (strEQ(name, "EFFECTIVE_URL")) return CURLINFO_EFFECTIVE_URL;
|
||||||
|
break;
|
||||||
|
case 'G':
|
||||||
|
case 'H':
|
||||||
|
if (strEQ(name, "HEADER_SIZE")) return CURLINFO_HEADER_SIZE;
|
||||||
|
if (strEQ(name, "HTTP_CODE")) return CURLINFO_HTTP_CODE;
|
||||||
|
break;
|
||||||
|
case 'I':
|
||||||
|
case 'J':
|
||||||
|
case 'K':
|
||||||
|
case 'L':
|
||||||
|
case 'M':
|
||||||
|
case 'N':
|
||||||
|
if (strEQ(name, "NAMELOOKUP_TIME")) return CURLINFO_NAMELOOKUP_TIME;
|
||||||
|
break;
|
||||||
|
case 'O':
|
||||||
|
case 'P':
|
||||||
|
if (strEQ(name, "PRETRANSFER_TIME")) return CURLINFO_PRETRANSFER_TIME;
|
||||||
|
break;
|
||||||
|
case 'Q':
|
||||||
|
case 'R':
|
||||||
|
if (strEQ(name, "REQUEST_SIZE")) return CURLINFO_REQUEST_SIZE;
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
case 'T':
|
||||||
|
if (strEQ(name, "SIZE_DOWNLOAD")) return CURLINFO_SIZE_DOWNLOAD;
|
||||||
|
if (strEQ(name, "SIZE_UPLOAD")) return CURLINFO_SIZE_UPLOAD;
|
||||||
|
if (strEQ(name, "SPEED_DOWNLOAD")) return CURLINFO_SPEED_DOWNLOAD;
|
||||||
|
if (strEQ(name, "SPEED_UPLOAD")) return CURLINFO_SPEED_UPLOAD;
|
||||||
|
if (strEQ(name, "TOTAL_TIME")) return CURLINFO_TOTAL_TIME;
|
||||||
|
break;
|
||||||
|
case 'U':
|
||||||
|
case 'V':
|
||||||
|
case 'W':
|
||||||
|
case 'X':
|
||||||
|
case 'Y':
|
||||||
|
case 'Z':
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if (strncmp(name, "CURLOPT_", 8) == 0) {
|
||||||
|
name += 8;
|
||||||
|
switch (*name) {
|
||||||
|
case 'A':
|
||||||
|
case 'B':
|
||||||
|
if (strEQ(name, "AUTOREFERER")) return CURLOPT_AUTOREFERER;
|
||||||
|
break;
|
||||||
|
case 'C':
|
||||||
|
case 'D':
|
||||||
|
if (strEQ(name, "COOKIE")) return CURLOPT_COOKIE;
|
||||||
|
if (strEQ(name, "COOKIEFILE")) return CURLOPT_COOKIEFILE;
|
||||||
|
if (strEQ(name, "CRLF")) return CURLOPT_CRLF;
|
||||||
|
if (strEQ(name, "CUSTOMREQUEST")) return CURLOPT_CUSTOMREQUEST;
|
||||||
|
break;
|
||||||
|
case 'E':
|
||||||
|
case 'F':
|
||||||
|
if (strEQ(name, "ERRORBUFFER")) return CURLOPT_ERRORBUFFER;
|
||||||
|
if (strEQ(name, "FAILONERROR")) return CURLOPT_FAILONERROR;
|
||||||
|
if (strEQ(name, "FILE")) return CURLOPT_FILE;
|
||||||
|
if (strEQ(name, "FOLLOWLOCATION")) return CURLOPT_FOLLOWLOCATION;
|
||||||
|
if (strEQ(name, "FTPAPPEND")) return CURLOPT_FTPAPPEND;
|
||||||
|
if (strEQ(name, "FTPASCII")) return CURLOPT_FTPASCII;
|
||||||
|
if (strEQ(name, "FTPLISTONLY")) return CURLOPT_FTPLISTONLY;
|
||||||
|
if (strEQ(name, "FTPPORT")) return CURLOPT_FTPPORT;
|
||||||
|
break;
|
||||||
|
case 'G':
|
||||||
|
case 'H':
|
||||||
|
if (strEQ(name, "HEADER")) return CURLOPT_HEADER;
|
||||||
|
if (strEQ(name, "HTTPHEADER")) return CURLOPT_HTTPHEADER;
|
||||||
|
if (strEQ(name, "HTTPPOST")) return CURLOPT_HTTPPOST;
|
||||||
|
if (strEQ(name, "HTTPPROXYTUNNEL")) return CURLOPT_HTTPPROXYTUNNEL;
|
||||||
|
if (strEQ(name, "HTTPREQUEST")) return CURLOPT_HTTPREQUEST;
|
||||||
|
break;
|
||||||
|
case 'I':
|
||||||
|
case 'J':
|
||||||
|
if (strEQ(name, "INFILE")) return CURLOPT_INFILE;
|
||||||
|
if (strEQ(name, "INFILESIZE")) return CURLOPT_INFILESIZE;
|
||||||
|
if (strEQ(name, "INTERFACE")) return CURLOPT_INTERFACE;
|
||||||
|
break;
|
||||||
|
case 'K':
|
||||||
|
case 'L':
|
||||||
|
if (strEQ(name, "KRB4LEVEL")) return CURLOPT_KRB4LEVEL;
|
||||||
|
if (strEQ(name, "LOW_SPEED_LIMIT")) return CURLOPT_LOW_SPEED_LIMIT;
|
||||||
|
if (strEQ(name, "LOW_SPEED_TIME")) return CURLOPT_LOW_SPEED_TIME;
|
||||||
|
break;
|
||||||
|
case 'M':
|
||||||
|
case 'N':
|
||||||
|
if (strEQ(name, "MUTE")) return CURLOPT_MUTE;
|
||||||
|
if (strEQ(name, "NETRC")) return CURLOPT_NETRC;
|
||||||
|
if (strEQ(name, "NOBODY")) return CURLOPT_NOBODY;
|
||||||
|
if (strEQ(name, "NOPROGRESS")) return CURLOPT_NOPROGRESS;
|
||||||
|
if (strEQ(name, "NOTHING")) return CURLOPT_NOTHING;
|
||||||
|
break;
|
||||||
|
case 'O':
|
||||||
|
case 'P':
|
||||||
|
if (strEQ(name, "PORT")) return CURLOPT_PORT;
|
||||||
|
if (strEQ(name, "POST")) return CURLOPT_POST;
|
||||||
|
if (strEQ(name, "POSTFIELDS")) return CURLOPT_POSTFIELDS;
|
||||||
|
if (strEQ(name, "POSTFIELDSIZE")) return CURLOPT_POSTFIELDSIZE;
|
||||||
|
if (strEQ(name, "POSTQUOTE")) return CURLOPT_POSTQUOTE;
|
||||||
|
if (strEQ(name, "PROGRESSDATA")) return CURLOPT_PROGRESSDATA;
|
||||||
|
if (strEQ(name, "PROGRESSFUNCTION")) return CURLOPT_PROGRESSFUNCTION;
|
||||||
|
if (strEQ(name, "PROXY")) return CURLOPT_PROXY;
|
||||||
|
if (strEQ(name, "PROXYPORT")) return CURLOPT_PROXYPORT;
|
||||||
|
if (strEQ(name, "PROXYUSERPWD")) return CURLOPT_PROXYUSERPWD;
|
||||||
|
if (strEQ(name, "PUT")) return CURLOPT_PUT;
|
||||||
|
break;
|
||||||
|
case 'Q':
|
||||||
|
case 'R':
|
||||||
|
if (strEQ(name, "QUOTE")) return CURLOPT_QUOTE;
|
||||||
|
if (strEQ(name, "RANGE")) return CURLOPT_RANGE;
|
||||||
|
if (strEQ(name, "READFUNCTION")) return CURLOPT_READFUNCTION;
|
||||||
|
if (strEQ(name, "REFERER")) return CURLOPT_REFERER;
|
||||||
|
if (strEQ(name, "RESUME_FROM")) return CURLOPT_RESUME_FROM;
|
||||||
|
break;
|
||||||
|
case 'S':
|
||||||
|
case 'T':
|
||||||
|
if (strEQ(name, "SSLCERT")) return CURLOPT_SSLCERT;
|
||||||
|
if (strEQ(name, "SSLCERTPASSWD")) return CURLOPT_SSLCERTPASSWD;
|
||||||
|
if (strEQ(name, "SSLVERSION")) return CURLOPT_SSLVERSION;
|
||||||
|
if (strEQ(name, "STDERR")) return CURLOPT_STDERR;
|
||||||
|
if (strEQ(name, "TIMECONDITION")) return CURLOPT_TIMECONDITION;
|
||||||
|
if (strEQ(name, "TIMEOUT")) return CURLOPT_TIMEOUT;
|
||||||
|
if (strEQ(name, "TIMEVALUE")) return CURLOPT_TIMEVALUE;
|
||||||
|
if (strEQ(name, "TRANSFERTEXT")) return CURLOPT_TRANSFERTEXT;
|
||||||
|
break;
|
||||||
|
case 'U':
|
||||||
|
case 'V':
|
||||||
|
if (strEQ(name, "UPLOAD")) return CURLOPT_UPLOAD;
|
||||||
|
if (strEQ(name, "URL")) return CURLOPT_URL;
|
||||||
|
if (strEQ(name, "USERAGENT")) return CURLOPT_USERAGENT;
|
||||||
|
if (strEQ(name, "USERPWD")) return CURLOPT_USERPWD;
|
||||||
|
if (strEQ(name, "VERBOSE")) return CURLOPT_VERBOSE;
|
||||||
|
break;
|
||||||
|
case 'W':
|
||||||
|
case 'X':
|
||||||
|
case 'Y':
|
||||||
|
case 'Z':
|
||||||
|
if (strEQ(name, "WRITEFUNCTION")) return CURLOPT_WRITEFUNCTION;
|
||||||
|
if (strEQ(name, "WRITEHEADER")) return CURLOPT_WRITEHEADER;
|
||||||
|
if (strEQ(name, "WRITEINFO")) return CURLOPT_WRITEINFO;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
errno = EINVAL;
|
||||||
|
return 0;
|
||||||
|
}
|
||||||
|
|
||||||
|
|
||||||
|
MODULE = Curl::easy PACKAGE = Curl::easy
|
||||||
|
|
||||||
|
int
|
||||||
|
constant(name,arg)
|
||||||
|
char * name
|
||||||
|
int arg
|
||||||
|
|
||||||
|
|
||||||
|
void *
|
||||||
|
curl_easy_init()
|
||||||
|
CODE:
|
||||||
|
if (errbufvarname) free(errbufvarname);
|
||||||
|
errbufvarname = NULL;
|
||||||
|
RETVAL = curl_easy_init();
|
||||||
|
OUTPUT:
|
||||||
|
RETVAL
|
||||||
|
|
||||||
|
|
||||||
|
int
|
||||||
|
curl_easy_setopt(curl, option, value)
|
||||||
|
void * curl
|
||||||
|
int option
|
||||||
|
char * value
|
||||||
|
CODE:
|
||||||
|
if (option < CURLOPTTYPE_OBJECTPOINT) {
|
||||||
|
/* This is an option specifying an integer value: */
|
||||||
|
long value = (long)SvIV(ST(2));
|
||||||
|
RETVAL = curl_easy_setopt(curl, option, value);
|
||||||
|
} else if (option == CURLOPT_FILE || option == CURLOPT_INFILE ||
|
||||||
|
option == CURLOPT_WRITEHEADER) {
|
||||||
|
/* This is an option specifying a FILE * value: */
|
||||||
|
FILE * value = IoIFP(sv_2io(ST(2)));
|
||||||
|
RETVAL = curl_easy_setopt(curl, option, value);
|
||||||
|
} else if (option == CURLOPT_ERRORBUFFER) {
|
||||||
|
SV *sv;
|
||||||
|
RETVAL = curl_easy_setopt(curl, option, errbuf);
|
||||||
|
if (errbufvarname) free(errbufvarname);
|
||||||
|
errbufvarname = strdup(value);
|
||||||
|
sv = perl_get_sv(errbufvarname, TRUE | GV_ADDMULTI);
|
||||||
|
} else if (option == CURLOPT_WRITEFUNCTION || option ==
|
||||||
|
CURLOPT_READFUNCTION || option == CURLOPT_PROGRESSFUNCTION) {
|
||||||
|
/* This is an option specifying a callback function */
|
||||||
|
/* not yet implemented */
|
||||||
|
RETVAL = -1;
|
||||||
|
} else {
|
||||||
|
/* default, option specifying a char * value: */
|
||||||
|
RETVAL = curl_easy_setopt(curl, option, value);
|
||||||
|
}
|
||||||
|
OUTPUT:
|
||||||
|
RETVAL
|
||||||
|
|
||||||
|
|
||||||
|
int
|
||||||
|
curl_easy_perform(curl)
|
||||||
|
void * curl
|
||||||
|
CODE:
|
||||||
|
RETVAL = curl_easy_perform(curl);
|
||||||
|
if (RETVAL && errbufvarname) {
|
||||||
|
SV *sv = perl_get_sv(errbufvarname, TRUE | GV_ADDMULTI);
|
||||||
|
sv_setpv(sv, errbuf);
|
||||||
|
}
|
||||||
|
OUTPUT:
|
||||||
|
RETVAL
|
||||||
|
|
||||||
|
|
||||||
|
int
|
||||||
|
curl_easy_getinfo(curl, option, value)
|
||||||
|
void * curl
|
||||||
|
int option
|
||||||
|
double value
|
||||||
|
CODE:
|
||||||
|
switch (option & CURLINFO_TYPEMASK) {
|
||||||
|
case CURLINFO_STRING: {
|
||||||
|
char * value = (char *)SvPV(ST(2), PL_na);
|
||||||
|
RETVAL = curl_easy_getinfo(curl, option, &value);
|
||||||
|
sv_setpv(ST(2), value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case CURLINFO_LONG: {
|
||||||
|
long value = (long)SvIV(ST(2));
|
||||||
|
RETVAL = curl_easy_getinfo(curl, option, &value);
|
||||||
|
sv_setiv(ST(2), value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case CURLINFO_DOUBLE: {
|
||||||
|
double value = (double)SvNV(ST(2));
|
||||||
|
RETVAL = curl_easy_getinfo(curl, option, &value);
|
||||||
|
sv_setnv(ST(2), value);
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
default: {
|
||||||
|
RETVAL = CURLE_BAD_FUNCTION_ARGUMENT;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
OUTPUT:
|
||||||
|
RETVAL
|
||||||
|
|
||||||
|
|
||||||
|
int
|
||||||
|
curl_easy_cleanup(curl)
|
||||||
|
void * curl
|
||||||
|
CODE:
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
if (errbufvarname) free(errbufvarname);
|
||||||
|
errbufvarname = NULL;
|
||||||
|
RETVAL = 0;
|
||||||
|
OUTPUT:
|
||||||
|
RETVAL
|
||||||
|
|
101
perl/Curl_easy/test.pl
Normal file
101
perl/Curl_easy/test.pl
Normal file
@@ -0,0 +1,101 @@
|
|||||||
|
# Test script for Perl extension Curl::easy.
|
||||||
|
# Check out the file README for more info.
|
||||||
|
|
||||||
|
# Before `make install' is performed this script should be runnable with
|
||||||
|
# `make test'. After `make install' it should work as `perl test.pl'
|
||||||
|
|
||||||
|
######################### We start with some black magic to print on failure.
|
||||||
|
|
||||||
|
# Change 1..1 below to 1..last_test_to_print .
|
||||||
|
# (It may become useful if the test is moved to ./t subdirectory.)
|
||||||
|
|
||||||
|
BEGIN { $| = 1; print "1..5\n"; }
|
||||||
|
END {print "not ok 1\n" unless $loaded;}
|
||||||
|
use Curl::easy;
|
||||||
|
$loaded = 1;
|
||||||
|
print "ok 1\n";
|
||||||
|
|
||||||
|
######################### End of black magic.
|
||||||
|
|
||||||
|
# Insert your test code below (better if it prints "ok 13"
|
||||||
|
# (correspondingly "not ok 13") depending on the success of chunk 13
|
||||||
|
# of the test code):
|
||||||
|
|
||||||
|
# Read URL to get
|
||||||
|
$defurl = "http://www/";
|
||||||
|
$url = "";
|
||||||
|
print "Please enter an URL to fetch [$defurl]: ";
|
||||||
|
$url = <STDIN>;
|
||||||
|
if ($url =~ /^\s*\n/) {
|
||||||
|
$url = $defurl;
|
||||||
|
}
|
||||||
|
|
||||||
|
# Use this for simple benchmarking
|
||||||
|
#for ($i=0; $i<1000; $i++) {
|
||||||
|
|
||||||
|
# Init the curl session
|
||||||
|
if (($curl = Curl::easy::curl_easy_init()) != 0) {
|
||||||
|
print "ok 2\n";
|
||||||
|
} else {
|
||||||
|
print "ko 2\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Set URL to get
|
||||||
|
if (Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_URL, $url) == 0) {
|
||||||
|
print "ok 3\n";
|
||||||
|
} else {
|
||||||
|
print "ko 3\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# No progress meter please
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_NOPROGRESS, 1);
|
||||||
|
|
||||||
|
# Shut up completely
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_MUTE, 1);
|
||||||
|
|
||||||
|
# Follow location headers
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_FOLLOWLOCATION, 1);
|
||||||
|
|
||||||
|
# Set timeout
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_TIMEOUT, 30);
|
||||||
|
|
||||||
|
# Set file where to read cookies from
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_COOKIEFILE, "cookies");
|
||||||
|
|
||||||
|
# Set file where to store the header
|
||||||
|
open HEAD, ">head.out";
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_WRITEHEADER, HEAD);
|
||||||
|
|
||||||
|
# Set file where to store the body
|
||||||
|
open BODY, ">body.out";
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_FILE, BODY);
|
||||||
|
|
||||||
|
# Store error messages in variable $errbuf
|
||||||
|
# NOTE: The name of the variable is passed as a string!
|
||||||
|
# curl_easy_setopt() creates a perl variable with that name, and
|
||||||
|
# curl_easy_perform() stores the errormessage into it if an error occurs.
|
||||||
|
Curl::easy::curl_easy_setopt($curl, Curl::easy::CURLOPT_ERRORBUFFER, "errbuf");
|
||||||
|
|
||||||
|
# Go get it
|
||||||
|
if (Curl::easy::curl_easy_perform($curl) == 0) {
|
||||||
|
Curl::easy::curl_easy_getinfo($curl, Curl::easy::CURLINFO_SIZE_DOWNLOAD, $bytes);
|
||||||
|
print "ok 4: $bytes bytes read\n";
|
||||||
|
print "check out the files head.out and body.out\n";
|
||||||
|
print "for the headers and content of the URL you just fetched...\n";
|
||||||
|
Curl::easy::curl_easy_getinfo($curl, Curl::easy::CURLINFO_EFFECTIVE_URL, $realurl);
|
||||||
|
Curl::easy::curl_easy_getinfo($curl, Curl::easy::CURLINFO_HTTP_CODE, $httpcode);
|
||||||
|
print "effective fetched url (http code: $httpcode) was: $url\n";
|
||||||
|
} else {
|
||||||
|
# We can acces the error message in $errbuf here
|
||||||
|
print "ko 4: '$errbuf'\n";
|
||||||
|
}
|
||||||
|
|
||||||
|
# Cleanup
|
||||||
|
close HEAD;
|
||||||
|
close BODY;
|
||||||
|
Curl::easy::curl_easy_cleanup($curl);
|
||||||
|
print "ok 5\n";
|
||||||
|
|
||||||
|
# Use this for simple benchmarking
|
||||||
|
#}
|
||||||
|
|
3
perl/Makefile.am
Normal file
3
perl/Makefile.am
Normal file
@@ -0,0 +1,3 @@
|
|||||||
|
SUBDIRS = Curl_easy
|
||||||
|
|
||||||
|
EXTRA_DIST = README
|
43
perl/README
43
perl/README
@@ -1,33 +1,18 @@
|
|||||||
This is just a small collection of perl scripts that use curl to do
|
_ _ ____ _
|
||||||
their jobs.
|
___| | | | _ \| |
|
||||||
|
/ __| | | | |_) | |
|
||||||
|
| (__| |_| | _ <| |___
|
||||||
|
\___|\___/|_| \_\_____|
|
||||||
|
|
||||||
If you need a proxy configuration in order to get HTTP or FTP
|
Perl
|
||||||
documents, do edit your .curlrc file in your HOME dir to contain:
|
|
||||||
-x <proxy host>:<proxy port>
|
|
||||||
|
|
||||||
These scripts are all written by Daniel Stenberg.
|
Perl's a great script language, not the least for quick prototyping. Curl is
|
||||||
|
elegantly used from within it. You can either invoke external curl command
|
||||||
|
line or use the curl interface.
|
||||||
|
|
||||||
checklinks.pl
|
Georg Horn's Perl interface to curl is available in the Curl_easy/
|
||||||
=============
|
subdirectory. Using the Curl::easy module is just straightforward and
|
||||||
This script fetches an HTML page, extracts all links and references to
|
works much like using libcurl in a C programm, so please refer to the
|
||||||
other documents and then goes through them to check that they work.
|
documentation of libcurl. Have a look at test.pl to get an idea of how
|
||||||
Reports progress in a format intended for machine-parsing.
|
to start.
|
||||||
|
|
||||||
getlinks.pl
|
|
||||||
===========
|
|
||||||
You ever wanted to download a bunch of programs a certain HTML page has
|
|
||||||
links to? This program extracts all links and references from a web page
|
|
||||||
and then compares them to the regex you supply. All matches will be
|
|
||||||
downloaded in the target directory of your choice.
|
|
||||||
|
|
||||||
recursiveftpget.pl
|
|
||||||
==================
|
|
||||||
This script recursively downloads all files from a directory on an ftp site
|
|
||||||
and all subdirectories it has. Optional depth-level.
|
|
||||||
|
|
||||||
formfind.pl
|
|
||||||
===========
|
|
||||||
Downloads an HTML page (or reads stdin) and reports a human readable report
|
|
||||||
about the FORM(s) present. What method, what URL, which input or select
|
|
||||||
field, what default values they have and what submit buttons there are. It
|
|
||||||
is useful if you intend to use curl to properly fake a form submission.
|
|
||||||
|
0
perl/checklinks.pl.in → perl/contrib/checklinks.pl.in
Executable file → Normal file
0
perl/checklinks.pl.in → perl/contrib/checklinks.pl.in
Executable file → Normal file
0
perl/getlinks.pl.in → perl/contrib/getlinks.pl.in
Executable file → Normal file
0
perl/getlinks.pl.in → perl/contrib/getlinks.pl.in
Executable file → Normal file
104
perl/contrib/mirror.pl
Normal file
104
perl/contrib/mirror.pl
Normal file
@@ -0,0 +1,104 @@
|
|||||||
|
#!/usr/bin/perl
|
||||||
|
|
||||||
|
#
|
||||||
|
# Author: Daniel Stenberg <daniel@haxx.se>
|
||||||
|
# Version: 0.1
|
||||||
|
# Date: October 10, 2000
|
||||||
|
#
|
||||||
|
# This is public domain. Feel free to do whatever you please with this script.
|
||||||
|
# There are no warranties whatsoever! It might work, it might ruin your hard
|
||||||
|
# disk. Use this on your own risk.
|
||||||
|
#
|
||||||
|
# PURPOSE
|
||||||
|
#
|
||||||
|
# This script uses a local directory to maintain a "mirror" of the curl
|
||||||
|
# packages listed in the remote curl web sites package list. Files present in
|
||||||
|
# the local directory that aren't present in the remote list will be removed.
|
||||||
|
# Files that are present in the remote list but not in the local directory
|
||||||
|
# will be downloaded and put there. Files present at both places will not
|
||||||
|
# be touched.
|
||||||
|
#
|
||||||
|
# WARNING: don't put other files in the mirror directory, they will be removed
|
||||||
|
# when this script runs if they don't exist in the remote package list!
|
||||||
|
#
|
||||||
|
|
||||||
|
# this is the directory to keep all the mirrored curl files in:
|
||||||
|
$some_dir = $ARGV[0];
|
||||||
|
|
||||||
|
if( ! -d $some_dir ) {
|
||||||
|
print "$some_dir is not a dir!\n";
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
# path to the curl binary
|
||||||
|
$curl = "/home/danste/bin/curl";
|
||||||
|
|
||||||
|
# this is the remote file list
|
||||||
|
$filelist = "http://curl.haxx.se/download/curldist.txt";
|
||||||
|
|
||||||
|
# prepend URL:
|
||||||
|
$prepend = "http://curl.haxx.se/download";
|
||||||
|
|
||||||
|
opendir(DIR, $some_dir) || die "can't opendir $some_dir: $!";
|
||||||
|
@existing = grep { /^[^\.]/ } readdir(DIR);
|
||||||
|
closedir DIR;
|
||||||
|
|
||||||
|
$LOCAL_FILE = 1;
|
||||||
|
$REMOTE_FILE = 2;
|
||||||
|
|
||||||
|
# create a hash array
|
||||||
|
for(@existing) {
|
||||||
|
$allfiles{$_} |= $LOCAL_FILE;
|
||||||
|
}
|
||||||
|
|
||||||
|
# get remote file list
|
||||||
|
print "Getting file list from $filelist\n";
|
||||||
|
@remotefiles=`$curl -s $filelist`;
|
||||||
|
|
||||||
|
# fill in the hash array
|
||||||
|
for(@remotefiles) {
|
||||||
|
chomp;
|
||||||
|
$allfiles{$_} |= $REMOTE_FILE;
|
||||||
|
$remote++;
|
||||||
|
}
|
||||||
|
if($remote < 10) {
|
||||||
|
print "There's something wrong. The remote file list seems too smallish!\n";
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
@sfiles = sort { $a cmp $b } keys %allfiles;
|
||||||
|
|
||||||
|
|
||||||
|
$leftalone = $downloaded = $removed = 0;
|
||||||
|
for(@sfiles) {
|
||||||
|
$file = $_;
|
||||||
|
$info = $allfiles{$file};
|
||||||
|
|
||||||
|
if($info == ($REMOTE_FILE|$LOCAL_FILE)) {
|
||||||
|
print "$file is LOCAL and REMOTE, left alone\n";
|
||||||
|
$leftalone++;
|
||||||
|
}
|
||||||
|
elsif($info == $REMOTE_FILE) {
|
||||||
|
print "$file is only REMOTE, getting it...\n";
|
||||||
|
system("$curl $prepend/$file -o $some_dir/$file");
|
||||||
|
$downloaded++;
|
||||||
|
}
|
||||||
|
elsif($info == $LOCAL_FILE) {
|
||||||
|
print "$file is only LOCAL, removing it...\n";
|
||||||
|
system("rm $some_dir/$file");
|
||||||
|
$removed++;
|
||||||
|
}
|
||||||
|
else {
|
||||||
|
print "Problem, file $file was marked $info\n";
|
||||||
|
}
|
||||||
|
$loops++;
|
||||||
|
}
|
||||||
|
|
||||||
|
if(!$loops) {
|
||||||
|
print "No remote or local files were found!\n";
|
||||||
|
exit;
|
||||||
|
}
|
||||||
|
|
||||||
|
print "$leftalone files were already present\n",
|
||||||
|
"$downloaded files were added\n",
|
||||||
|
"$removed files were removed\n";
|
0
perl/recursiveftpget.pl.in → perl/contrib/recursiveftpget.pl.in
Executable file → Normal file
0
perl/recursiveftpget.pl.in → perl/contrib/recursiveftpget.pl.in
Executable file → Normal file
2
php/Makefile.am
Normal file
2
php/Makefile.am
Normal file
@@ -0,0 +1,2 @@
|
|||||||
|
SUBDIRS = examples
|
||||||
|
EXTRA_DIST = README
|
15
php/README
Normal file
15
php/README
Normal file
@@ -0,0 +1,15 @@
|
|||||||
|
_ _ ____ _
|
||||||
|
___| | | | _ \| |
|
||||||
|
/ __| | | | |_) | |
|
||||||
|
| (__| |_| | _ <| |___
|
||||||
|
\___|\___/|_| \_\_____|
|
||||||
|
|
||||||
|
PHP
|
||||||
|
|
||||||
|
There's an excellent interface to curl written for PHP by Sterling Hughes. See
|
||||||
|
the subdirectory examples/ for some examples on how to program with it.
|
||||||
|
|
||||||
|
Unfortunately, we don't have much more information about the interface
|
||||||
|
included here yet, but there's a detailed online manual for it over at:
|
||||||
|
|
||||||
|
http://www.php.net/manual/ref.curl.php
|
1
php/examples/Makefile.am
Normal file
1
php/examples/Makefile.am
Normal file
@@ -0,0 +1 @@
|
|||||||
|
EXTRA_DIST = README getpageinvar.php simpleget.php simplepost.php
|
16
php/examples/README
Normal file
16
php/examples/README
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
_ _ ____ _
|
||||||
|
___| | | | _ \| |
|
||||||
|
/ __| | | | |_) | |
|
||||||
|
| (__| |_| | _ <| |___
|
||||||
|
\___|\___/|_| \_\_____|
|
||||||
|
|
||||||
|
PHP program examples
|
||||||
|
|
||||||
|
getpageinvar.php
|
||||||
|
- Fetch a single URL and return in a variable
|
||||||
|
|
||||||
|
simpleget.php
|
||||||
|
- A very simple example that gets a HTTP page
|
||||||
|
|
||||||
|
simplepost.php
|
||||||
|
- Example that sends a HTTP POST to a remote site
|
356
src/main.c
356
src/main.c
@@ -29,8 +29,6 @@
|
|||||||
#include <ctype.h>
|
#include <ctype.h>
|
||||||
|
|
||||||
#include <curl/curl.h>
|
#include <curl/curl.h>
|
||||||
#include <curl/types.h> /* new for v7 */
|
|
||||||
#include <curl/easy.h> /* new for v7 */
|
|
||||||
|
|
||||||
#define _MPRINTF_REPLACE /* we want curl-functions instead of native ones */
|
#define _MPRINTF_REPLACE /* we want curl-functions instead of native ones */
|
||||||
#include <curl/mprintf.h>
|
#include <curl/mprintf.h>
|
||||||
@@ -40,11 +38,6 @@
|
|||||||
|
|
||||||
#define CURLseparator "--_curl_--"
|
#define CURLseparator "--_curl_--"
|
||||||
|
|
||||||
/* This define make use of the "Curlseparator" as opposed to the
|
|
||||||
MIMEseparator. We might add support for the latter one in the
|
|
||||||
future, and that's why this is left in the source. */
|
|
||||||
#define CURL_SEPARATORS
|
|
||||||
|
|
||||||
/* This is now designed to have its own local setup.h */
|
/* This is now designed to have its own local setup.h */
|
||||||
#include "setup.h"
|
#include "setup.h"
|
||||||
|
|
||||||
@@ -242,61 +235,63 @@ static void help(void)
|
|||||||
" -a/--append Append to target file when uploading (F)\n"
|
" -a/--append Append to target file when uploading (F)\n"
|
||||||
" -A/--user-agent <string> User-Agent to send to server (H)\n"
|
" -A/--user-agent <string> User-Agent to send to server (H)\n"
|
||||||
" -b/--cookie <name=string/file> Cookie string or file to read cookies from (H)\n"
|
" -b/--cookie <name=string/file> Cookie string or file to read cookies from (H)\n"
|
||||||
" -B/--use-ascii Use ASCII/text transfer\n"
|
" -B/--use-ascii Use ASCII/text transfer\n",
|
||||||
" -C/--continue-at <offset> Specify absolute resume offset\n"
|
curl_version());
|
||||||
|
puts(" -C/--continue-at <offset> Specify absolute resume offset\n"
|
||||||
" -d/--data <data> HTTP POST data (H)\n"
|
" -d/--data <data> HTTP POST data (H)\n"
|
||||||
" --data-ascii <data> HTTP POST ASCII data (H)\n"
|
" --data-ascii <data> HTTP POST ASCII data (H)\n"
|
||||||
" --data-binary <data> HTTP POST binary data (H)\n"
|
" --data-binary <data> HTTP POST binary data (H)\n"
|
||||||
" -D/--dump-header <file> Write the headers to this file\n"
|
" -D/--dump-header <file> Write the headers to this file\n"
|
||||||
" -e/--referer Referer page (H)\n"
|
" --egd-file <file> EGD socket path for random data (SSL)\n"
|
||||||
" -E/--cert <cert[:passwd]> Specifies your certificate file and password (HTTPS)\n"
|
" -e/--referer Referer page (H)");
|
||||||
|
puts(" -E/--cert <cert[:passwd]> Specifies your certificate file and password (HTTPS)\n"
|
||||||
" --cacert <file> CA certifciate to verify peer against (HTTPS)\n"
|
" --cacert <file> CA certifciate to verify peer against (HTTPS)\n"
|
||||||
|
" --connect-timeout <seconds> Maximum time allowed for connection\n"
|
||||||
" -f/--fail Fail silently (no output at all) on errors (H)\n"
|
" -f/--fail Fail silently (no output at all) on errors (H)\n"
|
||||||
" -F/--form <name=content> Specify HTTP POST data (H)\n"
|
" -F/--form <name=content> Specify HTTP POST data (H)\n"
|
||||||
" -g/--globoff Disable URL sequences and ranges using {} and []\n"
|
" -g/--globoff Disable URL sequences and ranges using {} and []\n"
|
||||||
" -h/--help This help text\n"
|
" -h/--help This help text\n"
|
||||||
" -H/--header <line> Custom header to pass to server. (H)\n"
|
" -H/--header <line> Custom header to pass to server. (H)");
|
||||||
" -i/--include Include the HTTP-header in the output (H)\n"
|
puts(" -i/--include Include the HTTP-header in the output (H)\n"
|
||||||
" -I/--head Fetch document info only (HTTP HEAD/FTP SIZE)\n"
|
" -I/--head Fetch document info only (HTTP HEAD/FTP SIZE)\n"
|
||||||
" --interface <interface> Specify the interface to be used\n"
|
" --interface <interface> Specify the interface to be used\n"
|
||||||
" --krb4 <level> Enable krb4 with specified security level (F)\n"
|
" --krb4 <level> Enable krb4 with specified security level (F)\n"
|
||||||
" -K/--config Specify which config file to read\n"
|
" -K/--config Specify which config file to read\n"
|
||||||
" -l/--list-only List only names of an FTP directory (F)\n"
|
" -l/--list-only List only names of an FTP directory (F)");
|
||||||
" -L/--location Follow Location: hints (H)\n"
|
puts(" -L/--location Follow Location: hints (H)\n"
|
||||||
" -m/--max-time <seconds> Maximum time allowed for the transfer\n"
|
" -m/--max-time <seconds> Maximum time allowed for the transfer\n"
|
||||||
" -M/--manual Display huge help text\n"
|
" -M/--manual Display huge help text\n"
|
||||||
" -n/--netrc Read .netrc for user name and password\n"
|
" -n/--netrc Read .netrc for user name and password\n"
|
||||||
" -N/--no-buffer Disables the buffering of the output stream\n"
|
" -N/--no-buffer Disables the buffering of the output stream");
|
||||||
" -o/--output <file> Write output to <file> instead of stdout\n"
|
puts(" -o/--output <file> Write output to <file> instead of stdout\n"
|
||||||
" -O/--remote-name Write output to a file named as the remote file\n"
|
" -O/--remote-name Write output to a file named as the remote file\n"
|
||||||
" -p/--proxytunnel Perform non-HTTP services through a HTTP proxy\n"
|
" -p/--proxytunnel Perform non-HTTP services through a HTTP proxy\n"
|
||||||
" -P/--ftpport <address> Use PORT with address instead of PASV when ftping (F)\n"
|
" -P/--ftpport <address> Use PORT with address instead of PASV when ftping (F)\n"
|
||||||
" -q When used as the first parameter disables .curlrc\n"
|
" -q When used as the first parameter disables .curlrc\n"
|
||||||
" -Q/--quote <cmd> Send QUOTE command to FTP before file transfer (F)\n"
|
" -Q/--quote <cmd> Send QUOTE command to FTP before file transfer (F)");
|
||||||
" -r/--range <range> Retrieve a byte range from a HTTP/1.1 or FTP server\n"
|
puts(" -r/--range <range> Retrieve a byte range from a HTTP/1.1 or FTP server\n"
|
||||||
" -s/--silent Silent mode. Don't output anything\n"
|
" -s/--silent Silent mode. Don't output anything\n"
|
||||||
" -S/--show-error Show error. With -s, make curl show errors when they occur\n"
|
" -S/--show-error Show error. With -s, make curl show errors when they occur\n"
|
||||||
|
" --stderr <file> Where to redirect stderr. - means stdout.\n"
|
||||||
" -t/--telnet-option <OPT=val> Set telnet option\n"
|
" -t/--telnet-option <OPT=val> Set telnet option\n"
|
||||||
" -T/--upload-file <file> Transfer/upload <file> to remote site\n"
|
" -T/--upload-file <file> Transfer/upload <file> to remote site\n"
|
||||||
" --url <URL> Another way to specify URL to work with\n"
|
" --url <URL> Another way to specify URL to work with");
|
||||||
" -u/--user <user[:password]> Specify user and password to use\n"
|
puts(" -u/--user <user[:password]> Specify user and password to use\n"
|
||||||
" -U/--proxy-user <user[:password]> Specify Proxy authentication\n"
|
" -U/--proxy-user <user[:password]> Specify Proxy authentication\n"
|
||||||
" -v/--verbose Makes the operation more talkative\n"
|
" -v/--verbose Makes the operation more talkative\n"
|
||||||
" -V/--version Outputs version number then quits\n"
|
" -V/--version Outputs version number then quits\n"
|
||||||
" -w/--write-out [format] What to output after completion\n"
|
" -w/--write-out [format] What to output after completion\n"
|
||||||
" -x/--proxy <host[:port]> Use proxy. (Default port is 1080)\n"
|
" -x/--proxy <host[:port]> Use proxy. (Default port is 1080)\n"
|
||||||
" -X/--request <command> Specific request command to use\n"
|
" --random-file <file> File to use for reading random data from (SSL)\n"
|
||||||
" -y/--speed-time Time needed to trig speed-limit abort. Defaults to 30\n"
|
" -X/--request <command> Specific request command to use");
|
||||||
|
puts(" -y/--speed-time Time needed to trig speed-limit abort. Defaults to 30\n"
|
||||||
" -Y/--speed-limit Stop transfer if below speed-limit for 'speed-time' secs\n"
|
" -Y/--speed-limit Stop transfer if below speed-limit for 'speed-time' secs\n"
|
||||||
" -z/--time-cond <time> Includes a time condition to the server (H)\n"
|
" -z/--time-cond <time> Includes a time condition to the server (H)\n"
|
||||||
" -Z/--max-redirs <num> Set maximum number of redirections allowed (H)\n"
|
" -Z/--max-redirs <num> Set maximum number of redirections allowed (H)\n"
|
||||||
" -2/--sslv2 Force usage of SSLv2 (H)\n"
|
" -2/--sslv2 Force usage of SSLv2 (H)\n"
|
||||||
" -3/--sslv3 Force usage of SSLv3 (H)\n"
|
" -3/--sslv3 Force usage of SSLv3 (H)");
|
||||||
" -#/--progress-bar Display transfer progress as a progress bar\n"
|
puts(" -#/--progress-bar Display transfer progress as a progress bar\n"
|
||||||
" --crlf Convert LF to CRLF in upload. Useful for MVS (OS/390)\n"
|
" --crlf Convert LF to CRLF in upload. Useful for MVS (OS/390)");
|
||||||
" --stderr <file> Where to redirect stderr. - means stdout.\n",
|
|
||||||
curl_version()
|
|
||||||
);
|
|
||||||
}
|
}
|
||||||
|
|
||||||
struct LongShort {
|
struct LongShort {
|
||||||
@@ -306,6 +301,8 @@ struct LongShort {
|
|||||||
};
|
};
|
||||||
|
|
||||||
struct Configurable {
|
struct Configurable {
|
||||||
|
char *random_file;
|
||||||
|
char *egd_file;
|
||||||
char *useragent;
|
char *useragent;
|
||||||
char *cookie;
|
char *cookie;
|
||||||
bool use_resume;
|
bool use_resume;
|
||||||
@@ -314,6 +311,7 @@ struct Configurable {
|
|||||||
long postfieldsize;
|
long postfieldsize;
|
||||||
char *referer;
|
char *referer;
|
||||||
long timeout;
|
long timeout;
|
||||||
|
long connecttimeout;
|
||||||
long maxredirs;
|
long maxredirs;
|
||||||
char *headerfile;
|
char *headerfile;
|
||||||
char *ftpport;
|
char *ftpport;
|
||||||
@@ -525,6 +523,9 @@ static ParameterError getparameter(char *flag, /* f or -long-flag */
|
|||||||
{"7", "interface", TRUE},
|
{"7", "interface", TRUE},
|
||||||
{"6", "krb4", TRUE},
|
{"6", "krb4", TRUE},
|
||||||
{"5", "url", TRUE},
|
{"5", "url", TRUE},
|
||||||
|
{"5a", "random-file", TRUE},
|
||||||
|
{"5b", "egd-file", TRUE},
|
||||||
|
{"5c", "connect-timeout", TRUE},
|
||||||
|
|
||||||
{"2", "sslv2", FALSE},
|
{"2", "sslv2", FALSE},
|
||||||
{"3", "sslv3", FALSE},
|
{"3", "sslv3", FALSE},
|
||||||
@@ -674,29 +675,40 @@ static ParameterError getparameter(char *flag, /* f or -long-flag */
|
|||||||
GetStr(&config->krb4level, nextarg);
|
GetStr(&config->krb4level, nextarg);
|
||||||
break;
|
break;
|
||||||
case '5':
|
case '5':
|
||||||
/* the URL! */
|
switch(subletter) {
|
||||||
{
|
case 'a': /* random-file */
|
||||||
struct getout *url;
|
GetStr(&config->random_file, nextarg);
|
||||||
if(config->url_get || (config->url_get=config->url_list)) {
|
break;
|
||||||
/* there's a node here, if it already is filled-in continue to find
|
case 'b': /* egd-file */
|
||||||
an "empty" node */
|
GetStr(&config->egd_file, nextarg);
|
||||||
while(config->url_get && (config->url_get->flags&GETOUT_URL))
|
break;
|
||||||
config->url_get = config->url_get->next;
|
case 'c': /* connect-timeout */
|
||||||
}
|
config->connecttimeout=atoi(nextarg);
|
||||||
|
break;
|
||||||
|
default: /* the URL! */
|
||||||
|
{
|
||||||
|
struct getout *url;
|
||||||
|
if(config->url_get || (config->url_get=config->url_list)) {
|
||||||
|
/* there's a node here, if it already is filled-in continue to find
|
||||||
|
an "empty" node */
|
||||||
|
while(config->url_get && (config->url_get->flags&GETOUT_URL))
|
||||||
|
config->url_get = config->url_get->next;
|
||||||
|
}
|
||||||
|
|
||||||
/* now there might or might not be an available node to fill in! */
|
/* now there might or might not be an available node to fill in! */
|
||||||
|
|
||||||
if(config->url_get)
|
if(config->url_get)
|
||||||
/* existing node */
|
/* existing node */
|
||||||
url = config->url_get;
|
url = config->url_get;
|
||||||
else
|
else
|
||||||
/* there was no free node, create one! */
|
/* there was no free node, create one! */
|
||||||
url=new_getout(config);
|
url=new_getout(config);
|
||||||
|
|
||||||
if(url) {
|
if(url) {
|
||||||
/* fill in the URL */
|
/* fill in the URL */
|
||||||
GetStr(&url->url, nextarg);
|
GetStr(&url->url, nextarg);
|
||||||
url->flags |= GETOUT_URL;
|
url->flags |= GETOUT_URL;
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
break;
|
break;
|
||||||
@@ -1368,6 +1380,10 @@ void progressbarinit(struct ProgressData *bar)
|
|||||||
|
|
||||||
void free_config_fields(struct Configurable *config)
|
void free_config_fields(struct Configurable *config)
|
||||||
{
|
{
|
||||||
|
if(config->random_file)
|
||||||
|
free(config->random_file);
|
||||||
|
if(config->egd_file)
|
||||||
|
free(config->egd_file);
|
||||||
if(config->userpwd)
|
if(config->userpwd)
|
||||||
free(config->userpwd);
|
free(config->userpwd);
|
||||||
if(config->postfields)
|
if(config->postfields)
|
||||||
@@ -1441,12 +1457,10 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
curl_memdebug("memdump");
|
curl_memdebug("memdump");
|
||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
main_init(); /* inits winsock crap for windows */
|
||||||
|
|
||||||
config->showerror=TRUE;
|
config->showerror=TRUE;
|
||||||
config->conf=CONF_DEFAULT;
|
config->conf=CONF_DEFAULT;
|
||||||
#if 0
|
|
||||||
config->crlf=FALSE;
|
|
||||||
config->quote=NULL;
|
|
||||||
#endif
|
|
||||||
|
|
||||||
if(argc>1 &&
|
if(argc>1 &&
|
||||||
(!strnequal("--", argv[1], 2) && (argv[1][0] == '-')) &&
|
(!strnequal("--", argv[1], 2) && (argv[1][0] == '-')) &&
|
||||||
@@ -1455,9 +1469,6 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
* The first flag, that is not a verbose name, but a shortname
|
* The first flag, that is not a verbose name, but a shortname
|
||||||
* and it includes the 'q' flag!
|
* and it includes the 'q' flag!
|
||||||
*/
|
*/
|
||||||
#if 0
|
|
||||||
fprintf(stderr, "I TURNED OFF THE CRAP\n");
|
|
||||||
#endif
|
|
||||||
;
|
;
|
||||||
}
|
}
|
||||||
else {
|
else {
|
||||||
@@ -1537,6 +1548,15 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
else
|
else
|
||||||
allocuseragent = TRUE;
|
allocuseragent = TRUE;
|
||||||
|
|
||||||
|
/*
|
||||||
|
* Get a curl handle to use for all forthcoming curl transfers. Cleanup
|
||||||
|
* when all transfers are done. This is supported with libcurl 7.7 and
|
||||||
|
* should not be attempted on previous versions.
|
||||||
|
*/
|
||||||
|
curl = curl_easy_init();
|
||||||
|
if(!curl)
|
||||||
|
return CURLE_FAILED_INIT;
|
||||||
|
|
||||||
urlnode = config->url_list;
|
urlnode = config->url_list;
|
||||||
|
|
||||||
/* loop through the list of given URLs */
|
/* loop through the list of given URLs */
|
||||||
@@ -1722,121 +1742,114 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
#endif
|
#endif
|
||||||
|
|
||||||
|
|
||||||
main_init();
|
curl_easy_setopt(curl, CURLOPT_FILE, (FILE *)&outs); /* where to store */
|
||||||
|
/* what call to write: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, my_fwrite);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_INFILE, infd); /* for uploads */
|
||||||
|
/* size of uploaded file: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_INFILESIZE, infilesize);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_URL, url); /* what to fetch */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_PROXY, config->proxy); /* proxy to use */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_VERBOSE, config->conf&CONF_VERBOSE);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_HEADER, config->conf&CONF_HEADER);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, config->conf&CONF_NOPROGRESS);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_NOBODY, config->conf&CONF_NOBODY);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FAILONERROR,
|
||||||
|
config->conf&CONF_FAILONERROR);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_UPLOAD, config->conf&CONF_UPLOAD);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FTPLISTONLY,
|
||||||
|
config->conf&CONF_FTPLISTONLY);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FTPAPPEND, config->conf&CONF_FTPAPPEND);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_NETRC, config->conf&CONF_NETRC);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION,
|
||||||
|
config->conf&CONF_FOLLOWLOCATION);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TRANSFERTEXT, config->conf&CONF_GETTEXT);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_MUTE, config->conf&CONF_MUTE);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_USERPWD, config->userpwd);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_PROXYUSERPWD, config->proxyuserpwd);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_RANGE, config->range);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorbuffer);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TIMEOUT, config->timeout);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, config->postfields);
|
||||||
|
|
||||||
|
/* new in libcurl 7.2: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, config->postfieldsize);
|
||||||
|
|
||||||
|
curl_easy_setopt(curl, CURLOPT_REFERER, config->referer);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_AUTOREFERER,
|
||||||
|
config->conf&CONF_AUTO_REFERER);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_USERAGENT, config->useragent);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FTPPORT, config->ftpport);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_LOW_SPEED_LIMIT, config->low_speed_limit);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_LOW_SPEED_TIME, config->low_speed_time);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_RESUME_FROM,
|
||||||
|
config->use_resume?config->resume_from:0);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_COOKIE, config->cookie);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_HTTPHEADER, config->headers);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_HTTPPOST, config->httppost);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_SSLCERT, config->cert);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_SSLCERTPASSWD, config->cert_passwd);
|
||||||
|
|
||||||
curl = curl_easy_init();
|
if(config->cacert) {
|
||||||
if(curl) {
|
/* available from libcurl 7.5: */
|
||||||
curl_easy_setopt(curl, CURLOPT_FILE, (FILE *)&outs); /* where to store */
|
curl_easy_setopt(curl, CURLOPT_CAINFO, config->cacert);
|
||||||
/* what call to write: */
|
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, TRUE);
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEFUNCTION, my_fwrite);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_INFILE, infd); /* for uploads */
|
|
||||||
/* size of uploaded file: */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_INFILESIZE, infilesize);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_URL, url); /* what to fetch */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_PROXY, config->proxy); /* proxy to use */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_VERBOSE, config->conf&CONF_VERBOSE);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_HEADER, config->conf&CONF_HEADER);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_NOPROGRESS, config->conf&CONF_NOPROGRESS);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_NOBODY, config->conf&CONF_NOBODY);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FAILONERROR,
|
|
||||||
config->conf&CONF_FAILONERROR);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_UPLOAD, config->conf&CONF_UPLOAD);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FTPLISTONLY,
|
|
||||||
config->conf&CONF_FTPLISTONLY);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FTPAPPEND, config->conf&CONF_FTPAPPEND);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_NETRC, config->conf&CONF_NETRC);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FOLLOWLOCATION,
|
|
||||||
config->conf&CONF_FOLLOWLOCATION);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_TRANSFERTEXT, config->conf&CONF_GETTEXT);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_MUTE, config->conf&CONF_MUTE);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_USERPWD, config->userpwd);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_PROXYUSERPWD, config->proxyuserpwd);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_RANGE, config->range);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_ERRORBUFFER, errorbuffer);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_TIMEOUT, config->timeout);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_POSTFIELDS, config->postfields);
|
|
||||||
|
|
||||||
/* new in libcurl 7.2: */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, config->postfieldsize);
|
|
||||||
|
|
||||||
curl_easy_setopt(curl, CURLOPT_REFERER, config->referer);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_AUTOREFERER,
|
|
||||||
config->conf&CONF_AUTO_REFERER);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_USERAGENT, config->useragent);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FTPPORT, config->ftpport);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_LOW_SPEED_LIMIT, config->low_speed_limit);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_LOW_SPEED_TIME, config->low_speed_time);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_RESUME_FROM,
|
|
||||||
config->use_resume?config->resume_from:0);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_COOKIE, config->cookie);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_HTTPHEADER, config->headers);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_HTTPPOST, config->httppost);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_SSLCERT, config->cert);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_SSLCERTPASSWD, config->cert_passwd);
|
|
||||||
|
|
||||||
if(config->cacert) {
|
|
||||||
/* available from libcurl 7.5: */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_CAINFO, config->cacert);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_SSL_VERIFYPEER, TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
if(config->conf&(CONF_NOBODY|CONF_USEREMOTETIME)) {
|
|
||||||
/* no body or use remote time */
|
|
||||||
/* new in 7.5 */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_FILETIME, TRUE);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* 7.5 news: */
|
|
||||||
if (config->maxredirs)
|
|
||||||
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, config->maxredirs);
|
|
||||||
else
|
|
||||||
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, DEFAULT_MAXREDIRS);
|
|
||||||
|
|
||||||
curl_easy_setopt(curl, CURLOPT_CRLF, config->crlf);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_QUOTE, config->quote);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_POSTQUOTE, config->postquote);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_WRITEHEADER,
|
|
||||||
config->headerfile?&heads:NULL);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_COOKIEFILE, config->cookiefile);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_SSLVERSION, config->ssl_version);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_TIMECONDITION, config->timecond);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_TIMEVALUE, config->condtime);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, config->customrequest);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_STDERR, config->errors);
|
|
||||||
|
|
||||||
/* three new ones in libcurl 7.3: */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_HTTPPROXYTUNNEL, config->proxytunnel);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_INTERFACE, config->iface);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_KRB4LEVEL, config->krb4level);
|
|
||||||
|
|
||||||
if((config->progressmode == CURL_PROGRESS_BAR) &&
|
|
||||||
!(config->conf&(CONF_NOPROGRESS|CONF_MUTE))) {
|
|
||||||
/* we want the alternative style, then we have to implement it
|
|
||||||
ourselves! */
|
|
||||||
progressbarinit(&progressbar);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, myprogress);
|
|
||||||
curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, &progressbar);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* new in libcurl 7.6.2: */
|
|
||||||
curl_easy_setopt(curl, CURLOPT_TELNETOPTIONS, config->telnet_options);
|
|
||||||
|
|
||||||
res = curl_easy_perform(curl);
|
|
||||||
|
|
||||||
if(config->writeout) {
|
|
||||||
ourWriteOut(curl, config->writeout);
|
|
||||||
}
|
|
||||||
|
|
||||||
/* always cleanup */
|
|
||||||
curl_easy_cleanup(curl);
|
|
||||||
|
|
||||||
if((res!=CURLE_OK) && config->showerror)
|
|
||||||
fprintf(config->errors, "curl: (%d) %s\n", res, errorbuffer);
|
|
||||||
}
|
}
|
||||||
else
|
|
||||||
fprintf(config->errors, "curl: failed to init libcurl!\n");
|
if(config->conf&(CONF_NOBODY|CONF_USEREMOTETIME)) {
|
||||||
|
/* no body or use remote time */
|
||||||
|
/* new in 7.5 */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_FILETIME, TRUE);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* 7.5 news: */
|
||||||
|
if (config->maxredirs)
|
||||||
|
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, config->maxredirs);
|
||||||
|
else
|
||||||
|
curl_easy_setopt(curl, CURLOPT_MAXREDIRS, DEFAULT_MAXREDIRS);
|
||||||
|
|
||||||
|
curl_easy_setopt(curl, CURLOPT_CRLF, config->crlf);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_QUOTE, config->quote);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_POSTQUOTE, config->postquote);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_WRITEHEADER,
|
||||||
|
config->headerfile?&heads:NULL);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_COOKIEFILE, config->cookiefile);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_SSLVERSION, config->ssl_version);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TIMECONDITION, config->timecond);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TIMEVALUE, config->condtime);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_CUSTOMREQUEST, config->customrequest);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_STDERR, config->errors);
|
||||||
|
|
||||||
|
/* three new ones in libcurl 7.3: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_HTTPPROXYTUNNEL, config->proxytunnel);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_INTERFACE, config->iface);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_KRB4LEVEL, config->krb4level);
|
||||||
|
|
||||||
|
if((config->progressmode == CURL_PROGRESS_BAR) &&
|
||||||
|
!(config->conf&(CONF_NOPROGRESS|CONF_MUTE))) {
|
||||||
|
/* we want the alternative style, then we have to implement it
|
||||||
|
ourselves! */
|
||||||
|
progressbarinit(&progressbar);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_PROGRESSFUNCTION, myprogress);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_PROGRESSDATA, &progressbar);
|
||||||
|
}
|
||||||
|
|
||||||
|
/* new in libcurl 7.6.2: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_TELNETOPTIONS, config->telnet_options);
|
||||||
|
|
||||||
main_free();
|
/* new in libcurl 7.7: */
|
||||||
|
curl_easy_setopt(curl, CURLOPT_RANDOM_FILE, config->random_file);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_EGDSOCKET, config->egd_file);
|
||||||
|
curl_easy_setopt(curl, CURLOPT_CONNECTTIMEOUT, config->connecttimeout);
|
||||||
|
|
||||||
|
res = curl_easy_perform(curl);
|
||||||
|
|
||||||
|
if(config->writeout) {
|
||||||
|
ourWriteOut(curl, config->writeout);
|
||||||
|
}
|
||||||
|
|
||||||
|
if((res!=CURLE_OK) && config->showerror)
|
||||||
|
fprintf(config->errors, "curl: (%d) %s\n", res, errorbuffer);
|
||||||
|
|
||||||
if((config->errors != stderr) &&
|
if((config->errors != stderr) &&
|
||||||
(config->errors != stdout))
|
(config->errors != stdout))
|
||||||
@@ -1884,6 +1897,11 @@ operate(struct Configurable *config, int argc, char *argv[])
|
|||||||
if(allocuseragent)
|
if(allocuseragent)
|
||||||
free(config->useragent);
|
free(config->useragent);
|
||||||
|
|
||||||
|
/* cleanup the curl handle! */
|
||||||
|
curl_easy_cleanup(curl);
|
||||||
|
|
||||||
|
main_free(); /* cleanup the winsock stuff for windows */
|
||||||
|
|
||||||
return res;
|
return res;
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -75,18 +75,18 @@ for(@out) {
|
|||||||
|
|
||||||
$new = $_;
|
$new = $_;
|
||||||
|
|
||||||
$outsize += length($new);
|
$outsize += length($new)+1; # one for the newline
|
||||||
|
|
||||||
$new =~ s/\\/\\\\/g;
|
$new =~ s/\\/\\\\/g;
|
||||||
$new =~ s/\"/\\\"/g;
|
$new =~ s/\"/\\\"/g;
|
||||||
|
|
||||||
printf("\"%s\\n\"\n", $new);
|
# gcc 2.96 claims ISO C89 only is required to support 509 letter strings
|
||||||
|
if($outsize > 500) {
|
||||||
if($outsize > 10000) {
|
|
||||||
# terminate and make another puts() call here
|
# terminate and make another puts() call here
|
||||||
print ");\n puts(\n";
|
print ");\n puts(\n";
|
||||||
$outsize=0;
|
$outsize=length($new)+1;
|
||||||
}
|
}
|
||||||
|
printf("\"%s\\n\"\n", $new);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@@ -1,3 +1,3 @@
|
|||||||
#define CURL_NAME "curl"
|
#define CURL_NAME "curl"
|
||||||
#define CURL_VERSION "7.7-alpha2"
|
#define CURL_VERSION "7.7.1"
|
||||||
#define CURL_ID CURL_NAME " " CURL_VERSION " (" OS ") "
|
#define CURL_ID CURL_NAME " " CURL_VERSION " (" OS ") "
|
||||||
|
Some files were not shown because too many files have changed in this diff Show More
Reference in New Issue
Block a user