xref: /aosp_15_r20/external/libcups/cgi-bin/var.c (revision 5e7646d21f1134fb0638875d812ef646c12ab91e)
1*5e7646d2SAndroid Build Coastguard Worker /*
2*5e7646d2SAndroid Build Coastguard Worker  * CGI form variable and array functions for CUPS.
3*5e7646d2SAndroid Build Coastguard Worker  *
4*5e7646d2SAndroid Build Coastguard Worker  * Copyright © 2007-2020 by Apple Inc.
5*5e7646d2SAndroid Build Coastguard Worker  * Copyright © 1997-2005 by Easy Software Products.
6*5e7646d2SAndroid Build Coastguard Worker  *
7*5e7646d2SAndroid Build Coastguard Worker  * Licensed under Apache License v2.0.  See the file "LICENSE" for more
8*5e7646d2SAndroid Build Coastguard Worker  * information.
9*5e7646d2SAndroid Build Coastguard Worker  */
10*5e7646d2SAndroid Build Coastguard Worker 
11*5e7646d2SAndroid Build Coastguard Worker /*
12*5e7646d2SAndroid Build Coastguard Worker  * Include necessary headers...
13*5e7646d2SAndroid Build Coastguard Worker  */
14*5e7646d2SAndroid Build Coastguard Worker 
15*5e7646d2SAndroid Build Coastguard Worker /*#define DEBUG*/
16*5e7646d2SAndroid Build Coastguard Worker #include "cgi-private.h"
17*5e7646d2SAndroid Build Coastguard Worker #include <cups/http.h>
18*5e7646d2SAndroid Build Coastguard Worker 
19*5e7646d2SAndroid Build Coastguard Worker 
20*5e7646d2SAndroid Build Coastguard Worker /*
21*5e7646d2SAndroid Build Coastguard Worker  * Session ID name
22*5e7646d2SAndroid Build Coastguard Worker  */
23*5e7646d2SAndroid Build Coastguard Worker 
24*5e7646d2SAndroid Build Coastguard Worker #define CUPS_SID	"org.cups.sid"
25*5e7646d2SAndroid Build Coastguard Worker 
26*5e7646d2SAndroid Build Coastguard Worker 
27*5e7646d2SAndroid Build Coastguard Worker /*
28*5e7646d2SAndroid Build Coastguard Worker  * Data structure to hold all the CGI form variables and arrays...
29*5e7646d2SAndroid Build Coastguard Worker  */
30*5e7646d2SAndroid Build Coastguard Worker 
31*5e7646d2SAndroid Build Coastguard Worker typedef struct				/**** Form variable structure ****/
32*5e7646d2SAndroid Build Coastguard Worker {
33*5e7646d2SAndroid Build Coastguard Worker   char		*name;			/* Name of variable */
34*5e7646d2SAndroid Build Coastguard Worker   int		nvalues,		/* Number of values */
35*5e7646d2SAndroid Build Coastguard Worker 		avalues;		/* Number of values allocated */
36*5e7646d2SAndroid Build Coastguard Worker   char		**values;		/* Value(s) of variable */
37*5e7646d2SAndroid Build Coastguard Worker } _cgi_var_t;
38*5e7646d2SAndroid Build Coastguard Worker 
39*5e7646d2SAndroid Build Coastguard Worker 
40*5e7646d2SAndroid Build Coastguard Worker /*
41*5e7646d2SAndroid Build Coastguard Worker  * Local globals...
42*5e7646d2SAndroid Build Coastguard Worker  */
43*5e7646d2SAndroid Build Coastguard Worker 
44*5e7646d2SAndroid Build Coastguard Worker static int		num_cookies = 0;/* Number of cookies */
45*5e7646d2SAndroid Build Coastguard Worker static cups_option_t	*cookies = NULL;/* Cookies */
46*5e7646d2SAndroid Build Coastguard Worker static int		form_count = 0,	/* Form variable count */
47*5e7646d2SAndroid Build Coastguard Worker 			form_alloc = 0;	/* Number of variables allocated */
48*5e7646d2SAndroid Build Coastguard Worker static _cgi_var_t	*form_vars = NULL;
49*5e7646d2SAndroid Build Coastguard Worker 					/* Form variables */
50*5e7646d2SAndroid Build Coastguard Worker static cgi_file_t	*form_file = NULL;
51*5e7646d2SAndroid Build Coastguard Worker 					/* Uploaded file */
52*5e7646d2SAndroid Build Coastguard Worker 
53*5e7646d2SAndroid Build Coastguard Worker 
54*5e7646d2SAndroid Build Coastguard Worker /*
55*5e7646d2SAndroid Build Coastguard Worker  * Local functions...
56*5e7646d2SAndroid Build Coastguard Worker  */
57*5e7646d2SAndroid Build Coastguard Worker 
58*5e7646d2SAndroid Build Coastguard Worker static void		cgi_add_variable(const char *name, int element,
59*5e7646d2SAndroid Build Coastguard Worker 			                 const char *value);
60*5e7646d2SAndroid Build Coastguard Worker static int		cgi_compare_variables(const _cgi_var_t *v1,
61*5e7646d2SAndroid Build Coastguard Worker 			                      const _cgi_var_t *v2);
62*5e7646d2SAndroid Build Coastguard Worker static _cgi_var_t	*cgi_find_variable(const char *name);
63*5e7646d2SAndroid Build Coastguard Worker static void		cgi_initialize_cookies(void);
64*5e7646d2SAndroid Build Coastguard Worker static int		cgi_initialize_get(void);
65*5e7646d2SAndroid Build Coastguard Worker static int		cgi_initialize_multipart(const char *boundary);
66*5e7646d2SAndroid Build Coastguard Worker static int		cgi_initialize_post(void);
67*5e7646d2SAndroid Build Coastguard Worker static int		cgi_initialize_string(const char *data);
68*5e7646d2SAndroid Build Coastguard Worker static const char	*cgi_passwd(const char *prompt);
69*5e7646d2SAndroid Build Coastguard Worker static const char	*cgi_set_sid(void);
70*5e7646d2SAndroid Build Coastguard Worker static void		cgi_sort_variables(void);
71*5e7646d2SAndroid Build Coastguard Worker static void		cgi_unlink_file(void);
72*5e7646d2SAndroid Build Coastguard Worker 
73*5e7646d2SAndroid Build Coastguard Worker 
74*5e7646d2SAndroid Build Coastguard Worker /*
75*5e7646d2SAndroid Build Coastguard Worker  * 'cgiCheckVariables()' - Check for the presence of "required" variables.
76*5e7646d2SAndroid Build Coastguard Worker  *
77*5e7646d2SAndroid Build Coastguard Worker  * Names may be separated by spaces and/or commas.
78*5e7646d2SAndroid Build Coastguard Worker  */
79*5e7646d2SAndroid Build Coastguard Worker 
80*5e7646d2SAndroid Build Coastguard Worker int					/* O - 1 if all variables present, 0 otherwise */
cgiCheckVariables(const char * names)81*5e7646d2SAndroid Build Coastguard Worker cgiCheckVariables(const char *names)	/* I - Variables to look for */
82*5e7646d2SAndroid Build Coastguard Worker {
83*5e7646d2SAndroid Build Coastguard Worker   char		name[255],		/* Current variable name */
84*5e7646d2SAndroid Build Coastguard Worker 		*s;			/* Pointer in string */
85*5e7646d2SAndroid Build Coastguard Worker   const char	*val;			/* Value of variable */
86*5e7646d2SAndroid Build Coastguard Worker   int		element;		/* Array element number */
87*5e7646d2SAndroid Build Coastguard Worker 
88*5e7646d2SAndroid Build Coastguard Worker 
89*5e7646d2SAndroid Build Coastguard Worker   if (names == NULL)
90*5e7646d2SAndroid Build Coastguard Worker     return (1);
91*5e7646d2SAndroid Build Coastguard Worker 
92*5e7646d2SAndroid Build Coastguard Worker   while (*names != '\0')
93*5e7646d2SAndroid Build Coastguard Worker   {
94*5e7646d2SAndroid Build Coastguard Worker     while (*names == ' ' || *names == ',')
95*5e7646d2SAndroid Build Coastguard Worker       names ++;
96*5e7646d2SAndroid Build Coastguard Worker 
97*5e7646d2SAndroid Build Coastguard Worker     for (s = name; *names != '\0' && *names != ' ' && *names != ','; s ++, names ++)
98*5e7646d2SAndroid Build Coastguard Worker       *s = *names;
99*5e7646d2SAndroid Build Coastguard Worker 
100*5e7646d2SAndroid Build Coastguard Worker     *s = 0;
101*5e7646d2SAndroid Build Coastguard Worker     if (name[0] == '\0')
102*5e7646d2SAndroid Build Coastguard Worker       break;
103*5e7646d2SAndroid Build Coastguard Worker 
104*5e7646d2SAndroid Build Coastguard Worker     if ((s = strrchr(name, '-')) != NULL)
105*5e7646d2SAndroid Build Coastguard Worker     {
106*5e7646d2SAndroid Build Coastguard Worker       *s      = '\0';
107*5e7646d2SAndroid Build Coastguard Worker       element = atoi(s + 1) - 1;
108*5e7646d2SAndroid Build Coastguard Worker       val     = cgiGetArray(name, element);
109*5e7646d2SAndroid Build Coastguard Worker     }
110*5e7646d2SAndroid Build Coastguard Worker     else
111*5e7646d2SAndroid Build Coastguard Worker       val = cgiGetVariable(name);
112*5e7646d2SAndroid Build Coastguard Worker 
113*5e7646d2SAndroid Build Coastguard Worker     if (val == NULL)
114*5e7646d2SAndroid Build Coastguard Worker       return (0);
115*5e7646d2SAndroid Build Coastguard Worker 
116*5e7646d2SAndroid Build Coastguard Worker     if (*val == '\0')
117*5e7646d2SAndroid Build Coastguard Worker     {
118*5e7646d2SAndroid Build Coastguard Worker       free((void *)val);
119*5e7646d2SAndroid Build Coastguard Worker       return (0);	/* Can't be blank, either! */
120*5e7646d2SAndroid Build Coastguard Worker     }
121*5e7646d2SAndroid Build Coastguard Worker 
122*5e7646d2SAndroid Build Coastguard Worker     free((void *)val);
123*5e7646d2SAndroid Build Coastguard Worker   }
124*5e7646d2SAndroid Build Coastguard Worker 
125*5e7646d2SAndroid Build Coastguard Worker   return (1);
126*5e7646d2SAndroid Build Coastguard Worker }
127*5e7646d2SAndroid Build Coastguard Worker 
128*5e7646d2SAndroid Build Coastguard Worker 
129*5e7646d2SAndroid Build Coastguard Worker /*
130*5e7646d2SAndroid Build Coastguard Worker  * 'cgiClearVariables()' - Clear all form variables.
131*5e7646d2SAndroid Build Coastguard Worker  */
132*5e7646d2SAndroid Build Coastguard Worker 
133*5e7646d2SAndroid Build Coastguard Worker void
cgiClearVariables(void)134*5e7646d2SAndroid Build Coastguard Worker cgiClearVariables(void)
135*5e7646d2SAndroid Build Coastguard Worker {
136*5e7646d2SAndroid Build Coastguard Worker   int		i, j;			/* Looping vars */
137*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*v;			/* Current variable */
138*5e7646d2SAndroid Build Coastguard Worker 
139*5e7646d2SAndroid Build Coastguard Worker 
140*5e7646d2SAndroid Build Coastguard Worker   fputs("DEBUG: cgiClearVariables called.\n", stderr);
141*5e7646d2SAndroid Build Coastguard Worker 
142*5e7646d2SAndroid Build Coastguard Worker   for (v = form_vars, i = form_count; i > 0; v ++, i --)
143*5e7646d2SAndroid Build Coastguard Worker   {
144*5e7646d2SAndroid Build Coastguard Worker     free(v->name);
145*5e7646d2SAndroid Build Coastguard Worker     for (j = 0; j < v->nvalues; j ++)
146*5e7646d2SAndroid Build Coastguard Worker       if (v->values[j])
147*5e7646d2SAndroid Build Coastguard Worker         free(v->values[j]);
148*5e7646d2SAndroid Build Coastguard Worker   }
149*5e7646d2SAndroid Build Coastguard Worker 
150*5e7646d2SAndroid Build Coastguard Worker   form_count = 0;
151*5e7646d2SAndroid Build Coastguard Worker 
152*5e7646d2SAndroid Build Coastguard Worker   cgi_unlink_file();
153*5e7646d2SAndroid Build Coastguard Worker }
154*5e7646d2SAndroid Build Coastguard Worker 
155*5e7646d2SAndroid Build Coastguard Worker 
156*5e7646d2SAndroid Build Coastguard Worker /*
157*5e7646d2SAndroid Build Coastguard Worker  * 'cgiGetArray()' - Get an element from a form array.
158*5e7646d2SAndroid Build Coastguard Worker  */
159*5e7646d2SAndroid Build Coastguard Worker 
160*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Element value or NULL */
cgiGetArray(const char * name,int element)161*5e7646d2SAndroid Build Coastguard Worker cgiGetArray(const char *name,		/* I - Name of array variable */
162*5e7646d2SAndroid Build Coastguard Worker             int        element)		/* I - Element number (0 to N) */
163*5e7646d2SAndroid Build Coastguard Worker {
164*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*var;			/* Pointer to variable */
165*5e7646d2SAndroid Build Coastguard Worker 
166*5e7646d2SAndroid Build Coastguard Worker 
167*5e7646d2SAndroid Build Coastguard Worker   if ((var = cgi_find_variable(name)) == NULL)
168*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
169*5e7646d2SAndroid Build Coastguard Worker 
170*5e7646d2SAndroid Build Coastguard Worker   if (element < 0 || element >= var->nvalues)
171*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
172*5e7646d2SAndroid Build Coastguard Worker 
173*5e7646d2SAndroid Build Coastguard Worker   if (var->values[element] == NULL)
174*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
175*5e7646d2SAndroid Build Coastguard Worker 
176*5e7646d2SAndroid Build Coastguard Worker   return (strdup(var->values[element]));
177*5e7646d2SAndroid Build Coastguard Worker }
178*5e7646d2SAndroid Build Coastguard Worker 
179*5e7646d2SAndroid Build Coastguard Worker 
180*5e7646d2SAndroid Build Coastguard Worker /*
181*5e7646d2SAndroid Build Coastguard Worker  * 'cgiGetCookie()' - Get a cookie value.
182*5e7646d2SAndroid Build Coastguard Worker  */
183*5e7646d2SAndroid Build Coastguard Worker 
184*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Value or NULL */
cgiGetCookie(const char * name)185*5e7646d2SAndroid Build Coastguard Worker cgiGetCookie(const char *name)		/* I - Name of cookie */
186*5e7646d2SAndroid Build Coastguard Worker {
187*5e7646d2SAndroid Build Coastguard Worker   return (cupsGetOption(name, num_cookies, cookies));
188*5e7646d2SAndroid Build Coastguard Worker }
189*5e7646d2SAndroid Build Coastguard Worker 
190*5e7646d2SAndroid Build Coastguard Worker 
191*5e7646d2SAndroid Build Coastguard Worker /*
192*5e7646d2SAndroid Build Coastguard Worker  * 'cgiGetFile()' - Get the file (if any) that was submitted in the form.
193*5e7646d2SAndroid Build Coastguard Worker  */
194*5e7646d2SAndroid Build Coastguard Worker 
195*5e7646d2SAndroid Build Coastguard Worker const cgi_file_t *			/* O - Attached file or NULL */
cgiGetFile(void)196*5e7646d2SAndroid Build Coastguard Worker cgiGetFile(void)
197*5e7646d2SAndroid Build Coastguard Worker {
198*5e7646d2SAndroid Build Coastguard Worker   return (form_file);
199*5e7646d2SAndroid Build Coastguard Worker }
200*5e7646d2SAndroid Build Coastguard Worker 
201*5e7646d2SAndroid Build Coastguard Worker 
202*5e7646d2SAndroid Build Coastguard Worker /*
203*5e7646d2SAndroid Build Coastguard Worker  * 'cgiGetSize()' - Get the size of a form array value.
204*5e7646d2SAndroid Build Coastguard Worker  */
205*5e7646d2SAndroid Build Coastguard Worker 
206*5e7646d2SAndroid Build Coastguard Worker int					/* O - Number of elements */
cgiGetSize(const char * name)207*5e7646d2SAndroid Build Coastguard Worker cgiGetSize(const char *name)		/* I - Name of variable */
208*5e7646d2SAndroid Build Coastguard Worker {
209*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*var;			/* Pointer to variable */
210*5e7646d2SAndroid Build Coastguard Worker 
211*5e7646d2SAndroid Build Coastguard Worker 
212*5e7646d2SAndroid Build Coastguard Worker   if ((var = cgi_find_variable(name)) == NULL)
213*5e7646d2SAndroid Build Coastguard Worker     return (0);
214*5e7646d2SAndroid Build Coastguard Worker 
215*5e7646d2SAndroid Build Coastguard Worker   return (var->nvalues);
216*5e7646d2SAndroid Build Coastguard Worker }
217*5e7646d2SAndroid Build Coastguard Worker 
218*5e7646d2SAndroid Build Coastguard Worker 
219*5e7646d2SAndroid Build Coastguard Worker /*
220*5e7646d2SAndroid Build Coastguard Worker  * 'cgiGetVariable()' - Get a CGI variable from the database.
221*5e7646d2SAndroid Build Coastguard Worker  *
222*5e7646d2SAndroid Build Coastguard Worker  * Returns NULL if the variable doesn't exist.  If the variable is an
223*5e7646d2SAndroid Build Coastguard Worker  * array of values, returns the last element.
224*5e7646d2SAndroid Build Coastguard Worker  */
225*5e7646d2SAndroid Build Coastguard Worker 
226*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Value of variable */
cgiGetVariable(const char * name)227*5e7646d2SAndroid Build Coastguard Worker cgiGetVariable(const char *name)	/* I - Name of variable */
228*5e7646d2SAndroid Build Coastguard Worker {
229*5e7646d2SAndroid Build Coastguard Worker   const _cgi_var_t	*var;		/* Returned variable */
230*5e7646d2SAndroid Build Coastguard Worker 
231*5e7646d2SAndroid Build Coastguard Worker 
232*5e7646d2SAndroid Build Coastguard Worker   var = cgi_find_variable(name);
233*5e7646d2SAndroid Build Coastguard Worker 
234*5e7646d2SAndroid Build Coastguard Worker   return ((var == NULL) ? NULL : strdup(var->values[var->nvalues - 1]));
235*5e7646d2SAndroid Build Coastguard Worker }
236*5e7646d2SAndroid Build Coastguard Worker 
237*5e7646d2SAndroid Build Coastguard Worker 
238*5e7646d2SAndroid Build Coastguard Worker /*
239*5e7646d2SAndroid Build Coastguard Worker  * 'cgiInitialize()' - Initialize the CGI variable "database".
240*5e7646d2SAndroid Build Coastguard Worker  */
241*5e7646d2SAndroid Build Coastguard Worker 
242*5e7646d2SAndroid Build Coastguard Worker int					/* O - Non-zero if there was form data */
cgiInitialize(void)243*5e7646d2SAndroid Build Coastguard Worker cgiInitialize(void)
244*5e7646d2SAndroid Build Coastguard Worker {
245*5e7646d2SAndroid Build Coastguard Worker   const char	*method,		/* Form posting method */
246*5e7646d2SAndroid Build Coastguard Worker 		*content_type,		/* Content-Type of post data */
247*5e7646d2SAndroid Build Coastguard Worker 		*cups_sid_cookie,	/* SID cookie */
248*5e7646d2SAndroid Build Coastguard Worker 		*cups_sid_form;		/* SID form variable */
249*5e7646d2SAndroid Build Coastguard Worker 
250*5e7646d2SAndroid Build Coastguard Worker 
251*5e7646d2SAndroid Build Coastguard Worker  /*
252*5e7646d2SAndroid Build Coastguard Worker   * Setup a password callback for authentication...
253*5e7646d2SAndroid Build Coastguard Worker   */
254*5e7646d2SAndroid Build Coastguard Worker 
255*5e7646d2SAndroid Build Coastguard Worker   cupsSetPasswordCB(cgi_passwd);
256*5e7646d2SAndroid Build Coastguard Worker 
257*5e7646d2SAndroid Build Coastguard Worker  /*
258*5e7646d2SAndroid Build Coastguard Worker   * Set the locale so that times, etc. are formatted properly...
259*5e7646d2SAndroid Build Coastguard Worker   */
260*5e7646d2SAndroid Build Coastguard Worker 
261*5e7646d2SAndroid Build Coastguard Worker   setlocale(LC_ALL, "");
262*5e7646d2SAndroid Build Coastguard Worker 
263*5e7646d2SAndroid Build Coastguard Worker #ifdef DEBUG
264*5e7646d2SAndroid Build Coastguard Worker  /*
265*5e7646d2SAndroid Build Coastguard Worker   * Disable output buffering to find bugs...
266*5e7646d2SAndroid Build Coastguard Worker   */
267*5e7646d2SAndroid Build Coastguard Worker 
268*5e7646d2SAndroid Build Coastguard Worker   setbuf(stdout, NULL);
269*5e7646d2SAndroid Build Coastguard Worker #endif /* DEBUG */
270*5e7646d2SAndroid Build Coastguard Worker 
271*5e7646d2SAndroid Build Coastguard Worker  /*
272*5e7646d2SAndroid Build Coastguard Worker   * Get cookies...
273*5e7646d2SAndroid Build Coastguard Worker   */
274*5e7646d2SAndroid Build Coastguard Worker 
275*5e7646d2SAndroid Build Coastguard Worker   cgi_initialize_cookies();
276*5e7646d2SAndroid Build Coastguard Worker 
277*5e7646d2SAndroid Build Coastguard Worker   if ((cups_sid_cookie = cgiGetCookie(CUPS_SID)) == NULL)
278*5e7646d2SAndroid Build Coastguard Worker   {
279*5e7646d2SAndroid Build Coastguard Worker     fputs("DEBUG: " CUPS_SID " cookie not found, initializing!\n", stderr);
280*5e7646d2SAndroid Build Coastguard Worker     cups_sid_cookie = cgi_set_sid();
281*5e7646d2SAndroid Build Coastguard Worker   }
282*5e7646d2SAndroid Build Coastguard Worker 
283*5e7646d2SAndroid Build Coastguard Worker   fprintf(stderr, "DEBUG: " CUPS_SID " cookie is \"%s\"\n", cups_sid_cookie);
284*5e7646d2SAndroid Build Coastguard Worker 
285*5e7646d2SAndroid Build Coastguard Worker  /*
286*5e7646d2SAndroid Build Coastguard Worker   * Get the request method (GET or POST)...
287*5e7646d2SAndroid Build Coastguard Worker   */
288*5e7646d2SAndroid Build Coastguard Worker 
289*5e7646d2SAndroid Build Coastguard Worker   method       = getenv("REQUEST_METHOD");
290*5e7646d2SAndroid Build Coastguard Worker   content_type = getenv("CONTENT_TYPE");
291*5e7646d2SAndroid Build Coastguard Worker   if (!method)
292*5e7646d2SAndroid Build Coastguard Worker     return (0);
293*5e7646d2SAndroid Build Coastguard Worker 
294*5e7646d2SAndroid Build Coastguard Worker  /*
295*5e7646d2SAndroid Build Coastguard Worker   * Grab form data from the corresponding location...
296*5e7646d2SAndroid Build Coastguard Worker   */
297*5e7646d2SAndroid Build Coastguard Worker 
298*5e7646d2SAndroid Build Coastguard Worker   if (!_cups_strcasecmp(method, "GET"))
299*5e7646d2SAndroid Build Coastguard Worker     return (cgi_initialize_get());
300*5e7646d2SAndroid Build Coastguard Worker   else if (!_cups_strcasecmp(method, "POST") && content_type)
301*5e7646d2SAndroid Build Coastguard Worker   {
302*5e7646d2SAndroid Build Coastguard Worker     const char *boundary = strstr(content_type, "boundary=");
303*5e7646d2SAndroid Build Coastguard Worker 
304*5e7646d2SAndroid Build Coastguard Worker     if (boundary)
305*5e7646d2SAndroid Build Coastguard Worker       boundary += 9;
306*5e7646d2SAndroid Build Coastguard Worker 
307*5e7646d2SAndroid Build Coastguard Worker     if (!strncmp(content_type, "multipart/form-data; ", 21))
308*5e7646d2SAndroid Build Coastguard Worker     {
309*5e7646d2SAndroid Build Coastguard Worker       if (!cgi_initialize_multipart(boundary))
310*5e7646d2SAndroid Build Coastguard Worker         return (0);
311*5e7646d2SAndroid Build Coastguard Worker     }
312*5e7646d2SAndroid Build Coastguard Worker     else if (!cgi_initialize_post())
313*5e7646d2SAndroid Build Coastguard Worker       return (0);
314*5e7646d2SAndroid Build Coastguard Worker 
315*5e7646d2SAndroid Build Coastguard Worker     if ((cups_sid_form = cgiGetVariable(CUPS_SID)) == NULL ||
316*5e7646d2SAndroid Build Coastguard Worker 	strcmp(cups_sid_cookie, cups_sid_form))
317*5e7646d2SAndroid Build Coastguard Worker     {
318*5e7646d2SAndroid Build Coastguard Worker       if (cups_sid_form)
319*5e7646d2SAndroid Build Coastguard Worker 	fprintf(stderr, "DEBUG: " CUPS_SID " form variable is \"%s\"\n",
320*5e7646d2SAndroid Build Coastguard Worker 	        cups_sid_form);
321*5e7646d2SAndroid Build Coastguard Worker       else
322*5e7646d2SAndroid Build Coastguard Worker 	fputs("DEBUG: " CUPS_SID " form variable is not present.\n", stderr);
323*5e7646d2SAndroid Build Coastguard Worker 
324*5e7646d2SAndroid Build Coastguard Worker       free((void *)cups_sid_form);
325*5e7646d2SAndroid Build Coastguard Worker 
326*5e7646d2SAndroid Build Coastguard Worker       cgiClearVariables();
327*5e7646d2SAndroid Build Coastguard Worker 
328*5e7646d2SAndroid Build Coastguard Worker       return (0);
329*5e7646d2SAndroid Build Coastguard Worker     }
330*5e7646d2SAndroid Build Coastguard Worker     else
331*5e7646d2SAndroid Build Coastguard Worker     {
332*5e7646d2SAndroid Build Coastguard Worker       free((void *)cups_sid_form);
333*5e7646d2SAndroid Build Coastguard Worker 
334*5e7646d2SAndroid Build Coastguard Worker       return (1);
335*5e7646d2SAndroid Build Coastguard Worker     }
336*5e7646d2SAndroid Build Coastguard Worker   }
337*5e7646d2SAndroid Build Coastguard Worker   else
338*5e7646d2SAndroid Build Coastguard Worker     return (0);
339*5e7646d2SAndroid Build Coastguard Worker }
340*5e7646d2SAndroid Build Coastguard Worker 
341*5e7646d2SAndroid Build Coastguard Worker 
342*5e7646d2SAndroid Build Coastguard Worker /*
343*5e7646d2SAndroid Build Coastguard Worker  * 'cgiIsPOST()' - Determine whether this page was POSTed.
344*5e7646d2SAndroid Build Coastguard Worker  */
345*5e7646d2SAndroid Build Coastguard Worker 
346*5e7646d2SAndroid Build Coastguard Worker int					/* O - 1 if POST, 0 if GET */
cgiIsPOST(void)347*5e7646d2SAndroid Build Coastguard Worker cgiIsPOST(void)
348*5e7646d2SAndroid Build Coastguard Worker {
349*5e7646d2SAndroid Build Coastguard Worker   const char	*method;		/* REQUEST_METHOD environment variable */
350*5e7646d2SAndroid Build Coastguard Worker 
351*5e7646d2SAndroid Build Coastguard Worker 
352*5e7646d2SAndroid Build Coastguard Worker   if ((method = getenv("REQUEST_METHOD")) == NULL)
353*5e7646d2SAndroid Build Coastguard Worker     return (0);
354*5e7646d2SAndroid Build Coastguard Worker   else
355*5e7646d2SAndroid Build Coastguard Worker     return (!strcmp(method, "POST"));
356*5e7646d2SAndroid Build Coastguard Worker }
357*5e7646d2SAndroid Build Coastguard Worker 
358*5e7646d2SAndroid Build Coastguard Worker 
359*5e7646d2SAndroid Build Coastguard Worker /*
360*5e7646d2SAndroid Build Coastguard Worker  * 'cgiSetArray()' - Set array element N to the specified string.
361*5e7646d2SAndroid Build Coastguard Worker  *
362*5e7646d2SAndroid Build Coastguard Worker  * If the variable array is smaller than (element + 1), the intervening
363*5e7646d2SAndroid Build Coastguard Worker  * elements are set to NULL.
364*5e7646d2SAndroid Build Coastguard Worker  */
365*5e7646d2SAndroid Build Coastguard Worker 
366*5e7646d2SAndroid Build Coastguard Worker void
cgiSetArray(const char * name,int element,const char * value)367*5e7646d2SAndroid Build Coastguard Worker cgiSetArray(const char *name,		/* I - Name of variable */
368*5e7646d2SAndroid Build Coastguard Worker             int        element,		/* I - Element number (0 to N) */
369*5e7646d2SAndroid Build Coastguard Worker             const char *value)		/* I - Value of variable */
370*5e7646d2SAndroid Build Coastguard Worker {
371*5e7646d2SAndroid Build Coastguard Worker   int		i;			/* Looping var */
372*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*var;			/* Returned variable */
373*5e7646d2SAndroid Build Coastguard Worker 
374*5e7646d2SAndroid Build Coastguard Worker 
375*5e7646d2SAndroid Build Coastguard Worker   if (name == NULL || value == NULL || element < 0 || element > 100000)
376*5e7646d2SAndroid Build Coastguard Worker     return;
377*5e7646d2SAndroid Build Coastguard Worker 
378*5e7646d2SAndroid Build Coastguard Worker   fprintf(stderr, "DEBUG: cgiSetArray: %s[%d]=\"%s\"\n", name, element, value);
379*5e7646d2SAndroid Build Coastguard Worker 
380*5e7646d2SAndroid Build Coastguard Worker   if ((var = cgi_find_variable(name)) == NULL)
381*5e7646d2SAndroid Build Coastguard Worker   {
382*5e7646d2SAndroid Build Coastguard Worker     cgi_add_variable(name, element, value);
383*5e7646d2SAndroid Build Coastguard Worker     cgi_sort_variables();
384*5e7646d2SAndroid Build Coastguard Worker   }
385*5e7646d2SAndroid Build Coastguard Worker   else
386*5e7646d2SAndroid Build Coastguard Worker   {
387*5e7646d2SAndroid Build Coastguard Worker     if (element >= var->avalues)
388*5e7646d2SAndroid Build Coastguard Worker     {
389*5e7646d2SAndroid Build Coastguard Worker       char **temp;			/* Temporary pointer */
390*5e7646d2SAndroid Build Coastguard Worker 
391*5e7646d2SAndroid Build Coastguard Worker       temp = (char **)realloc((void *)(var->values), sizeof(char *) * (size_t)(element + 16));
392*5e7646d2SAndroid Build Coastguard Worker       if (!temp)
393*5e7646d2SAndroid Build Coastguard Worker         return;
394*5e7646d2SAndroid Build Coastguard Worker 
395*5e7646d2SAndroid Build Coastguard Worker       var->avalues = element + 16;
396*5e7646d2SAndroid Build Coastguard Worker       var->values  = temp;
397*5e7646d2SAndroid Build Coastguard Worker     }
398*5e7646d2SAndroid Build Coastguard Worker 
399*5e7646d2SAndroid Build Coastguard Worker     if (element >= var->nvalues)
400*5e7646d2SAndroid Build Coastguard Worker     {
401*5e7646d2SAndroid Build Coastguard Worker       for (i = var->nvalues; i < element; i ++)
402*5e7646d2SAndroid Build Coastguard Worker 	var->values[i] = NULL;
403*5e7646d2SAndroid Build Coastguard Worker 
404*5e7646d2SAndroid Build Coastguard Worker       var->nvalues = element + 1;
405*5e7646d2SAndroid Build Coastguard Worker     }
406*5e7646d2SAndroid Build Coastguard Worker     else if (var->values[element])
407*5e7646d2SAndroid Build Coastguard Worker       free((char *)var->values[element]);
408*5e7646d2SAndroid Build Coastguard Worker 
409*5e7646d2SAndroid Build Coastguard Worker     var->values[element] = strdup(value);
410*5e7646d2SAndroid Build Coastguard Worker   }
411*5e7646d2SAndroid Build Coastguard Worker }
412*5e7646d2SAndroid Build Coastguard Worker 
413*5e7646d2SAndroid Build Coastguard Worker 
414*5e7646d2SAndroid Build Coastguard Worker /*
415*5e7646d2SAndroid Build Coastguard Worker  * 'cgiSetCookie()' - Set a cookie value.
416*5e7646d2SAndroid Build Coastguard Worker  */
417*5e7646d2SAndroid Build Coastguard Worker 
418*5e7646d2SAndroid Build Coastguard Worker void
cgiSetCookie(const char * name,const char * value,const char * path,const char * domain,time_t expires,int secure)419*5e7646d2SAndroid Build Coastguard Worker cgiSetCookie(const char *name,		/* I - Name */
420*5e7646d2SAndroid Build Coastguard Worker              const char *value,		/* I - Value */
421*5e7646d2SAndroid Build Coastguard Worker              const char *path,		/* I - Path (typically "/") */
422*5e7646d2SAndroid Build Coastguard Worker 	     const char *domain,	/* I - Domain name */
423*5e7646d2SAndroid Build Coastguard Worker 	     time_t     expires,	/* I - Expiration date (0 for session) */
424*5e7646d2SAndroid Build Coastguard Worker 	     int        secure)		/* I - Require SSL */
425*5e7646d2SAndroid Build Coastguard Worker {
426*5e7646d2SAndroid Build Coastguard Worker   num_cookies = cupsAddOption(name, value, num_cookies, &cookies);
427*5e7646d2SAndroid Build Coastguard Worker 
428*5e7646d2SAndroid Build Coastguard Worker   printf("Set-Cookie: %s=%s;", name, value);
429*5e7646d2SAndroid Build Coastguard Worker   if (path)
430*5e7646d2SAndroid Build Coastguard Worker     printf(" path=%s;", path);
431*5e7646d2SAndroid Build Coastguard Worker   if (domain)
432*5e7646d2SAndroid Build Coastguard Worker     printf(" domain=%s;", domain);
433*5e7646d2SAndroid Build Coastguard Worker   if (expires)
434*5e7646d2SAndroid Build Coastguard Worker   {
435*5e7646d2SAndroid Build Coastguard Worker     char	date[256];		/* Date string */
436*5e7646d2SAndroid Build Coastguard Worker 
437*5e7646d2SAndroid Build Coastguard Worker     printf(" expires=%s;", httpGetDateString2(expires, date, sizeof(date)));
438*5e7646d2SAndroid Build Coastguard Worker   }
439*5e7646d2SAndroid Build Coastguard Worker   if (secure)
440*5e7646d2SAndroid Build Coastguard Worker     puts(" httponly; secure;");
441*5e7646d2SAndroid Build Coastguard Worker   else
442*5e7646d2SAndroid Build Coastguard Worker     puts(" httponly;");
443*5e7646d2SAndroid Build Coastguard Worker }
444*5e7646d2SAndroid Build Coastguard Worker 
445*5e7646d2SAndroid Build Coastguard Worker 
446*5e7646d2SAndroid Build Coastguard Worker /*
447*5e7646d2SAndroid Build Coastguard Worker  * 'cgiSetSize()' - Set the array size.
448*5e7646d2SAndroid Build Coastguard Worker  */
449*5e7646d2SAndroid Build Coastguard Worker 
450*5e7646d2SAndroid Build Coastguard Worker void
cgiSetSize(const char * name,int size)451*5e7646d2SAndroid Build Coastguard Worker cgiSetSize(const char *name,		/* I - Name of variable */
452*5e7646d2SAndroid Build Coastguard Worker            int        size)		/* I - Number of elements (0 to N) */
453*5e7646d2SAndroid Build Coastguard Worker {
454*5e7646d2SAndroid Build Coastguard Worker   int		i;			/* Looping var */
455*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*var;			/* Returned variable */
456*5e7646d2SAndroid Build Coastguard Worker 
457*5e7646d2SAndroid Build Coastguard Worker 
458*5e7646d2SAndroid Build Coastguard Worker   if (name == NULL || size < 0 || size > 100000)
459*5e7646d2SAndroid Build Coastguard Worker     return;
460*5e7646d2SAndroid Build Coastguard Worker 
461*5e7646d2SAndroid Build Coastguard Worker   if ((var = cgi_find_variable(name)) == NULL)
462*5e7646d2SAndroid Build Coastguard Worker     return;
463*5e7646d2SAndroid Build Coastguard Worker 
464*5e7646d2SAndroid Build Coastguard Worker   if (size >= var->avalues)
465*5e7646d2SAndroid Build Coastguard Worker   {
466*5e7646d2SAndroid Build Coastguard Worker     char **temp;			/* Temporary pointer */
467*5e7646d2SAndroid Build Coastguard Worker 
468*5e7646d2SAndroid Build Coastguard Worker     temp = (char **)realloc((void *)(var->values), sizeof(char *) * (size_t)(size + 16));
469*5e7646d2SAndroid Build Coastguard Worker     if (!temp)
470*5e7646d2SAndroid Build Coastguard Worker       return;
471*5e7646d2SAndroid Build Coastguard Worker 
472*5e7646d2SAndroid Build Coastguard Worker     var->avalues = size + 16;
473*5e7646d2SAndroid Build Coastguard Worker     var->values  = temp;
474*5e7646d2SAndroid Build Coastguard Worker   }
475*5e7646d2SAndroid Build Coastguard Worker 
476*5e7646d2SAndroid Build Coastguard Worker   if (size > var->nvalues)
477*5e7646d2SAndroid Build Coastguard Worker   {
478*5e7646d2SAndroid Build Coastguard Worker     for (i = var->nvalues; i < size; i ++)
479*5e7646d2SAndroid Build Coastguard Worker       var->values[i] = NULL;
480*5e7646d2SAndroid Build Coastguard Worker   }
481*5e7646d2SAndroid Build Coastguard Worker   else if (size < var->nvalues)
482*5e7646d2SAndroid Build Coastguard Worker   {
483*5e7646d2SAndroid Build Coastguard Worker     for (i = size; i < var->nvalues; i ++)
484*5e7646d2SAndroid Build Coastguard Worker       if (var->values[i])
485*5e7646d2SAndroid Build Coastguard Worker         free((void *)(var->values[i]));
486*5e7646d2SAndroid Build Coastguard Worker   }
487*5e7646d2SAndroid Build Coastguard Worker 
488*5e7646d2SAndroid Build Coastguard Worker   var->nvalues = size;
489*5e7646d2SAndroid Build Coastguard Worker }
490*5e7646d2SAndroid Build Coastguard Worker 
491*5e7646d2SAndroid Build Coastguard Worker 
492*5e7646d2SAndroid Build Coastguard Worker /*
493*5e7646d2SAndroid Build Coastguard Worker  * 'cgiSetVariable()' - Set a CGI variable in the database.
494*5e7646d2SAndroid Build Coastguard Worker  *
495*5e7646d2SAndroid Build Coastguard Worker  * If the variable is an array, this truncates the array to a single element.
496*5e7646d2SAndroid Build Coastguard Worker  */
497*5e7646d2SAndroid Build Coastguard Worker 
498*5e7646d2SAndroid Build Coastguard Worker void
cgiSetVariable(const char * name,const char * value)499*5e7646d2SAndroid Build Coastguard Worker cgiSetVariable(const char *name,	/* I - Name of variable */
500*5e7646d2SAndroid Build Coastguard Worker                const char *value)	/* I - Value of variable */
501*5e7646d2SAndroid Build Coastguard Worker {
502*5e7646d2SAndroid Build Coastguard Worker   int		i;			/* Looping var */
503*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*var;			/* Returned variable */
504*5e7646d2SAndroid Build Coastguard Worker 
505*5e7646d2SAndroid Build Coastguard Worker 
506*5e7646d2SAndroid Build Coastguard Worker   if (name == NULL || value == NULL)
507*5e7646d2SAndroid Build Coastguard Worker     return;
508*5e7646d2SAndroid Build Coastguard Worker 
509*5e7646d2SAndroid Build Coastguard Worker   fprintf(stderr, "cgiSetVariable: %s=\"%s\"\n", name, value);
510*5e7646d2SAndroid Build Coastguard Worker 
511*5e7646d2SAndroid Build Coastguard Worker   if ((var = cgi_find_variable(name)) == NULL)
512*5e7646d2SAndroid Build Coastguard Worker   {
513*5e7646d2SAndroid Build Coastguard Worker     cgi_add_variable(name, 0, value);
514*5e7646d2SAndroid Build Coastguard Worker     cgi_sort_variables();
515*5e7646d2SAndroid Build Coastguard Worker   }
516*5e7646d2SAndroid Build Coastguard Worker   else
517*5e7646d2SAndroid Build Coastguard Worker   {
518*5e7646d2SAndroid Build Coastguard Worker     for (i = 0; i < var->nvalues; i ++)
519*5e7646d2SAndroid Build Coastguard Worker       if (var->values[i])
520*5e7646d2SAndroid Build Coastguard Worker         free((char *)var->values[i]);
521*5e7646d2SAndroid Build Coastguard Worker 
522*5e7646d2SAndroid Build Coastguard Worker     var->values[0] = strdup(value);
523*5e7646d2SAndroid Build Coastguard Worker     var->nvalues   = 1;
524*5e7646d2SAndroid Build Coastguard Worker   }
525*5e7646d2SAndroid Build Coastguard Worker }
526*5e7646d2SAndroid Build Coastguard Worker 
527*5e7646d2SAndroid Build Coastguard Worker 
528*5e7646d2SAndroid Build Coastguard Worker /*
529*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_add_variable()' - Add a form variable.
530*5e7646d2SAndroid Build Coastguard Worker  */
531*5e7646d2SAndroid Build Coastguard Worker 
532*5e7646d2SAndroid Build Coastguard Worker static void
cgi_add_variable(const char * name,int element,const char * value)533*5e7646d2SAndroid Build Coastguard Worker cgi_add_variable(const char *name,	/* I - Variable name */
534*5e7646d2SAndroid Build Coastguard Worker 		 int        element,	/* I - Array element number */
535*5e7646d2SAndroid Build Coastguard Worker                  const char *value)	/* I - Variable value */
536*5e7646d2SAndroid Build Coastguard Worker {
537*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	*var;			/* New variable */
538*5e7646d2SAndroid Build Coastguard Worker 
539*5e7646d2SAndroid Build Coastguard Worker 
540*5e7646d2SAndroid Build Coastguard Worker   if (name == NULL || value == NULL || element < 0 || element > 100000)
541*5e7646d2SAndroid Build Coastguard Worker     return;
542*5e7646d2SAndroid Build Coastguard Worker 
543*5e7646d2SAndroid Build Coastguard Worker   if (form_count >= form_alloc)
544*5e7646d2SAndroid Build Coastguard Worker   {
545*5e7646d2SAndroid Build Coastguard Worker     _cgi_var_t	*temp_vars;		/* Temporary form pointer */
546*5e7646d2SAndroid Build Coastguard Worker 
547*5e7646d2SAndroid Build Coastguard Worker 
548*5e7646d2SAndroid Build Coastguard Worker     if (form_alloc == 0)
549*5e7646d2SAndroid Build Coastguard Worker       temp_vars = malloc(sizeof(_cgi_var_t) * 16);
550*5e7646d2SAndroid Build Coastguard Worker     else
551*5e7646d2SAndroid Build Coastguard Worker       temp_vars = realloc(form_vars, (size_t)(form_alloc + 16) * sizeof(_cgi_var_t));
552*5e7646d2SAndroid Build Coastguard Worker 
553*5e7646d2SAndroid Build Coastguard Worker     if (!temp_vars)
554*5e7646d2SAndroid Build Coastguard Worker       return;
555*5e7646d2SAndroid Build Coastguard Worker 
556*5e7646d2SAndroid Build Coastguard Worker     form_vars  = temp_vars;
557*5e7646d2SAndroid Build Coastguard Worker     form_alloc += 16;
558*5e7646d2SAndroid Build Coastguard Worker   }
559*5e7646d2SAndroid Build Coastguard Worker 
560*5e7646d2SAndroid Build Coastguard Worker   var = form_vars + form_count;
561*5e7646d2SAndroid Build Coastguard Worker 
562*5e7646d2SAndroid Build Coastguard Worker   if ((var->values = calloc((size_t)element + 1, sizeof(char *))) == NULL)
563*5e7646d2SAndroid Build Coastguard Worker     return;
564*5e7646d2SAndroid Build Coastguard Worker 
565*5e7646d2SAndroid Build Coastguard Worker   var->name            = strdup(name);
566*5e7646d2SAndroid Build Coastguard Worker   var->nvalues         = element + 1;
567*5e7646d2SAndroid Build Coastguard Worker   var->avalues         = element + 1;
568*5e7646d2SAndroid Build Coastguard Worker   var->values[element] = strdup(value);
569*5e7646d2SAndroid Build Coastguard Worker 
570*5e7646d2SAndroid Build Coastguard Worker   form_count ++;
571*5e7646d2SAndroid Build Coastguard Worker }
572*5e7646d2SAndroid Build Coastguard Worker 
573*5e7646d2SAndroid Build Coastguard Worker 
574*5e7646d2SAndroid Build Coastguard Worker /*
575*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_compare_variables()' - Compare two variables.
576*5e7646d2SAndroid Build Coastguard Worker  */
577*5e7646d2SAndroid Build Coastguard Worker 
578*5e7646d2SAndroid Build Coastguard Worker static int				/* O - Result of comparison */
cgi_compare_variables(const _cgi_var_t * v1,const _cgi_var_t * v2)579*5e7646d2SAndroid Build Coastguard Worker cgi_compare_variables(
580*5e7646d2SAndroid Build Coastguard Worker     const _cgi_var_t *v1,		/* I - First variable */
581*5e7646d2SAndroid Build Coastguard Worker     const _cgi_var_t *v2)		/* I - Second variable */
582*5e7646d2SAndroid Build Coastguard Worker {
583*5e7646d2SAndroid Build Coastguard Worker   return (_cups_strcasecmp(v1->name, v2->name));
584*5e7646d2SAndroid Build Coastguard Worker }
585*5e7646d2SAndroid Build Coastguard Worker 
586*5e7646d2SAndroid Build Coastguard Worker 
587*5e7646d2SAndroid Build Coastguard Worker /*
588*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_find_variable()' - Find a variable.
589*5e7646d2SAndroid Build Coastguard Worker  */
590*5e7646d2SAndroid Build Coastguard Worker 
591*5e7646d2SAndroid Build Coastguard Worker static _cgi_var_t *			/* O - Variable pointer or NULL */
cgi_find_variable(const char * name)592*5e7646d2SAndroid Build Coastguard Worker cgi_find_variable(const char *name)	/* I - Name of variable */
593*5e7646d2SAndroid Build Coastguard Worker {
594*5e7646d2SAndroid Build Coastguard Worker   _cgi_var_t	key;			/* Search key */
595*5e7646d2SAndroid Build Coastguard Worker 
596*5e7646d2SAndroid Build Coastguard Worker 
597*5e7646d2SAndroid Build Coastguard Worker   if (form_count < 1 || name == NULL)
598*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
599*5e7646d2SAndroid Build Coastguard Worker 
600*5e7646d2SAndroid Build Coastguard Worker   key.name = (char *)name;
601*5e7646d2SAndroid Build Coastguard Worker 
602*5e7646d2SAndroid Build Coastguard Worker   return ((_cgi_var_t *)bsearch(&key, form_vars, (size_t)form_count, sizeof(_cgi_var_t),
603*5e7646d2SAndroid Build Coastguard Worker                            (int (*)(const void *, const void *))cgi_compare_variables));
604*5e7646d2SAndroid Build Coastguard Worker }
605*5e7646d2SAndroid Build Coastguard Worker 
606*5e7646d2SAndroid Build Coastguard Worker 
607*5e7646d2SAndroid Build Coastguard Worker /*
608*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_initialize_cookies()' - Initialize cookies.
609*5e7646d2SAndroid Build Coastguard Worker  */
610*5e7646d2SAndroid Build Coastguard Worker 
611*5e7646d2SAndroid Build Coastguard Worker static void
cgi_initialize_cookies(void)612*5e7646d2SAndroid Build Coastguard Worker cgi_initialize_cookies(void)
613*5e7646d2SAndroid Build Coastguard Worker {
614*5e7646d2SAndroid Build Coastguard Worker   const char	*cookie;		/* HTTP_COOKIE environment variable */
615*5e7646d2SAndroid Build Coastguard Worker   char		name[128],		/* Name string */
616*5e7646d2SAndroid Build Coastguard Worker 		value[512],		/* Value string */
617*5e7646d2SAndroid Build Coastguard Worker 		*ptr;			/* Pointer into name/value */
618*5e7646d2SAndroid Build Coastguard Worker 
619*5e7646d2SAndroid Build Coastguard Worker 
620*5e7646d2SAndroid Build Coastguard Worker   if ((cookie = getenv("HTTP_COOKIE")) == NULL)
621*5e7646d2SAndroid Build Coastguard Worker     return;
622*5e7646d2SAndroid Build Coastguard Worker 
623*5e7646d2SAndroid Build Coastguard Worker   while (*cookie)
624*5e7646d2SAndroid Build Coastguard Worker   {
625*5e7646d2SAndroid Build Coastguard Worker     int	skip = 0;			/* Skip this cookie? */
626*5e7646d2SAndroid Build Coastguard Worker 
627*5e7646d2SAndroid Build Coastguard Worker    /*
628*5e7646d2SAndroid Build Coastguard Worker     * Skip leading whitespace...
629*5e7646d2SAndroid Build Coastguard Worker     */
630*5e7646d2SAndroid Build Coastguard Worker 
631*5e7646d2SAndroid Build Coastguard Worker     while (isspace(*cookie & 255))
632*5e7646d2SAndroid Build Coastguard Worker       cookie ++;
633*5e7646d2SAndroid Build Coastguard Worker     if (!*cookie)
634*5e7646d2SAndroid Build Coastguard Worker       break;
635*5e7646d2SAndroid Build Coastguard Worker 
636*5e7646d2SAndroid Build Coastguard Worker    /*
637*5e7646d2SAndroid Build Coastguard Worker     * Copy the name...
638*5e7646d2SAndroid Build Coastguard Worker     */
639*5e7646d2SAndroid Build Coastguard Worker 
640*5e7646d2SAndroid Build Coastguard Worker     for (ptr = name; *cookie && *cookie != '=';)
641*5e7646d2SAndroid Build Coastguard Worker       if (ptr < (name + sizeof(name) - 1))
642*5e7646d2SAndroid Build Coastguard Worker       {
643*5e7646d2SAndroid Build Coastguard Worker         *ptr++ = *cookie++;
644*5e7646d2SAndroid Build Coastguard Worker       }
645*5e7646d2SAndroid Build Coastguard Worker       else
646*5e7646d2SAndroid Build Coastguard Worker       {
647*5e7646d2SAndroid Build Coastguard Worker         skip = 1;
648*5e7646d2SAndroid Build Coastguard Worker 	cookie ++;
649*5e7646d2SAndroid Build Coastguard Worker       }
650*5e7646d2SAndroid Build Coastguard Worker 
651*5e7646d2SAndroid Build Coastguard Worker     if (*cookie != '=')
652*5e7646d2SAndroid Build Coastguard Worker       break;
653*5e7646d2SAndroid Build Coastguard Worker 
654*5e7646d2SAndroid Build Coastguard Worker     *ptr = '\0';
655*5e7646d2SAndroid Build Coastguard Worker     cookie ++;
656*5e7646d2SAndroid Build Coastguard Worker 
657*5e7646d2SAndroid Build Coastguard Worker    /*
658*5e7646d2SAndroid Build Coastguard Worker     * Then the value...
659*5e7646d2SAndroid Build Coastguard Worker     */
660*5e7646d2SAndroid Build Coastguard Worker 
661*5e7646d2SAndroid Build Coastguard Worker     if (*cookie == '\"')
662*5e7646d2SAndroid Build Coastguard Worker     {
663*5e7646d2SAndroid Build Coastguard Worker       for (cookie ++, ptr = value; *cookie && *cookie != '\"';)
664*5e7646d2SAndroid Build Coastguard Worker         if (ptr < (value + sizeof(value) - 1))
665*5e7646d2SAndroid Build Coastguard Worker 	{
666*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = *cookie++;
667*5e7646d2SAndroid Build Coastguard Worker 	}
668*5e7646d2SAndroid Build Coastguard Worker 	else
669*5e7646d2SAndroid Build Coastguard Worker 	{
670*5e7646d2SAndroid Build Coastguard Worker 	  skip = 1;
671*5e7646d2SAndroid Build Coastguard Worker 	  cookie ++;
672*5e7646d2SAndroid Build Coastguard Worker 	}
673*5e7646d2SAndroid Build Coastguard Worker 
674*5e7646d2SAndroid Build Coastguard Worker       if (*cookie == '\"')
675*5e7646d2SAndroid Build Coastguard Worker         cookie ++;
676*5e7646d2SAndroid Build Coastguard Worker       else
677*5e7646d2SAndroid Build Coastguard Worker         skip = 1;
678*5e7646d2SAndroid Build Coastguard Worker     }
679*5e7646d2SAndroid Build Coastguard Worker     else
680*5e7646d2SAndroid Build Coastguard Worker     {
681*5e7646d2SAndroid Build Coastguard Worker       for (ptr = value; *cookie && *cookie != ';';)
682*5e7646d2SAndroid Build Coastguard Worker         if (ptr < (value + sizeof(value) - 1))
683*5e7646d2SAndroid Build Coastguard Worker 	{
684*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = *cookie++;
685*5e7646d2SAndroid Build Coastguard Worker 	}
686*5e7646d2SAndroid Build Coastguard Worker 	else
687*5e7646d2SAndroid Build Coastguard Worker 	{
688*5e7646d2SAndroid Build Coastguard Worker 	  skip = 1;
689*5e7646d2SAndroid Build Coastguard Worker 	  cookie ++;
690*5e7646d2SAndroid Build Coastguard Worker 	}
691*5e7646d2SAndroid Build Coastguard Worker     }
692*5e7646d2SAndroid Build Coastguard Worker 
693*5e7646d2SAndroid Build Coastguard Worker     if (*cookie == ';')
694*5e7646d2SAndroid Build Coastguard Worker       cookie ++;
695*5e7646d2SAndroid Build Coastguard Worker     else if (*cookie)
696*5e7646d2SAndroid Build Coastguard Worker       skip = 1;
697*5e7646d2SAndroid Build Coastguard Worker 
698*5e7646d2SAndroid Build Coastguard Worker     *ptr = '\0';
699*5e7646d2SAndroid Build Coastguard Worker 
700*5e7646d2SAndroid Build Coastguard Worker    /*
701*5e7646d2SAndroid Build Coastguard Worker     * Then add the cookie to an array as long as the name doesn't start with
702*5e7646d2SAndroid Build Coastguard Worker     * "$"...
703*5e7646d2SAndroid Build Coastguard Worker     */
704*5e7646d2SAndroid Build Coastguard Worker 
705*5e7646d2SAndroid Build Coastguard Worker     if (name[0] != '$' && !skip)
706*5e7646d2SAndroid Build Coastguard Worker       num_cookies = cupsAddOption(name, value, num_cookies, &cookies);
707*5e7646d2SAndroid Build Coastguard Worker   }
708*5e7646d2SAndroid Build Coastguard Worker }
709*5e7646d2SAndroid Build Coastguard Worker 
710*5e7646d2SAndroid Build Coastguard Worker 
711*5e7646d2SAndroid Build Coastguard Worker /*
712*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_initialize_get()' - Initialize form variables using the GET method.
713*5e7646d2SAndroid Build Coastguard Worker  */
714*5e7646d2SAndroid Build Coastguard Worker 
715*5e7646d2SAndroid Build Coastguard Worker static int				/* O - 1 if form data read */
cgi_initialize_get(void)716*5e7646d2SAndroid Build Coastguard Worker cgi_initialize_get(void)
717*5e7646d2SAndroid Build Coastguard Worker {
718*5e7646d2SAndroid Build Coastguard Worker   char	*data;				/* Pointer to form data string */
719*5e7646d2SAndroid Build Coastguard Worker 
720*5e7646d2SAndroid Build Coastguard Worker 
721*5e7646d2SAndroid Build Coastguard Worker  /*
722*5e7646d2SAndroid Build Coastguard Worker   * Check to see if there is anything for us to read...
723*5e7646d2SAndroid Build Coastguard Worker   */
724*5e7646d2SAndroid Build Coastguard Worker 
725*5e7646d2SAndroid Build Coastguard Worker   data = getenv("QUERY_STRING");
726*5e7646d2SAndroid Build Coastguard Worker   if (data == NULL || strlen(data) == 0)
727*5e7646d2SAndroid Build Coastguard Worker     return (0);
728*5e7646d2SAndroid Build Coastguard Worker 
729*5e7646d2SAndroid Build Coastguard Worker  /*
730*5e7646d2SAndroid Build Coastguard Worker   * Parse it out and return...
731*5e7646d2SAndroid Build Coastguard Worker   */
732*5e7646d2SAndroid Build Coastguard Worker 
733*5e7646d2SAndroid Build Coastguard Worker   return (cgi_initialize_string(data));
734*5e7646d2SAndroid Build Coastguard Worker }
735*5e7646d2SAndroid Build Coastguard Worker 
736*5e7646d2SAndroid Build Coastguard Worker 
737*5e7646d2SAndroid Build Coastguard Worker /*
738*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_initialize_multipart()' - Initialize variables and file using the POST
739*5e7646d2SAndroid Build Coastguard Worker  *                                method.
740*5e7646d2SAndroid Build Coastguard Worker  *
741*5e7646d2SAndroid Build Coastguard Worker  * TODO: Update to support files > 2GB.
742*5e7646d2SAndroid Build Coastguard Worker  */
743*5e7646d2SAndroid Build Coastguard Worker 
744*5e7646d2SAndroid Build Coastguard Worker static int				/* O - 1 if form data was read */
cgi_initialize_multipart(const char * boundary)745*5e7646d2SAndroid Build Coastguard Worker cgi_initialize_multipart(
746*5e7646d2SAndroid Build Coastguard Worker     const char *boundary)		/* I - Boundary string */
747*5e7646d2SAndroid Build Coastguard Worker {
748*5e7646d2SAndroid Build Coastguard Worker   char		line[10240],		/* MIME header line */
749*5e7646d2SAndroid Build Coastguard Worker 		name[1024],		/* Form variable name */
750*5e7646d2SAndroid Build Coastguard Worker 		filename[1024],		/* Form filename */
751*5e7646d2SAndroid Build Coastguard Worker 		mimetype[1024],		/* MIME media type */
752*5e7646d2SAndroid Build Coastguard Worker 		bstring[256],		/* Boundary string to look for */
753*5e7646d2SAndroid Build Coastguard Worker 		*ptr,			/* Pointer into name/filename */
754*5e7646d2SAndroid Build Coastguard Worker 		*end;			/* End of buffer */
755*5e7646d2SAndroid Build Coastguard Worker   int		ch,			/* Character from file */
756*5e7646d2SAndroid Build Coastguard Worker 		fd;			/* Temporary file descriptor */
757*5e7646d2SAndroid Build Coastguard Worker   size_t	blen;			/* Length of boundary string */
758*5e7646d2SAndroid Build Coastguard Worker 
759*5e7646d2SAndroid Build Coastguard Worker 
760*5e7646d2SAndroid Build Coastguard Worker  /*
761*5e7646d2SAndroid Build Coastguard Worker   * Read multipart form data until we run out...
762*5e7646d2SAndroid Build Coastguard Worker   */
763*5e7646d2SAndroid Build Coastguard Worker 
764*5e7646d2SAndroid Build Coastguard Worker   name[0]     = '\0';
765*5e7646d2SAndroid Build Coastguard Worker   filename[0] = '\0';
766*5e7646d2SAndroid Build Coastguard Worker   mimetype[0] = '\0';
767*5e7646d2SAndroid Build Coastguard Worker 
768*5e7646d2SAndroid Build Coastguard Worker   snprintf(bstring, sizeof(bstring), "\r\n--%s", boundary);
769*5e7646d2SAndroid Build Coastguard Worker   blen = strlen(bstring);
770*5e7646d2SAndroid Build Coastguard Worker 
771*5e7646d2SAndroid Build Coastguard Worker   while (fgets(line, sizeof(line), stdin))
772*5e7646d2SAndroid Build Coastguard Worker   {
773*5e7646d2SAndroid Build Coastguard Worker     if (!strcmp(line, "\r\n"))
774*5e7646d2SAndroid Build Coastguard Worker     {
775*5e7646d2SAndroid Build Coastguard Worker      /*
776*5e7646d2SAndroid Build Coastguard Worker       * End of headers, grab value...
777*5e7646d2SAndroid Build Coastguard Worker       */
778*5e7646d2SAndroid Build Coastguard Worker 
779*5e7646d2SAndroid Build Coastguard Worker       if (filename[0])
780*5e7646d2SAndroid Build Coastguard Worker       {
781*5e7646d2SAndroid Build Coastguard Worker        /*
782*5e7646d2SAndroid Build Coastguard Worker         * Read an embedded file...
783*5e7646d2SAndroid Build Coastguard Worker 	*/
784*5e7646d2SAndroid Build Coastguard Worker 
785*5e7646d2SAndroid Build Coastguard Worker         if (form_file)
786*5e7646d2SAndroid Build Coastguard Worker 	{
787*5e7646d2SAndroid Build Coastguard Worker 	 /*
788*5e7646d2SAndroid Build Coastguard Worker 	  * Remove previous file...
789*5e7646d2SAndroid Build Coastguard Worker 	  */
790*5e7646d2SAndroid Build Coastguard Worker 
791*5e7646d2SAndroid Build Coastguard Worker 	  cgi_unlink_file();
792*5e7646d2SAndroid Build Coastguard Worker 	}
793*5e7646d2SAndroid Build Coastguard Worker 
794*5e7646d2SAndroid Build Coastguard Worker        /*
795*5e7646d2SAndroid Build Coastguard Worker         * Allocate memory for the new file...
796*5e7646d2SAndroid Build Coastguard Worker 	*/
797*5e7646d2SAndroid Build Coastguard Worker 
798*5e7646d2SAndroid Build Coastguard Worker 	if ((form_file = calloc(1, sizeof(cgi_file_t))) == NULL)
799*5e7646d2SAndroid Build Coastguard Worker 	  return (0);
800*5e7646d2SAndroid Build Coastguard Worker 
801*5e7646d2SAndroid Build Coastguard Worker         form_file->name     = strdup(name);
802*5e7646d2SAndroid Build Coastguard Worker 	form_file->filename = strdup(filename);
803*5e7646d2SAndroid Build Coastguard Worker 	form_file->mimetype = strdup(mimetype);
804*5e7646d2SAndroid Build Coastguard Worker 
805*5e7646d2SAndroid Build Coastguard Worker         fd = cupsTempFd(form_file->tempfile, sizeof(form_file->tempfile));
806*5e7646d2SAndroid Build Coastguard Worker 
807*5e7646d2SAndroid Build Coastguard Worker         if (fd < 0)
808*5e7646d2SAndroid Build Coastguard Worker 	  return (0);
809*5e7646d2SAndroid Build Coastguard Worker 
810*5e7646d2SAndroid Build Coastguard Worker         atexit(cgi_unlink_file);
811*5e7646d2SAndroid Build Coastguard Worker 
812*5e7646d2SAndroid Build Coastguard Worker        /*
813*5e7646d2SAndroid Build Coastguard Worker         * Copy file data to the temp file...
814*5e7646d2SAndroid Build Coastguard Worker 	*/
815*5e7646d2SAndroid Build Coastguard Worker 
816*5e7646d2SAndroid Build Coastguard Worker         ptr = line;
817*5e7646d2SAndroid Build Coastguard Worker 
818*5e7646d2SAndroid Build Coastguard Worker 	while ((ch = getchar()) != EOF)
819*5e7646d2SAndroid Build Coastguard Worker 	{
820*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = (char)ch;
821*5e7646d2SAndroid Build Coastguard Worker 
822*5e7646d2SAndroid Build Coastguard Worker           if ((size_t)(ptr - line) >= blen && !memcmp(ptr - blen, bstring, blen))
823*5e7646d2SAndroid Build Coastguard Worker 	  {
824*5e7646d2SAndroid Build Coastguard Worker 	    ptr -= blen;
825*5e7646d2SAndroid Build Coastguard Worker 	    break;
826*5e7646d2SAndroid Build Coastguard Worker 	  }
827*5e7646d2SAndroid Build Coastguard Worker 
828*5e7646d2SAndroid Build Coastguard Worker           if ((ptr - line - (int)blen) >= 8192)
829*5e7646d2SAndroid Build Coastguard Worker 	  {
830*5e7646d2SAndroid Build Coastguard Worker 	   /*
831*5e7646d2SAndroid Build Coastguard Worker 	    * Write out the first 8k of the buffer...
832*5e7646d2SAndroid Build Coastguard Worker 	    */
833*5e7646d2SAndroid Build Coastguard Worker 
834*5e7646d2SAndroid Build Coastguard Worker 	    write(fd, line, 8192);
835*5e7646d2SAndroid Build Coastguard Worker 	    memmove(line, line + 8192, (size_t)(ptr - line - 8192));
836*5e7646d2SAndroid Build Coastguard Worker 	    ptr -= 8192;
837*5e7646d2SAndroid Build Coastguard Worker 	  }
838*5e7646d2SAndroid Build Coastguard Worker 	}
839*5e7646d2SAndroid Build Coastguard Worker 
840*5e7646d2SAndroid Build Coastguard Worker        /*
841*5e7646d2SAndroid Build Coastguard Worker         * Write the rest of the data and close the temp file...
842*5e7646d2SAndroid Build Coastguard Worker 	*/
843*5e7646d2SAndroid Build Coastguard Worker 
844*5e7646d2SAndroid Build Coastguard Worker 	if (ptr > line)
845*5e7646d2SAndroid Build Coastguard Worker           write(fd, line, (size_t)(ptr - line));
846*5e7646d2SAndroid Build Coastguard Worker 
847*5e7646d2SAndroid Build Coastguard Worker 	close(fd);
848*5e7646d2SAndroid Build Coastguard Worker       }
849*5e7646d2SAndroid Build Coastguard Worker       else
850*5e7646d2SAndroid Build Coastguard Worker       {
851*5e7646d2SAndroid Build Coastguard Worker        /*
852*5e7646d2SAndroid Build Coastguard Worker         * Just get a form variable; the current code only handles
853*5e7646d2SAndroid Build Coastguard Worker 	* form values up to 10k in size...
854*5e7646d2SAndroid Build Coastguard Worker 	*/
855*5e7646d2SAndroid Build Coastguard Worker 
856*5e7646d2SAndroid Build Coastguard Worker         ptr = line;
857*5e7646d2SAndroid Build Coastguard Worker 	end = line + sizeof(line) - 1;
858*5e7646d2SAndroid Build Coastguard Worker 
859*5e7646d2SAndroid Build Coastguard Worker 	while ((ch = getchar()) != EOF)
860*5e7646d2SAndroid Build Coastguard Worker 	{
861*5e7646d2SAndroid Build Coastguard Worker 	  if (ptr < end)
862*5e7646d2SAndroid Build Coastguard Worker 	    *ptr++ = (char)ch;
863*5e7646d2SAndroid Build Coastguard Worker 
864*5e7646d2SAndroid Build Coastguard Worker           if ((size_t)(ptr - line) >= blen && !memcmp(ptr - blen, bstring, blen))
865*5e7646d2SAndroid Build Coastguard Worker 	  {
866*5e7646d2SAndroid Build Coastguard Worker 	    ptr -= blen;
867*5e7646d2SAndroid Build Coastguard Worker 	    break;
868*5e7646d2SAndroid Build Coastguard Worker 	  }
869*5e7646d2SAndroid Build Coastguard Worker 	}
870*5e7646d2SAndroid Build Coastguard Worker 
871*5e7646d2SAndroid Build Coastguard Worker 	*ptr = '\0';
872*5e7646d2SAndroid Build Coastguard Worker 
873*5e7646d2SAndroid Build Coastguard Worker        /*
874*5e7646d2SAndroid Build Coastguard Worker         * Set the form variable...
875*5e7646d2SAndroid Build Coastguard Worker 	*/
876*5e7646d2SAndroid Build Coastguard Worker 
877*5e7646d2SAndroid Build Coastguard Worker 	if ((ptr = strrchr(name, '-')) != NULL && isdigit(ptr[1] & 255))
878*5e7646d2SAndroid Build Coastguard Worker 	{
879*5e7646d2SAndroid Build Coastguard Worker 	 /*
880*5e7646d2SAndroid Build Coastguard Worker 	  * Set a specific index in the array...
881*5e7646d2SAndroid Build Coastguard Worker 	  */
882*5e7646d2SAndroid Build Coastguard Worker 
883*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = '\0';
884*5e7646d2SAndroid Build Coastguard Worker 	  if (line[0])
885*5e7646d2SAndroid Build Coastguard Worker             cgiSetArray(name, atoi(ptr) - 1, line);
886*5e7646d2SAndroid Build Coastguard Worker 	}
887*5e7646d2SAndroid Build Coastguard Worker 	else if ((ptr = cgiGetVariable(name)) != NULL)
888*5e7646d2SAndroid Build Coastguard Worker 	{
889*5e7646d2SAndroid Build Coastguard Worker 	 /*
890*5e7646d2SAndroid Build Coastguard Worker 	  * Add another element in the array...
891*5e7646d2SAndroid Build Coastguard Worker 	  */
892*5e7646d2SAndroid Build Coastguard Worker 
893*5e7646d2SAndroid Build Coastguard Worker           free(ptr);
894*5e7646d2SAndroid Build Coastguard Worker 	  cgiSetArray(name, cgiGetSize(name), line);
895*5e7646d2SAndroid Build Coastguard Worker 	}
896*5e7646d2SAndroid Build Coastguard Worker 	else
897*5e7646d2SAndroid Build Coastguard Worker 	{
898*5e7646d2SAndroid Build Coastguard Worker 	 /*
899*5e7646d2SAndroid Build Coastguard Worker 	  * Just set the line...
900*5e7646d2SAndroid Build Coastguard Worker 	  */
901*5e7646d2SAndroid Build Coastguard Worker 
902*5e7646d2SAndroid Build Coastguard Worker 	  cgiSetVariable(name, line);
903*5e7646d2SAndroid Build Coastguard Worker 	}
904*5e7646d2SAndroid Build Coastguard Worker       }
905*5e7646d2SAndroid Build Coastguard Worker 
906*5e7646d2SAndroid Build Coastguard Worker      /*
907*5e7646d2SAndroid Build Coastguard Worker       * Read the rest of the current line...
908*5e7646d2SAndroid Build Coastguard Worker       */
909*5e7646d2SAndroid Build Coastguard Worker 
910*5e7646d2SAndroid Build Coastguard Worker       fgets(line, sizeof(line), stdin);
911*5e7646d2SAndroid Build Coastguard Worker 
912*5e7646d2SAndroid Build Coastguard Worker      /*
913*5e7646d2SAndroid Build Coastguard Worker       * Clear the state vars...
914*5e7646d2SAndroid Build Coastguard Worker       */
915*5e7646d2SAndroid Build Coastguard Worker 
916*5e7646d2SAndroid Build Coastguard Worker       name[0]     = '\0';
917*5e7646d2SAndroid Build Coastguard Worker       filename[0] = '\0';
918*5e7646d2SAndroid Build Coastguard Worker       mimetype[0] = '\0';
919*5e7646d2SAndroid Build Coastguard Worker     }
920*5e7646d2SAndroid Build Coastguard Worker     else if (!_cups_strncasecmp(line, "Content-Disposition:", 20))
921*5e7646d2SAndroid Build Coastguard Worker     {
922*5e7646d2SAndroid Build Coastguard Worker       if ((ptr = strstr(line + 20, " name=\"")) != NULL)
923*5e7646d2SAndroid Build Coastguard Worker       {
924*5e7646d2SAndroid Build Coastguard Worker         strlcpy(name, ptr + 7, sizeof(name));
925*5e7646d2SAndroid Build Coastguard Worker 
926*5e7646d2SAndroid Build Coastguard Worker 	if ((ptr = strchr(name, '\"')) != NULL)
927*5e7646d2SAndroid Build Coastguard Worker 	  *ptr = '\0';
928*5e7646d2SAndroid Build Coastguard Worker       }
929*5e7646d2SAndroid Build Coastguard Worker 
930*5e7646d2SAndroid Build Coastguard Worker       if ((ptr = strstr(line + 20, " filename=\"")) != NULL)
931*5e7646d2SAndroid Build Coastguard Worker       {
932*5e7646d2SAndroid Build Coastguard Worker         strlcpy(filename, ptr + 11, sizeof(filename));
933*5e7646d2SAndroid Build Coastguard Worker 
934*5e7646d2SAndroid Build Coastguard Worker 	if ((ptr = strchr(filename, '\"')) != NULL)
935*5e7646d2SAndroid Build Coastguard Worker 	  *ptr = '\0';
936*5e7646d2SAndroid Build Coastguard Worker       }
937*5e7646d2SAndroid Build Coastguard Worker     }
938*5e7646d2SAndroid Build Coastguard Worker     else if (!_cups_strncasecmp(line, "Content-Type:", 13))
939*5e7646d2SAndroid Build Coastguard Worker     {
940*5e7646d2SAndroid Build Coastguard Worker       for (ptr = line + 13; isspace(*ptr & 255); ptr ++);
941*5e7646d2SAndroid Build Coastguard Worker 
942*5e7646d2SAndroid Build Coastguard Worker       strlcpy(mimetype, ptr, sizeof(mimetype));
943*5e7646d2SAndroid Build Coastguard Worker 
944*5e7646d2SAndroid Build Coastguard Worker       for (ptr = mimetype + strlen(mimetype) - 1;
945*5e7646d2SAndroid Build Coastguard Worker            ptr > mimetype && isspace(*ptr & 255);
946*5e7646d2SAndroid Build Coastguard Worker 	   *ptr-- = '\0');
947*5e7646d2SAndroid Build Coastguard Worker     }
948*5e7646d2SAndroid Build Coastguard Worker   }
949*5e7646d2SAndroid Build Coastguard Worker 
950*5e7646d2SAndroid Build Coastguard Worker  /*
951*5e7646d2SAndroid Build Coastguard Worker   * Return 1 for "form data found"...
952*5e7646d2SAndroid Build Coastguard Worker   */
953*5e7646d2SAndroid Build Coastguard Worker 
954*5e7646d2SAndroid Build Coastguard Worker   return (1);
955*5e7646d2SAndroid Build Coastguard Worker }
956*5e7646d2SAndroid Build Coastguard Worker 
957*5e7646d2SAndroid Build Coastguard Worker 
958*5e7646d2SAndroid Build Coastguard Worker /*
959*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_initialize_post()' - Initialize variables using the POST method.
960*5e7646d2SAndroid Build Coastguard Worker  */
961*5e7646d2SAndroid Build Coastguard Worker 
962*5e7646d2SAndroid Build Coastguard Worker static int				/* O - 1 if form data was read */
cgi_initialize_post(void)963*5e7646d2SAndroid Build Coastguard Worker cgi_initialize_post(void)
964*5e7646d2SAndroid Build Coastguard Worker {
965*5e7646d2SAndroid Build Coastguard Worker   char		*content_length,	/* Length of input data (string) */
966*5e7646d2SAndroid Build Coastguard Worker 		*data;			/* Pointer to form data string */
967*5e7646d2SAndroid Build Coastguard Worker   size_t	length,			/* Length of input data */
968*5e7646d2SAndroid Build Coastguard Worker 		tbytes;			/* Total number of bytes read */
969*5e7646d2SAndroid Build Coastguard Worker   ssize_t	nbytes;			/* Number of bytes read this read() */
970*5e7646d2SAndroid Build Coastguard Worker   int		status;			/* Return status */
971*5e7646d2SAndroid Build Coastguard Worker 
972*5e7646d2SAndroid Build Coastguard Worker 
973*5e7646d2SAndroid Build Coastguard Worker  /*
974*5e7646d2SAndroid Build Coastguard Worker   * Check to see if there is anything for us to read...
975*5e7646d2SAndroid Build Coastguard Worker   */
976*5e7646d2SAndroid Build Coastguard Worker 
977*5e7646d2SAndroid Build Coastguard Worker   content_length = getenv("CONTENT_LENGTH");
978*5e7646d2SAndroid Build Coastguard Worker   if (content_length == NULL || atoi(content_length) <= 0)
979*5e7646d2SAndroid Build Coastguard Worker     return (0);
980*5e7646d2SAndroid Build Coastguard Worker 
981*5e7646d2SAndroid Build Coastguard Worker  /*
982*5e7646d2SAndroid Build Coastguard Worker   * Get the length of the input stream and allocate a buffer for it...
983*5e7646d2SAndroid Build Coastguard Worker   */
984*5e7646d2SAndroid Build Coastguard Worker 
985*5e7646d2SAndroid Build Coastguard Worker   length = (size_t)strtol(content_length, NULL, 10);
986*5e7646d2SAndroid Build Coastguard Worker   data   = malloc(length + 1);		// lgtm [cpp/uncontrolled-allocation-size]
987*5e7646d2SAndroid Build Coastguard Worker 
988*5e7646d2SAndroid Build Coastguard Worker   if (data == NULL)
989*5e7646d2SAndroid Build Coastguard Worker     return (0);
990*5e7646d2SAndroid Build Coastguard Worker 
991*5e7646d2SAndroid Build Coastguard Worker  /*
992*5e7646d2SAndroid Build Coastguard Worker   * Read the data into the buffer...
993*5e7646d2SAndroid Build Coastguard Worker   */
994*5e7646d2SAndroid Build Coastguard Worker 
995*5e7646d2SAndroid Build Coastguard Worker   for (tbytes = 0; tbytes < length; tbytes += (size_t)nbytes)
996*5e7646d2SAndroid Build Coastguard Worker     if ((nbytes = read(0, data + tbytes, (size_t)(length - tbytes))) < 0)
997*5e7646d2SAndroid Build Coastguard Worker     {
998*5e7646d2SAndroid Build Coastguard Worker       if (errno != EAGAIN)
999*5e7646d2SAndroid Build Coastguard Worker       {
1000*5e7646d2SAndroid Build Coastguard Worker         free(data);
1001*5e7646d2SAndroid Build Coastguard Worker         return (0);
1002*5e7646d2SAndroid Build Coastguard Worker       }
1003*5e7646d2SAndroid Build Coastguard Worker       else
1004*5e7646d2SAndroid Build Coastguard Worker         nbytes = 0;
1005*5e7646d2SAndroid Build Coastguard Worker     }
1006*5e7646d2SAndroid Build Coastguard Worker     else if (nbytes == 0)
1007*5e7646d2SAndroid Build Coastguard Worker     {
1008*5e7646d2SAndroid Build Coastguard Worker      /*
1009*5e7646d2SAndroid Build Coastguard Worker       * CUPS STR #3176: OpenBSD: Early end-of-file on POST data causes 100% CPU
1010*5e7646d2SAndroid Build Coastguard Worker       *
1011*5e7646d2SAndroid Build Coastguard Worker       * This should never happen, but does on OpenBSD.  If we see early end-of-
1012*5e7646d2SAndroid Build Coastguard Worker       * file, treat this as an error and process no data.
1013*5e7646d2SAndroid Build Coastguard Worker       */
1014*5e7646d2SAndroid Build Coastguard Worker 
1015*5e7646d2SAndroid Build Coastguard Worker       free(data);
1016*5e7646d2SAndroid Build Coastguard Worker       return (0);
1017*5e7646d2SAndroid Build Coastguard Worker     }
1018*5e7646d2SAndroid Build Coastguard Worker 
1019*5e7646d2SAndroid Build Coastguard Worker   data[length] = '\0';
1020*5e7646d2SAndroid Build Coastguard Worker 
1021*5e7646d2SAndroid Build Coastguard Worker  /*
1022*5e7646d2SAndroid Build Coastguard Worker   * Parse it out...
1023*5e7646d2SAndroid Build Coastguard Worker   */
1024*5e7646d2SAndroid Build Coastguard Worker 
1025*5e7646d2SAndroid Build Coastguard Worker   status = cgi_initialize_string(data);
1026*5e7646d2SAndroid Build Coastguard Worker 
1027*5e7646d2SAndroid Build Coastguard Worker  /*
1028*5e7646d2SAndroid Build Coastguard Worker   * Free the data and return...
1029*5e7646d2SAndroid Build Coastguard Worker   */
1030*5e7646d2SAndroid Build Coastguard Worker 
1031*5e7646d2SAndroid Build Coastguard Worker   free(data);
1032*5e7646d2SAndroid Build Coastguard Worker 
1033*5e7646d2SAndroid Build Coastguard Worker   return (status);
1034*5e7646d2SAndroid Build Coastguard Worker }
1035*5e7646d2SAndroid Build Coastguard Worker 
1036*5e7646d2SAndroid Build Coastguard Worker 
1037*5e7646d2SAndroid Build Coastguard Worker /*
1038*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_initialize_string()' - Initialize form variables from a string.
1039*5e7646d2SAndroid Build Coastguard Worker  */
1040*5e7646d2SAndroid Build Coastguard Worker 
1041*5e7646d2SAndroid Build Coastguard Worker static int				/* O - 1 if form data was processed */
cgi_initialize_string(const char * data)1042*5e7646d2SAndroid Build Coastguard Worker cgi_initialize_string(const char *data)	/* I - Form data string */
1043*5e7646d2SAndroid Build Coastguard Worker {
1044*5e7646d2SAndroid Build Coastguard Worker   int	done;				/* True if we're done reading a form variable */
1045*5e7646d2SAndroid Build Coastguard Worker   char	*s,				/* Pointer to current form string */
1046*5e7646d2SAndroid Build Coastguard Worker 	ch,				/* Temporary character */
1047*5e7646d2SAndroid Build Coastguard Worker 	name[255],			/* Name of form variable */
1048*5e7646d2SAndroid Build Coastguard Worker 	value[65536],			/* Variable value */
1049*5e7646d2SAndroid Build Coastguard Worker 	*temp;				/* Temporary pointer */
1050*5e7646d2SAndroid Build Coastguard Worker 
1051*5e7646d2SAndroid Build Coastguard Worker 
1052*5e7646d2SAndroid Build Coastguard Worker  /*
1053*5e7646d2SAndroid Build Coastguard Worker   * Check input...
1054*5e7646d2SAndroid Build Coastguard Worker   */
1055*5e7646d2SAndroid Build Coastguard Worker 
1056*5e7646d2SAndroid Build Coastguard Worker   if (data == NULL)
1057*5e7646d2SAndroid Build Coastguard Worker     return (0);
1058*5e7646d2SAndroid Build Coastguard Worker 
1059*5e7646d2SAndroid Build Coastguard Worker  /*
1060*5e7646d2SAndroid Build Coastguard Worker   * Loop until we've read all the form data...
1061*5e7646d2SAndroid Build Coastguard Worker   */
1062*5e7646d2SAndroid Build Coastguard Worker 
1063*5e7646d2SAndroid Build Coastguard Worker   while (*data != '\0')
1064*5e7646d2SAndroid Build Coastguard Worker   {
1065*5e7646d2SAndroid Build Coastguard Worker    /*
1066*5e7646d2SAndroid Build Coastguard Worker     * Get the variable name...
1067*5e7646d2SAndroid Build Coastguard Worker     */
1068*5e7646d2SAndroid Build Coastguard Worker 
1069*5e7646d2SAndroid Build Coastguard Worker     for (s = name; *data != '\0'; data ++)
1070*5e7646d2SAndroid Build Coastguard Worker       if (*data == '=')
1071*5e7646d2SAndroid Build Coastguard Worker         break;
1072*5e7646d2SAndroid Build Coastguard Worker       else if (*data >= ' ' && s < (name + sizeof(name) - 1))
1073*5e7646d2SAndroid Build Coastguard Worker         *s++ = *data;
1074*5e7646d2SAndroid Build Coastguard Worker 
1075*5e7646d2SAndroid Build Coastguard Worker     *s = '\0';
1076*5e7646d2SAndroid Build Coastguard Worker     if (*data == '=')
1077*5e7646d2SAndroid Build Coastguard Worker       data ++;
1078*5e7646d2SAndroid Build Coastguard Worker     else
1079*5e7646d2SAndroid Build Coastguard Worker       return (0);
1080*5e7646d2SAndroid Build Coastguard Worker 
1081*5e7646d2SAndroid Build Coastguard Worker    /*
1082*5e7646d2SAndroid Build Coastguard Worker     * Read the variable value...
1083*5e7646d2SAndroid Build Coastguard Worker     */
1084*5e7646d2SAndroid Build Coastguard Worker 
1085*5e7646d2SAndroid Build Coastguard Worker     for (s = value, done = 0; !done && *data != '\0'; data ++)
1086*5e7646d2SAndroid Build Coastguard Worker       switch (*data)
1087*5e7646d2SAndroid Build Coastguard Worker       {
1088*5e7646d2SAndroid Build Coastguard Worker 	case '&' :	/* End of data... */
1089*5e7646d2SAndroid Build Coastguard Worker             done = 1;
1090*5e7646d2SAndroid Build Coastguard Worker             break;
1091*5e7646d2SAndroid Build Coastguard Worker 
1092*5e7646d2SAndroid Build Coastguard Worker 	case '+' :	/* Escaped space character */
1093*5e7646d2SAndroid Build Coastguard Worker             if (s < (value + sizeof(value) - 1))
1094*5e7646d2SAndroid Build Coastguard Worker               *s++ = ' ';
1095*5e7646d2SAndroid Build Coastguard Worker             break;
1096*5e7646d2SAndroid Build Coastguard Worker 
1097*5e7646d2SAndroid Build Coastguard Worker 	case '%' :	/* Escaped control character */
1098*5e7646d2SAndroid Build Coastguard Worker 	   /*
1099*5e7646d2SAndroid Build Coastguard Worker 	    * Read the hex code...
1100*5e7646d2SAndroid Build Coastguard Worker 	    */
1101*5e7646d2SAndroid Build Coastguard Worker 
1102*5e7646d2SAndroid Build Coastguard Worker             if (!isxdigit(data[1] & 255) || !isxdigit(data[2] & 255))
1103*5e7646d2SAndroid Build Coastguard Worker 	      return (0);
1104*5e7646d2SAndroid Build Coastguard Worker 
1105*5e7646d2SAndroid Build Coastguard Worker             if (s < (value + sizeof(value) - 1))
1106*5e7646d2SAndroid Build Coastguard Worker 	    {
1107*5e7646d2SAndroid Build Coastguard Worker               data ++;
1108*5e7646d2SAndroid Build Coastguard Worker               ch = *data - '0';
1109*5e7646d2SAndroid Build Coastguard Worker               if (ch > 9)
1110*5e7646d2SAndroid Build Coastguard Worker         	ch -= 7;
1111*5e7646d2SAndroid Build Coastguard Worker               *s = (char)(ch << 4);
1112*5e7646d2SAndroid Build Coastguard Worker 
1113*5e7646d2SAndroid Build Coastguard Worker               data ++;
1114*5e7646d2SAndroid Build Coastguard Worker               ch = *data - '0';
1115*5e7646d2SAndroid Build Coastguard Worker               if (ch > 9)
1116*5e7646d2SAndroid Build Coastguard Worker         	ch -= 7;
1117*5e7646d2SAndroid Build Coastguard Worker               *s++ |= ch;
1118*5e7646d2SAndroid Build Coastguard Worker             }
1119*5e7646d2SAndroid Build Coastguard Worker 	    else
1120*5e7646d2SAndroid Build Coastguard Worker 	      data += 2;
1121*5e7646d2SAndroid Build Coastguard Worker             break;
1122*5e7646d2SAndroid Build Coastguard Worker 
1123*5e7646d2SAndroid Build Coastguard Worker 	default :	/* Other characters come straight through */
1124*5e7646d2SAndroid Build Coastguard Worker 	    if (*data >= ' ' && s < (value + sizeof(value) - 1))
1125*5e7646d2SAndroid Build Coastguard Worker               *s++ = *data;
1126*5e7646d2SAndroid Build Coastguard Worker             break;
1127*5e7646d2SAndroid Build Coastguard Worker       }
1128*5e7646d2SAndroid Build Coastguard Worker 
1129*5e7646d2SAndroid Build Coastguard Worker     *s = '\0';		/* nul terminate the string */
1130*5e7646d2SAndroid Build Coastguard Worker 
1131*5e7646d2SAndroid Build Coastguard Worker    /*
1132*5e7646d2SAndroid Build Coastguard Worker     * Remove trailing whitespace...
1133*5e7646d2SAndroid Build Coastguard Worker     */
1134*5e7646d2SAndroid Build Coastguard Worker 
1135*5e7646d2SAndroid Build Coastguard Worker     if (s > value)
1136*5e7646d2SAndroid Build Coastguard Worker       s --;
1137*5e7646d2SAndroid Build Coastguard Worker 
1138*5e7646d2SAndroid Build Coastguard Worker     while (s >= value && isspace(*s & 255))
1139*5e7646d2SAndroid Build Coastguard Worker       *s-- = '\0';
1140*5e7646d2SAndroid Build Coastguard Worker 
1141*5e7646d2SAndroid Build Coastguard Worker    /*
1142*5e7646d2SAndroid Build Coastguard Worker     * Add the string to the variable "database"...
1143*5e7646d2SAndroid Build Coastguard Worker     */
1144*5e7646d2SAndroid Build Coastguard Worker 
1145*5e7646d2SAndroid Build Coastguard Worker     if ((s = strrchr(name, '-')) != NULL && isdigit(s[1] & 255))
1146*5e7646d2SAndroid Build Coastguard Worker     {
1147*5e7646d2SAndroid Build Coastguard Worker       *s++ = '\0';
1148*5e7646d2SAndroid Build Coastguard Worker       if (value[0])
1149*5e7646d2SAndroid Build Coastguard Worker         cgiSetArray(name, atoi(s) - 1, value);
1150*5e7646d2SAndroid Build Coastguard Worker     }
1151*5e7646d2SAndroid Build Coastguard Worker     else if ((temp = cgiGetVariable(name)) != NULL)
1152*5e7646d2SAndroid Build Coastguard Worker     {
1153*5e7646d2SAndroid Build Coastguard Worker       free(temp);
1154*5e7646d2SAndroid Build Coastguard Worker       cgiSetArray(name, cgiGetSize(name), value);
1155*5e7646d2SAndroid Build Coastguard Worker     }
1156*5e7646d2SAndroid Build Coastguard Worker     else
1157*5e7646d2SAndroid Build Coastguard Worker       cgiSetVariable(name, value);
1158*5e7646d2SAndroid Build Coastguard Worker   }
1159*5e7646d2SAndroid Build Coastguard Worker 
1160*5e7646d2SAndroid Build Coastguard Worker   return (1);
1161*5e7646d2SAndroid Build Coastguard Worker }
1162*5e7646d2SAndroid Build Coastguard Worker 
1163*5e7646d2SAndroid Build Coastguard Worker 
1164*5e7646d2SAndroid Build Coastguard Worker /*
1165*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_passwd()' - Catch authentication requests and notify the server.
1166*5e7646d2SAndroid Build Coastguard Worker  *
1167*5e7646d2SAndroid Build Coastguard Worker  * This function sends a Status header and exits, forcing authentication
1168*5e7646d2SAndroid Build Coastguard Worker  * for this request.
1169*5e7646d2SAndroid Build Coastguard Worker  */
1170*5e7646d2SAndroid Build Coastguard Worker 
1171*5e7646d2SAndroid Build Coastguard Worker static const char *			/* O - NULL (no return) */
cgi_passwd(const char * prompt)1172*5e7646d2SAndroid Build Coastguard Worker cgi_passwd(const char *prompt)		/* I - Prompt (not used) */
1173*5e7646d2SAndroid Build Coastguard Worker {
1174*5e7646d2SAndroid Build Coastguard Worker   (void)prompt;
1175*5e7646d2SAndroid Build Coastguard Worker 
1176*5e7646d2SAndroid Build Coastguard Worker   fprintf(stderr, "DEBUG: cgi_passwd(prompt=\"%s\") called!\n",
1177*5e7646d2SAndroid Build Coastguard Worker           prompt ? prompt : "(null)");
1178*5e7646d2SAndroid Build Coastguard Worker 
1179*5e7646d2SAndroid Build Coastguard Worker  /*
1180*5e7646d2SAndroid Build Coastguard Worker   * Send a 401 (unauthorized) status to the server, so it can notify
1181*5e7646d2SAndroid Build Coastguard Worker   * the client that authentication is required.
1182*5e7646d2SAndroid Build Coastguard Worker   */
1183*5e7646d2SAndroid Build Coastguard Worker 
1184*5e7646d2SAndroid Build Coastguard Worker   puts("Status: 401\n");
1185*5e7646d2SAndroid Build Coastguard Worker   exit(0);
1186*5e7646d2SAndroid Build Coastguard Worker 
1187*5e7646d2SAndroid Build Coastguard Worker  /*
1188*5e7646d2SAndroid Build Coastguard Worker   * This code is never executed, but is present to satisfy the compiler.
1189*5e7646d2SAndroid Build Coastguard Worker   */
1190*5e7646d2SAndroid Build Coastguard Worker 
1191*5e7646d2SAndroid Build Coastguard Worker   return (NULL);
1192*5e7646d2SAndroid Build Coastguard Worker }
1193*5e7646d2SAndroid Build Coastguard Worker 
1194*5e7646d2SAndroid Build Coastguard Worker 
1195*5e7646d2SAndroid Build Coastguard Worker /*
1196*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_set_sid()' - Set the CUPS session ID.
1197*5e7646d2SAndroid Build Coastguard Worker  */
1198*5e7646d2SAndroid Build Coastguard Worker 
1199*5e7646d2SAndroid Build Coastguard Worker static const char *			/* O - New session ID */
cgi_set_sid(void)1200*5e7646d2SAndroid Build Coastguard Worker cgi_set_sid(void)
1201*5e7646d2SAndroid Build Coastguard Worker {
1202*5e7646d2SAndroid Build Coastguard Worker   char			buffer[512],	/* SID data */
1203*5e7646d2SAndroid Build Coastguard Worker 			sid[33];	/* SID string */
1204*5e7646d2SAndroid Build Coastguard Worker   unsigned char		sum[16];	/* MD5 sum */
1205*5e7646d2SAndroid Build Coastguard Worker   const char		*remote_addr,	/* REMOTE_ADDR */
1206*5e7646d2SAndroid Build Coastguard Worker 			*server_name,	/* SERVER_NAME */
1207*5e7646d2SAndroid Build Coastguard Worker 			*server_port;	/* SERVER_PORT */
1208*5e7646d2SAndroid Build Coastguard Worker   struct timeval	curtime;	/* Current time */
1209*5e7646d2SAndroid Build Coastguard Worker 
1210*5e7646d2SAndroid Build Coastguard Worker 
1211*5e7646d2SAndroid Build Coastguard Worker   if ((remote_addr = getenv("REMOTE_ADDR")) == NULL)
1212*5e7646d2SAndroid Build Coastguard Worker     remote_addr = "REMOTE_ADDR";
1213*5e7646d2SAndroid Build Coastguard Worker   if ((server_name = getenv("SERVER_NAME")) == NULL)
1214*5e7646d2SAndroid Build Coastguard Worker     server_name = "SERVER_NAME";
1215*5e7646d2SAndroid Build Coastguard Worker   if ((server_port = getenv("SERVER_PORT")) == NULL)
1216*5e7646d2SAndroid Build Coastguard Worker     server_port = "SERVER_PORT";
1217*5e7646d2SAndroid Build Coastguard Worker 
1218*5e7646d2SAndroid Build Coastguard Worker   gettimeofday(&curtime, NULL);
1219*5e7646d2SAndroid Build Coastguard Worker   CUPS_SRAND(curtime.tv_sec + curtime.tv_usec);
1220*5e7646d2SAndroid Build Coastguard Worker   snprintf(buffer, sizeof(buffer), "%s:%s:%s:%02X%02X%02X%02X%02X%02X%02X%02X",
1221*5e7646d2SAndroid Build Coastguard Worker            remote_addr, server_name, server_port,
1222*5e7646d2SAndroid Build Coastguard Worker 	   (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255,
1223*5e7646d2SAndroid Build Coastguard Worker 	   (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255,
1224*5e7646d2SAndroid Build Coastguard Worker 	   (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255,
1225*5e7646d2SAndroid Build Coastguard Worker 	   (unsigned)CUPS_RAND() & 255, (unsigned)CUPS_RAND() & 255);
1226*5e7646d2SAndroid Build Coastguard Worker   cupsHashData("md5", (unsigned char *)buffer, strlen(buffer), sum, sizeof(sum));
1227*5e7646d2SAndroid Build Coastguard Worker 
1228*5e7646d2SAndroid Build Coastguard Worker   cgiSetCookie(CUPS_SID, cupsHashString(sum, sizeof(sum), sid, sizeof(sid)), "/", NULL, 0, 0);
1229*5e7646d2SAndroid Build Coastguard Worker 
1230*5e7646d2SAndroid Build Coastguard Worker   return (cupsGetOption(CUPS_SID, num_cookies, cookies));
1231*5e7646d2SAndroid Build Coastguard Worker }
1232*5e7646d2SAndroid Build Coastguard Worker 
1233*5e7646d2SAndroid Build Coastguard Worker 
1234*5e7646d2SAndroid Build Coastguard Worker /*
1235*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_sort_variables()' - Sort all form variables for faster lookup.
1236*5e7646d2SAndroid Build Coastguard Worker  */
1237*5e7646d2SAndroid Build Coastguard Worker 
1238*5e7646d2SAndroid Build Coastguard Worker static void
cgi_sort_variables(void)1239*5e7646d2SAndroid Build Coastguard Worker cgi_sort_variables(void)
1240*5e7646d2SAndroid Build Coastguard Worker {
1241*5e7646d2SAndroid Build Coastguard Worker   if (form_count < 2)
1242*5e7646d2SAndroid Build Coastguard Worker     return;
1243*5e7646d2SAndroid Build Coastguard Worker 
1244*5e7646d2SAndroid Build Coastguard Worker   qsort(form_vars, (size_t)form_count, sizeof(_cgi_var_t),
1245*5e7646d2SAndroid Build Coastguard Worker         (int (*)(const void *, const void *))cgi_compare_variables);
1246*5e7646d2SAndroid Build Coastguard Worker }
1247*5e7646d2SAndroid Build Coastguard Worker 
1248*5e7646d2SAndroid Build Coastguard Worker 
1249*5e7646d2SAndroid Build Coastguard Worker /*
1250*5e7646d2SAndroid Build Coastguard Worker  * 'cgi_unlink_file()' - Remove the uploaded form.
1251*5e7646d2SAndroid Build Coastguard Worker  */
1252*5e7646d2SAndroid Build Coastguard Worker 
1253*5e7646d2SAndroid Build Coastguard Worker static void
cgi_unlink_file(void)1254*5e7646d2SAndroid Build Coastguard Worker cgi_unlink_file(void)
1255*5e7646d2SAndroid Build Coastguard Worker {
1256*5e7646d2SAndroid Build Coastguard Worker   if (form_file)
1257*5e7646d2SAndroid Build Coastguard Worker   {
1258*5e7646d2SAndroid Build Coastguard Worker    /*
1259*5e7646d2SAndroid Build Coastguard Worker     * Remove the temporary file...
1260*5e7646d2SAndroid Build Coastguard Worker     */
1261*5e7646d2SAndroid Build Coastguard Worker 
1262*5e7646d2SAndroid Build Coastguard Worker     unlink(form_file->tempfile);
1263*5e7646d2SAndroid Build Coastguard Worker 
1264*5e7646d2SAndroid Build Coastguard Worker    /*
1265*5e7646d2SAndroid Build Coastguard Worker     * Free memory used...
1266*5e7646d2SAndroid Build Coastguard Worker     */
1267*5e7646d2SAndroid Build Coastguard Worker 
1268*5e7646d2SAndroid Build Coastguard Worker     free(form_file->name);
1269*5e7646d2SAndroid Build Coastguard Worker     free(form_file->filename);
1270*5e7646d2SAndroid Build Coastguard Worker     free(form_file->mimetype);
1271*5e7646d2SAndroid Build Coastguard Worker     free(form_file);
1272*5e7646d2SAndroid Build Coastguard Worker 
1273*5e7646d2SAndroid Build Coastguard Worker     form_file = NULL;
1274*5e7646d2SAndroid Build Coastguard Worker   }
1275*5e7646d2SAndroid Build Coastguard Worker }
1276