xref: /aosp_15_r20/external/curl/docs/examples/post-callback.c (revision 6236dae45794135f37c4eb022389c904c8b0090d)
1*6236dae4SAndroid Build Coastguard Worker /***************************************************************************
2*6236dae4SAndroid Build Coastguard Worker  *                                  _   _ ____  _
3*6236dae4SAndroid Build Coastguard Worker  *  Project                     ___| | | |  _ \| |
4*6236dae4SAndroid Build Coastguard Worker  *                             / __| | | | |_) | |
5*6236dae4SAndroid Build Coastguard Worker  *                            | (__| |_| |  _ <| |___
6*6236dae4SAndroid Build Coastguard Worker  *                             \___|\___/|_| \_\_____|
7*6236dae4SAndroid Build Coastguard Worker  *
8*6236dae4SAndroid Build Coastguard Worker  * Copyright (C) Daniel Stenberg, <[email protected]>, et al.
9*6236dae4SAndroid Build Coastguard Worker  *
10*6236dae4SAndroid Build Coastguard Worker  * This software is licensed as described in the file COPYING, which
11*6236dae4SAndroid Build Coastguard Worker  * you should have received as part of this distribution. The terms
12*6236dae4SAndroid Build Coastguard Worker  * are also available at https://curl.se/docs/copyright.html.
13*6236dae4SAndroid Build Coastguard Worker  *
14*6236dae4SAndroid Build Coastguard Worker  * You may opt to use, copy, modify, merge, publish, distribute and/or sell
15*6236dae4SAndroid Build Coastguard Worker  * copies of the Software, and permit persons to whom the Software is
16*6236dae4SAndroid Build Coastguard Worker  * furnished to do so, under the terms of the COPYING file.
17*6236dae4SAndroid Build Coastguard Worker  *
18*6236dae4SAndroid Build Coastguard Worker  * This software is distributed on an "AS IS" basis, WITHOUT WARRANTY OF ANY
19*6236dae4SAndroid Build Coastguard Worker  * KIND, either express or implied.
20*6236dae4SAndroid Build Coastguard Worker  *
21*6236dae4SAndroid Build Coastguard Worker  * SPDX-License-Identifier: curl
22*6236dae4SAndroid Build Coastguard Worker  *
23*6236dae4SAndroid Build Coastguard Worker  ***************************************************************************/
24*6236dae4SAndroid Build Coastguard Worker /* <DESC>
25*6236dae4SAndroid Build Coastguard Worker  * Issue an HTTP POST and provide the data through the read callback.
26*6236dae4SAndroid Build Coastguard Worker  * </DESC>
27*6236dae4SAndroid Build Coastguard Worker  */
28*6236dae4SAndroid Build Coastguard Worker #include <stdio.h>
29*6236dae4SAndroid Build Coastguard Worker #include <string.h>
30*6236dae4SAndroid Build Coastguard Worker #include <curl/curl.h>
31*6236dae4SAndroid Build Coastguard Worker 
32*6236dae4SAndroid Build Coastguard Worker /* silly test data to POST */
33*6236dae4SAndroid Build Coastguard Worker static const char data[]="Lorem ipsum dolor sit amet, consectetur adipiscing "
34*6236dae4SAndroid Build Coastguard Worker   "elit. Sed vel urna neque. Ut quis leo metus. Quisque eleifend, ex at "
35*6236dae4SAndroid Build Coastguard Worker   "laoreet rhoncus, odio ipsum semper metus, at tempus ante urna in mauris. "
36*6236dae4SAndroid Build Coastguard Worker   "Suspendisse ornare tempor venenatis. Ut dui neque, pellentesque a varius "
37*6236dae4SAndroid Build Coastguard Worker   "eget, mattis vitae ligula. Fusce ut pharetra est. Ut ullamcorper mi ac "
38*6236dae4SAndroid Build Coastguard Worker   "sollicitudin semper. Praesent sit amet tellus varius, posuere nulla non, "
39*6236dae4SAndroid Build Coastguard Worker   "rhoncus ipsum.";
40*6236dae4SAndroid Build Coastguard Worker 
41*6236dae4SAndroid Build Coastguard Worker struct WriteThis {
42*6236dae4SAndroid Build Coastguard Worker   const char *readptr;
43*6236dae4SAndroid Build Coastguard Worker   size_t sizeleft;
44*6236dae4SAndroid Build Coastguard Worker };
45*6236dae4SAndroid Build Coastguard Worker 
read_callback(char * dest,size_t size,size_t nmemb,void * userp)46*6236dae4SAndroid Build Coastguard Worker static size_t read_callback(char *dest, size_t size, size_t nmemb, void *userp)
47*6236dae4SAndroid Build Coastguard Worker {
48*6236dae4SAndroid Build Coastguard Worker   struct WriteThis *wt = (struct WriteThis *)userp;
49*6236dae4SAndroid Build Coastguard Worker   size_t buffer_size = size*nmemb;
50*6236dae4SAndroid Build Coastguard Worker 
51*6236dae4SAndroid Build Coastguard Worker   if(wt->sizeleft) {
52*6236dae4SAndroid Build Coastguard Worker     /* copy as much as possible from the source to the destination */
53*6236dae4SAndroid Build Coastguard Worker     size_t copy_this_much = wt->sizeleft;
54*6236dae4SAndroid Build Coastguard Worker     if(copy_this_much > buffer_size)
55*6236dae4SAndroid Build Coastguard Worker       copy_this_much = buffer_size;
56*6236dae4SAndroid Build Coastguard Worker     memcpy(dest, wt->readptr, copy_this_much);
57*6236dae4SAndroid Build Coastguard Worker 
58*6236dae4SAndroid Build Coastguard Worker     wt->readptr += copy_this_much;
59*6236dae4SAndroid Build Coastguard Worker     wt->sizeleft -= copy_this_much;
60*6236dae4SAndroid Build Coastguard Worker     return copy_this_much; /* we copied this many bytes */
61*6236dae4SAndroid Build Coastguard Worker   }
62*6236dae4SAndroid Build Coastguard Worker 
63*6236dae4SAndroid Build Coastguard Worker   return 0; /* no more data left to deliver */
64*6236dae4SAndroid Build Coastguard Worker }
65*6236dae4SAndroid Build Coastguard Worker 
main(void)66*6236dae4SAndroid Build Coastguard Worker int main(void)
67*6236dae4SAndroid Build Coastguard Worker {
68*6236dae4SAndroid Build Coastguard Worker   CURL *curl;
69*6236dae4SAndroid Build Coastguard Worker   CURLcode res;
70*6236dae4SAndroid Build Coastguard Worker 
71*6236dae4SAndroid Build Coastguard Worker   struct WriteThis wt;
72*6236dae4SAndroid Build Coastguard Worker 
73*6236dae4SAndroid Build Coastguard Worker   wt.readptr = data;
74*6236dae4SAndroid Build Coastguard Worker   wt.sizeleft = strlen(data);
75*6236dae4SAndroid Build Coastguard Worker 
76*6236dae4SAndroid Build Coastguard Worker   /* In Windows, this inits the Winsock stuff */
77*6236dae4SAndroid Build Coastguard Worker   res = curl_global_init(CURL_GLOBAL_DEFAULT);
78*6236dae4SAndroid Build Coastguard Worker   /* Check for errors */
79*6236dae4SAndroid Build Coastguard Worker   if(res != CURLE_OK) {
80*6236dae4SAndroid Build Coastguard Worker     fprintf(stderr, "curl_global_init() failed: %s\n",
81*6236dae4SAndroid Build Coastguard Worker             curl_easy_strerror(res));
82*6236dae4SAndroid Build Coastguard Worker     return 1;
83*6236dae4SAndroid Build Coastguard Worker   }
84*6236dae4SAndroid Build Coastguard Worker 
85*6236dae4SAndroid Build Coastguard Worker   /* get a curl handle */
86*6236dae4SAndroid Build Coastguard Worker   curl = curl_easy_init();
87*6236dae4SAndroid Build Coastguard Worker   if(curl) {
88*6236dae4SAndroid Build Coastguard Worker     /* First set the URL that is about to receive our POST. */
89*6236dae4SAndroid Build Coastguard Worker     curl_easy_setopt(curl, CURLOPT_URL, "https://example.com/index.cgi");
90*6236dae4SAndroid Build Coastguard Worker 
91*6236dae4SAndroid Build Coastguard Worker     /* Now specify we want to POST data */
92*6236dae4SAndroid Build Coastguard Worker     curl_easy_setopt(curl, CURLOPT_POST, 1L);
93*6236dae4SAndroid Build Coastguard Worker 
94*6236dae4SAndroid Build Coastguard Worker     /* we want to use our own read function */
95*6236dae4SAndroid Build Coastguard Worker     curl_easy_setopt(curl, CURLOPT_READFUNCTION, read_callback);
96*6236dae4SAndroid Build Coastguard Worker 
97*6236dae4SAndroid Build Coastguard Worker     /* pointer to pass to our read function */
98*6236dae4SAndroid Build Coastguard Worker     curl_easy_setopt(curl, CURLOPT_READDATA, &wt);
99*6236dae4SAndroid Build Coastguard Worker 
100*6236dae4SAndroid Build Coastguard Worker     /* get verbose debug output please */
101*6236dae4SAndroid Build Coastguard Worker     curl_easy_setopt(curl, CURLOPT_VERBOSE, 1L);
102*6236dae4SAndroid Build Coastguard Worker 
103*6236dae4SAndroid Build Coastguard Worker     /*
104*6236dae4SAndroid Build Coastguard Worker       If you use POST to an HTTP 1.1 server, you can send data without knowing
105*6236dae4SAndroid Build Coastguard Worker       the size before starting the POST if you use chunked encoding. You
106*6236dae4SAndroid Build Coastguard Worker       enable this by adding a header like "Transfer-Encoding: chunked" with
107*6236dae4SAndroid Build Coastguard Worker       CURLOPT_HTTPHEADER. With HTTP 1.0 or without chunked transfer, you must
108*6236dae4SAndroid Build Coastguard Worker       specify the size in the request.
109*6236dae4SAndroid Build Coastguard Worker     */
110*6236dae4SAndroid Build Coastguard Worker #ifdef USE_CHUNKED
111*6236dae4SAndroid Build Coastguard Worker     {
112*6236dae4SAndroid Build Coastguard Worker       struct curl_slist *chunk = NULL;
113*6236dae4SAndroid Build Coastguard Worker 
114*6236dae4SAndroid Build Coastguard Worker       chunk = curl_slist_append(chunk, "Transfer-Encoding: chunked");
115*6236dae4SAndroid Build Coastguard Worker       res = curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
116*6236dae4SAndroid Build Coastguard Worker       /* use curl_slist_free_all() after the *perform() call to free this
117*6236dae4SAndroid Build Coastguard Worker          list again */
118*6236dae4SAndroid Build Coastguard Worker     }
119*6236dae4SAndroid Build Coastguard Worker #else
120*6236dae4SAndroid Build Coastguard Worker     /* Set the expected POST size. If you want to POST large amounts of data,
121*6236dae4SAndroid Build Coastguard Worker        consider CURLOPT_POSTFIELDSIZE_LARGE */
122*6236dae4SAndroid Build Coastguard Worker     curl_easy_setopt(curl, CURLOPT_POSTFIELDSIZE, (long)wt.sizeleft);
123*6236dae4SAndroid Build Coastguard Worker #endif
124*6236dae4SAndroid Build Coastguard Worker 
125*6236dae4SAndroid Build Coastguard Worker #ifdef DISABLE_EXPECT
126*6236dae4SAndroid Build Coastguard Worker     /*
127*6236dae4SAndroid Build Coastguard Worker       Using POST with HTTP 1.1 implies the use of a "Expect: 100-continue"
128*6236dae4SAndroid Build Coastguard Worker       header.  You can disable this header with CURLOPT_HTTPHEADER as usual.
129*6236dae4SAndroid Build Coastguard Worker       NOTE: if you want chunked transfer too, you need to combine these two
130*6236dae4SAndroid Build Coastguard Worker       since you can only set one list of headers with CURLOPT_HTTPHEADER. */
131*6236dae4SAndroid Build Coastguard Worker 
132*6236dae4SAndroid Build Coastguard Worker     /* A less good option would be to enforce HTTP 1.0, but that might also
133*6236dae4SAndroid Build Coastguard Worker        have other implications. */
134*6236dae4SAndroid Build Coastguard Worker     {
135*6236dae4SAndroid Build Coastguard Worker       struct curl_slist *chunk = NULL;
136*6236dae4SAndroid Build Coastguard Worker 
137*6236dae4SAndroid Build Coastguard Worker       chunk = curl_slist_append(chunk, "Expect:");
138*6236dae4SAndroid Build Coastguard Worker       res = curl_easy_setopt(curl, CURLOPT_HTTPHEADER, chunk);
139*6236dae4SAndroid Build Coastguard Worker       /* use curl_slist_free_all() after the *perform() call to free this
140*6236dae4SAndroid Build Coastguard Worker          list again */
141*6236dae4SAndroid Build Coastguard Worker     }
142*6236dae4SAndroid Build Coastguard Worker #endif
143*6236dae4SAndroid Build Coastguard Worker 
144*6236dae4SAndroid Build Coastguard Worker     /* Perform the request, res gets the return code */
145*6236dae4SAndroid Build Coastguard Worker     res = curl_easy_perform(curl);
146*6236dae4SAndroid Build Coastguard Worker     /* Check for errors */
147*6236dae4SAndroid Build Coastguard Worker     if(res != CURLE_OK)
148*6236dae4SAndroid Build Coastguard Worker       fprintf(stderr, "curl_easy_perform() failed: %s\n",
149*6236dae4SAndroid Build Coastguard Worker               curl_easy_strerror(res));
150*6236dae4SAndroid Build Coastguard Worker 
151*6236dae4SAndroid Build Coastguard Worker     /* always cleanup */
152*6236dae4SAndroid Build Coastguard Worker     curl_easy_cleanup(curl);
153*6236dae4SAndroid Build Coastguard Worker   }
154*6236dae4SAndroid Build Coastguard Worker   curl_global_cleanup();
155*6236dae4SAndroid Build Coastguard Worker   return 0;
156*6236dae4SAndroid Build Coastguard Worker }
157