xref: /aosp_15_r20/external/curl/docs/examples/http2-pushinmemory.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  * HTTP/2 server push. Receive all data in memory.
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 <stdlib.h>
30*6236dae4SAndroid Build Coastguard Worker #include <string.h>
31*6236dae4SAndroid Build Coastguard Worker 
32*6236dae4SAndroid Build Coastguard Worker /* curl stuff */
33*6236dae4SAndroid Build Coastguard Worker #include <curl/curl.h>
34*6236dae4SAndroid Build Coastguard Worker 
35*6236dae4SAndroid Build Coastguard Worker struct Memory {
36*6236dae4SAndroid Build Coastguard Worker   char *memory;
37*6236dae4SAndroid Build Coastguard Worker   size_t size;
38*6236dae4SAndroid Build Coastguard Worker };
39*6236dae4SAndroid Build Coastguard Worker 
40*6236dae4SAndroid Build Coastguard Worker static size_t
write_cb(void * contents,size_t size,size_t nmemb,void * userp)41*6236dae4SAndroid Build Coastguard Worker write_cb(void *contents, size_t size, size_t nmemb, void *userp)
42*6236dae4SAndroid Build Coastguard Worker {
43*6236dae4SAndroid Build Coastguard Worker   size_t realsize = size * nmemb;
44*6236dae4SAndroid Build Coastguard Worker   struct Memory *mem = (struct Memory *)userp;
45*6236dae4SAndroid Build Coastguard Worker   char *ptr = realloc(mem->memory, mem->size + realsize + 1);
46*6236dae4SAndroid Build Coastguard Worker   if(!ptr) {
47*6236dae4SAndroid Build Coastguard Worker     /* out of memory! */
48*6236dae4SAndroid Build Coastguard Worker     printf("not enough memory (realloc returned NULL)\n");
49*6236dae4SAndroid Build Coastguard Worker     return 0;
50*6236dae4SAndroid Build Coastguard Worker   }
51*6236dae4SAndroid Build Coastguard Worker 
52*6236dae4SAndroid Build Coastguard Worker   mem->memory = ptr;
53*6236dae4SAndroid Build Coastguard Worker   memcpy(&(mem->memory[mem->size]), contents, realsize);
54*6236dae4SAndroid Build Coastguard Worker   mem->size += realsize;
55*6236dae4SAndroid Build Coastguard Worker   mem->memory[mem->size] = 0;
56*6236dae4SAndroid Build Coastguard Worker 
57*6236dae4SAndroid Build Coastguard Worker   return realsize;
58*6236dae4SAndroid Build Coastguard Worker }
59*6236dae4SAndroid Build Coastguard Worker 
60*6236dae4SAndroid Build Coastguard Worker #define MAX_FILES 10
61*6236dae4SAndroid Build Coastguard Worker static struct Memory files[MAX_FILES];
62*6236dae4SAndroid Build Coastguard Worker static int pushindex = 1;
63*6236dae4SAndroid Build Coastguard Worker 
init_memory(struct Memory * chunk)64*6236dae4SAndroid Build Coastguard Worker static void init_memory(struct Memory *chunk)
65*6236dae4SAndroid Build Coastguard Worker {
66*6236dae4SAndroid Build Coastguard Worker   chunk->memory = malloc(1);  /* grown as needed with realloc */
67*6236dae4SAndroid Build Coastguard Worker   chunk->size = 0;            /* no data at this point */
68*6236dae4SAndroid Build Coastguard Worker }
69*6236dae4SAndroid Build Coastguard Worker 
setup(CURL * hnd)70*6236dae4SAndroid Build Coastguard Worker static void setup(CURL *hnd)
71*6236dae4SAndroid Build Coastguard Worker {
72*6236dae4SAndroid Build Coastguard Worker   /* set the same URL */
73*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_URL, "https://localhost:8443/index.html");
74*6236dae4SAndroid Build Coastguard Worker 
75*6236dae4SAndroid Build Coastguard Worker   /* HTTP/2 please */
76*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_HTTP_VERSION, CURL_HTTP_VERSION_2_0);
77*6236dae4SAndroid Build Coastguard Worker 
78*6236dae4SAndroid Build Coastguard Worker   /* we use a self-signed test server, skip verification during debugging */
79*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_SSL_VERIFYPEER, 0L);
80*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_SSL_VERIFYHOST, 0L);
81*6236dae4SAndroid Build Coastguard Worker 
82*6236dae4SAndroid Build Coastguard Worker   /* write data to a struct  */
83*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_WRITEFUNCTION, write_cb);
84*6236dae4SAndroid Build Coastguard Worker   init_memory(&files[0]);
85*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_WRITEDATA, &files[0]);
86*6236dae4SAndroid Build Coastguard Worker 
87*6236dae4SAndroid Build Coastguard Worker   /* wait for pipe connection to confirm */
88*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(hnd, CURLOPT_PIPEWAIT, 1L);
89*6236dae4SAndroid Build Coastguard Worker }
90*6236dae4SAndroid Build Coastguard Worker 
91*6236dae4SAndroid Build Coastguard Worker /* called when there is an incoming push */
server_push_callback(CURL * parent,CURL * easy,size_t num_headers,struct curl_pushheaders * headers,void * userp)92*6236dae4SAndroid Build Coastguard Worker static int server_push_callback(CURL *parent,
93*6236dae4SAndroid Build Coastguard Worker                                 CURL *easy,
94*6236dae4SAndroid Build Coastguard Worker                                 size_t num_headers,
95*6236dae4SAndroid Build Coastguard Worker                                 struct curl_pushheaders *headers,
96*6236dae4SAndroid Build Coastguard Worker                                 void *userp)
97*6236dae4SAndroid Build Coastguard Worker {
98*6236dae4SAndroid Build Coastguard Worker   char *headp;
99*6236dae4SAndroid Build Coastguard Worker   int *transfers = (int *)userp;
100*6236dae4SAndroid Build Coastguard Worker   (void)parent; /* we have no use for this */
101*6236dae4SAndroid Build Coastguard Worker   (void)num_headers; /* unused */
102*6236dae4SAndroid Build Coastguard Worker 
103*6236dae4SAndroid Build Coastguard Worker   if(pushindex == MAX_FILES)
104*6236dae4SAndroid Build Coastguard Worker     /* cannot fit anymore */
105*6236dae4SAndroid Build Coastguard Worker     return CURL_PUSH_DENY;
106*6236dae4SAndroid Build Coastguard Worker 
107*6236dae4SAndroid Build Coastguard Worker   /* write to this buffer */
108*6236dae4SAndroid Build Coastguard Worker   init_memory(&files[pushindex]);
109*6236dae4SAndroid Build Coastguard Worker   curl_easy_setopt(easy, CURLOPT_WRITEDATA, &files[pushindex]);
110*6236dae4SAndroid Build Coastguard Worker   pushindex++;
111*6236dae4SAndroid Build Coastguard Worker 
112*6236dae4SAndroid Build Coastguard Worker   headp = curl_pushheader_byname(headers, ":path");
113*6236dae4SAndroid Build Coastguard Worker   if(headp)
114*6236dae4SAndroid Build Coastguard Worker     fprintf(stderr, "* Pushed :path '%s'\n", headp /* skip :path + colon */);
115*6236dae4SAndroid Build Coastguard Worker 
116*6236dae4SAndroid Build Coastguard Worker   (*transfers)++; /* one more */
117*6236dae4SAndroid Build Coastguard Worker   return CURL_PUSH_OK;
118*6236dae4SAndroid Build Coastguard Worker }
119*6236dae4SAndroid Build Coastguard Worker 
120*6236dae4SAndroid Build Coastguard Worker 
121*6236dae4SAndroid Build Coastguard Worker /*
122*6236dae4SAndroid Build Coastguard Worker  * Download a file over HTTP/2, take care of server push.
123*6236dae4SAndroid Build Coastguard Worker  */
main(void)124*6236dae4SAndroid Build Coastguard Worker int main(void)
125*6236dae4SAndroid Build Coastguard Worker {
126*6236dae4SAndroid Build Coastguard Worker   CURL *easy;
127*6236dae4SAndroid Build Coastguard Worker   CURLM *multi;
128*6236dae4SAndroid Build Coastguard Worker   int still_running; /* keep number of running handles */
129*6236dae4SAndroid Build Coastguard Worker   int transfers = 1; /* we start with one */
130*6236dae4SAndroid Build Coastguard Worker   int i;
131*6236dae4SAndroid Build Coastguard Worker   struct CURLMsg *m;
132*6236dae4SAndroid Build Coastguard Worker 
133*6236dae4SAndroid Build Coastguard Worker   /* init a multi stack */
134*6236dae4SAndroid Build Coastguard Worker   multi = curl_multi_init();
135*6236dae4SAndroid Build Coastguard Worker 
136*6236dae4SAndroid Build Coastguard Worker   easy = curl_easy_init();
137*6236dae4SAndroid Build Coastguard Worker 
138*6236dae4SAndroid Build Coastguard Worker   /* set options */
139*6236dae4SAndroid Build Coastguard Worker   setup(easy);
140*6236dae4SAndroid Build Coastguard Worker 
141*6236dae4SAndroid Build Coastguard Worker   /* add the easy transfer */
142*6236dae4SAndroid Build Coastguard Worker   curl_multi_add_handle(multi, easy);
143*6236dae4SAndroid Build Coastguard Worker 
144*6236dae4SAndroid Build Coastguard Worker   curl_multi_setopt(multi, CURLMOPT_PIPELINING, CURLPIPE_MULTIPLEX);
145*6236dae4SAndroid Build Coastguard Worker   curl_multi_setopt(multi, CURLMOPT_PUSHFUNCTION, server_push_callback);
146*6236dae4SAndroid Build Coastguard Worker   curl_multi_setopt(multi, CURLMOPT_PUSHDATA, &transfers);
147*6236dae4SAndroid Build Coastguard Worker 
148*6236dae4SAndroid Build Coastguard Worker   while(transfers) {
149*6236dae4SAndroid Build Coastguard Worker     int rc;
150*6236dae4SAndroid Build Coastguard Worker     CURLMcode mcode = curl_multi_perform(multi, &still_running);
151*6236dae4SAndroid Build Coastguard Worker     if(mcode)
152*6236dae4SAndroid Build Coastguard Worker       break;
153*6236dae4SAndroid Build Coastguard Worker 
154*6236dae4SAndroid Build Coastguard Worker     mcode = curl_multi_wait(multi, NULL, 0, 1000, &rc);
155*6236dae4SAndroid Build Coastguard Worker     if(mcode)
156*6236dae4SAndroid Build Coastguard Worker       break;
157*6236dae4SAndroid Build Coastguard Worker 
158*6236dae4SAndroid Build Coastguard Worker 
159*6236dae4SAndroid Build Coastguard Worker     /*
160*6236dae4SAndroid Build Coastguard Worker      * When doing server push, libcurl itself created and added one or more
161*6236dae4SAndroid Build Coastguard Worker      * easy handles but *we* need to clean them up when they are done.
162*6236dae4SAndroid Build Coastguard Worker      */
163*6236dae4SAndroid Build Coastguard Worker     do {
164*6236dae4SAndroid Build Coastguard Worker       int msgq = 0;
165*6236dae4SAndroid Build Coastguard Worker       m = curl_multi_info_read(multi, &msgq);
166*6236dae4SAndroid Build Coastguard Worker       if(m && (m->msg == CURLMSG_DONE)) {
167*6236dae4SAndroid Build Coastguard Worker         CURL *e = m->easy_handle;
168*6236dae4SAndroid Build Coastguard Worker         transfers--;
169*6236dae4SAndroid Build Coastguard Worker         curl_multi_remove_handle(multi, e);
170*6236dae4SAndroid Build Coastguard Worker         curl_easy_cleanup(e);
171*6236dae4SAndroid Build Coastguard Worker       }
172*6236dae4SAndroid Build Coastguard Worker     } while(m);
173*6236dae4SAndroid Build Coastguard Worker 
174*6236dae4SAndroid Build Coastguard Worker   }
175*6236dae4SAndroid Build Coastguard Worker 
176*6236dae4SAndroid Build Coastguard Worker 
177*6236dae4SAndroid Build Coastguard Worker   curl_multi_cleanup(multi);
178*6236dae4SAndroid Build Coastguard Worker 
179*6236dae4SAndroid Build Coastguard Worker   /* 'pushindex' is now the number of received transfers */
180*6236dae4SAndroid Build Coastguard Worker   for(i = 0; i < pushindex; i++) {
181*6236dae4SAndroid Build Coastguard Worker     /* do something fun with the data, and then free it when done */
182*6236dae4SAndroid Build Coastguard Worker     free(files[i].memory);
183*6236dae4SAndroid Build Coastguard Worker   }
184*6236dae4SAndroid Build Coastguard Worker 
185*6236dae4SAndroid Build Coastguard Worker   return 0;
186*6236dae4SAndroid Build Coastguard Worker }
187