xref: /aosp_15_r20/external/libcups/cups/http-support.c (revision 5e7646d21f1134fb0638875d812ef646c12ab91e)
1*5e7646d2SAndroid Build Coastguard Worker /*
2*5e7646d2SAndroid Build Coastguard Worker  * HTTP support routines for CUPS.
3*5e7646d2SAndroid Build Coastguard Worker  *
4*5e7646d2SAndroid Build Coastguard Worker  * Copyright 2007-2019 by Apple Inc.
5*5e7646d2SAndroid Build Coastguard Worker  * Copyright 1997-2007 by Easy Software Products, all rights reserved.
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 #include "cups-private.h"
16*5e7646d2SAndroid Build Coastguard Worker #include "debug-internal.h"
17*5e7646d2SAndroid Build Coastguard Worker #ifdef HAVE_DNSSD
18*5e7646d2SAndroid Build Coastguard Worker #  include <dns_sd.h>
19*5e7646d2SAndroid Build Coastguard Worker #  ifdef _WIN32
20*5e7646d2SAndroid Build Coastguard Worker #    include <io.h>
21*5e7646d2SAndroid Build Coastguard Worker #  elif defined(HAVE_POLL)
22*5e7646d2SAndroid Build Coastguard Worker #    include <poll.h>
23*5e7646d2SAndroid Build Coastguard Worker #  else
24*5e7646d2SAndroid Build Coastguard Worker #    include <sys/select.h>
25*5e7646d2SAndroid Build Coastguard Worker #  endif /* _WIN32 */
26*5e7646d2SAndroid Build Coastguard Worker #elif defined(HAVE_AVAHI)
27*5e7646d2SAndroid Build Coastguard Worker #  include <avahi-client/client.h>
28*5e7646d2SAndroid Build Coastguard Worker #  include <avahi-client/lookup.h>
29*5e7646d2SAndroid Build Coastguard Worker #  include <avahi-common/simple-watch.h>
30*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_DNSSD */
31*5e7646d2SAndroid Build Coastguard Worker 
32*5e7646d2SAndroid Build Coastguard Worker 
33*5e7646d2SAndroid Build Coastguard Worker /*
34*5e7646d2SAndroid Build Coastguard Worker  * Local types...
35*5e7646d2SAndroid Build Coastguard Worker  */
36*5e7646d2SAndroid Build Coastguard Worker 
37*5e7646d2SAndroid Build Coastguard Worker typedef struct _http_uribuf_s		/* URI buffer */
38*5e7646d2SAndroid Build Coastguard Worker {
39*5e7646d2SAndroid Build Coastguard Worker #ifdef HAVE_AVAHI
40*5e7646d2SAndroid Build Coastguard Worker   AvahiSimplePoll	*poll;		/* Poll state */
41*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_AVAHI */
42*5e7646d2SAndroid Build Coastguard Worker   char			*buffer;	/* Pointer to buffer */
43*5e7646d2SAndroid Build Coastguard Worker   size_t		bufsize;	/* Size of buffer */
44*5e7646d2SAndroid Build Coastguard Worker   int			options;	/* Options passed to _httpResolveURI */
45*5e7646d2SAndroid Build Coastguard Worker   const char		*resource;	/* Resource from URI */
46*5e7646d2SAndroid Build Coastguard Worker   const char		*uuid;		/* UUID from URI */
47*5e7646d2SAndroid Build Coastguard Worker } _http_uribuf_t;
48*5e7646d2SAndroid Build Coastguard Worker 
49*5e7646d2SAndroid Build Coastguard Worker 
50*5e7646d2SAndroid Build Coastguard Worker /*
51*5e7646d2SAndroid Build Coastguard Worker  * Local globals...
52*5e7646d2SAndroid Build Coastguard Worker  */
53*5e7646d2SAndroid Build Coastguard Worker 
54*5e7646d2SAndroid Build Coastguard Worker static const char * const http_days[7] =/* Days of the week */
55*5e7646d2SAndroid Build Coastguard Worker 			{
56*5e7646d2SAndroid Build Coastguard Worker 			  "Sun",
57*5e7646d2SAndroid Build Coastguard Worker 			  "Mon",
58*5e7646d2SAndroid Build Coastguard Worker 			  "Tue",
59*5e7646d2SAndroid Build Coastguard Worker 			  "Wed",
60*5e7646d2SAndroid Build Coastguard Worker 			  "Thu",
61*5e7646d2SAndroid Build Coastguard Worker 			  "Fri",
62*5e7646d2SAndroid Build Coastguard Worker 			  "Sat"
63*5e7646d2SAndroid Build Coastguard Worker 			};
64*5e7646d2SAndroid Build Coastguard Worker static const char * const http_months[12] =
65*5e7646d2SAndroid Build Coastguard Worker 			{		/* Months of the year */
66*5e7646d2SAndroid Build Coastguard Worker 			  "Jan",
67*5e7646d2SAndroid Build Coastguard Worker 			  "Feb",
68*5e7646d2SAndroid Build Coastguard Worker 			  "Mar",
69*5e7646d2SAndroid Build Coastguard Worker 			  "Apr",
70*5e7646d2SAndroid Build Coastguard Worker 			  "May",
71*5e7646d2SAndroid Build Coastguard Worker 			  "Jun",
72*5e7646d2SAndroid Build Coastguard Worker 		          "Jul",
73*5e7646d2SAndroid Build Coastguard Worker 			  "Aug",
74*5e7646d2SAndroid Build Coastguard Worker 			  "Sep",
75*5e7646d2SAndroid Build Coastguard Worker 			  "Oct",
76*5e7646d2SAndroid Build Coastguard Worker 			  "Nov",
77*5e7646d2SAndroid Build Coastguard Worker 			  "Dec"
78*5e7646d2SAndroid Build Coastguard Worker 			};
79*5e7646d2SAndroid Build Coastguard Worker static const char * const http_states[] =
80*5e7646d2SAndroid Build Coastguard Worker 			{		/* HTTP state strings */
81*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_ERROR",
82*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_WAITING",
83*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_OPTIONS",
84*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_GET",
85*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_GET_SEND",
86*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_HEAD",
87*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_POST",
88*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_POST_RECV",
89*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_POST_SEND",
90*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_PUT",
91*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_PUT_RECV",
92*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_DELETE",
93*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_TRACE",
94*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_CONNECT",
95*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_STATUS",
96*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_UNKNOWN_METHOD",
97*5e7646d2SAndroid Build Coastguard Worker 			  "HTTP_STATE_UNKNOWN_VERSION"
98*5e7646d2SAndroid Build Coastguard Worker 			};
99*5e7646d2SAndroid Build Coastguard Worker 
100*5e7646d2SAndroid Build Coastguard Worker 
101*5e7646d2SAndroid Build Coastguard Worker /*
102*5e7646d2SAndroid Build Coastguard Worker  * Local functions...
103*5e7646d2SAndroid Build Coastguard Worker  */
104*5e7646d2SAndroid Build Coastguard Worker 
105*5e7646d2SAndroid Build Coastguard Worker static const char	*http_copy_decode(char *dst, const char *src,
106*5e7646d2SAndroid Build Coastguard Worker 			                  int dstsize, const char *term,
107*5e7646d2SAndroid Build Coastguard Worker 					  int decode);
108*5e7646d2SAndroid Build Coastguard Worker static char		*http_copy_encode(char *dst, const char *src,
109*5e7646d2SAndroid Build Coastguard Worker 			                  char *dstend, const char *reserved,
110*5e7646d2SAndroid Build Coastguard Worker 					  const char *term, int encode);
111*5e7646d2SAndroid Build Coastguard Worker #ifdef HAVE_DNSSD
112*5e7646d2SAndroid Build Coastguard Worker static void DNSSD_API	http_resolve_cb(DNSServiceRef sdRef,
113*5e7646d2SAndroid Build Coastguard Worker 					DNSServiceFlags flags,
114*5e7646d2SAndroid Build Coastguard Worker 					uint32_t interfaceIndex,
115*5e7646d2SAndroid Build Coastguard Worker 					DNSServiceErrorType errorCode,
116*5e7646d2SAndroid Build Coastguard Worker 					const char *fullName,
117*5e7646d2SAndroid Build Coastguard Worker 					const char *hostTarget,
118*5e7646d2SAndroid Build Coastguard Worker 					uint16_t port, uint16_t txtLen,
119*5e7646d2SAndroid Build Coastguard Worker 					const unsigned char *txtRecord,
120*5e7646d2SAndroid Build Coastguard Worker 					void *context);
121*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_DNSSD */
122*5e7646d2SAndroid Build Coastguard Worker 
123*5e7646d2SAndroid Build Coastguard Worker #ifdef HAVE_AVAHI
124*5e7646d2SAndroid Build Coastguard Worker static void	http_client_cb(AvahiClient *client,
125*5e7646d2SAndroid Build Coastguard Worker 			       AvahiClientState state, void *simple_poll);
126*5e7646d2SAndroid Build Coastguard Worker static int	http_poll_cb(struct pollfd *pollfds, unsigned int num_pollfds,
127*5e7646d2SAndroid Build Coastguard Worker 		             int timeout, void *context);
128*5e7646d2SAndroid Build Coastguard Worker static void	http_resolve_cb(AvahiServiceResolver *resolver,
129*5e7646d2SAndroid Build Coastguard Worker 				AvahiIfIndex interface,
130*5e7646d2SAndroid Build Coastguard Worker 				AvahiProtocol protocol,
131*5e7646d2SAndroid Build Coastguard Worker 				AvahiResolverEvent event,
132*5e7646d2SAndroid Build Coastguard Worker 				const char *name, const char *type,
133*5e7646d2SAndroid Build Coastguard Worker 				const char *domain, const char *host_name,
134*5e7646d2SAndroid Build Coastguard Worker 				const AvahiAddress *address, uint16_t port,
135*5e7646d2SAndroid Build Coastguard Worker 				AvahiStringList *txt,
136*5e7646d2SAndroid Build Coastguard Worker 				AvahiLookupResultFlags flags, void *context);
137*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_AVAHI */
138*5e7646d2SAndroid Build Coastguard Worker 
139*5e7646d2SAndroid Build Coastguard Worker 
140*5e7646d2SAndroid Build Coastguard Worker /*
141*5e7646d2SAndroid Build Coastguard Worker  * 'httpAssembleURI()' - Assemble a uniform resource identifier from its
142*5e7646d2SAndroid Build Coastguard Worker  *                       components.
143*5e7646d2SAndroid Build Coastguard Worker  *
144*5e7646d2SAndroid Build Coastguard Worker  * This function escapes reserved characters in the URI depending on the
145*5e7646d2SAndroid Build Coastguard Worker  * value of the "encoding" argument.  You should use this function in
146*5e7646d2SAndroid Build Coastguard Worker  * place of traditional string functions whenever you need to create a
147*5e7646d2SAndroid Build Coastguard Worker  * URI string.
148*5e7646d2SAndroid Build Coastguard Worker  *
149*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.2/macOS 10.5@
150*5e7646d2SAndroid Build Coastguard Worker  */
151*5e7646d2SAndroid Build Coastguard Worker 
152*5e7646d2SAndroid Build Coastguard Worker http_uri_status_t			/* O - URI status */
httpAssembleURI(http_uri_coding_t encoding,char * uri,int urilen,const char * scheme,const char * username,const char * host,int port,const char * resource)153*5e7646d2SAndroid Build Coastguard Worker httpAssembleURI(
154*5e7646d2SAndroid Build Coastguard Worker     http_uri_coding_t encoding,		/* I - Encoding flags */
155*5e7646d2SAndroid Build Coastguard Worker     char              *uri,		/* I - URI buffer */
156*5e7646d2SAndroid Build Coastguard Worker     int               urilen,		/* I - Size of URI buffer */
157*5e7646d2SAndroid Build Coastguard Worker     const char        *scheme,		/* I - Scheme name */
158*5e7646d2SAndroid Build Coastguard Worker     const char        *username,	/* I - Username */
159*5e7646d2SAndroid Build Coastguard Worker     const char        *host,		/* I - Hostname or address */
160*5e7646d2SAndroid Build Coastguard Worker     int               port,		/* I - Port number */
161*5e7646d2SAndroid Build Coastguard Worker     const char        *resource)	/* I - Resource */
162*5e7646d2SAndroid Build Coastguard Worker {
163*5e7646d2SAndroid Build Coastguard Worker   char		*ptr,			/* Pointer into URI buffer */
164*5e7646d2SAndroid Build Coastguard Worker 		*end;			/* End of URI buffer */
165*5e7646d2SAndroid Build Coastguard Worker 
166*5e7646d2SAndroid Build Coastguard Worker 
167*5e7646d2SAndroid Build Coastguard Worker  /*
168*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
169*5e7646d2SAndroid Build Coastguard Worker   */
170*5e7646d2SAndroid Build Coastguard Worker 
171*5e7646d2SAndroid Build Coastguard Worker   if (!uri || urilen < 1 || !scheme || port < 0)
172*5e7646d2SAndroid Build Coastguard Worker   {
173*5e7646d2SAndroid Build Coastguard Worker     if (uri)
174*5e7646d2SAndroid Build Coastguard Worker       *uri = '\0';
175*5e7646d2SAndroid Build Coastguard Worker 
176*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_URI_STATUS_BAD_ARGUMENTS);
177*5e7646d2SAndroid Build Coastguard Worker   }
178*5e7646d2SAndroid Build Coastguard Worker 
179*5e7646d2SAndroid Build Coastguard Worker  /*
180*5e7646d2SAndroid Build Coastguard Worker   * Assemble the URI starting with the scheme...
181*5e7646d2SAndroid Build Coastguard Worker   */
182*5e7646d2SAndroid Build Coastguard Worker 
183*5e7646d2SAndroid Build Coastguard Worker   end = uri + urilen - 1;
184*5e7646d2SAndroid Build Coastguard Worker   ptr = http_copy_encode(uri, scheme, end, NULL, NULL, 0);
185*5e7646d2SAndroid Build Coastguard Worker 
186*5e7646d2SAndroid Build Coastguard Worker   if (!ptr)
187*5e7646d2SAndroid Build Coastguard Worker     goto assemble_overflow;
188*5e7646d2SAndroid Build Coastguard Worker 
189*5e7646d2SAndroid Build Coastguard Worker   if (!strcmp(scheme, "geo") || !strcmp(scheme, "mailto") || !strcmp(scheme, "tel"))
190*5e7646d2SAndroid Build Coastguard Worker   {
191*5e7646d2SAndroid Build Coastguard Worker    /*
192*5e7646d2SAndroid Build Coastguard Worker     * geo:, mailto:, and tel: only have :, no //...
193*5e7646d2SAndroid Build Coastguard Worker     */
194*5e7646d2SAndroid Build Coastguard Worker 
195*5e7646d2SAndroid Build Coastguard Worker     if (ptr < end)
196*5e7646d2SAndroid Build Coastguard Worker       *ptr++ = ':';
197*5e7646d2SAndroid Build Coastguard Worker     else
198*5e7646d2SAndroid Build Coastguard Worker       goto assemble_overflow;
199*5e7646d2SAndroid Build Coastguard Worker   }
200*5e7646d2SAndroid Build Coastguard Worker   else
201*5e7646d2SAndroid Build Coastguard Worker   {
202*5e7646d2SAndroid Build Coastguard Worker    /*
203*5e7646d2SAndroid Build Coastguard Worker     * Schemes other than geo:, mailto:, and tel: typically have //...
204*5e7646d2SAndroid Build Coastguard Worker     */
205*5e7646d2SAndroid Build Coastguard Worker 
206*5e7646d2SAndroid Build Coastguard Worker     if ((ptr + 2) < end)
207*5e7646d2SAndroid Build Coastguard Worker     {
208*5e7646d2SAndroid Build Coastguard Worker       *ptr++ = ':';
209*5e7646d2SAndroid Build Coastguard Worker       *ptr++ = '/';
210*5e7646d2SAndroid Build Coastguard Worker       *ptr++ = '/';
211*5e7646d2SAndroid Build Coastguard Worker     }
212*5e7646d2SAndroid Build Coastguard Worker     else
213*5e7646d2SAndroid Build Coastguard Worker       goto assemble_overflow;
214*5e7646d2SAndroid Build Coastguard Worker   }
215*5e7646d2SAndroid Build Coastguard Worker 
216*5e7646d2SAndroid Build Coastguard Worker  /*
217*5e7646d2SAndroid Build Coastguard Worker   * Next the username and hostname, if any...
218*5e7646d2SAndroid Build Coastguard Worker   */
219*5e7646d2SAndroid Build Coastguard Worker 
220*5e7646d2SAndroid Build Coastguard Worker   if (host)
221*5e7646d2SAndroid Build Coastguard Worker   {
222*5e7646d2SAndroid Build Coastguard Worker     const char	*hostptr;		/* Pointer into hostname */
223*5e7646d2SAndroid Build Coastguard Worker     int		have_ipv6;		/* Do we have an IPv6 address? */
224*5e7646d2SAndroid Build Coastguard Worker 
225*5e7646d2SAndroid Build Coastguard Worker     if (username && *username)
226*5e7646d2SAndroid Build Coastguard Worker     {
227*5e7646d2SAndroid Build Coastguard Worker      /*
228*5e7646d2SAndroid Build Coastguard Worker       * Add username@ first...
229*5e7646d2SAndroid Build Coastguard Worker       */
230*5e7646d2SAndroid Build Coastguard Worker 
231*5e7646d2SAndroid Build Coastguard Worker       ptr = http_copy_encode(ptr, username, end, "/?#[]@", NULL,
232*5e7646d2SAndroid Build Coastguard Worker                              encoding & HTTP_URI_CODING_USERNAME);
233*5e7646d2SAndroid Build Coastguard Worker 
234*5e7646d2SAndroid Build Coastguard Worker       if (!ptr)
235*5e7646d2SAndroid Build Coastguard Worker         goto assemble_overflow;
236*5e7646d2SAndroid Build Coastguard Worker 
237*5e7646d2SAndroid Build Coastguard Worker       if (ptr < end)
238*5e7646d2SAndroid Build Coastguard Worker 	*ptr++ = '@';
239*5e7646d2SAndroid Build Coastguard Worker       else
240*5e7646d2SAndroid Build Coastguard Worker         goto assemble_overflow;
241*5e7646d2SAndroid Build Coastguard Worker     }
242*5e7646d2SAndroid Build Coastguard Worker 
243*5e7646d2SAndroid Build Coastguard Worker    /*
244*5e7646d2SAndroid Build Coastguard Worker     * Then add the hostname.  Since IPv6 is a particular pain to deal
245*5e7646d2SAndroid Build Coastguard Worker     * with, we have several special cases to deal with.  If we get
246*5e7646d2SAndroid Build Coastguard Worker     * an IPv6 address with brackets around it, assume it is already in
247*5e7646d2SAndroid Build Coastguard Worker     * URI format.  Since DNS-SD service names can sometimes look like
248*5e7646d2SAndroid Build Coastguard Worker     * raw IPv6 addresses, we specifically look for "._tcp" in the name,
249*5e7646d2SAndroid Build Coastguard Worker     * too...
250*5e7646d2SAndroid Build Coastguard Worker     */
251*5e7646d2SAndroid Build Coastguard Worker 
252*5e7646d2SAndroid Build Coastguard Worker     for (hostptr = host,
253*5e7646d2SAndroid Build Coastguard Worker              have_ipv6 = strchr(host, ':') && !strstr(host, "._tcp");
254*5e7646d2SAndroid Build Coastguard Worker          *hostptr && have_ipv6;
255*5e7646d2SAndroid Build Coastguard Worker          hostptr ++)
256*5e7646d2SAndroid Build Coastguard Worker       if (*hostptr != ':' && !isxdigit(*hostptr & 255))
257*5e7646d2SAndroid Build Coastguard Worker       {
258*5e7646d2SAndroid Build Coastguard Worker         have_ipv6 = *hostptr == '%';
259*5e7646d2SAndroid Build Coastguard Worker         break;
260*5e7646d2SAndroid Build Coastguard Worker       }
261*5e7646d2SAndroid Build Coastguard Worker 
262*5e7646d2SAndroid Build Coastguard Worker     if (have_ipv6)
263*5e7646d2SAndroid Build Coastguard Worker     {
264*5e7646d2SAndroid Build Coastguard Worker      /*
265*5e7646d2SAndroid Build Coastguard Worker       * We have a raw IPv6 address...
266*5e7646d2SAndroid Build Coastguard Worker       */
267*5e7646d2SAndroid Build Coastguard Worker 
268*5e7646d2SAndroid Build Coastguard Worker       if (strchr(host, '%') && !(encoding & HTTP_URI_CODING_RFC6874))
269*5e7646d2SAndroid Build Coastguard Worker       {
270*5e7646d2SAndroid Build Coastguard Worker        /*
271*5e7646d2SAndroid Build Coastguard Worker         * We have a link-local address, add "[v1." prefix...
272*5e7646d2SAndroid Build Coastguard Worker 	*/
273*5e7646d2SAndroid Build Coastguard Worker 
274*5e7646d2SAndroid Build Coastguard Worker 	if ((ptr + 4) < end)
275*5e7646d2SAndroid Build Coastguard Worker 	{
276*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = '[';
277*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = 'v';
278*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = '1';
279*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = '.';
280*5e7646d2SAndroid Build Coastguard Worker 	}
281*5e7646d2SAndroid Build Coastguard Worker 	else
282*5e7646d2SAndroid Build Coastguard Worker           goto assemble_overflow;
283*5e7646d2SAndroid Build Coastguard Worker       }
284*5e7646d2SAndroid Build Coastguard Worker       else
285*5e7646d2SAndroid Build Coastguard Worker       {
286*5e7646d2SAndroid Build Coastguard Worker        /*
287*5e7646d2SAndroid Build Coastguard Worker         * We have a normal (or RFC 6874 link-local) address, add "[" prefix...
288*5e7646d2SAndroid Build Coastguard Worker 	*/
289*5e7646d2SAndroid Build Coastguard Worker 
290*5e7646d2SAndroid Build Coastguard Worker 	if (ptr < end)
291*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = '[';
292*5e7646d2SAndroid Build Coastguard Worker 	else
293*5e7646d2SAndroid Build Coastguard Worker           goto assemble_overflow;
294*5e7646d2SAndroid Build Coastguard Worker       }
295*5e7646d2SAndroid Build Coastguard Worker 
296*5e7646d2SAndroid Build Coastguard Worker      /*
297*5e7646d2SAndroid Build Coastguard Worker       * Copy the rest of the IPv6 address, and terminate with "]".
298*5e7646d2SAndroid Build Coastguard Worker       */
299*5e7646d2SAndroid Build Coastguard Worker 
300*5e7646d2SAndroid Build Coastguard Worker       while (ptr < end && *host)
301*5e7646d2SAndroid Build Coastguard Worker       {
302*5e7646d2SAndroid Build Coastguard Worker         if (*host == '%')
303*5e7646d2SAndroid Build Coastguard Worker         {
304*5e7646d2SAndroid Build Coastguard Worker          /*
305*5e7646d2SAndroid Build Coastguard Worker           * Convert/encode zone separator
306*5e7646d2SAndroid Build Coastguard Worker           */
307*5e7646d2SAndroid Build Coastguard Worker 
308*5e7646d2SAndroid Build Coastguard Worker           if (encoding & HTTP_URI_CODING_RFC6874)
309*5e7646d2SAndroid Build Coastguard Worker           {
310*5e7646d2SAndroid Build Coastguard Worker             if (ptr >= (end - 2))
311*5e7646d2SAndroid Build Coastguard Worker               goto assemble_overflow;
312*5e7646d2SAndroid Build Coastguard Worker 
313*5e7646d2SAndroid Build Coastguard Worker             *ptr++ = '%';
314*5e7646d2SAndroid Build Coastguard Worker             *ptr++ = '2';
315*5e7646d2SAndroid Build Coastguard Worker             *ptr++ = '5';
316*5e7646d2SAndroid Build Coastguard Worker           }
317*5e7646d2SAndroid Build Coastguard Worker           else
318*5e7646d2SAndroid Build Coastguard Worker 	    *ptr++ = '+';
319*5e7646d2SAndroid Build Coastguard Worker 
320*5e7646d2SAndroid Build Coastguard Worker 	  host ++;
321*5e7646d2SAndroid Build Coastguard Worker 	}
322*5e7646d2SAndroid Build Coastguard Worker 	else
323*5e7646d2SAndroid Build Coastguard Worker 	  *ptr++ = *host++;
324*5e7646d2SAndroid Build Coastguard Worker       }
325*5e7646d2SAndroid Build Coastguard Worker 
326*5e7646d2SAndroid Build Coastguard Worker       if (*host)
327*5e7646d2SAndroid Build Coastguard Worker         goto assemble_overflow;
328*5e7646d2SAndroid Build Coastguard Worker 
329*5e7646d2SAndroid Build Coastguard Worker       if (ptr < end)
330*5e7646d2SAndroid Build Coastguard Worker 	*ptr++ = ']';
331*5e7646d2SAndroid Build Coastguard Worker       else
332*5e7646d2SAndroid Build Coastguard Worker         goto assemble_overflow;
333*5e7646d2SAndroid Build Coastguard Worker     }
334*5e7646d2SAndroid Build Coastguard Worker     else
335*5e7646d2SAndroid Build Coastguard Worker     {
336*5e7646d2SAndroid Build Coastguard Worker      /*
337*5e7646d2SAndroid Build Coastguard Worker       * Otherwise, just copy the host string (the extra chars are not in the
338*5e7646d2SAndroid Build Coastguard Worker       * "reg-name" ABNF rule; anything <= SP or >= DEL plus % gets automatically
339*5e7646d2SAndroid Build Coastguard Worker       * percent-encoded.
340*5e7646d2SAndroid Build Coastguard Worker       */
341*5e7646d2SAndroid Build Coastguard Worker 
342*5e7646d2SAndroid Build Coastguard Worker       ptr = http_copy_encode(ptr, host, end, "\"#/:<>?@[\\]^`{|}", NULL,
343*5e7646d2SAndroid Build Coastguard Worker                              encoding & HTTP_URI_CODING_HOSTNAME);
344*5e7646d2SAndroid Build Coastguard Worker 
345*5e7646d2SAndroid Build Coastguard Worker       if (!ptr)
346*5e7646d2SAndroid Build Coastguard Worker         goto assemble_overflow;
347*5e7646d2SAndroid Build Coastguard Worker     }
348*5e7646d2SAndroid Build Coastguard Worker 
349*5e7646d2SAndroid Build Coastguard Worker    /*
350*5e7646d2SAndroid Build Coastguard Worker     * Finish things off with the port number...
351*5e7646d2SAndroid Build Coastguard Worker     */
352*5e7646d2SAndroid Build Coastguard Worker 
353*5e7646d2SAndroid Build Coastguard Worker     if (port > 0)
354*5e7646d2SAndroid Build Coastguard Worker     {
355*5e7646d2SAndroid Build Coastguard Worker       snprintf(ptr, (size_t)(end - ptr + 1), ":%d", port);
356*5e7646d2SAndroid Build Coastguard Worker       ptr += strlen(ptr);
357*5e7646d2SAndroid Build Coastguard Worker 
358*5e7646d2SAndroid Build Coastguard Worker       if (ptr >= end)
359*5e7646d2SAndroid Build Coastguard Worker 	goto assemble_overflow;
360*5e7646d2SAndroid Build Coastguard Worker     }
361*5e7646d2SAndroid Build Coastguard Worker   }
362*5e7646d2SAndroid Build Coastguard Worker 
363*5e7646d2SAndroid Build Coastguard Worker  /*
364*5e7646d2SAndroid Build Coastguard Worker   * Last but not least, add the resource string...
365*5e7646d2SAndroid Build Coastguard Worker   */
366*5e7646d2SAndroid Build Coastguard Worker 
367*5e7646d2SAndroid Build Coastguard Worker   if (resource)
368*5e7646d2SAndroid Build Coastguard Worker   {
369*5e7646d2SAndroid Build Coastguard Worker     char	*query;			/* Pointer to query string */
370*5e7646d2SAndroid Build Coastguard Worker 
371*5e7646d2SAndroid Build Coastguard Worker 
372*5e7646d2SAndroid Build Coastguard Worker    /*
373*5e7646d2SAndroid Build Coastguard Worker     * Copy the resource string up to the query string if present...
374*5e7646d2SAndroid Build Coastguard Worker     */
375*5e7646d2SAndroid Build Coastguard Worker 
376*5e7646d2SAndroid Build Coastguard Worker     query = strchr(resource, '?');
377*5e7646d2SAndroid Build Coastguard Worker     ptr   = http_copy_encode(ptr, resource, end, NULL, "?",
378*5e7646d2SAndroid Build Coastguard Worker                              encoding & HTTP_URI_CODING_RESOURCE);
379*5e7646d2SAndroid Build Coastguard Worker     if (!ptr)
380*5e7646d2SAndroid Build Coastguard Worker       goto assemble_overflow;
381*5e7646d2SAndroid Build Coastguard Worker 
382*5e7646d2SAndroid Build Coastguard Worker     if (query)
383*5e7646d2SAndroid Build Coastguard Worker     {
384*5e7646d2SAndroid Build Coastguard Worker      /*
385*5e7646d2SAndroid Build Coastguard Worker       * Copy query string without encoding...
386*5e7646d2SAndroid Build Coastguard Worker       */
387*5e7646d2SAndroid Build Coastguard Worker 
388*5e7646d2SAndroid Build Coastguard Worker       ptr = http_copy_encode(ptr, query, end, NULL, NULL,
389*5e7646d2SAndroid Build Coastguard Worker 			     encoding & HTTP_URI_CODING_QUERY);
390*5e7646d2SAndroid Build Coastguard Worker       if (!ptr)
391*5e7646d2SAndroid Build Coastguard Worker 	goto assemble_overflow;
392*5e7646d2SAndroid Build Coastguard Worker     }
393*5e7646d2SAndroid Build Coastguard Worker   }
394*5e7646d2SAndroid Build Coastguard Worker   else if (ptr < end)
395*5e7646d2SAndroid Build Coastguard Worker     *ptr++ = '/';
396*5e7646d2SAndroid Build Coastguard Worker   else
397*5e7646d2SAndroid Build Coastguard Worker     goto assemble_overflow;
398*5e7646d2SAndroid Build Coastguard Worker 
399*5e7646d2SAndroid Build Coastguard Worker  /*
400*5e7646d2SAndroid Build Coastguard Worker   * Nul-terminate the URI buffer and return with no errors...
401*5e7646d2SAndroid Build Coastguard Worker   */
402*5e7646d2SAndroid Build Coastguard Worker 
403*5e7646d2SAndroid Build Coastguard Worker   *ptr = '\0';
404*5e7646d2SAndroid Build Coastguard Worker 
405*5e7646d2SAndroid Build Coastguard Worker   return (HTTP_URI_STATUS_OK);
406*5e7646d2SAndroid Build Coastguard Worker 
407*5e7646d2SAndroid Build Coastguard Worker  /*
408*5e7646d2SAndroid Build Coastguard Worker   * Clear the URI string and return an overflow error; I don't usually
409*5e7646d2SAndroid Build Coastguard Worker   * like goto's, but in this case it makes sense...
410*5e7646d2SAndroid Build Coastguard Worker   */
411*5e7646d2SAndroid Build Coastguard Worker 
412*5e7646d2SAndroid Build Coastguard Worker   assemble_overflow:
413*5e7646d2SAndroid Build Coastguard Worker 
414*5e7646d2SAndroid Build Coastguard Worker   *uri = '\0';
415*5e7646d2SAndroid Build Coastguard Worker   return (HTTP_URI_STATUS_OVERFLOW);
416*5e7646d2SAndroid Build Coastguard Worker }
417*5e7646d2SAndroid Build Coastguard Worker 
418*5e7646d2SAndroid Build Coastguard Worker 
419*5e7646d2SAndroid Build Coastguard Worker /*
420*5e7646d2SAndroid Build Coastguard Worker  * 'httpAssembleURIf()' - Assemble a uniform resource identifier from its
421*5e7646d2SAndroid Build Coastguard Worker  *                        components with a formatted resource.
422*5e7646d2SAndroid Build Coastguard Worker  *
423*5e7646d2SAndroid Build Coastguard Worker  * This function creates a formatted version of the resource string
424*5e7646d2SAndroid Build Coastguard Worker  * argument "resourcef" and escapes reserved characters in the URI
425*5e7646d2SAndroid Build Coastguard Worker  * depending on the value of the "encoding" argument.  You should use
426*5e7646d2SAndroid Build Coastguard Worker  * this function in place of traditional string functions whenever
427*5e7646d2SAndroid Build Coastguard Worker  * you need to create a URI string.
428*5e7646d2SAndroid Build Coastguard Worker  *
429*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.2/macOS 10.5@
430*5e7646d2SAndroid Build Coastguard Worker  */
431*5e7646d2SAndroid Build Coastguard Worker 
432*5e7646d2SAndroid Build Coastguard Worker http_uri_status_t			/* O - URI status */
httpAssembleURIf(http_uri_coding_t encoding,char * uri,int urilen,const char * scheme,const char * username,const char * host,int port,const char * resourcef,...)433*5e7646d2SAndroid Build Coastguard Worker httpAssembleURIf(
434*5e7646d2SAndroid Build Coastguard Worker     http_uri_coding_t encoding,		/* I - Encoding flags */
435*5e7646d2SAndroid Build Coastguard Worker     char              *uri,		/* I - URI buffer */
436*5e7646d2SAndroid Build Coastguard Worker     int               urilen,		/* I - Size of URI buffer */
437*5e7646d2SAndroid Build Coastguard Worker     const char        *scheme,		/* I - Scheme name */
438*5e7646d2SAndroid Build Coastguard Worker     const char        *username,	/* I - Username */
439*5e7646d2SAndroid Build Coastguard Worker     const char        *host,		/* I - Hostname or address */
440*5e7646d2SAndroid Build Coastguard Worker     int               port,		/* I - Port number */
441*5e7646d2SAndroid Build Coastguard Worker     const char        *resourcef,	/* I - Printf-style resource */
442*5e7646d2SAndroid Build Coastguard Worker     ...)				/* I - Additional arguments as needed */
443*5e7646d2SAndroid Build Coastguard Worker {
444*5e7646d2SAndroid Build Coastguard Worker   va_list	ap;			/* Pointer to additional arguments */
445*5e7646d2SAndroid Build Coastguard Worker   char		resource[1024];		/* Formatted resource string */
446*5e7646d2SAndroid Build Coastguard Worker   int		bytes;			/* Bytes in formatted string */
447*5e7646d2SAndroid Build Coastguard Worker 
448*5e7646d2SAndroid Build Coastguard Worker 
449*5e7646d2SAndroid Build Coastguard Worker  /*
450*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
451*5e7646d2SAndroid Build Coastguard Worker   */
452*5e7646d2SAndroid Build Coastguard Worker 
453*5e7646d2SAndroid Build Coastguard Worker   if (!uri || urilen < 1 || !scheme || port < 0 || !resourcef)
454*5e7646d2SAndroid Build Coastguard Worker   {
455*5e7646d2SAndroid Build Coastguard Worker     if (uri)
456*5e7646d2SAndroid Build Coastguard Worker       *uri = '\0';
457*5e7646d2SAndroid Build Coastguard Worker 
458*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_URI_STATUS_BAD_ARGUMENTS);
459*5e7646d2SAndroid Build Coastguard Worker   }
460*5e7646d2SAndroid Build Coastguard Worker 
461*5e7646d2SAndroid Build Coastguard Worker  /*
462*5e7646d2SAndroid Build Coastguard Worker   * Format the resource string and assemble the URI...
463*5e7646d2SAndroid Build Coastguard Worker   */
464*5e7646d2SAndroid Build Coastguard Worker 
465*5e7646d2SAndroid Build Coastguard Worker   va_start(ap, resourcef);
466*5e7646d2SAndroid Build Coastguard Worker   bytes = vsnprintf(resource, sizeof(resource), resourcef, ap);
467*5e7646d2SAndroid Build Coastguard Worker   va_end(ap);
468*5e7646d2SAndroid Build Coastguard Worker 
469*5e7646d2SAndroid Build Coastguard Worker   if ((size_t)bytes >= sizeof(resource))
470*5e7646d2SAndroid Build Coastguard Worker   {
471*5e7646d2SAndroid Build Coastguard Worker     *uri = '\0';
472*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_URI_STATUS_OVERFLOW);
473*5e7646d2SAndroid Build Coastguard Worker   }
474*5e7646d2SAndroid Build Coastguard Worker   else
475*5e7646d2SAndroid Build Coastguard Worker     return (httpAssembleURI(encoding,  uri, urilen, scheme, username, host,
476*5e7646d2SAndroid Build Coastguard Worker                             port, resource));
477*5e7646d2SAndroid Build Coastguard Worker }
478*5e7646d2SAndroid Build Coastguard Worker 
479*5e7646d2SAndroid Build Coastguard Worker 
480*5e7646d2SAndroid Build Coastguard Worker /*
481*5e7646d2SAndroid Build Coastguard Worker  * 'httpAssembleUUID()' - Assemble a name-based UUID URN conforming to RFC 4122.
482*5e7646d2SAndroid Build Coastguard Worker  *
483*5e7646d2SAndroid Build Coastguard Worker  * This function creates a unique 128-bit identifying number using the server
484*5e7646d2SAndroid Build Coastguard Worker  * name, port number, random data, and optionally an object name and/or object
485*5e7646d2SAndroid Build Coastguard Worker  * number.  The result is formatted as a UUID URN as defined in RFC 4122.
486*5e7646d2SAndroid Build Coastguard Worker  *
487*5e7646d2SAndroid Build Coastguard Worker  * The buffer needs to be at least 46 bytes in size.
488*5e7646d2SAndroid Build Coastguard Worker  *
489*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.7/macOS 10.9@
490*5e7646d2SAndroid Build Coastguard Worker  */
491*5e7646d2SAndroid Build Coastguard Worker 
492*5e7646d2SAndroid Build Coastguard Worker char *					/* I - UUID string */
httpAssembleUUID(const char * server,int port,const char * name,int number,char * buffer,size_t bufsize)493*5e7646d2SAndroid Build Coastguard Worker httpAssembleUUID(const char *server,	/* I - Server name */
494*5e7646d2SAndroid Build Coastguard Worker 		 int        port,	/* I - Port number */
495*5e7646d2SAndroid Build Coastguard Worker 		 const char *name,	/* I - Object name or NULL */
496*5e7646d2SAndroid Build Coastguard Worker 		 int        number,	/* I - Object number or 0 */
497*5e7646d2SAndroid Build Coastguard Worker 		 char       *buffer,	/* I - String buffer */
498*5e7646d2SAndroid Build Coastguard Worker 		 size_t     bufsize)	/* I - Size of buffer */
499*5e7646d2SAndroid Build Coastguard Worker {
500*5e7646d2SAndroid Build Coastguard Worker   char			data[1024];	/* Source string for MD5 */
501*5e7646d2SAndroid Build Coastguard Worker   unsigned char		md5sum[16];	/* MD5 digest/sum */
502*5e7646d2SAndroid Build Coastguard Worker 
503*5e7646d2SAndroid Build Coastguard Worker 
504*5e7646d2SAndroid Build Coastguard Worker  /*
505*5e7646d2SAndroid Build Coastguard Worker   * Build a version 3 UUID conforming to RFC 4122.
506*5e7646d2SAndroid Build Coastguard Worker   *
507*5e7646d2SAndroid Build Coastguard Worker   * Start with the MD5 sum of the server, port, object name and
508*5e7646d2SAndroid Build Coastguard Worker   * number, and some random data on the end.
509*5e7646d2SAndroid Build Coastguard Worker   */
510*5e7646d2SAndroid Build Coastguard Worker 
511*5e7646d2SAndroid Build Coastguard Worker   snprintf(data, sizeof(data), "%s:%d:%s:%d:%04x:%04x", server,
512*5e7646d2SAndroid Build Coastguard Worker            port, name ? name : server, number,
513*5e7646d2SAndroid Build Coastguard Worker 	   (unsigned)CUPS_RAND() & 0xffff, (unsigned)CUPS_RAND() & 0xffff);
514*5e7646d2SAndroid Build Coastguard Worker 
515*5e7646d2SAndroid Build Coastguard Worker   cupsHashData("md5", (unsigned char *)data, strlen(data), md5sum, sizeof(md5sum));
516*5e7646d2SAndroid Build Coastguard Worker 
517*5e7646d2SAndroid Build Coastguard Worker  /*
518*5e7646d2SAndroid Build Coastguard Worker   * Generate the UUID from the MD5...
519*5e7646d2SAndroid Build Coastguard Worker   */
520*5e7646d2SAndroid Build Coastguard Worker 
521*5e7646d2SAndroid Build Coastguard Worker   snprintf(buffer, bufsize,
522*5e7646d2SAndroid Build Coastguard Worker            "urn:uuid:%02x%02x%02x%02x-%02x%02x-%02x%02x-%02x%02x-"
523*5e7646d2SAndroid Build Coastguard Worker 	   "%02x%02x%02x%02x%02x%02x",
524*5e7646d2SAndroid Build Coastguard Worker 	   md5sum[0], md5sum[1], md5sum[2], md5sum[3], md5sum[4], md5sum[5],
525*5e7646d2SAndroid Build Coastguard Worker 	   (md5sum[6] & 15) | 0x30, md5sum[7], (md5sum[8] & 0x3f) | 0x40,
526*5e7646d2SAndroid Build Coastguard Worker 	   md5sum[9], md5sum[10], md5sum[11], md5sum[12], md5sum[13],
527*5e7646d2SAndroid Build Coastguard Worker 	   md5sum[14], md5sum[15]);
528*5e7646d2SAndroid Build Coastguard Worker 
529*5e7646d2SAndroid Build Coastguard Worker   return (buffer);
530*5e7646d2SAndroid Build Coastguard Worker }
531*5e7646d2SAndroid Build Coastguard Worker 
532*5e7646d2SAndroid Build Coastguard Worker 
533*5e7646d2SAndroid Build Coastguard Worker /*
534*5e7646d2SAndroid Build Coastguard Worker  * 'httpDecode64()' - Base64-decode a string.
535*5e7646d2SAndroid Build Coastguard Worker  *
536*5e7646d2SAndroid Build Coastguard Worker  * This function is deprecated. Use the httpDecode64_2() function instead
537*5e7646d2SAndroid Build Coastguard Worker  * which provides buffer length arguments.
538*5e7646d2SAndroid Build Coastguard Worker  *
539*5e7646d2SAndroid Build Coastguard Worker  * @deprecated@ @exclude all@
540*5e7646d2SAndroid Build Coastguard Worker  */
541*5e7646d2SAndroid Build Coastguard Worker 
542*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Decoded string */
httpDecode64(char * out,const char * in)543*5e7646d2SAndroid Build Coastguard Worker httpDecode64(char       *out,		/* I - String to write to */
544*5e7646d2SAndroid Build Coastguard Worker              const char *in)		/* I - String to read from */
545*5e7646d2SAndroid Build Coastguard Worker {
546*5e7646d2SAndroid Build Coastguard Worker   int	outlen;				/* Output buffer length */
547*5e7646d2SAndroid Build Coastguard Worker 
548*5e7646d2SAndroid Build Coastguard Worker 
549*5e7646d2SAndroid Build Coastguard Worker  /*
550*5e7646d2SAndroid Build Coastguard Worker   * Use the old maximum buffer size for binary compatibility...
551*5e7646d2SAndroid Build Coastguard Worker   */
552*5e7646d2SAndroid Build Coastguard Worker 
553*5e7646d2SAndroid Build Coastguard Worker   outlen = 512;
554*5e7646d2SAndroid Build Coastguard Worker 
555*5e7646d2SAndroid Build Coastguard Worker   return (httpDecode64_2(out, &outlen, in));
556*5e7646d2SAndroid Build Coastguard Worker }
557*5e7646d2SAndroid Build Coastguard Worker 
558*5e7646d2SAndroid Build Coastguard Worker 
559*5e7646d2SAndroid Build Coastguard Worker /*
560*5e7646d2SAndroid Build Coastguard Worker  * 'httpDecode64_2()' - Base64-decode a string.
561*5e7646d2SAndroid Build Coastguard Worker  *
562*5e7646d2SAndroid Build Coastguard Worker  * The caller must initialize "outlen" to the maximum size of the decoded
563*5e7646d2SAndroid Build Coastguard Worker  * string before calling @code httpDecode64_2@.  On return "outlen" contains the
564*5e7646d2SAndroid Build Coastguard Worker  * decoded length of the string.
565*5e7646d2SAndroid Build Coastguard Worker  *
566*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.1.21/macOS 10.4@
567*5e7646d2SAndroid Build Coastguard Worker  */
568*5e7646d2SAndroid Build Coastguard Worker 
569*5e7646d2SAndroid Build Coastguard Worker char *					/* O  - Decoded string */
httpDecode64_2(char * out,int * outlen,const char * in)570*5e7646d2SAndroid Build Coastguard Worker httpDecode64_2(char       *out,		/* I  - String to write to */
571*5e7646d2SAndroid Build Coastguard Worker 	       int        *outlen,	/* IO - Size of output string */
572*5e7646d2SAndroid Build Coastguard Worker                const char *in)		/* I  - String to read from */
573*5e7646d2SAndroid Build Coastguard Worker {
574*5e7646d2SAndroid Build Coastguard Worker   int		pos;			/* Bit position */
575*5e7646d2SAndroid Build Coastguard Worker   unsigned	base64;			/* Value of this character */
576*5e7646d2SAndroid Build Coastguard Worker   char		*outptr,		/* Output pointer */
577*5e7646d2SAndroid Build Coastguard Worker 		*outend;		/* End of output buffer */
578*5e7646d2SAndroid Build Coastguard Worker 
579*5e7646d2SAndroid Build Coastguard Worker 
580*5e7646d2SAndroid Build Coastguard Worker  /*
581*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
582*5e7646d2SAndroid Build Coastguard Worker   */
583*5e7646d2SAndroid Build Coastguard Worker 
584*5e7646d2SAndroid Build Coastguard Worker   if (!out || !outlen || *outlen < 1 || !in)
585*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
586*5e7646d2SAndroid Build Coastguard Worker 
587*5e7646d2SAndroid Build Coastguard Worker   if (!*in)
588*5e7646d2SAndroid Build Coastguard Worker   {
589*5e7646d2SAndroid Build Coastguard Worker     *out    = '\0';
590*5e7646d2SAndroid Build Coastguard Worker     *outlen = 0;
591*5e7646d2SAndroid Build Coastguard Worker 
592*5e7646d2SAndroid Build Coastguard Worker     return (out);
593*5e7646d2SAndroid Build Coastguard Worker   }
594*5e7646d2SAndroid Build Coastguard Worker 
595*5e7646d2SAndroid Build Coastguard Worker  /*
596*5e7646d2SAndroid Build Coastguard Worker   * Convert from base-64 to bytes...
597*5e7646d2SAndroid Build Coastguard Worker   */
598*5e7646d2SAndroid Build Coastguard Worker 
599*5e7646d2SAndroid Build Coastguard Worker   for (outptr = out, outend = out + *outlen - 1, pos = 0; *in != '\0'; in ++)
600*5e7646d2SAndroid Build Coastguard Worker   {
601*5e7646d2SAndroid Build Coastguard Worker    /*
602*5e7646d2SAndroid Build Coastguard Worker     * Decode this character into a number from 0 to 63...
603*5e7646d2SAndroid Build Coastguard Worker     */
604*5e7646d2SAndroid Build Coastguard Worker 
605*5e7646d2SAndroid Build Coastguard Worker     if (*in >= 'A' && *in <= 'Z')
606*5e7646d2SAndroid Build Coastguard Worker       base64 = (unsigned)(*in - 'A');
607*5e7646d2SAndroid Build Coastguard Worker     else if (*in >= 'a' && *in <= 'z')
608*5e7646d2SAndroid Build Coastguard Worker       base64 = (unsigned)(*in - 'a' + 26);
609*5e7646d2SAndroid Build Coastguard Worker     else if (*in >= '0' && *in <= '9')
610*5e7646d2SAndroid Build Coastguard Worker       base64 = (unsigned)(*in - '0' + 52);
611*5e7646d2SAndroid Build Coastguard Worker     else if (*in == '+')
612*5e7646d2SAndroid Build Coastguard Worker       base64 = 62;
613*5e7646d2SAndroid Build Coastguard Worker     else if (*in == '/')
614*5e7646d2SAndroid Build Coastguard Worker       base64 = 63;
615*5e7646d2SAndroid Build Coastguard Worker     else if (*in == '=')
616*5e7646d2SAndroid Build Coastguard Worker       break;
617*5e7646d2SAndroid Build Coastguard Worker     else
618*5e7646d2SAndroid Build Coastguard Worker       continue;
619*5e7646d2SAndroid Build Coastguard Worker 
620*5e7646d2SAndroid Build Coastguard Worker    /*
621*5e7646d2SAndroid Build Coastguard Worker     * Store the result in the appropriate chars...
622*5e7646d2SAndroid Build Coastguard Worker     */
623*5e7646d2SAndroid Build Coastguard Worker 
624*5e7646d2SAndroid Build Coastguard Worker     switch (pos)
625*5e7646d2SAndroid Build Coastguard Worker     {
626*5e7646d2SAndroid Build Coastguard Worker       case 0 :
627*5e7646d2SAndroid Build Coastguard Worker           if (outptr < outend)
628*5e7646d2SAndroid Build Coastguard Worker             *outptr = (char)(base64 << 2);
629*5e7646d2SAndroid Build Coastguard Worker 	  pos ++;
630*5e7646d2SAndroid Build Coastguard Worker 	  break;
631*5e7646d2SAndroid Build Coastguard Worker       case 1 :
632*5e7646d2SAndroid Build Coastguard Worker           if (outptr < outend)
633*5e7646d2SAndroid Build Coastguard Worker             *outptr++ |= (char)((base64 >> 4) & 3);
634*5e7646d2SAndroid Build Coastguard Worker           if (outptr < outend)
635*5e7646d2SAndroid Build Coastguard Worker 	    *outptr = (char)((base64 << 4) & 255);
636*5e7646d2SAndroid Build Coastguard Worker 	  pos ++;
637*5e7646d2SAndroid Build Coastguard Worker 	  break;
638*5e7646d2SAndroid Build Coastguard Worker       case 2 :
639*5e7646d2SAndroid Build Coastguard Worker           if (outptr < outend)
640*5e7646d2SAndroid Build Coastguard Worker             *outptr++ |= (char)((base64 >> 2) & 15);
641*5e7646d2SAndroid Build Coastguard Worker           if (outptr < outend)
642*5e7646d2SAndroid Build Coastguard Worker 	    *outptr = (char)((base64 << 6) & 255);
643*5e7646d2SAndroid Build Coastguard Worker 	  pos ++;
644*5e7646d2SAndroid Build Coastguard Worker 	  break;
645*5e7646d2SAndroid Build Coastguard Worker       case 3 :
646*5e7646d2SAndroid Build Coastguard Worker           if (outptr < outend)
647*5e7646d2SAndroid Build Coastguard Worker             *outptr++ |= (char)base64;
648*5e7646d2SAndroid Build Coastguard Worker 	  pos = 0;
649*5e7646d2SAndroid Build Coastguard Worker 	  break;
650*5e7646d2SAndroid Build Coastguard Worker     }
651*5e7646d2SAndroid Build Coastguard Worker   }
652*5e7646d2SAndroid Build Coastguard Worker 
653*5e7646d2SAndroid Build Coastguard Worker   *outptr = '\0';
654*5e7646d2SAndroid Build Coastguard Worker 
655*5e7646d2SAndroid Build Coastguard Worker  /*
656*5e7646d2SAndroid Build Coastguard Worker   * Return the decoded string and size...
657*5e7646d2SAndroid Build Coastguard Worker   */
658*5e7646d2SAndroid Build Coastguard Worker 
659*5e7646d2SAndroid Build Coastguard Worker   *outlen = (int)(outptr - out);
660*5e7646d2SAndroid Build Coastguard Worker 
661*5e7646d2SAndroid Build Coastguard Worker   return (out);
662*5e7646d2SAndroid Build Coastguard Worker }
663*5e7646d2SAndroid Build Coastguard Worker 
664*5e7646d2SAndroid Build Coastguard Worker 
665*5e7646d2SAndroid Build Coastguard Worker /*
666*5e7646d2SAndroid Build Coastguard Worker  * 'httpEncode64()' - Base64-encode a string.
667*5e7646d2SAndroid Build Coastguard Worker  *
668*5e7646d2SAndroid Build Coastguard Worker  * This function is deprecated. Use the httpEncode64_2() function instead
669*5e7646d2SAndroid Build Coastguard Worker  * which provides buffer length arguments.
670*5e7646d2SAndroid Build Coastguard Worker  *
671*5e7646d2SAndroid Build Coastguard Worker  * @deprecated@ @exclude all@
672*5e7646d2SAndroid Build Coastguard Worker  */
673*5e7646d2SAndroid Build Coastguard Worker 
674*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Encoded string */
httpEncode64(char * out,const char * in)675*5e7646d2SAndroid Build Coastguard Worker httpEncode64(char       *out,		/* I - String to write to */
676*5e7646d2SAndroid Build Coastguard Worker              const char *in)		/* I - String to read from */
677*5e7646d2SAndroid Build Coastguard Worker {
678*5e7646d2SAndroid Build Coastguard Worker   return (httpEncode64_2(out, 512, in, (int)strlen(in)));
679*5e7646d2SAndroid Build Coastguard Worker }
680*5e7646d2SAndroid Build Coastguard Worker 
681*5e7646d2SAndroid Build Coastguard Worker 
682*5e7646d2SAndroid Build Coastguard Worker /*
683*5e7646d2SAndroid Build Coastguard Worker  * 'httpEncode64_2()' - Base64-encode a string.
684*5e7646d2SAndroid Build Coastguard Worker  *
685*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.1.21/macOS 10.4@
686*5e7646d2SAndroid Build Coastguard Worker  */
687*5e7646d2SAndroid Build Coastguard Worker 
688*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Encoded string */
httpEncode64_2(char * out,int outlen,const char * in,int inlen)689*5e7646d2SAndroid Build Coastguard Worker httpEncode64_2(char       *out,		/* I - String to write to */
690*5e7646d2SAndroid Build Coastguard Worker 	       int        outlen,	/* I - Maximum size of output string */
691*5e7646d2SAndroid Build Coastguard Worker                const char *in,		/* I - String to read from */
692*5e7646d2SAndroid Build Coastguard Worker 	       int        inlen)	/* I - Size of input string */
693*5e7646d2SAndroid Build Coastguard Worker {
694*5e7646d2SAndroid Build Coastguard Worker   char		*outptr,		/* Output pointer */
695*5e7646d2SAndroid Build Coastguard Worker 		*outend;		/* End of output buffer */
696*5e7646d2SAndroid Build Coastguard Worker   static const char base64[] =		/* Base64 characters... */
697*5e7646d2SAndroid Build Coastguard Worker   		{
698*5e7646d2SAndroid Build Coastguard Worker 		  "ABCDEFGHIJKLMNOPQRSTUVWXYZ"
699*5e7646d2SAndroid Build Coastguard Worker 		  "abcdefghijklmnopqrstuvwxyz"
700*5e7646d2SAndroid Build Coastguard Worker 		  "0123456789"
701*5e7646d2SAndroid Build Coastguard Worker 		  "+/"
702*5e7646d2SAndroid Build Coastguard Worker   		};
703*5e7646d2SAndroid Build Coastguard Worker 
704*5e7646d2SAndroid Build Coastguard Worker 
705*5e7646d2SAndroid Build Coastguard Worker  /*
706*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
707*5e7646d2SAndroid Build Coastguard Worker   */
708*5e7646d2SAndroid Build Coastguard Worker 
709*5e7646d2SAndroid Build Coastguard Worker   if (!out || outlen < 1 || !in)
710*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
711*5e7646d2SAndroid Build Coastguard Worker 
712*5e7646d2SAndroid Build Coastguard Worker  /*
713*5e7646d2SAndroid Build Coastguard Worker   * Convert bytes to base-64...
714*5e7646d2SAndroid Build Coastguard Worker   */
715*5e7646d2SAndroid Build Coastguard Worker 
716*5e7646d2SAndroid Build Coastguard Worker   for (outptr = out, outend = out + outlen - 1; inlen > 0; in ++, inlen --)
717*5e7646d2SAndroid Build Coastguard Worker   {
718*5e7646d2SAndroid Build Coastguard Worker    /*
719*5e7646d2SAndroid Build Coastguard Worker     * Encode the up to 3 characters as 4 Base64 numbers...
720*5e7646d2SAndroid Build Coastguard Worker     */
721*5e7646d2SAndroid Build Coastguard Worker 
722*5e7646d2SAndroid Build Coastguard Worker     if (outptr < outend)
723*5e7646d2SAndroid Build Coastguard Worker       *outptr ++ = base64[(in[0] & 255) >> 2];
724*5e7646d2SAndroid Build Coastguard Worker 
725*5e7646d2SAndroid Build Coastguard Worker     if (outptr < outend)
726*5e7646d2SAndroid Build Coastguard Worker     {
727*5e7646d2SAndroid Build Coastguard Worker       if (inlen > 1)
728*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = base64[(((in[0] & 255) << 4) | ((in[1] & 255) >> 4)) & 63];
729*5e7646d2SAndroid Build Coastguard Worker       else
730*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = base64[((in[0] & 255) << 4) & 63];
731*5e7646d2SAndroid Build Coastguard Worker     }
732*5e7646d2SAndroid Build Coastguard Worker 
733*5e7646d2SAndroid Build Coastguard Worker     in ++;
734*5e7646d2SAndroid Build Coastguard Worker     inlen --;
735*5e7646d2SAndroid Build Coastguard Worker     if (inlen <= 0)
736*5e7646d2SAndroid Build Coastguard Worker     {
737*5e7646d2SAndroid Build Coastguard Worker       if (outptr < outend)
738*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = '=';
739*5e7646d2SAndroid Build Coastguard Worker       if (outptr < outend)
740*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = '=';
741*5e7646d2SAndroid Build Coastguard Worker       break;
742*5e7646d2SAndroid Build Coastguard Worker     }
743*5e7646d2SAndroid Build Coastguard Worker 
744*5e7646d2SAndroid Build Coastguard Worker     if (outptr < outend)
745*5e7646d2SAndroid Build Coastguard Worker     {
746*5e7646d2SAndroid Build Coastguard Worker       if (inlen > 1)
747*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = base64[(((in[0] & 255) << 2) | ((in[1] & 255) >> 6)) & 63];
748*5e7646d2SAndroid Build Coastguard Worker       else
749*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = base64[((in[0] & 255) << 2) & 63];
750*5e7646d2SAndroid Build Coastguard Worker     }
751*5e7646d2SAndroid Build Coastguard Worker 
752*5e7646d2SAndroid Build Coastguard Worker     in ++;
753*5e7646d2SAndroid Build Coastguard Worker     inlen --;
754*5e7646d2SAndroid Build Coastguard Worker     if (inlen <= 0)
755*5e7646d2SAndroid Build Coastguard Worker     {
756*5e7646d2SAndroid Build Coastguard Worker       if (outptr < outend)
757*5e7646d2SAndroid Build Coastguard Worker         *outptr ++ = '=';
758*5e7646d2SAndroid Build Coastguard Worker       break;
759*5e7646d2SAndroid Build Coastguard Worker     }
760*5e7646d2SAndroid Build Coastguard Worker 
761*5e7646d2SAndroid Build Coastguard Worker     if (outptr < outend)
762*5e7646d2SAndroid Build Coastguard Worker       *outptr ++ = base64[in[0] & 63];
763*5e7646d2SAndroid Build Coastguard Worker   }
764*5e7646d2SAndroid Build Coastguard Worker 
765*5e7646d2SAndroid Build Coastguard Worker   *outptr = '\0';
766*5e7646d2SAndroid Build Coastguard Worker 
767*5e7646d2SAndroid Build Coastguard Worker  /*
768*5e7646d2SAndroid Build Coastguard Worker   * Return the encoded string...
769*5e7646d2SAndroid Build Coastguard Worker   */
770*5e7646d2SAndroid Build Coastguard Worker 
771*5e7646d2SAndroid Build Coastguard Worker   return (out);
772*5e7646d2SAndroid Build Coastguard Worker }
773*5e7646d2SAndroid Build Coastguard Worker 
774*5e7646d2SAndroid Build Coastguard Worker 
775*5e7646d2SAndroid Build Coastguard Worker /*
776*5e7646d2SAndroid Build Coastguard Worker  * 'httpGetDateString()' - Get a formatted date/time string from a time value.
777*5e7646d2SAndroid Build Coastguard Worker  *
778*5e7646d2SAndroid Build Coastguard Worker  * @deprecated@ @exclude all@
779*5e7646d2SAndroid Build Coastguard Worker  */
780*5e7646d2SAndroid Build Coastguard Worker 
781*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Date/time string */
httpGetDateString(time_t t)782*5e7646d2SAndroid Build Coastguard Worker httpGetDateString(time_t t)		/* I - Time in seconds */
783*5e7646d2SAndroid Build Coastguard Worker {
784*5e7646d2SAndroid Build Coastguard Worker   _cups_globals_t *cg = _cupsGlobals();	/* Pointer to library globals */
785*5e7646d2SAndroid Build Coastguard Worker 
786*5e7646d2SAndroid Build Coastguard Worker 
787*5e7646d2SAndroid Build Coastguard Worker   return (httpGetDateString2(t, cg->http_date, sizeof(cg->http_date)));
788*5e7646d2SAndroid Build Coastguard Worker }
789*5e7646d2SAndroid Build Coastguard Worker 
790*5e7646d2SAndroid Build Coastguard Worker 
791*5e7646d2SAndroid Build Coastguard Worker /*
792*5e7646d2SAndroid Build Coastguard Worker  * 'httpGetDateString2()' - Get a formatted date/time string from a time value.
793*5e7646d2SAndroid Build Coastguard Worker  *
794*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.2/macOS 10.5@
795*5e7646d2SAndroid Build Coastguard Worker  */
796*5e7646d2SAndroid Build Coastguard Worker 
797*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Date/time string */
httpGetDateString2(time_t t,char * s,int slen)798*5e7646d2SAndroid Build Coastguard Worker httpGetDateString2(time_t t,		/* I - Time in seconds */
799*5e7646d2SAndroid Build Coastguard Worker                    char   *s,		/* I - String buffer */
800*5e7646d2SAndroid Build Coastguard Worker 		   int    slen)		/* I - Size of string buffer */
801*5e7646d2SAndroid Build Coastguard Worker {
802*5e7646d2SAndroid Build Coastguard Worker   struct tm	tdate;			/* UNIX date/time data */
803*5e7646d2SAndroid Build Coastguard Worker 
804*5e7646d2SAndroid Build Coastguard Worker 
805*5e7646d2SAndroid Build Coastguard Worker   gmtime_r(&t, &tdate);
806*5e7646d2SAndroid Build Coastguard Worker 
807*5e7646d2SAndroid Build Coastguard Worker   snprintf(s, (size_t)slen, "%s, %02d %s %d %02d:%02d:%02d GMT", http_days[tdate.tm_wday], tdate.tm_mday, http_months[tdate.tm_mon], tdate.tm_year + 1900, tdate.tm_hour, tdate.tm_min, tdate.tm_sec);
808*5e7646d2SAndroid Build Coastguard Worker 
809*5e7646d2SAndroid Build Coastguard Worker   return (s);
810*5e7646d2SAndroid Build Coastguard Worker }
811*5e7646d2SAndroid Build Coastguard Worker 
812*5e7646d2SAndroid Build Coastguard Worker 
813*5e7646d2SAndroid Build Coastguard Worker /*
814*5e7646d2SAndroid Build Coastguard Worker  * 'httpGetDateTime()' - Get a time value from a formatted date/time string.
815*5e7646d2SAndroid Build Coastguard Worker  */
816*5e7646d2SAndroid Build Coastguard Worker 
817*5e7646d2SAndroid Build Coastguard Worker time_t					/* O - Time in seconds */
httpGetDateTime(const char * s)818*5e7646d2SAndroid Build Coastguard Worker httpGetDateTime(const char *s)		/* I - Date/time string */
819*5e7646d2SAndroid Build Coastguard Worker {
820*5e7646d2SAndroid Build Coastguard Worker   int		i;			/* Looping var */
821*5e7646d2SAndroid Build Coastguard Worker   char		mon[16];		/* Abbreviated month name */
822*5e7646d2SAndroid Build Coastguard Worker   int		day, year;		/* Day of month and year */
823*5e7646d2SAndroid Build Coastguard Worker   int		hour, min, sec;		/* Time */
824*5e7646d2SAndroid Build Coastguard Worker   int		days;			/* Number of days since 1970 */
825*5e7646d2SAndroid Build Coastguard Worker   static const int normal_days[] =	/* Days to a month, normal years */
826*5e7646d2SAndroid Build Coastguard Worker 		{ 0, 31, 59, 90, 120, 151, 181, 212, 243, 273, 304, 334, 365 };
827*5e7646d2SAndroid Build Coastguard Worker   static const int leap_days[] =	/* Days to a month, leap years */
828*5e7646d2SAndroid Build Coastguard Worker 		{ 0, 31, 60, 91, 121, 152, 182, 213, 244, 274, 305, 335, 366 };
829*5e7646d2SAndroid Build Coastguard Worker 
830*5e7646d2SAndroid Build Coastguard Worker 
831*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2httpGetDateTime(s=\"%s\")", s));
832*5e7646d2SAndroid Build Coastguard Worker 
833*5e7646d2SAndroid Build Coastguard Worker  /*
834*5e7646d2SAndroid Build Coastguard Worker   * Extract the date and time from the formatted string...
835*5e7646d2SAndroid Build Coastguard Worker   */
836*5e7646d2SAndroid Build Coastguard Worker 
837*5e7646d2SAndroid Build Coastguard Worker   if (sscanf(s, "%*s%d%15s%d%d:%d:%d", &day, mon, &year, &hour, &min, &sec) < 6)
838*5e7646d2SAndroid Build Coastguard Worker     return (0);
839*5e7646d2SAndroid Build Coastguard Worker 
840*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4httpGetDateTime: day=%d, mon=\"%s\", year=%d, hour=%d, "
841*5e7646d2SAndroid Build Coastguard Worker                 "min=%d, sec=%d", day, mon, year, hour, min, sec));
842*5e7646d2SAndroid Build Coastguard Worker 
843*5e7646d2SAndroid Build Coastguard Worker  /*
844*5e7646d2SAndroid Build Coastguard Worker   * Check for invalid year (RFC 7231 says it's 4DIGIT)
845*5e7646d2SAndroid Build Coastguard Worker   */
846*5e7646d2SAndroid Build Coastguard Worker 
847*5e7646d2SAndroid Build Coastguard Worker   if (year > 9999)
848*5e7646d2SAndroid Build Coastguard Worker     return (0);
849*5e7646d2SAndroid Build Coastguard Worker 
850*5e7646d2SAndroid Build Coastguard Worker  /*
851*5e7646d2SAndroid Build Coastguard Worker   * Convert the month name to a number from 0 to 11.
852*5e7646d2SAndroid Build Coastguard Worker   */
853*5e7646d2SAndroid Build Coastguard Worker 
854*5e7646d2SAndroid Build Coastguard Worker   for (i = 0; i < 12; i ++)
855*5e7646d2SAndroid Build Coastguard Worker     if (!_cups_strcasecmp(mon, http_months[i]))
856*5e7646d2SAndroid Build Coastguard Worker       break;
857*5e7646d2SAndroid Build Coastguard Worker 
858*5e7646d2SAndroid Build Coastguard Worker   if (i >= 12)
859*5e7646d2SAndroid Build Coastguard Worker     return (0);
860*5e7646d2SAndroid Build Coastguard Worker 
861*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4httpGetDateTime: i=%d", i));
862*5e7646d2SAndroid Build Coastguard Worker 
863*5e7646d2SAndroid Build Coastguard Worker  /*
864*5e7646d2SAndroid Build Coastguard Worker   * Now convert the date and time to a UNIX time value in seconds since
865*5e7646d2SAndroid Build Coastguard Worker   * 1970.  We can't use mktime() since the timezone may not be UTC but
866*5e7646d2SAndroid Build Coastguard Worker   * the date/time string *is* UTC.
867*5e7646d2SAndroid Build Coastguard Worker   */
868*5e7646d2SAndroid Build Coastguard Worker 
869*5e7646d2SAndroid Build Coastguard Worker   if ((year & 3) == 0 && ((year % 100) != 0 || (year % 400) == 0))
870*5e7646d2SAndroid Build Coastguard Worker     days = leap_days[i] + day - 1;
871*5e7646d2SAndroid Build Coastguard Worker   else
872*5e7646d2SAndroid Build Coastguard Worker     days = normal_days[i] + day - 1;
873*5e7646d2SAndroid Build Coastguard Worker 
874*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4httpGetDateTime: days=%d", days));
875*5e7646d2SAndroid Build Coastguard Worker 
876*5e7646d2SAndroid Build Coastguard Worker   days += (year - 1970) * 365 +		/* 365 days per year (normally) */
877*5e7646d2SAndroid Build Coastguard Worker           ((year - 1) / 4 - 492) -	/* + leap days */
878*5e7646d2SAndroid Build Coastguard Worker 	  ((year - 1) / 100 - 19) +	/* - 100 year days */
879*5e7646d2SAndroid Build Coastguard Worker           ((year - 1) / 400 - 4);	/* + 400 year days */
880*5e7646d2SAndroid Build Coastguard Worker 
881*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4httpGetDateTime: days=%d\n", days));
882*5e7646d2SAndroid Build Coastguard Worker 
883*5e7646d2SAndroid Build Coastguard Worker   return (days * 86400 + hour * 3600 + min * 60 + sec);
884*5e7646d2SAndroid Build Coastguard Worker }
885*5e7646d2SAndroid Build Coastguard Worker 
886*5e7646d2SAndroid Build Coastguard Worker 
887*5e7646d2SAndroid Build Coastguard Worker /*
888*5e7646d2SAndroid Build Coastguard Worker  * 'httpSeparate()' - Separate a Universal Resource Identifier into its
889*5e7646d2SAndroid Build Coastguard Worker  *                    components.
890*5e7646d2SAndroid Build Coastguard Worker  *
891*5e7646d2SAndroid Build Coastguard Worker  * This function is deprecated; use the httpSeparateURI() function instead.
892*5e7646d2SAndroid Build Coastguard Worker  *
893*5e7646d2SAndroid Build Coastguard Worker  * @deprecated@ @exclude all@
894*5e7646d2SAndroid Build Coastguard Worker  */
895*5e7646d2SAndroid Build Coastguard Worker 
896*5e7646d2SAndroid Build Coastguard Worker void
httpSeparate(const char * uri,char * scheme,char * username,char * host,int * port,char * resource)897*5e7646d2SAndroid Build Coastguard Worker httpSeparate(const char *uri,		/* I - Universal Resource Identifier */
898*5e7646d2SAndroid Build Coastguard Worker              char       *scheme,	/* O - Scheme [32] (http, https, etc.) */
899*5e7646d2SAndroid Build Coastguard Worker 	     char       *username,	/* O - Username [1024] */
900*5e7646d2SAndroid Build Coastguard Worker 	     char       *host,		/* O - Hostname [1024] */
901*5e7646d2SAndroid Build Coastguard Worker 	     int        *port,		/* O - Port number to use */
902*5e7646d2SAndroid Build Coastguard Worker              char       *resource)	/* O - Resource/filename [1024] */
903*5e7646d2SAndroid Build Coastguard Worker {
904*5e7646d2SAndroid Build Coastguard Worker   httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, 32, username,
905*5e7646d2SAndroid Build Coastguard Worker                   HTTP_MAX_URI, host, HTTP_MAX_URI, port, resource,
906*5e7646d2SAndroid Build Coastguard Worker 		  HTTP_MAX_URI);
907*5e7646d2SAndroid Build Coastguard Worker }
908*5e7646d2SAndroid Build Coastguard Worker 
909*5e7646d2SAndroid Build Coastguard Worker 
910*5e7646d2SAndroid Build Coastguard Worker /*
911*5e7646d2SAndroid Build Coastguard Worker  * 'httpSeparate2()' - Separate a Universal Resource Identifier into its
912*5e7646d2SAndroid Build Coastguard Worker  *                     components.
913*5e7646d2SAndroid Build Coastguard Worker  *
914*5e7646d2SAndroid Build Coastguard Worker  * This function is deprecated; use the httpSeparateURI() function instead.
915*5e7646d2SAndroid Build Coastguard Worker  *
916*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.1.21/macOS 10.4@
917*5e7646d2SAndroid Build Coastguard Worker  * @deprecated@ @exclude all@
918*5e7646d2SAndroid Build Coastguard Worker  */
919*5e7646d2SAndroid Build Coastguard Worker 
920*5e7646d2SAndroid Build Coastguard Worker void
httpSeparate2(const char * uri,char * scheme,int schemelen,char * username,int usernamelen,char * host,int hostlen,int * port,char * resource,int resourcelen)921*5e7646d2SAndroid Build Coastguard Worker httpSeparate2(const char *uri,		/* I - Universal Resource Identifier */
922*5e7646d2SAndroid Build Coastguard Worker               char       *scheme,	/* O - Scheme (http, https, etc.) */
923*5e7646d2SAndroid Build Coastguard Worker 	      int        schemelen,	/* I - Size of scheme buffer */
924*5e7646d2SAndroid Build Coastguard Worker 	      char       *username,	/* O - Username */
925*5e7646d2SAndroid Build Coastguard Worker 	      int        usernamelen,	/* I - Size of username buffer */
926*5e7646d2SAndroid Build Coastguard Worker 	      char       *host,		/* O - Hostname */
927*5e7646d2SAndroid Build Coastguard Worker 	      int        hostlen,	/* I - Size of hostname buffer */
928*5e7646d2SAndroid Build Coastguard Worker 	      int        *port,		/* O - Port number to use */
929*5e7646d2SAndroid Build Coastguard Worker               char       *resource,	/* O - Resource/filename */
930*5e7646d2SAndroid Build Coastguard Worker 	      int        resourcelen)	/* I - Size of resource buffer */
931*5e7646d2SAndroid Build Coastguard Worker {
932*5e7646d2SAndroid Build Coastguard Worker   httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme, schemelen, username,
933*5e7646d2SAndroid Build Coastguard Worker                   usernamelen, host, hostlen, port, resource, resourcelen);
934*5e7646d2SAndroid Build Coastguard Worker }
935*5e7646d2SAndroid Build Coastguard Worker 
936*5e7646d2SAndroid Build Coastguard Worker 
937*5e7646d2SAndroid Build Coastguard Worker /*
938*5e7646d2SAndroid Build Coastguard Worker  * 'httpSeparateURI()' - Separate a Universal Resource Identifier into its
939*5e7646d2SAndroid Build Coastguard Worker  *                       components.
940*5e7646d2SAndroid Build Coastguard Worker  *
941*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.2/macOS 10.5@
942*5e7646d2SAndroid Build Coastguard Worker  */
943*5e7646d2SAndroid Build Coastguard Worker 
944*5e7646d2SAndroid Build Coastguard Worker http_uri_status_t			/* O - Result of separation */
httpSeparateURI(http_uri_coding_t decoding,const char * uri,char * scheme,int schemelen,char * username,int usernamelen,char * host,int hostlen,int * port,char * resource,int resourcelen)945*5e7646d2SAndroid Build Coastguard Worker httpSeparateURI(
946*5e7646d2SAndroid Build Coastguard Worker     http_uri_coding_t decoding,		/* I - Decoding flags */
947*5e7646d2SAndroid Build Coastguard Worker     const char        *uri,		/* I - Universal Resource Identifier */
948*5e7646d2SAndroid Build Coastguard Worker     char              *scheme,		/* O - Scheme (http, https, etc.) */
949*5e7646d2SAndroid Build Coastguard Worker     int               schemelen,	/* I - Size of scheme buffer */
950*5e7646d2SAndroid Build Coastguard Worker     char              *username,	/* O - Username */
951*5e7646d2SAndroid Build Coastguard Worker     int               usernamelen,	/* I - Size of username buffer */
952*5e7646d2SAndroid Build Coastguard Worker     char              *host,		/* O - Hostname */
953*5e7646d2SAndroid Build Coastguard Worker     int               hostlen,		/* I - Size of hostname buffer */
954*5e7646d2SAndroid Build Coastguard Worker     int               *port,		/* O - Port number to use */
955*5e7646d2SAndroid Build Coastguard Worker     char              *resource,	/* O - Resource/filename */
956*5e7646d2SAndroid Build Coastguard Worker     int               resourcelen)	/* I - Size of resource buffer */
957*5e7646d2SAndroid Build Coastguard Worker {
958*5e7646d2SAndroid Build Coastguard Worker   char			*ptr,		/* Pointer into string... */
959*5e7646d2SAndroid Build Coastguard Worker 			*end;		/* End of string */
960*5e7646d2SAndroid Build Coastguard Worker   const char		*sep;		/* Separator character */
961*5e7646d2SAndroid Build Coastguard Worker   http_uri_status_t	status;		/* Result of separation */
962*5e7646d2SAndroid Build Coastguard Worker 
963*5e7646d2SAndroid Build Coastguard Worker 
964*5e7646d2SAndroid Build Coastguard Worker  /*
965*5e7646d2SAndroid Build Coastguard Worker   * Initialize everything to blank...
966*5e7646d2SAndroid Build Coastguard Worker   */
967*5e7646d2SAndroid Build Coastguard Worker 
968*5e7646d2SAndroid Build Coastguard Worker   if (scheme && schemelen > 0)
969*5e7646d2SAndroid Build Coastguard Worker     *scheme = '\0';
970*5e7646d2SAndroid Build Coastguard Worker 
971*5e7646d2SAndroid Build Coastguard Worker   if (username && usernamelen > 0)
972*5e7646d2SAndroid Build Coastguard Worker     *username = '\0';
973*5e7646d2SAndroid Build Coastguard Worker 
974*5e7646d2SAndroid Build Coastguard Worker   if (host && hostlen > 0)
975*5e7646d2SAndroid Build Coastguard Worker     *host = '\0';
976*5e7646d2SAndroid Build Coastguard Worker 
977*5e7646d2SAndroid Build Coastguard Worker   if (port)
978*5e7646d2SAndroid Build Coastguard Worker     *port = 0;
979*5e7646d2SAndroid Build Coastguard Worker 
980*5e7646d2SAndroid Build Coastguard Worker   if (resource && resourcelen > 0)
981*5e7646d2SAndroid Build Coastguard Worker     *resource = '\0';
982*5e7646d2SAndroid Build Coastguard Worker 
983*5e7646d2SAndroid Build Coastguard Worker  /*
984*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
985*5e7646d2SAndroid Build Coastguard Worker   */
986*5e7646d2SAndroid Build Coastguard Worker 
987*5e7646d2SAndroid Build Coastguard Worker   if (!uri || !port || !scheme || schemelen <= 0 || !username ||
988*5e7646d2SAndroid Build Coastguard Worker       usernamelen <= 0 || !host || hostlen <= 0 || !resource ||
989*5e7646d2SAndroid Build Coastguard Worker       resourcelen <= 0)
990*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_URI_STATUS_BAD_ARGUMENTS);
991*5e7646d2SAndroid Build Coastguard Worker 
992*5e7646d2SAndroid Build Coastguard Worker   if (!*uri)
993*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_URI_STATUS_BAD_URI);
994*5e7646d2SAndroid Build Coastguard Worker 
995*5e7646d2SAndroid Build Coastguard Worker  /*
996*5e7646d2SAndroid Build Coastguard Worker   * Grab the scheme portion of the URI...
997*5e7646d2SAndroid Build Coastguard Worker   */
998*5e7646d2SAndroid Build Coastguard Worker 
999*5e7646d2SAndroid Build Coastguard Worker   status = HTTP_URI_STATUS_OK;
1000*5e7646d2SAndroid Build Coastguard Worker 
1001*5e7646d2SAndroid Build Coastguard Worker   if (!strncmp(uri, "//", 2))
1002*5e7646d2SAndroid Build Coastguard Worker   {
1003*5e7646d2SAndroid Build Coastguard Worker    /*
1004*5e7646d2SAndroid Build Coastguard Worker     * Workaround for HP IPP client bug...
1005*5e7646d2SAndroid Build Coastguard Worker     */
1006*5e7646d2SAndroid Build Coastguard Worker 
1007*5e7646d2SAndroid Build Coastguard Worker     strlcpy(scheme, "ipp", (size_t)schemelen);
1008*5e7646d2SAndroid Build Coastguard Worker     status = HTTP_URI_STATUS_MISSING_SCHEME;
1009*5e7646d2SAndroid Build Coastguard Worker   }
1010*5e7646d2SAndroid Build Coastguard Worker   else if (*uri == '/')
1011*5e7646d2SAndroid Build Coastguard Worker   {
1012*5e7646d2SAndroid Build Coastguard Worker    /*
1013*5e7646d2SAndroid Build Coastguard Worker     * Filename...
1014*5e7646d2SAndroid Build Coastguard Worker     */
1015*5e7646d2SAndroid Build Coastguard Worker 
1016*5e7646d2SAndroid Build Coastguard Worker     strlcpy(scheme, "file", (size_t)schemelen);
1017*5e7646d2SAndroid Build Coastguard Worker     status = HTTP_URI_STATUS_MISSING_SCHEME;
1018*5e7646d2SAndroid Build Coastguard Worker   }
1019*5e7646d2SAndroid Build Coastguard Worker   else
1020*5e7646d2SAndroid Build Coastguard Worker   {
1021*5e7646d2SAndroid Build Coastguard Worker    /*
1022*5e7646d2SAndroid Build Coastguard Worker     * Standard URI with scheme...
1023*5e7646d2SAndroid Build Coastguard Worker     */
1024*5e7646d2SAndroid Build Coastguard Worker 
1025*5e7646d2SAndroid Build Coastguard Worker     for (ptr = scheme, end = scheme + schemelen - 1;
1026*5e7646d2SAndroid Build Coastguard Worker          *uri && *uri != ':' && ptr < end;)
1027*5e7646d2SAndroid Build Coastguard Worker       if (strchr("ABCDEFGHIJKLMNOPQRSTUVWXYZ"
1028*5e7646d2SAndroid Build Coastguard Worker                  "abcdefghijklmnopqrstuvwxyz"
1029*5e7646d2SAndroid Build Coastguard Worker 		 "0123456789-+.", *uri) != NULL)
1030*5e7646d2SAndroid Build Coastguard Worker         *ptr++ = *uri++;
1031*5e7646d2SAndroid Build Coastguard Worker       else
1032*5e7646d2SAndroid Build Coastguard Worker         break;
1033*5e7646d2SAndroid Build Coastguard Worker 
1034*5e7646d2SAndroid Build Coastguard Worker     *ptr = '\0';
1035*5e7646d2SAndroid Build Coastguard Worker 
1036*5e7646d2SAndroid Build Coastguard Worker     if (*uri != ':' || *scheme == '.' || !*scheme)
1037*5e7646d2SAndroid Build Coastguard Worker     {
1038*5e7646d2SAndroid Build Coastguard Worker       *scheme = '\0';
1039*5e7646d2SAndroid Build Coastguard Worker       return (HTTP_URI_STATUS_BAD_SCHEME);
1040*5e7646d2SAndroid Build Coastguard Worker     }
1041*5e7646d2SAndroid Build Coastguard Worker 
1042*5e7646d2SAndroid Build Coastguard Worker     uri ++;
1043*5e7646d2SAndroid Build Coastguard Worker   }
1044*5e7646d2SAndroid Build Coastguard Worker 
1045*5e7646d2SAndroid Build Coastguard Worker  /*
1046*5e7646d2SAndroid Build Coastguard Worker   * Set the default port number...
1047*5e7646d2SAndroid Build Coastguard Worker   */
1048*5e7646d2SAndroid Build Coastguard Worker 
1049*5e7646d2SAndroid Build Coastguard Worker   if (!strcmp(scheme, "http"))
1050*5e7646d2SAndroid Build Coastguard Worker     *port = 80;
1051*5e7646d2SAndroid Build Coastguard Worker   else if (!strcmp(scheme, "https"))
1052*5e7646d2SAndroid Build Coastguard Worker     *port = 443;
1053*5e7646d2SAndroid Build Coastguard Worker   else if (!strcmp(scheme, "ipp") || !strcmp(scheme, "ipps"))
1054*5e7646d2SAndroid Build Coastguard Worker     *port = 631;
1055*5e7646d2SAndroid Build Coastguard Worker   else if (!_cups_strcasecmp(scheme, "lpd"))
1056*5e7646d2SAndroid Build Coastguard Worker     *port = 515;
1057*5e7646d2SAndroid Build Coastguard Worker   else if (!strcmp(scheme, "socket"))	/* Not yet registered with IANA... */
1058*5e7646d2SAndroid Build Coastguard Worker     *port = 9100;
1059*5e7646d2SAndroid Build Coastguard Worker   else if (strcmp(scheme, "file") && strcmp(scheme, "mailto") && strcmp(scheme, "tel"))
1060*5e7646d2SAndroid Build Coastguard Worker     status = HTTP_URI_STATUS_UNKNOWN_SCHEME;
1061*5e7646d2SAndroid Build Coastguard Worker 
1062*5e7646d2SAndroid Build Coastguard Worker  /*
1063*5e7646d2SAndroid Build Coastguard Worker   * Now see if we have a hostname...
1064*5e7646d2SAndroid Build Coastguard Worker   */
1065*5e7646d2SAndroid Build Coastguard Worker 
1066*5e7646d2SAndroid Build Coastguard Worker   if (!strncmp(uri, "//", 2))
1067*5e7646d2SAndroid Build Coastguard Worker   {
1068*5e7646d2SAndroid Build Coastguard Worker    /*
1069*5e7646d2SAndroid Build Coastguard Worker     * Yes, extract it...
1070*5e7646d2SAndroid Build Coastguard Worker     */
1071*5e7646d2SAndroid Build Coastguard Worker 
1072*5e7646d2SAndroid Build Coastguard Worker     uri += 2;
1073*5e7646d2SAndroid Build Coastguard Worker 
1074*5e7646d2SAndroid Build Coastguard Worker    /*
1075*5e7646d2SAndroid Build Coastguard Worker     * Grab the username, if any...
1076*5e7646d2SAndroid Build Coastguard Worker     */
1077*5e7646d2SAndroid Build Coastguard Worker 
1078*5e7646d2SAndroid Build Coastguard Worker     if ((sep = strpbrk(uri, "@/")) != NULL && *sep == '@')
1079*5e7646d2SAndroid Build Coastguard Worker     {
1080*5e7646d2SAndroid Build Coastguard Worker      /*
1081*5e7646d2SAndroid Build Coastguard Worker       * Get a username:password combo...
1082*5e7646d2SAndroid Build Coastguard Worker       */
1083*5e7646d2SAndroid Build Coastguard Worker 
1084*5e7646d2SAndroid Build Coastguard Worker       uri = http_copy_decode(username, uri, usernamelen, "@",
1085*5e7646d2SAndroid Build Coastguard Worker                              decoding & HTTP_URI_CODING_USERNAME);
1086*5e7646d2SAndroid Build Coastguard Worker 
1087*5e7646d2SAndroid Build Coastguard Worker       if (!uri)
1088*5e7646d2SAndroid Build Coastguard Worker       {
1089*5e7646d2SAndroid Build Coastguard Worker         *username = '\0';
1090*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_USERNAME);
1091*5e7646d2SAndroid Build Coastguard Worker       }
1092*5e7646d2SAndroid Build Coastguard Worker 
1093*5e7646d2SAndroid Build Coastguard Worker       uri ++;
1094*5e7646d2SAndroid Build Coastguard Worker     }
1095*5e7646d2SAndroid Build Coastguard Worker 
1096*5e7646d2SAndroid Build Coastguard Worker    /*
1097*5e7646d2SAndroid Build Coastguard Worker     * Then the hostname/IP address...
1098*5e7646d2SAndroid Build Coastguard Worker     */
1099*5e7646d2SAndroid Build Coastguard Worker 
1100*5e7646d2SAndroid Build Coastguard Worker     if (*uri == '[')
1101*5e7646d2SAndroid Build Coastguard Worker     {
1102*5e7646d2SAndroid Build Coastguard Worker      /*
1103*5e7646d2SAndroid Build Coastguard Worker       * Grab IPv6 address...
1104*5e7646d2SAndroid Build Coastguard Worker       */
1105*5e7646d2SAndroid Build Coastguard Worker 
1106*5e7646d2SAndroid Build Coastguard Worker       uri ++;
1107*5e7646d2SAndroid Build Coastguard Worker       if (*uri == 'v')
1108*5e7646d2SAndroid Build Coastguard Worker       {
1109*5e7646d2SAndroid Build Coastguard Worker        /*
1110*5e7646d2SAndroid Build Coastguard Worker         * Skip IPvFuture ("vXXXX.") prefix...
1111*5e7646d2SAndroid Build Coastguard Worker         */
1112*5e7646d2SAndroid Build Coastguard Worker 
1113*5e7646d2SAndroid Build Coastguard Worker         uri ++;
1114*5e7646d2SAndroid Build Coastguard Worker 
1115*5e7646d2SAndroid Build Coastguard Worker         while (isxdigit(*uri & 255))
1116*5e7646d2SAndroid Build Coastguard Worker           uri ++;
1117*5e7646d2SAndroid Build Coastguard Worker 
1118*5e7646d2SAndroid Build Coastguard Worker         if (*uri != '.')
1119*5e7646d2SAndroid Build Coastguard Worker         {
1120*5e7646d2SAndroid Build Coastguard Worker 	  *host = '\0';
1121*5e7646d2SAndroid Build Coastguard Worker 	  return (HTTP_URI_STATUS_BAD_HOSTNAME);
1122*5e7646d2SAndroid Build Coastguard Worker         }
1123*5e7646d2SAndroid Build Coastguard Worker 
1124*5e7646d2SAndroid Build Coastguard Worker         uri ++;
1125*5e7646d2SAndroid Build Coastguard Worker       }
1126*5e7646d2SAndroid Build Coastguard Worker 
1127*5e7646d2SAndroid Build Coastguard Worker       uri = http_copy_decode(host, uri, hostlen, "]",
1128*5e7646d2SAndroid Build Coastguard Worker                              decoding & HTTP_URI_CODING_HOSTNAME);
1129*5e7646d2SAndroid Build Coastguard Worker 
1130*5e7646d2SAndroid Build Coastguard Worker       if (!uri)
1131*5e7646d2SAndroid Build Coastguard Worker       {
1132*5e7646d2SAndroid Build Coastguard Worker         *host = '\0';
1133*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_HOSTNAME);
1134*5e7646d2SAndroid Build Coastguard Worker       }
1135*5e7646d2SAndroid Build Coastguard Worker 
1136*5e7646d2SAndroid Build Coastguard Worker      /*
1137*5e7646d2SAndroid Build Coastguard Worker       * Validate value...
1138*5e7646d2SAndroid Build Coastguard Worker       */
1139*5e7646d2SAndroid Build Coastguard Worker 
1140*5e7646d2SAndroid Build Coastguard Worker       if (*uri != ']')
1141*5e7646d2SAndroid Build Coastguard Worker       {
1142*5e7646d2SAndroid Build Coastguard Worker         *host = '\0';
1143*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_HOSTNAME);
1144*5e7646d2SAndroid Build Coastguard Worker       }
1145*5e7646d2SAndroid Build Coastguard Worker 
1146*5e7646d2SAndroid Build Coastguard Worker       uri ++;
1147*5e7646d2SAndroid Build Coastguard Worker 
1148*5e7646d2SAndroid Build Coastguard Worker       for (ptr = host; *ptr; ptr ++)
1149*5e7646d2SAndroid Build Coastguard Worker         if (*ptr == '+')
1150*5e7646d2SAndroid Build Coastguard Worker 	{
1151*5e7646d2SAndroid Build Coastguard Worker 	 /*
1152*5e7646d2SAndroid Build Coastguard Worker 	  * Convert zone separator to % and stop here...
1153*5e7646d2SAndroid Build Coastguard Worker 	  */
1154*5e7646d2SAndroid Build Coastguard Worker 
1155*5e7646d2SAndroid Build Coastguard Worker 	  *ptr = '%';
1156*5e7646d2SAndroid Build Coastguard Worker 	  break;
1157*5e7646d2SAndroid Build Coastguard Worker 	}
1158*5e7646d2SAndroid Build Coastguard Worker 	else if (*ptr == '%')
1159*5e7646d2SAndroid Build Coastguard Worker 	{
1160*5e7646d2SAndroid Build Coastguard Worker 	 /*
1161*5e7646d2SAndroid Build Coastguard Worker 	  * Stop at zone separator (RFC 6874)
1162*5e7646d2SAndroid Build Coastguard Worker 	  */
1163*5e7646d2SAndroid Build Coastguard Worker 
1164*5e7646d2SAndroid Build Coastguard Worker 	  break;
1165*5e7646d2SAndroid Build Coastguard Worker 	}
1166*5e7646d2SAndroid Build Coastguard Worker 	else if (*ptr != ':' && *ptr != '.' && !isxdigit(*ptr & 255))
1167*5e7646d2SAndroid Build Coastguard Worker 	{
1168*5e7646d2SAndroid Build Coastguard Worker 	  *host = '\0';
1169*5e7646d2SAndroid Build Coastguard Worker 	  return (HTTP_URI_STATUS_BAD_HOSTNAME);
1170*5e7646d2SAndroid Build Coastguard Worker 	}
1171*5e7646d2SAndroid Build Coastguard Worker     }
1172*5e7646d2SAndroid Build Coastguard Worker     else
1173*5e7646d2SAndroid Build Coastguard Worker     {
1174*5e7646d2SAndroid Build Coastguard Worker      /*
1175*5e7646d2SAndroid Build Coastguard Worker       * Validate the hostname or IPv4 address first...
1176*5e7646d2SAndroid Build Coastguard Worker       */
1177*5e7646d2SAndroid Build Coastguard Worker 
1178*5e7646d2SAndroid Build Coastguard Worker       for (ptr = (char *)uri; *ptr; ptr ++)
1179*5e7646d2SAndroid Build Coastguard Worker         if (strchr(":?/", *ptr))
1180*5e7646d2SAndroid Build Coastguard Worker 	  break;
1181*5e7646d2SAndroid Build Coastguard Worker         else if (!strchr("abcdefghijklmnopqrstuvwxyz"	/* unreserved */
1182*5e7646d2SAndroid Build Coastguard Worker 			 "ABCDEFGHIJKLMNOPQRSTUVWXYZ"	/* unreserved */
1183*5e7646d2SAndroid Build Coastguard Worker 			 "0123456789"			/* unreserved */
1184*5e7646d2SAndroid Build Coastguard Worker 	        	 "-._~"				/* unreserved */
1185*5e7646d2SAndroid Build Coastguard Worker 			 "%"				/* pct-encoded */
1186*5e7646d2SAndroid Build Coastguard Worker 			 "!$&'()*+,;="			/* sub-delims */
1187*5e7646d2SAndroid Build Coastguard Worker 			 "\\", *ptr))			/* SMB domain */
1188*5e7646d2SAndroid Build Coastguard Worker 	{
1189*5e7646d2SAndroid Build Coastguard Worker 	  *host = '\0';
1190*5e7646d2SAndroid Build Coastguard Worker 	  return (HTTP_URI_STATUS_BAD_HOSTNAME);
1191*5e7646d2SAndroid Build Coastguard Worker 	}
1192*5e7646d2SAndroid Build Coastguard Worker 
1193*5e7646d2SAndroid Build Coastguard Worker      /*
1194*5e7646d2SAndroid Build Coastguard Worker       * Then copy the hostname or IPv4 address to the buffer...
1195*5e7646d2SAndroid Build Coastguard Worker       */
1196*5e7646d2SAndroid Build Coastguard Worker 
1197*5e7646d2SAndroid Build Coastguard Worker       uri = http_copy_decode(host, uri, hostlen, ":?/",
1198*5e7646d2SAndroid Build Coastguard Worker                              decoding & HTTP_URI_CODING_HOSTNAME);
1199*5e7646d2SAndroid Build Coastguard Worker 
1200*5e7646d2SAndroid Build Coastguard Worker       if (!uri)
1201*5e7646d2SAndroid Build Coastguard Worker       {
1202*5e7646d2SAndroid Build Coastguard Worker         *host = '\0';
1203*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_HOSTNAME);
1204*5e7646d2SAndroid Build Coastguard Worker       }
1205*5e7646d2SAndroid Build Coastguard Worker     }
1206*5e7646d2SAndroid Build Coastguard Worker 
1207*5e7646d2SAndroid Build Coastguard Worker    /*
1208*5e7646d2SAndroid Build Coastguard Worker     * Validate hostname for file scheme - only empty and localhost are
1209*5e7646d2SAndroid Build Coastguard Worker     * acceptable.
1210*5e7646d2SAndroid Build Coastguard Worker     */
1211*5e7646d2SAndroid Build Coastguard Worker 
1212*5e7646d2SAndroid Build Coastguard Worker     if (!strcmp(scheme, "file") && strcmp(host, "localhost") && host[0])
1213*5e7646d2SAndroid Build Coastguard Worker     {
1214*5e7646d2SAndroid Build Coastguard Worker       *host = '\0';
1215*5e7646d2SAndroid Build Coastguard Worker       return (HTTP_URI_STATUS_BAD_HOSTNAME);
1216*5e7646d2SAndroid Build Coastguard Worker     }
1217*5e7646d2SAndroid Build Coastguard Worker 
1218*5e7646d2SAndroid Build Coastguard Worker    /*
1219*5e7646d2SAndroid Build Coastguard Worker     * See if we have a port number...
1220*5e7646d2SAndroid Build Coastguard Worker     */
1221*5e7646d2SAndroid Build Coastguard Worker 
1222*5e7646d2SAndroid Build Coastguard Worker     if (*uri == ':')
1223*5e7646d2SAndroid Build Coastguard Worker     {
1224*5e7646d2SAndroid Build Coastguard Worker      /*
1225*5e7646d2SAndroid Build Coastguard Worker       * Yes, collect the port number...
1226*5e7646d2SAndroid Build Coastguard Worker       */
1227*5e7646d2SAndroid Build Coastguard Worker 
1228*5e7646d2SAndroid Build Coastguard Worker       if (!isdigit(uri[1] & 255))
1229*5e7646d2SAndroid Build Coastguard Worker       {
1230*5e7646d2SAndroid Build Coastguard Worker         *port = 0;
1231*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_PORT);
1232*5e7646d2SAndroid Build Coastguard Worker       }
1233*5e7646d2SAndroid Build Coastguard Worker 
1234*5e7646d2SAndroid Build Coastguard Worker       *port = (int)strtol(uri + 1, (char **)&uri, 10);
1235*5e7646d2SAndroid Build Coastguard Worker 
1236*5e7646d2SAndroid Build Coastguard Worker       if (*port <= 0 || *port > 65535)
1237*5e7646d2SAndroid Build Coastguard Worker       {
1238*5e7646d2SAndroid Build Coastguard Worker         *port = 0;
1239*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_PORT);
1240*5e7646d2SAndroid Build Coastguard Worker       }
1241*5e7646d2SAndroid Build Coastguard Worker 
1242*5e7646d2SAndroid Build Coastguard Worker       if (*uri != '/' && *uri)
1243*5e7646d2SAndroid Build Coastguard Worker       {
1244*5e7646d2SAndroid Build Coastguard Worker         *port = 0;
1245*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_URI_STATUS_BAD_PORT);
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  /*
1251*5e7646d2SAndroid Build Coastguard Worker   * The remaining portion is the resource string...
1252*5e7646d2SAndroid Build Coastguard Worker   */
1253*5e7646d2SAndroid Build Coastguard Worker 
1254*5e7646d2SAndroid Build Coastguard Worker   if (*uri == '?' || !*uri)
1255*5e7646d2SAndroid Build Coastguard Worker   {
1256*5e7646d2SAndroid Build Coastguard Worker    /*
1257*5e7646d2SAndroid Build Coastguard Worker     * Hostname but no path...
1258*5e7646d2SAndroid Build Coastguard Worker     */
1259*5e7646d2SAndroid Build Coastguard Worker 
1260*5e7646d2SAndroid Build Coastguard Worker     status    = HTTP_URI_STATUS_MISSING_RESOURCE;
1261*5e7646d2SAndroid Build Coastguard Worker     *resource = '/';
1262*5e7646d2SAndroid Build Coastguard Worker 
1263*5e7646d2SAndroid Build Coastguard Worker    /*
1264*5e7646d2SAndroid Build Coastguard Worker     * Copy any query string...
1265*5e7646d2SAndroid Build Coastguard Worker     */
1266*5e7646d2SAndroid Build Coastguard Worker 
1267*5e7646d2SAndroid Build Coastguard Worker     if (*uri == '?')
1268*5e7646d2SAndroid Build Coastguard Worker       uri = http_copy_decode(resource + 1, uri, resourcelen - 1, NULL,
1269*5e7646d2SAndroid Build Coastguard Worker                              decoding & HTTP_URI_CODING_QUERY);
1270*5e7646d2SAndroid Build Coastguard Worker     else
1271*5e7646d2SAndroid Build Coastguard Worker       resource[1] = '\0';
1272*5e7646d2SAndroid Build Coastguard Worker   }
1273*5e7646d2SAndroid Build Coastguard Worker   else
1274*5e7646d2SAndroid Build Coastguard Worker   {
1275*5e7646d2SAndroid Build Coastguard Worker     uri = http_copy_decode(resource, uri, resourcelen, "?",
1276*5e7646d2SAndroid Build Coastguard Worker                            decoding & HTTP_URI_CODING_RESOURCE);
1277*5e7646d2SAndroid Build Coastguard Worker 
1278*5e7646d2SAndroid Build Coastguard Worker     if (uri && *uri == '?')
1279*5e7646d2SAndroid Build Coastguard Worker     {
1280*5e7646d2SAndroid Build Coastguard Worker      /*
1281*5e7646d2SAndroid Build Coastguard Worker       * Concatenate any query string...
1282*5e7646d2SAndroid Build Coastguard Worker       */
1283*5e7646d2SAndroid Build Coastguard Worker 
1284*5e7646d2SAndroid Build Coastguard Worker       char *resptr = resource + strlen(resource);
1285*5e7646d2SAndroid Build Coastguard Worker 
1286*5e7646d2SAndroid Build Coastguard Worker       uri = http_copy_decode(resptr, uri,
1287*5e7646d2SAndroid Build Coastguard Worker                              resourcelen - (int)(resptr - resource), NULL,
1288*5e7646d2SAndroid Build Coastguard Worker                              decoding & HTTP_URI_CODING_QUERY);
1289*5e7646d2SAndroid Build Coastguard Worker     }
1290*5e7646d2SAndroid Build Coastguard Worker   }
1291*5e7646d2SAndroid Build Coastguard Worker 
1292*5e7646d2SAndroid Build Coastguard Worker   if (!uri)
1293*5e7646d2SAndroid Build Coastguard Worker   {
1294*5e7646d2SAndroid Build Coastguard Worker     *resource = '\0';
1295*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_URI_STATUS_BAD_RESOURCE);
1296*5e7646d2SAndroid Build Coastguard Worker   }
1297*5e7646d2SAndroid Build Coastguard Worker 
1298*5e7646d2SAndroid Build Coastguard Worker  /*
1299*5e7646d2SAndroid Build Coastguard Worker   * Return the URI separation status...
1300*5e7646d2SAndroid Build Coastguard Worker   */
1301*5e7646d2SAndroid Build Coastguard Worker 
1302*5e7646d2SAndroid Build Coastguard Worker   return (status);
1303*5e7646d2SAndroid Build Coastguard Worker }
1304*5e7646d2SAndroid Build Coastguard Worker 
1305*5e7646d2SAndroid Build Coastguard Worker 
1306*5e7646d2SAndroid Build Coastguard Worker /*
1307*5e7646d2SAndroid Build Coastguard Worker  * '_httpSetDigestAuthString()' - Calculate a Digest authentication response
1308*5e7646d2SAndroid Build Coastguard Worker  *                                using the appropriate RFC 2068/2617/7616
1309*5e7646d2SAndroid Build Coastguard Worker  *                                algorithm.
1310*5e7646d2SAndroid Build Coastguard Worker  */
1311*5e7646d2SAndroid Build Coastguard Worker 
1312*5e7646d2SAndroid Build Coastguard Worker int					/* O - 1 on success, 0 on failure */
_httpSetDigestAuthString(http_t * http,const char * nonce,const char * method,const char * resource)1313*5e7646d2SAndroid Build Coastguard Worker _httpSetDigestAuthString(
1314*5e7646d2SAndroid Build Coastguard Worker     http_t     *http,			/* I - HTTP connection */
1315*5e7646d2SAndroid Build Coastguard Worker     const char *nonce,			/* I - Nonce value */
1316*5e7646d2SAndroid Build Coastguard Worker     const char *method,			/* I - HTTP method */
1317*5e7646d2SAndroid Build Coastguard Worker     const char *resource)		/* I - HTTP resource path */
1318*5e7646d2SAndroid Build Coastguard Worker {
1319*5e7646d2SAndroid Build Coastguard Worker   char		kd[65],			/* Final MD5/SHA-256 digest */
1320*5e7646d2SAndroid Build Coastguard Worker 		ha1[65],		/* Hash of username:realm:password */
1321*5e7646d2SAndroid Build Coastguard Worker 		ha2[65],		/* Hash of method:request-uri */
1322*5e7646d2SAndroid Build Coastguard Worker 		username[HTTP_MAX_VALUE],
1323*5e7646d2SAndroid Build Coastguard Worker 					/* username:password */
1324*5e7646d2SAndroid Build Coastguard Worker 		*password,		/* Pointer to password */
1325*5e7646d2SAndroid Build Coastguard Worker 		temp[1024],		/* Temporary string */
1326*5e7646d2SAndroid Build Coastguard Worker 		digest[1024];		/* Digest auth data */
1327*5e7646d2SAndroid Build Coastguard Worker   unsigned char	hash[32];		/* Hash buffer */
1328*5e7646d2SAndroid Build Coastguard Worker   size_t	hashsize;		/* Size of hash */
1329*5e7646d2SAndroid Build Coastguard Worker   _cups_globals_t *cg = _cupsGlobals();	/* Per-thread globals */
1330*5e7646d2SAndroid Build Coastguard Worker 
1331*5e7646d2SAndroid Build Coastguard Worker 
1332*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2_httpSetDigestAuthString(http=%p, nonce=\"%s\", method=\"%s\", resource=\"%s\")", (void *)http, nonce, method, resource));
1333*5e7646d2SAndroid Build Coastguard Worker 
1334*5e7646d2SAndroid Build Coastguard Worker   if (nonce && *nonce && strcmp(nonce, http->nonce))
1335*5e7646d2SAndroid Build Coastguard Worker   {
1336*5e7646d2SAndroid Build Coastguard Worker     strlcpy(http->nonce, nonce, sizeof(http->nonce));
1337*5e7646d2SAndroid Build Coastguard Worker 
1338*5e7646d2SAndroid Build Coastguard Worker     if (nonce == http->nextnonce)
1339*5e7646d2SAndroid Build Coastguard Worker       http->nextnonce[0] = '\0';
1340*5e7646d2SAndroid Build Coastguard Worker 
1341*5e7646d2SAndroid Build Coastguard Worker     http->nonce_count = 1;
1342*5e7646d2SAndroid Build Coastguard Worker   }
1343*5e7646d2SAndroid Build Coastguard Worker   else
1344*5e7646d2SAndroid Build Coastguard Worker     http->nonce_count ++;
1345*5e7646d2SAndroid Build Coastguard Worker 
1346*5e7646d2SAndroid Build Coastguard Worker   strlcpy(username, http->userpass, sizeof(username));
1347*5e7646d2SAndroid Build Coastguard Worker   if ((password = strchr(username, ':')) != NULL)
1348*5e7646d2SAndroid Build Coastguard Worker     *password++ = '\0';
1349*5e7646d2SAndroid Build Coastguard Worker   else
1350*5e7646d2SAndroid Build Coastguard Worker     return (0);
1351*5e7646d2SAndroid Build Coastguard Worker 
1352*5e7646d2SAndroid Build Coastguard Worker   if (http->algorithm[0])
1353*5e7646d2SAndroid Build Coastguard Worker   {
1354*5e7646d2SAndroid Build Coastguard Worker    /*
1355*5e7646d2SAndroid Build Coastguard Worker     * Follow RFC 2617/7616...
1356*5e7646d2SAndroid Build Coastguard Worker     */
1357*5e7646d2SAndroid Build Coastguard Worker 
1358*5e7646d2SAndroid Build Coastguard Worker     int		i;			/* Looping var */
1359*5e7646d2SAndroid Build Coastguard Worker     char	cnonce[65];		/* cnonce value */
1360*5e7646d2SAndroid Build Coastguard Worker     const char	*hashalg;		/* Hashing algorithm */
1361*5e7646d2SAndroid Build Coastguard Worker 
1362*5e7646d2SAndroid Build Coastguard Worker     for (i = 0; i < 64; i ++)
1363*5e7646d2SAndroid Build Coastguard Worker       cnonce[i] = "0123456789ABCDEF"[CUPS_RAND() & 15];
1364*5e7646d2SAndroid Build Coastguard Worker     cnonce[64] = '\0';
1365*5e7646d2SAndroid Build Coastguard Worker 
1366*5e7646d2SAndroid Build Coastguard Worker     if (!_cups_strcasecmp(http->algorithm, "MD5"))
1367*5e7646d2SAndroid Build Coastguard Worker     {
1368*5e7646d2SAndroid Build Coastguard Worker      /*
1369*5e7646d2SAndroid Build Coastguard Worker       * RFC 2617 Digest with MD5
1370*5e7646d2SAndroid Build Coastguard Worker       */
1371*5e7646d2SAndroid Build Coastguard Worker 
1372*5e7646d2SAndroid Build Coastguard Worker       if (cg->digestoptions == _CUPS_DIGESTOPTIONS_DENYMD5)
1373*5e7646d2SAndroid Build Coastguard Worker       {
1374*5e7646d2SAndroid Build Coastguard Worker 	DEBUG_puts("3_httpSetDigestAuthString: MD5 Digest is disabled.");
1375*5e7646d2SAndroid Build Coastguard Worker 	return (0);
1376*5e7646d2SAndroid Build Coastguard Worker       }
1377*5e7646d2SAndroid Build Coastguard Worker 
1378*5e7646d2SAndroid Build Coastguard Worker       hashalg = "md5";
1379*5e7646d2SAndroid Build Coastguard Worker     }
1380*5e7646d2SAndroid Build Coastguard Worker     else if (!_cups_strcasecmp(http->algorithm, "SHA-256"))
1381*5e7646d2SAndroid Build Coastguard Worker     {
1382*5e7646d2SAndroid Build Coastguard Worker      /*
1383*5e7646d2SAndroid Build Coastguard Worker       * RFC 7616 Digest with SHA-256
1384*5e7646d2SAndroid Build Coastguard Worker       */
1385*5e7646d2SAndroid Build Coastguard Worker 
1386*5e7646d2SAndroid Build Coastguard Worker       hashalg = "sha2-256";
1387*5e7646d2SAndroid Build Coastguard Worker     }
1388*5e7646d2SAndroid Build Coastguard Worker     else
1389*5e7646d2SAndroid Build Coastguard Worker     {
1390*5e7646d2SAndroid Build Coastguard Worker      /*
1391*5e7646d2SAndroid Build Coastguard Worker       * Some other algorithm we don't support, skip this one...
1392*5e7646d2SAndroid Build Coastguard Worker       */
1393*5e7646d2SAndroid Build Coastguard Worker 
1394*5e7646d2SAndroid Build Coastguard Worker       return (0);
1395*5e7646d2SAndroid Build Coastguard Worker     }
1396*5e7646d2SAndroid Build Coastguard Worker 
1397*5e7646d2SAndroid Build Coastguard Worker    /*
1398*5e7646d2SAndroid Build Coastguard Worker     * Calculate digest value...
1399*5e7646d2SAndroid Build Coastguard Worker     */
1400*5e7646d2SAndroid Build Coastguard Worker 
1401*5e7646d2SAndroid Build Coastguard Worker     /* H(A1) = H(username:realm:password) */
1402*5e7646d2SAndroid Build Coastguard Worker     snprintf(temp, sizeof(temp), "%s:%s:%s", username, http->realm, password);
1403*5e7646d2SAndroid Build Coastguard Worker     hashsize = (size_t)cupsHashData(hashalg, (unsigned char *)temp, strlen(temp), hash, sizeof(hash));
1404*5e7646d2SAndroid Build Coastguard Worker     cupsHashString(hash, hashsize, ha1, sizeof(ha1));
1405*5e7646d2SAndroid Build Coastguard Worker 
1406*5e7646d2SAndroid Build Coastguard Worker     /* H(A2) = H(method:uri) */
1407*5e7646d2SAndroid Build Coastguard Worker     snprintf(temp, sizeof(temp), "%s:%s", method, resource);
1408*5e7646d2SAndroid Build Coastguard Worker     hashsize = (size_t)cupsHashData(hashalg, (unsigned char *)temp, strlen(temp), hash, sizeof(hash));
1409*5e7646d2SAndroid Build Coastguard Worker     cupsHashString(hash, hashsize, ha2, sizeof(ha2));
1410*5e7646d2SAndroid Build Coastguard Worker 
1411*5e7646d2SAndroid Build Coastguard Worker     /* KD = H(H(A1):nonce:nc:cnonce:qop:H(A2)) */
1412*5e7646d2SAndroid Build Coastguard Worker     snprintf(temp, sizeof(temp), "%s:%s:%08x:%s:%s:%s", ha1, http->nonce, http->nonce_count, cnonce, "auth", ha2);
1413*5e7646d2SAndroid Build Coastguard Worker     hashsize = (size_t)cupsHashData(hashalg, (unsigned char *)temp, strlen(temp), hash, sizeof(hash));
1414*5e7646d2SAndroid Build Coastguard Worker     cupsHashString(hash, hashsize, kd, sizeof(kd));
1415*5e7646d2SAndroid Build Coastguard Worker 
1416*5e7646d2SAndroid Build Coastguard Worker    /*
1417*5e7646d2SAndroid Build Coastguard Worker     * Pass the RFC 2617/7616 WWW-Authenticate header...
1418*5e7646d2SAndroid Build Coastguard Worker     */
1419*5e7646d2SAndroid Build Coastguard Worker 
1420*5e7646d2SAndroid Build Coastguard Worker     if (http->opaque[0])
1421*5e7646d2SAndroid Build Coastguard Worker       snprintf(digest, sizeof(digest), "username=\"%s\", realm=\"%s\", nonce=\"%s\", algorithm=%s, qop=auth, opaque=\"%s\", cnonce=\"%s\", nc=%08x, uri=\"%s\", response=\"%s\"", cupsUser(), http->realm, http->nonce, http->algorithm, http->opaque, cnonce, http->nonce_count, resource, kd);
1422*5e7646d2SAndroid Build Coastguard Worker     else
1423*5e7646d2SAndroid Build Coastguard Worker       snprintf(digest, sizeof(digest), "username=\"%s\", realm=\"%s\", nonce=\"%s\", algorithm=%s, qop=auth, cnonce=\"%s\", nc=%08x, uri=\"%s\", response=\"%s\"", username, http->realm, http->nonce, http->algorithm, cnonce, http->nonce_count, resource, kd);
1424*5e7646d2SAndroid Build Coastguard Worker   }
1425*5e7646d2SAndroid Build Coastguard Worker   else
1426*5e7646d2SAndroid Build Coastguard Worker   {
1427*5e7646d2SAndroid Build Coastguard Worker    /*
1428*5e7646d2SAndroid Build Coastguard Worker     * Use old RFC 2069 Digest method...
1429*5e7646d2SAndroid Build Coastguard Worker     */
1430*5e7646d2SAndroid Build Coastguard Worker 
1431*5e7646d2SAndroid Build Coastguard Worker     /* H(A1) = H(username:realm:password) */
1432*5e7646d2SAndroid Build Coastguard Worker     snprintf(temp, sizeof(temp), "%s:%s:%s", username, http->realm, password);
1433*5e7646d2SAndroid Build Coastguard Worker     hashsize = (size_t)cupsHashData("md5", (unsigned char *)temp, strlen(temp), hash, sizeof(hash));
1434*5e7646d2SAndroid Build Coastguard Worker     cupsHashString(hash, hashsize, ha1, sizeof(ha1));
1435*5e7646d2SAndroid Build Coastguard Worker 
1436*5e7646d2SAndroid Build Coastguard Worker     /* H(A2) = H(method:uri) */
1437*5e7646d2SAndroid Build Coastguard Worker     snprintf(temp, sizeof(temp), "%s:%s", method, resource);
1438*5e7646d2SAndroid Build Coastguard Worker     hashsize = (size_t)cupsHashData("md5", (unsigned char *)temp, strlen(temp), hash, sizeof(hash));
1439*5e7646d2SAndroid Build Coastguard Worker     cupsHashString(hash, hashsize, ha2, sizeof(ha2));
1440*5e7646d2SAndroid Build Coastguard Worker 
1441*5e7646d2SAndroid Build Coastguard Worker     /* KD = H(H(A1):nonce:H(A2)) */
1442*5e7646d2SAndroid Build Coastguard Worker     snprintf(temp, sizeof(temp), "%s:%s:%s", ha1, http->nonce, ha2);
1443*5e7646d2SAndroid Build Coastguard Worker     hashsize = (size_t)cupsHashData("md5", (unsigned char *)temp, strlen(temp), hash, sizeof(hash));
1444*5e7646d2SAndroid Build Coastguard Worker     cupsHashString(hash, hashsize, kd, sizeof(kd));
1445*5e7646d2SAndroid Build Coastguard Worker 
1446*5e7646d2SAndroid Build Coastguard Worker    /*
1447*5e7646d2SAndroid Build Coastguard Worker     * Pass the old RFC 2069 WWW-Authenticate header...
1448*5e7646d2SAndroid Build Coastguard Worker     */
1449*5e7646d2SAndroid Build Coastguard Worker 
1450*5e7646d2SAndroid Build Coastguard Worker     snprintf(digest, sizeof(digest), "username=\"%s\", realm=\"%s\", nonce=\"%s\", uri=\"%s\", response=\"%s\"", username, http->realm, http->nonce, resource, kd);
1451*5e7646d2SAndroid Build Coastguard Worker   }
1452*5e7646d2SAndroid Build Coastguard Worker 
1453*5e7646d2SAndroid Build Coastguard Worker   httpSetAuthString(http, "Digest", digest);
1454*5e7646d2SAndroid Build Coastguard Worker 
1455*5e7646d2SAndroid Build Coastguard Worker   return (1);
1456*5e7646d2SAndroid Build Coastguard Worker }
1457*5e7646d2SAndroid Build Coastguard Worker 
1458*5e7646d2SAndroid Build Coastguard Worker 
1459*5e7646d2SAndroid Build Coastguard Worker /*
1460*5e7646d2SAndroid Build Coastguard Worker  * 'httpStateString()' - Return the string describing a HTTP state value.
1461*5e7646d2SAndroid Build Coastguard Worker  *
1462*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 2.0/OS 10.10@
1463*5e7646d2SAndroid Build Coastguard Worker  */
1464*5e7646d2SAndroid Build Coastguard Worker 
1465*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - State string */
httpStateString(http_state_t state)1466*5e7646d2SAndroid Build Coastguard Worker httpStateString(http_state_t state)	/* I - HTTP state value */
1467*5e7646d2SAndroid Build Coastguard Worker {
1468*5e7646d2SAndroid Build Coastguard Worker   if (state < HTTP_STATE_ERROR || state > HTTP_STATE_UNKNOWN_VERSION)
1469*5e7646d2SAndroid Build Coastguard Worker     return ("HTTP_STATE_???");
1470*5e7646d2SAndroid Build Coastguard Worker   else
1471*5e7646d2SAndroid Build Coastguard Worker     return (http_states[state - HTTP_STATE_ERROR]);
1472*5e7646d2SAndroid Build Coastguard Worker }
1473*5e7646d2SAndroid Build Coastguard Worker 
1474*5e7646d2SAndroid Build Coastguard Worker 
1475*5e7646d2SAndroid Build Coastguard Worker /*
1476*5e7646d2SAndroid Build Coastguard Worker  * '_httpStatus()' - Return the localized string describing a HTTP status code.
1477*5e7646d2SAndroid Build Coastguard Worker  *
1478*5e7646d2SAndroid Build Coastguard Worker  * The returned string is localized using the passed message catalog.
1479*5e7646d2SAndroid Build Coastguard Worker  */
1480*5e7646d2SAndroid Build Coastguard Worker 
1481*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Localized status string */
_httpStatus(cups_lang_t * lang,http_status_t status)1482*5e7646d2SAndroid Build Coastguard Worker _httpStatus(cups_lang_t   *lang,	/* I - Language */
1483*5e7646d2SAndroid Build Coastguard Worker             http_status_t status)	/* I - HTTP status code */
1484*5e7646d2SAndroid Build Coastguard Worker {
1485*5e7646d2SAndroid Build Coastguard Worker   const char	*s;			/* Status string */
1486*5e7646d2SAndroid Build Coastguard Worker 
1487*5e7646d2SAndroid Build Coastguard Worker 
1488*5e7646d2SAndroid Build Coastguard Worker   switch (status)
1489*5e7646d2SAndroid Build Coastguard Worker   {
1490*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_ERROR :
1491*5e7646d2SAndroid Build Coastguard Worker         s = strerror(errno);
1492*5e7646d2SAndroid Build Coastguard Worker         break;
1493*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_CONTINUE :
1494*5e7646d2SAndroid Build Coastguard Worker         s = _("Continue");
1495*5e7646d2SAndroid Build Coastguard Worker 	break;
1496*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_SWITCHING_PROTOCOLS :
1497*5e7646d2SAndroid Build Coastguard Worker         s = _("Switching Protocols");
1498*5e7646d2SAndroid Build Coastguard Worker 	break;
1499*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_OK :
1500*5e7646d2SAndroid Build Coastguard Worker         s = _("OK");
1501*5e7646d2SAndroid Build Coastguard Worker 	break;
1502*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_CREATED :
1503*5e7646d2SAndroid Build Coastguard Worker         s = _("Created");
1504*5e7646d2SAndroid Build Coastguard Worker 	break;
1505*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_ACCEPTED :
1506*5e7646d2SAndroid Build Coastguard Worker         s = _("Accepted");
1507*5e7646d2SAndroid Build Coastguard Worker 	break;
1508*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_NO_CONTENT :
1509*5e7646d2SAndroid Build Coastguard Worker         s = _("No Content");
1510*5e7646d2SAndroid Build Coastguard Worker 	break;
1511*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_MOVED_PERMANENTLY :
1512*5e7646d2SAndroid Build Coastguard Worker         s = _("Moved Permanently");
1513*5e7646d2SAndroid Build Coastguard Worker 	break;
1514*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_FOUND :
1515*5e7646d2SAndroid Build Coastguard Worker         s = _("Found");
1516*5e7646d2SAndroid Build Coastguard Worker 	break;
1517*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_SEE_OTHER :
1518*5e7646d2SAndroid Build Coastguard Worker         s = _("See Other");
1519*5e7646d2SAndroid Build Coastguard Worker 	break;
1520*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_NOT_MODIFIED :
1521*5e7646d2SAndroid Build Coastguard Worker         s = _("Not Modified");
1522*5e7646d2SAndroid Build Coastguard Worker 	break;
1523*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_BAD_REQUEST :
1524*5e7646d2SAndroid Build Coastguard Worker         s = _("Bad Request");
1525*5e7646d2SAndroid Build Coastguard Worker 	break;
1526*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_UNAUTHORIZED :
1527*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_CUPS_AUTHORIZATION_CANCELED :
1528*5e7646d2SAndroid Build Coastguard Worker         s = _("Unauthorized");
1529*5e7646d2SAndroid Build Coastguard Worker 	break;
1530*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_FORBIDDEN :
1531*5e7646d2SAndroid Build Coastguard Worker         s = _("Forbidden");
1532*5e7646d2SAndroid Build Coastguard Worker 	break;
1533*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_NOT_FOUND :
1534*5e7646d2SAndroid Build Coastguard Worker         s = _("Not Found");
1535*5e7646d2SAndroid Build Coastguard Worker 	break;
1536*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_REQUEST_TOO_LARGE :
1537*5e7646d2SAndroid Build Coastguard Worker         s = _("Request Entity Too Large");
1538*5e7646d2SAndroid Build Coastguard Worker 	break;
1539*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_URI_TOO_LONG :
1540*5e7646d2SAndroid Build Coastguard Worker         s = _("URI Too Long");
1541*5e7646d2SAndroid Build Coastguard Worker 	break;
1542*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_UPGRADE_REQUIRED :
1543*5e7646d2SAndroid Build Coastguard Worker         s = _("Upgrade Required");
1544*5e7646d2SAndroid Build Coastguard Worker 	break;
1545*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_NOT_IMPLEMENTED :
1546*5e7646d2SAndroid Build Coastguard Worker         s = _("Not Implemented");
1547*5e7646d2SAndroid Build Coastguard Worker 	break;
1548*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_NOT_SUPPORTED :
1549*5e7646d2SAndroid Build Coastguard Worker         s = _("Not Supported");
1550*5e7646d2SAndroid Build Coastguard Worker 	break;
1551*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_EXPECTATION_FAILED :
1552*5e7646d2SAndroid Build Coastguard Worker         s = _("Expectation Failed");
1553*5e7646d2SAndroid Build Coastguard Worker 	break;
1554*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_SERVICE_UNAVAILABLE :
1555*5e7646d2SAndroid Build Coastguard Worker         s = _("Service Unavailable");
1556*5e7646d2SAndroid Build Coastguard Worker 	break;
1557*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_SERVER_ERROR :
1558*5e7646d2SAndroid Build Coastguard Worker         s = _("Internal Server Error");
1559*5e7646d2SAndroid Build Coastguard Worker 	break;
1560*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_CUPS_PKI_ERROR :
1561*5e7646d2SAndroid Build Coastguard Worker         s = _("SSL/TLS Negotiation Error");
1562*5e7646d2SAndroid Build Coastguard Worker 	break;
1563*5e7646d2SAndroid Build Coastguard Worker     case HTTP_STATUS_CUPS_WEBIF_DISABLED :
1564*5e7646d2SAndroid Build Coastguard Worker         s = _("Web Interface is Disabled");
1565*5e7646d2SAndroid Build Coastguard Worker 	break;
1566*5e7646d2SAndroid Build Coastguard Worker 
1567*5e7646d2SAndroid Build Coastguard Worker     default :
1568*5e7646d2SAndroid Build Coastguard Worker         s = _("Unknown");
1569*5e7646d2SAndroid Build Coastguard Worker 	break;
1570*5e7646d2SAndroid Build Coastguard Worker   }
1571*5e7646d2SAndroid Build Coastguard Worker 
1572*5e7646d2SAndroid Build Coastguard Worker   return (_cupsLangString(lang, s));
1573*5e7646d2SAndroid Build Coastguard Worker }
1574*5e7646d2SAndroid Build Coastguard Worker 
1575*5e7646d2SAndroid Build Coastguard Worker 
1576*5e7646d2SAndroid Build Coastguard Worker /*
1577*5e7646d2SAndroid Build Coastguard Worker  * 'httpStatus()' - Return a short string describing a HTTP status code.
1578*5e7646d2SAndroid Build Coastguard Worker  *
1579*5e7646d2SAndroid Build Coastguard Worker  * The returned string is localized to the current POSIX locale and is based
1580*5e7646d2SAndroid Build Coastguard Worker  * on the status strings defined in RFC 7231.
1581*5e7646d2SAndroid Build Coastguard Worker  */
1582*5e7646d2SAndroid Build Coastguard Worker 
1583*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Localized status string */
httpStatus(http_status_t status)1584*5e7646d2SAndroid Build Coastguard Worker httpStatus(http_status_t status)	/* I - HTTP status code */
1585*5e7646d2SAndroid Build Coastguard Worker {
1586*5e7646d2SAndroid Build Coastguard Worker   _cups_globals_t *cg = _cupsGlobals();	/* Global data */
1587*5e7646d2SAndroid Build Coastguard Worker 
1588*5e7646d2SAndroid Build Coastguard Worker 
1589*5e7646d2SAndroid Build Coastguard Worker   if (!cg->lang_default)
1590*5e7646d2SAndroid Build Coastguard Worker     cg->lang_default = cupsLangDefault();
1591*5e7646d2SAndroid Build Coastguard Worker 
1592*5e7646d2SAndroid Build Coastguard Worker   return (_httpStatus(cg->lang_default, status));
1593*5e7646d2SAndroid Build Coastguard Worker }
1594*5e7646d2SAndroid Build Coastguard Worker 
1595*5e7646d2SAndroid Build Coastguard Worker /*
1596*5e7646d2SAndroid Build Coastguard Worker  * 'httpURIStatusString()' - Return a string describing a URI status code.
1597*5e7646d2SAndroid Build Coastguard Worker  *
1598*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 2.0/OS 10.10@
1599*5e7646d2SAndroid Build Coastguard Worker  */
1600*5e7646d2SAndroid Build Coastguard Worker 
1601*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Localized status string */
httpURIStatusString(http_uri_status_t status)1602*5e7646d2SAndroid Build Coastguard Worker httpURIStatusString(
1603*5e7646d2SAndroid Build Coastguard Worker     http_uri_status_t status)		/* I - URI status code */
1604*5e7646d2SAndroid Build Coastguard Worker {
1605*5e7646d2SAndroid Build Coastguard Worker   const char	*s;			/* Status string */
1606*5e7646d2SAndroid Build Coastguard Worker   _cups_globals_t *cg = _cupsGlobals();	/* Global data */
1607*5e7646d2SAndroid Build Coastguard Worker 
1608*5e7646d2SAndroid Build Coastguard Worker 
1609*5e7646d2SAndroid Build Coastguard Worker   if (!cg->lang_default)
1610*5e7646d2SAndroid Build Coastguard Worker     cg->lang_default = cupsLangDefault();
1611*5e7646d2SAndroid Build Coastguard Worker 
1612*5e7646d2SAndroid Build Coastguard Worker   switch (status)
1613*5e7646d2SAndroid Build Coastguard Worker   {
1614*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_OVERFLOW :
1615*5e7646d2SAndroid Build Coastguard Worker 	s = _("URI too large");
1616*5e7646d2SAndroid Build Coastguard Worker 	break;
1617*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_ARGUMENTS :
1618*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad arguments to function");
1619*5e7646d2SAndroid Build Coastguard Worker 	break;
1620*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_RESOURCE :
1621*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad resource in URI");
1622*5e7646d2SAndroid Build Coastguard Worker 	break;
1623*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_PORT :
1624*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad port number in URI");
1625*5e7646d2SAndroid Build Coastguard Worker 	break;
1626*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_HOSTNAME :
1627*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad hostname/address in URI");
1628*5e7646d2SAndroid Build Coastguard Worker 	break;
1629*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_USERNAME :
1630*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad username in URI");
1631*5e7646d2SAndroid Build Coastguard Worker 	break;
1632*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_SCHEME :
1633*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad scheme in URI");
1634*5e7646d2SAndroid Build Coastguard Worker 	break;
1635*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_BAD_URI :
1636*5e7646d2SAndroid Build Coastguard Worker 	s = _("Bad/empty URI");
1637*5e7646d2SAndroid Build Coastguard Worker 	break;
1638*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_OK :
1639*5e7646d2SAndroid Build Coastguard Worker 	s = _("OK");
1640*5e7646d2SAndroid Build Coastguard Worker 	break;
1641*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_MISSING_SCHEME :
1642*5e7646d2SAndroid Build Coastguard Worker 	s = _("Missing scheme in URI");
1643*5e7646d2SAndroid Build Coastguard Worker 	break;
1644*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_UNKNOWN_SCHEME :
1645*5e7646d2SAndroid Build Coastguard Worker 	s = _("Unknown scheme in URI");
1646*5e7646d2SAndroid Build Coastguard Worker 	break;
1647*5e7646d2SAndroid Build Coastguard Worker     case HTTP_URI_STATUS_MISSING_RESOURCE :
1648*5e7646d2SAndroid Build Coastguard Worker 	s = _("Missing resource in URI");
1649*5e7646d2SAndroid Build Coastguard Worker 	break;
1650*5e7646d2SAndroid Build Coastguard Worker 
1651*5e7646d2SAndroid Build Coastguard Worker     default:
1652*5e7646d2SAndroid Build Coastguard Worker         s = _("Unknown");
1653*5e7646d2SAndroid Build Coastguard Worker 	break;
1654*5e7646d2SAndroid Build Coastguard Worker   }
1655*5e7646d2SAndroid Build Coastguard Worker 
1656*5e7646d2SAndroid Build Coastguard Worker   return (_cupsLangString(cg->lang_default, s));
1657*5e7646d2SAndroid Build Coastguard Worker }
1658*5e7646d2SAndroid Build Coastguard Worker 
1659*5e7646d2SAndroid Build Coastguard Worker 
1660*5e7646d2SAndroid Build Coastguard Worker #ifndef HAVE_HSTRERROR
1661*5e7646d2SAndroid Build Coastguard Worker /*
1662*5e7646d2SAndroid Build Coastguard Worker  * '_cups_hstrerror()' - hstrerror() emulation function for Solaris and others.
1663*5e7646d2SAndroid Build Coastguard Worker  */
1664*5e7646d2SAndroid Build Coastguard Worker 
1665*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Error string */
_cups_hstrerror(int error)1666*5e7646d2SAndroid Build Coastguard Worker _cups_hstrerror(int error)		/* I - Error number */
1667*5e7646d2SAndroid Build Coastguard Worker {
1668*5e7646d2SAndroid Build Coastguard Worker   static const char * const errors[] =	/* Error strings */
1669*5e7646d2SAndroid Build Coastguard Worker 		{
1670*5e7646d2SAndroid Build Coastguard Worker 		  "OK",
1671*5e7646d2SAndroid Build Coastguard Worker 		  "Host not found.",
1672*5e7646d2SAndroid Build Coastguard Worker 		  "Try again.",
1673*5e7646d2SAndroid Build Coastguard Worker 		  "Unrecoverable lookup error.",
1674*5e7646d2SAndroid Build Coastguard Worker 		  "No data associated with name."
1675*5e7646d2SAndroid Build Coastguard Worker 		};
1676*5e7646d2SAndroid Build Coastguard Worker 
1677*5e7646d2SAndroid Build Coastguard Worker 
1678*5e7646d2SAndroid Build Coastguard Worker   if (error < 0 || error > 4)
1679*5e7646d2SAndroid Build Coastguard Worker     return ("Unknown hostname lookup error.");
1680*5e7646d2SAndroid Build Coastguard Worker   else
1681*5e7646d2SAndroid Build Coastguard Worker     return (errors[error]);
1682*5e7646d2SAndroid Build Coastguard Worker }
1683*5e7646d2SAndroid Build Coastguard Worker #endif /* !HAVE_HSTRERROR */
1684*5e7646d2SAndroid Build Coastguard Worker 
1685*5e7646d2SAndroid Build Coastguard Worker 
1686*5e7646d2SAndroid Build Coastguard Worker /*
1687*5e7646d2SAndroid Build Coastguard Worker  * '_httpDecodeURI()' - Percent-decode a HTTP request URI.
1688*5e7646d2SAndroid Build Coastguard Worker  */
1689*5e7646d2SAndroid Build Coastguard Worker 
1690*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Decoded URI or NULL on error */
_httpDecodeURI(char * dst,const char * src,size_t dstsize)1691*5e7646d2SAndroid Build Coastguard Worker _httpDecodeURI(char       *dst,		/* I - Destination buffer */
1692*5e7646d2SAndroid Build Coastguard Worker                const char *src,		/* I - Source URI */
1693*5e7646d2SAndroid Build Coastguard Worker 	       size_t     dstsize)	/* I - Size of destination buffer */
1694*5e7646d2SAndroid Build Coastguard Worker {
1695*5e7646d2SAndroid Build Coastguard Worker   if (http_copy_decode(dst, src, (int)dstsize, NULL, 1))
1696*5e7646d2SAndroid Build Coastguard Worker     return (dst);
1697*5e7646d2SAndroid Build Coastguard Worker   else
1698*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
1699*5e7646d2SAndroid Build Coastguard Worker }
1700*5e7646d2SAndroid Build Coastguard Worker 
1701*5e7646d2SAndroid Build Coastguard Worker 
1702*5e7646d2SAndroid Build Coastguard Worker /*
1703*5e7646d2SAndroid Build Coastguard Worker  * '_httpEncodeURI()' - Percent-encode a HTTP request URI.
1704*5e7646d2SAndroid Build Coastguard Worker  */
1705*5e7646d2SAndroid Build Coastguard Worker 
1706*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Encoded URI */
_httpEncodeURI(char * dst,const char * src,size_t dstsize)1707*5e7646d2SAndroid Build Coastguard Worker _httpEncodeURI(char       *dst,		/* I - Destination buffer */
1708*5e7646d2SAndroid Build Coastguard Worker                const char *src,		/* I - Source URI */
1709*5e7646d2SAndroid Build Coastguard Worker 	       size_t     dstsize)	/* I - Size of destination buffer */
1710*5e7646d2SAndroid Build Coastguard Worker {
1711*5e7646d2SAndroid Build Coastguard Worker   http_copy_encode(dst, src, dst + dstsize - 1, NULL, NULL, 1);
1712*5e7646d2SAndroid Build Coastguard Worker   return (dst);
1713*5e7646d2SAndroid Build Coastguard Worker }
1714*5e7646d2SAndroid Build Coastguard Worker 
1715*5e7646d2SAndroid Build Coastguard Worker 
1716*5e7646d2SAndroid Build Coastguard Worker /*
1717*5e7646d2SAndroid Build Coastguard Worker  * '_httpResolveURI()' - Resolve a DNS-SD URI.
1718*5e7646d2SAndroid Build Coastguard Worker  */
1719*5e7646d2SAndroid Build Coastguard Worker 
1720*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Resolved URI */
_httpResolveURI(const char * uri,char * resolved_uri,size_t resolved_size,int options,int (* cb)(void * context),void * context)1721*5e7646d2SAndroid Build Coastguard Worker _httpResolveURI(
1722*5e7646d2SAndroid Build Coastguard Worker     const char *uri,			/* I - DNS-SD URI */
1723*5e7646d2SAndroid Build Coastguard Worker     char       *resolved_uri,		/* I - Buffer for resolved URI */
1724*5e7646d2SAndroid Build Coastguard Worker     size_t     resolved_size,		/* I - Size of URI buffer */
1725*5e7646d2SAndroid Build Coastguard Worker     int        options,			/* I - Resolve options */
1726*5e7646d2SAndroid Build Coastguard Worker     int        (*cb)(void *context),	/* I - Continue callback function */
1727*5e7646d2SAndroid Build Coastguard Worker     void       *context)		/* I - Context pointer for callback */
1728*5e7646d2SAndroid Build Coastguard Worker {
1729*5e7646d2SAndroid Build Coastguard Worker   char			scheme[32],	/* URI components... */
1730*5e7646d2SAndroid Build Coastguard Worker 			userpass[256],
1731*5e7646d2SAndroid Build Coastguard Worker 			hostname[1024],
1732*5e7646d2SAndroid Build Coastguard Worker 			resource[1024];
1733*5e7646d2SAndroid Build Coastguard Worker   int			port;
1734*5e7646d2SAndroid Build Coastguard Worker #ifdef DEBUG
1735*5e7646d2SAndroid Build Coastguard Worker   http_uri_status_t	status;		/* URI decode status */
1736*5e7646d2SAndroid Build Coastguard Worker #endif /* DEBUG */
1737*5e7646d2SAndroid Build Coastguard Worker 
1738*5e7646d2SAndroid Build Coastguard Worker 
1739*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("_httpResolveURI(uri=\"%s\", resolved_uri=%p, resolved_size=" CUPS_LLFMT ", options=0x%x, cb=%p, context=%p)", uri, (void *)resolved_uri, CUPS_LLCAST resolved_size, options, (void *)cb, context));
1740*5e7646d2SAndroid Build Coastguard Worker 
1741*5e7646d2SAndroid Build Coastguard Worker  /*
1742*5e7646d2SAndroid Build Coastguard Worker   * Get the device URI...
1743*5e7646d2SAndroid Build Coastguard Worker   */
1744*5e7646d2SAndroid Build Coastguard Worker 
1745*5e7646d2SAndroid Build Coastguard Worker #ifdef DEBUG
1746*5e7646d2SAndroid Build Coastguard Worker   if ((status = httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme,
1747*5e7646d2SAndroid Build Coastguard Worker                                 sizeof(scheme), userpass, sizeof(userpass),
1748*5e7646d2SAndroid Build Coastguard Worker 				hostname, sizeof(hostname), &port, resource,
1749*5e7646d2SAndroid Build Coastguard Worker 				sizeof(resource))) < HTTP_URI_STATUS_OK)
1750*5e7646d2SAndroid Build Coastguard Worker #else
1751*5e7646d2SAndroid Build Coastguard Worker   if (httpSeparateURI(HTTP_URI_CODING_ALL, uri, scheme,
1752*5e7646d2SAndroid Build Coastguard Worker 		      sizeof(scheme), userpass, sizeof(userpass),
1753*5e7646d2SAndroid Build Coastguard Worker 		      hostname, sizeof(hostname), &port, resource,
1754*5e7646d2SAndroid Build Coastguard Worker 		      sizeof(resource)) < HTTP_URI_STATUS_OK)
1755*5e7646d2SAndroid Build Coastguard Worker #endif /* DEBUG */
1756*5e7646d2SAndroid Build Coastguard Worker   {
1757*5e7646d2SAndroid Build Coastguard Worker     if (options & _HTTP_RESOLVE_STDERR)
1758*5e7646d2SAndroid Build Coastguard Worker       _cupsLangPrintFilter(stderr, "ERROR", _("Bad device-uri \"%s\"."), uri);
1759*5e7646d2SAndroid Build Coastguard Worker 
1760*5e7646d2SAndroid Build Coastguard Worker     DEBUG_printf(("2_httpResolveURI: httpSeparateURI returned %d!", status));
1761*5e7646d2SAndroid Build Coastguard Worker     DEBUG_puts("2_httpResolveURI: Returning NULL");
1762*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
1763*5e7646d2SAndroid Build Coastguard Worker   }
1764*5e7646d2SAndroid Build Coastguard Worker 
1765*5e7646d2SAndroid Build Coastguard Worker  /*
1766*5e7646d2SAndroid Build Coastguard Worker   * Resolve it as needed...
1767*5e7646d2SAndroid Build Coastguard Worker   */
1768*5e7646d2SAndroid Build Coastguard Worker 
1769*5e7646d2SAndroid Build Coastguard Worker   if (strstr(hostname, "._tcp"))
1770*5e7646d2SAndroid Build Coastguard Worker   {
1771*5e7646d2SAndroid Build Coastguard Worker #if defined(HAVE_DNSSD) || defined(HAVE_AVAHI)
1772*5e7646d2SAndroid Build Coastguard Worker     char		*regtype,	/* Pointer to type in hostname */
1773*5e7646d2SAndroid Build Coastguard Worker 			*domain,	/* Pointer to domain in hostname */
1774*5e7646d2SAndroid Build Coastguard Worker 			*uuid,		/* Pointer to UUID in URI */
1775*5e7646d2SAndroid Build Coastguard Worker 			*uuidend;	/* Pointer to end of UUID in URI */
1776*5e7646d2SAndroid Build Coastguard Worker     _http_uribuf_t	uribuf;		/* URI buffer */
1777*5e7646d2SAndroid Build Coastguard Worker     int			offline = 0;	/* offline-report state set? */
1778*5e7646d2SAndroid Build Coastguard Worker #  ifdef HAVE_DNSSD
1779*5e7646d2SAndroid Build Coastguard Worker     DNSServiceRef	ref,		/* DNS-SD master service reference */
1780*5e7646d2SAndroid Build Coastguard Worker 			domainref = NULL,/* DNS-SD service reference for domain */
1781*5e7646d2SAndroid Build Coastguard Worker 			ippref = NULL,	/* DNS-SD service reference for network IPP */
1782*5e7646d2SAndroid Build Coastguard Worker 			ippsref = NULL,	/* DNS-SD service reference for network IPPS */
1783*5e7646d2SAndroid Build Coastguard Worker 			localref;	/* DNS-SD service reference for .local */
1784*5e7646d2SAndroid Build Coastguard Worker     int			extrasent = 0;	/* Send the domain/IPP/IPPS resolves? */
1785*5e7646d2SAndroid Build Coastguard Worker #    ifdef HAVE_POLL
1786*5e7646d2SAndroid Build Coastguard Worker     struct pollfd	polldata;	/* Polling data */
1787*5e7646d2SAndroid Build Coastguard Worker #    else /* select() */
1788*5e7646d2SAndroid Build Coastguard Worker     fd_set		input_set;	/* Input set for select() */
1789*5e7646d2SAndroid Build Coastguard Worker     struct timeval	stimeout;	/* Timeout value for select() */
1790*5e7646d2SAndroid Build Coastguard Worker #    endif /* HAVE_POLL */
1791*5e7646d2SAndroid Build Coastguard Worker #  elif defined(HAVE_AVAHI)
1792*5e7646d2SAndroid Build Coastguard Worker     AvahiClient		*client;	/* Client information */
1793*5e7646d2SAndroid Build Coastguard Worker     int			error;		/* Status */
1794*5e7646d2SAndroid Build Coastguard Worker #  endif /* HAVE_DNSSD */
1795*5e7646d2SAndroid Build Coastguard Worker 
1796*5e7646d2SAndroid Build Coastguard Worker     if (options & _HTTP_RESOLVE_STDERR)
1797*5e7646d2SAndroid Build Coastguard Worker       fprintf(stderr, "DEBUG: Resolving \"%s\"...\n", hostname);
1798*5e7646d2SAndroid Build Coastguard Worker 
1799*5e7646d2SAndroid Build Coastguard Worker    /*
1800*5e7646d2SAndroid Build Coastguard Worker     * Separate the hostname into service name, registration type, and domain...
1801*5e7646d2SAndroid Build Coastguard Worker     */
1802*5e7646d2SAndroid Build Coastguard Worker 
1803*5e7646d2SAndroid Build Coastguard Worker     for (regtype = strstr(hostname, "._tcp") - 2;
1804*5e7646d2SAndroid Build Coastguard Worker          regtype > hostname;
1805*5e7646d2SAndroid Build Coastguard Worker 	 regtype --)
1806*5e7646d2SAndroid Build Coastguard Worker       if (regtype[0] == '.' && regtype[1] == '_')
1807*5e7646d2SAndroid Build Coastguard Worker       {
1808*5e7646d2SAndroid Build Coastguard Worker        /*
1809*5e7646d2SAndroid Build Coastguard Worker         * Found ._servicetype in front of ._tcp...
1810*5e7646d2SAndroid Build Coastguard Worker 	*/
1811*5e7646d2SAndroid Build Coastguard Worker 
1812*5e7646d2SAndroid Build Coastguard Worker         *regtype++ = '\0';
1813*5e7646d2SAndroid Build Coastguard Worker 	break;
1814*5e7646d2SAndroid Build Coastguard Worker       }
1815*5e7646d2SAndroid Build Coastguard Worker 
1816*5e7646d2SAndroid Build Coastguard Worker     if (regtype <= hostname)
1817*5e7646d2SAndroid Build Coastguard Worker     {
1818*5e7646d2SAndroid Build Coastguard Worker       DEBUG_puts("2_httpResolveURI: Bad hostname, returning NULL");
1819*5e7646d2SAndroid Build Coastguard Worker       return (NULL);
1820*5e7646d2SAndroid Build Coastguard Worker     }
1821*5e7646d2SAndroid Build Coastguard Worker 
1822*5e7646d2SAndroid Build Coastguard Worker     for (domain = strchr(regtype, '.');
1823*5e7646d2SAndroid Build Coastguard Worker          domain;
1824*5e7646d2SAndroid Build Coastguard Worker 	 domain = strchr(domain + 1, '.'))
1825*5e7646d2SAndroid Build Coastguard Worker       if (domain[1] != '_')
1826*5e7646d2SAndroid Build Coastguard Worker         break;
1827*5e7646d2SAndroid Build Coastguard Worker 
1828*5e7646d2SAndroid Build Coastguard Worker     if (domain)
1829*5e7646d2SAndroid Build Coastguard Worker       *domain++ = '\0';
1830*5e7646d2SAndroid Build Coastguard Worker 
1831*5e7646d2SAndroid Build Coastguard Worker     if ((uuid = strstr(resource, "?uuid=")) != NULL)
1832*5e7646d2SAndroid Build Coastguard Worker     {
1833*5e7646d2SAndroid Build Coastguard Worker       *uuid = '\0';
1834*5e7646d2SAndroid Build Coastguard Worker       uuid  += 6;
1835*5e7646d2SAndroid Build Coastguard Worker       if ((uuidend = strchr(uuid, '&')) != NULL)
1836*5e7646d2SAndroid Build Coastguard Worker         *uuidend = '\0';
1837*5e7646d2SAndroid Build Coastguard Worker     }
1838*5e7646d2SAndroid Build Coastguard Worker 
1839*5e7646d2SAndroid Build Coastguard Worker     resolved_uri[0] = '\0';
1840*5e7646d2SAndroid Build Coastguard Worker 
1841*5e7646d2SAndroid Build Coastguard Worker     uribuf.buffer   = resolved_uri;
1842*5e7646d2SAndroid Build Coastguard Worker     uribuf.bufsize  = resolved_size;
1843*5e7646d2SAndroid Build Coastguard Worker     uribuf.options  = options;
1844*5e7646d2SAndroid Build Coastguard Worker     uribuf.resource = resource;
1845*5e7646d2SAndroid Build Coastguard Worker     uribuf.uuid     = uuid;
1846*5e7646d2SAndroid Build Coastguard Worker 
1847*5e7646d2SAndroid Build Coastguard Worker     DEBUG_printf(("2_httpResolveURI: Resolving hostname=\"%s\", regtype=\"%s\", "
1848*5e7646d2SAndroid Build Coastguard Worker                   "domain=\"%s\"\n", hostname, regtype, domain));
1849*5e7646d2SAndroid Build Coastguard Worker     if (options & _HTTP_RESOLVE_STDERR)
1850*5e7646d2SAndroid Build Coastguard Worker     {
1851*5e7646d2SAndroid Build Coastguard Worker       fputs("STATE: +connecting-to-device\n", stderr);
1852*5e7646d2SAndroid Build Coastguard Worker       fprintf(stderr, "DEBUG: Resolving \"%s\", regtype=\"%s\", "
1853*5e7646d2SAndroid Build Coastguard Worker                       "domain=\"local.\"...\n", hostname, regtype);
1854*5e7646d2SAndroid Build Coastguard Worker     }
1855*5e7646d2SAndroid Build Coastguard Worker 
1856*5e7646d2SAndroid Build Coastguard Worker     uri = NULL;
1857*5e7646d2SAndroid Build Coastguard Worker 
1858*5e7646d2SAndroid Build Coastguard Worker #  ifdef HAVE_DNSSD
1859*5e7646d2SAndroid Build Coastguard Worker     if (DNSServiceCreateConnection(&ref) == kDNSServiceErr_NoError)
1860*5e7646d2SAndroid Build Coastguard Worker     {
1861*5e7646d2SAndroid Build Coastguard Worker       uint32_t myinterface = kDNSServiceInterfaceIndexAny;
1862*5e7646d2SAndroid Build Coastguard Worker 					/* Lookup on any interface */
1863*5e7646d2SAndroid Build Coastguard Worker 
1864*5e7646d2SAndroid Build Coastguard Worker       if (!strcmp(scheme, "ippusb"))
1865*5e7646d2SAndroid Build Coastguard Worker         myinterface = kDNSServiceInterfaceIndexLocalOnly;
1866*5e7646d2SAndroid Build Coastguard Worker 
1867*5e7646d2SAndroid Build Coastguard Worker       localref = ref;
1868*5e7646d2SAndroid Build Coastguard Worker       if (DNSServiceResolve(&localref,
1869*5e7646d2SAndroid Build Coastguard Worker                             kDNSServiceFlagsShareConnection, myinterface,
1870*5e7646d2SAndroid Build Coastguard Worker                             hostname, regtype, "local.", http_resolve_cb,
1871*5e7646d2SAndroid Build Coastguard Worker 			    &uribuf) == kDNSServiceErr_NoError)
1872*5e7646d2SAndroid Build Coastguard Worker       {
1873*5e7646d2SAndroid Build Coastguard Worker 	int	fds;			/* Number of ready descriptors */
1874*5e7646d2SAndroid Build Coastguard Worker 	time_t	timeout,		/* Poll timeout */
1875*5e7646d2SAndroid Build Coastguard Worker 		start_time = time(NULL),/* Start time */
1876*5e7646d2SAndroid Build Coastguard Worker 		end_time = start_time + 90;
1877*5e7646d2SAndroid Build Coastguard Worker 					/* End time */
1878*5e7646d2SAndroid Build Coastguard Worker 
1879*5e7646d2SAndroid Build Coastguard Worker 	while (time(NULL) < end_time)
1880*5e7646d2SAndroid Build Coastguard Worker 	{
1881*5e7646d2SAndroid Build Coastguard Worker 	  if (options & _HTTP_RESOLVE_STDERR)
1882*5e7646d2SAndroid Build Coastguard Worker 	    _cupsLangPrintFilter(stderr, "INFO", _("Looking for printer."));
1883*5e7646d2SAndroid Build Coastguard Worker 
1884*5e7646d2SAndroid Build Coastguard Worker 	  if (cb && !(*cb)(context))
1885*5e7646d2SAndroid Build Coastguard Worker 	  {
1886*5e7646d2SAndroid Build Coastguard Worker 	    DEBUG_puts("2_httpResolveURI: callback returned 0 (stop)");
1887*5e7646d2SAndroid Build Coastguard Worker 	    break;
1888*5e7646d2SAndroid Build Coastguard Worker 	  }
1889*5e7646d2SAndroid Build Coastguard Worker 
1890*5e7646d2SAndroid Build Coastguard Worker 	 /*
1891*5e7646d2SAndroid Build Coastguard Worker 	  * Wakeup every 2 seconds to emit a "looking for printer" message...
1892*5e7646d2SAndroid Build Coastguard Worker 	  */
1893*5e7646d2SAndroid Build Coastguard Worker 
1894*5e7646d2SAndroid Build Coastguard Worker 	  if ((timeout = end_time - time(NULL)) > 2)
1895*5e7646d2SAndroid Build Coastguard Worker 	    timeout = 2;
1896*5e7646d2SAndroid Build Coastguard Worker 
1897*5e7646d2SAndroid Build Coastguard Worker #    ifdef HAVE_POLL
1898*5e7646d2SAndroid Build Coastguard Worker 	  polldata.fd     = DNSServiceRefSockFD(ref);
1899*5e7646d2SAndroid Build Coastguard Worker 	  polldata.events = POLLIN;
1900*5e7646d2SAndroid Build Coastguard Worker 
1901*5e7646d2SAndroid Build Coastguard Worker 	  fds = poll(&polldata, 1, (int)(1000 * timeout));
1902*5e7646d2SAndroid Build Coastguard Worker 
1903*5e7646d2SAndroid Build Coastguard Worker #    else /* select() */
1904*5e7646d2SAndroid Build Coastguard Worker 	  FD_ZERO(&input_set);
1905*5e7646d2SAndroid Build Coastguard Worker 	  FD_SET(DNSServiceRefSockFD(ref), &input_set);
1906*5e7646d2SAndroid Build Coastguard Worker 
1907*5e7646d2SAndroid Build Coastguard Worker #      ifdef _WIN32
1908*5e7646d2SAndroid Build Coastguard Worker 	  stimeout.tv_sec  = (long)timeout;
1909*5e7646d2SAndroid Build Coastguard Worker #      else
1910*5e7646d2SAndroid Build Coastguard Worker 	  stimeout.tv_sec  = timeout;
1911*5e7646d2SAndroid Build Coastguard Worker #      endif /* _WIN32 */
1912*5e7646d2SAndroid Build Coastguard Worker 	  stimeout.tv_usec = 0;
1913*5e7646d2SAndroid Build Coastguard Worker 
1914*5e7646d2SAndroid Build Coastguard Worker 	  fds = select(DNSServiceRefSockFD(ref)+1, &input_set, NULL, NULL,
1915*5e7646d2SAndroid Build Coastguard Worker 		       &stimeout);
1916*5e7646d2SAndroid Build Coastguard Worker #    endif /* HAVE_POLL */
1917*5e7646d2SAndroid Build Coastguard Worker 
1918*5e7646d2SAndroid Build Coastguard Worker 	  if (fds < 0)
1919*5e7646d2SAndroid Build Coastguard Worker 	  {
1920*5e7646d2SAndroid Build Coastguard Worker 	    if (errno != EINTR && errno != EAGAIN)
1921*5e7646d2SAndroid Build Coastguard Worker 	    {
1922*5e7646d2SAndroid Build Coastguard Worker 	      DEBUG_printf(("2_httpResolveURI: poll error: %s", strerror(errno)));
1923*5e7646d2SAndroid Build Coastguard Worker 	      break;
1924*5e7646d2SAndroid Build Coastguard Worker 	    }
1925*5e7646d2SAndroid Build Coastguard Worker 	  }
1926*5e7646d2SAndroid Build Coastguard Worker 	  else if (fds == 0)
1927*5e7646d2SAndroid Build Coastguard Worker 	  {
1928*5e7646d2SAndroid Build Coastguard Worker 	   /*
1929*5e7646d2SAndroid Build Coastguard Worker 	    * Wait 2 seconds for a response to the local resolve; if nothing
1930*5e7646d2SAndroid Build Coastguard Worker 	    * comes in, do an additional domain resolution...
1931*5e7646d2SAndroid Build Coastguard Worker 	    */
1932*5e7646d2SAndroid Build Coastguard Worker 
1933*5e7646d2SAndroid Build Coastguard Worker 	    if (extrasent == 0 && domain && _cups_strcasecmp(domain, "local."))
1934*5e7646d2SAndroid Build Coastguard Worker 	    {
1935*5e7646d2SAndroid Build Coastguard Worker 	      if (options & _HTTP_RESOLVE_STDERR)
1936*5e7646d2SAndroid Build Coastguard Worker 		fprintf(stderr,
1937*5e7646d2SAndroid Build Coastguard Worker 		        "DEBUG: Resolving \"%s\", regtype=\"%s\", "
1938*5e7646d2SAndroid Build Coastguard Worker 			"domain=\"%s\"...\n", hostname, regtype,
1939*5e7646d2SAndroid Build Coastguard Worker 			domain ? domain : "");
1940*5e7646d2SAndroid Build Coastguard Worker 
1941*5e7646d2SAndroid Build Coastguard Worker 	      domainref = ref;
1942*5e7646d2SAndroid Build Coastguard Worker 	      if (DNSServiceResolve(&domainref,
1943*5e7646d2SAndroid Build Coastguard Worker 	                            kDNSServiceFlagsShareConnection,
1944*5e7646d2SAndroid Build Coastguard Worker 	                            myinterface, hostname, regtype, domain,
1945*5e7646d2SAndroid Build Coastguard Worker 				    http_resolve_cb,
1946*5e7646d2SAndroid Build Coastguard Worker 				    &uribuf) == kDNSServiceErr_NoError)
1947*5e7646d2SAndroid Build Coastguard Worker 		extrasent = 1;
1948*5e7646d2SAndroid Build Coastguard Worker 	    }
1949*5e7646d2SAndroid Build Coastguard Worker 	    else if (extrasent == 0 && !strcmp(scheme, "ippusb"))
1950*5e7646d2SAndroid Build Coastguard Worker 	    {
1951*5e7646d2SAndroid Build Coastguard Worker 	      if (options & _HTTP_RESOLVE_STDERR)
1952*5e7646d2SAndroid Build Coastguard Worker 		fprintf(stderr, "DEBUG: Resolving \"%s\", regtype=\"_ipps._tcp\", domain=\"local.\"...\n", hostname);
1953*5e7646d2SAndroid Build Coastguard Worker 
1954*5e7646d2SAndroid Build Coastguard Worker 	      ippsref = ref;
1955*5e7646d2SAndroid Build Coastguard Worker 	      if (DNSServiceResolve(&ippsref,
1956*5e7646d2SAndroid Build Coastguard Worker 	                            kDNSServiceFlagsShareConnection,
1957*5e7646d2SAndroid Build Coastguard Worker 	                            kDNSServiceInterfaceIndexAny, hostname,
1958*5e7646d2SAndroid Build Coastguard Worker 	                            "_ipps._tcp", domain, http_resolve_cb,
1959*5e7646d2SAndroid Build Coastguard Worker 				    &uribuf) == kDNSServiceErr_NoError)
1960*5e7646d2SAndroid Build Coastguard Worker 		extrasent = 1;
1961*5e7646d2SAndroid Build Coastguard Worker 	    }
1962*5e7646d2SAndroid Build Coastguard Worker 	    else if (extrasent == 1 && !strcmp(scheme, "ippusb"))
1963*5e7646d2SAndroid Build Coastguard Worker 	    {
1964*5e7646d2SAndroid Build Coastguard Worker 	      if (options & _HTTP_RESOLVE_STDERR)
1965*5e7646d2SAndroid Build Coastguard Worker 		fprintf(stderr, "DEBUG: Resolving \"%s\", regtype=\"_ipp._tcp\", domain=\"local.\"...\n", hostname);
1966*5e7646d2SAndroid Build Coastguard Worker 
1967*5e7646d2SAndroid Build Coastguard Worker 	      ippref = ref;
1968*5e7646d2SAndroid Build Coastguard Worker 	      if (DNSServiceResolve(&ippref,
1969*5e7646d2SAndroid Build Coastguard Worker 	                            kDNSServiceFlagsShareConnection,
1970*5e7646d2SAndroid Build Coastguard Worker 	                            kDNSServiceInterfaceIndexAny, hostname,
1971*5e7646d2SAndroid Build Coastguard Worker 	                            "_ipp._tcp", domain, http_resolve_cb,
1972*5e7646d2SAndroid Build Coastguard Worker 				    &uribuf) == kDNSServiceErr_NoError)
1973*5e7646d2SAndroid Build Coastguard Worker 		extrasent = 2;
1974*5e7646d2SAndroid Build Coastguard Worker 	    }
1975*5e7646d2SAndroid Build Coastguard Worker 
1976*5e7646d2SAndroid Build Coastguard Worker 	   /*
1977*5e7646d2SAndroid Build Coastguard Worker 	    * If it hasn't resolved within 5 seconds set the offline-report
1978*5e7646d2SAndroid Build Coastguard Worker 	    * printer-state-reason...
1979*5e7646d2SAndroid Build Coastguard Worker 	    */
1980*5e7646d2SAndroid Build Coastguard Worker 
1981*5e7646d2SAndroid Build Coastguard Worker 	    if ((options & _HTTP_RESOLVE_STDERR) && offline == 0 &&
1982*5e7646d2SAndroid Build Coastguard Worker 	        time(NULL) > (start_time + 5))
1983*5e7646d2SAndroid Build Coastguard Worker 	    {
1984*5e7646d2SAndroid Build Coastguard Worker 	      fputs("STATE: +offline-report\n", stderr);
1985*5e7646d2SAndroid Build Coastguard Worker 	      offline = 1;
1986*5e7646d2SAndroid Build Coastguard Worker 	    }
1987*5e7646d2SAndroid Build Coastguard Worker 	  }
1988*5e7646d2SAndroid Build Coastguard Worker 	  else
1989*5e7646d2SAndroid Build Coastguard Worker 	  {
1990*5e7646d2SAndroid Build Coastguard Worker 	    if (DNSServiceProcessResult(ref) == kDNSServiceErr_NoError &&
1991*5e7646d2SAndroid Build Coastguard Worker 	        resolved_uri[0])
1992*5e7646d2SAndroid Build Coastguard Worker 	    {
1993*5e7646d2SAndroid Build Coastguard Worker 	      uri = resolved_uri;
1994*5e7646d2SAndroid Build Coastguard Worker 	      break;
1995*5e7646d2SAndroid Build Coastguard Worker 	    }
1996*5e7646d2SAndroid Build Coastguard Worker 	  }
1997*5e7646d2SAndroid Build Coastguard Worker 	}
1998*5e7646d2SAndroid Build Coastguard Worker 
1999*5e7646d2SAndroid Build Coastguard Worker 	if (extrasent)
2000*5e7646d2SAndroid Build Coastguard Worker 	{
2001*5e7646d2SAndroid Build Coastguard Worker 	  if (domainref)
2002*5e7646d2SAndroid Build Coastguard Worker 	    DNSServiceRefDeallocate(domainref);
2003*5e7646d2SAndroid Build Coastguard Worker 	  if (ippref)
2004*5e7646d2SAndroid Build Coastguard Worker 	    DNSServiceRefDeallocate(ippref);
2005*5e7646d2SAndroid Build Coastguard Worker 	  if (ippsref)
2006*5e7646d2SAndroid Build Coastguard Worker 	    DNSServiceRefDeallocate(ippsref);
2007*5e7646d2SAndroid Build Coastguard Worker 	}
2008*5e7646d2SAndroid Build Coastguard Worker 
2009*5e7646d2SAndroid Build Coastguard Worker 	DNSServiceRefDeallocate(localref);
2010*5e7646d2SAndroid Build Coastguard Worker       }
2011*5e7646d2SAndroid Build Coastguard Worker 
2012*5e7646d2SAndroid Build Coastguard Worker       DNSServiceRefDeallocate(ref);
2013*5e7646d2SAndroid Build Coastguard Worker     }
2014*5e7646d2SAndroid Build Coastguard Worker #  else /* HAVE_AVAHI */
2015*5e7646d2SAndroid Build Coastguard Worker     if ((uribuf.poll = avahi_simple_poll_new()) != NULL)
2016*5e7646d2SAndroid Build Coastguard Worker     {
2017*5e7646d2SAndroid Build Coastguard Worker       avahi_simple_poll_set_func(uribuf.poll, http_poll_cb, NULL);
2018*5e7646d2SAndroid Build Coastguard Worker 
2019*5e7646d2SAndroid Build Coastguard Worker       if ((client = avahi_client_new(avahi_simple_poll_get(uribuf.poll),
2020*5e7646d2SAndroid Build Coastguard Worker 				      0, http_client_cb,
2021*5e7646d2SAndroid Build Coastguard Worker 				      &uribuf, &error)) != NULL)
2022*5e7646d2SAndroid Build Coastguard Worker       {
2023*5e7646d2SAndroid Build Coastguard Worker 	if (avahi_service_resolver_new(client, AVAHI_IF_UNSPEC,
2024*5e7646d2SAndroid Build Coastguard Worker 				       AVAHI_PROTO_UNSPEC, hostname,
2025*5e7646d2SAndroid Build Coastguard Worker 				       regtype, "local.", AVAHI_PROTO_UNSPEC, 0,
2026*5e7646d2SAndroid Build Coastguard Worker 				       http_resolve_cb, &uribuf) != NULL)
2027*5e7646d2SAndroid Build Coastguard Worker 	{
2028*5e7646d2SAndroid Build Coastguard Worker 	  time_t	start_time = time(NULL),
2029*5e7646d2SAndroid Build Coastguard Worker 	  				/* Start time */
2030*5e7646d2SAndroid Build Coastguard Worker 			end_time = start_time + 90;
2031*5e7646d2SAndroid Build Coastguard Worker 					/* End time */
2032*5e7646d2SAndroid Build Coastguard Worker           int           pstatus;	/* Poll status */
2033*5e7646d2SAndroid Build Coastguard Worker 
2034*5e7646d2SAndroid Build Coastguard Worker 	  pstatus = avahi_simple_poll_iterate(uribuf.poll, 2000);
2035*5e7646d2SAndroid Build Coastguard Worker 
2036*5e7646d2SAndroid Build Coastguard Worker 	  if (pstatus == 0 && !resolved_uri[0] && domain &&
2037*5e7646d2SAndroid Build Coastguard Worker 	      _cups_strcasecmp(domain, "local."))
2038*5e7646d2SAndroid Build Coastguard Worker 	  {
2039*5e7646d2SAndroid Build Coastguard Worker 	   /*
2040*5e7646d2SAndroid Build Coastguard Worker 	    * Resolve for .local hasn't returned anything, try the listed
2041*5e7646d2SAndroid Build Coastguard Worker 	    * domain...
2042*5e7646d2SAndroid Build Coastguard Worker 	    */
2043*5e7646d2SAndroid Build Coastguard Worker 
2044*5e7646d2SAndroid Build Coastguard Worker 	    avahi_service_resolver_new(client, AVAHI_IF_UNSPEC,
2045*5e7646d2SAndroid Build Coastguard Worker 				       AVAHI_PROTO_UNSPEC, hostname,
2046*5e7646d2SAndroid Build Coastguard Worker 				       regtype, domain, AVAHI_PROTO_UNSPEC, 0,
2047*5e7646d2SAndroid Build Coastguard Worker 				       http_resolve_cb, &uribuf);
2048*5e7646d2SAndroid Build Coastguard Worker           }
2049*5e7646d2SAndroid Build Coastguard Worker 
2050*5e7646d2SAndroid Build Coastguard Worker 	  while (!pstatus && !resolved_uri[0] && time(NULL) < end_time)
2051*5e7646d2SAndroid Build Coastguard Worker           {
2052*5e7646d2SAndroid Build Coastguard Worker   	    if ((pstatus = avahi_simple_poll_iterate(uribuf.poll, 2000)) != 0)
2053*5e7646d2SAndroid Build Coastguard Worker   	      break;
2054*5e7646d2SAndroid Build Coastguard Worker 
2055*5e7646d2SAndroid Build Coastguard Worker 	   /*
2056*5e7646d2SAndroid Build Coastguard Worker 	    * If it hasn't resolved within 5 seconds set the offline-report
2057*5e7646d2SAndroid Build Coastguard Worker 	    * printer-state-reason...
2058*5e7646d2SAndroid Build Coastguard Worker 	    */
2059*5e7646d2SAndroid Build Coastguard Worker 
2060*5e7646d2SAndroid Build Coastguard Worker 	    if ((options & _HTTP_RESOLVE_STDERR) && offline == 0 &&
2061*5e7646d2SAndroid Build Coastguard Worker 	        time(NULL) > (start_time + 5))
2062*5e7646d2SAndroid Build Coastguard Worker 	    {
2063*5e7646d2SAndroid Build Coastguard Worker 	      fputs("STATE: +offline-report\n", stderr);
2064*5e7646d2SAndroid Build Coastguard Worker 	      offline = 1;
2065*5e7646d2SAndroid Build Coastguard Worker 	    }
2066*5e7646d2SAndroid Build Coastguard Worker           }
2067*5e7646d2SAndroid Build Coastguard Worker 
2068*5e7646d2SAndroid Build Coastguard Worker 	 /*
2069*5e7646d2SAndroid Build Coastguard Worker 	  * Collect the result (if we got one).
2070*5e7646d2SAndroid Build Coastguard Worker 	  */
2071*5e7646d2SAndroid Build Coastguard Worker 
2072*5e7646d2SAndroid Build Coastguard Worker 	  if (resolved_uri[0])
2073*5e7646d2SAndroid Build Coastguard Worker 	    uri = resolved_uri;
2074*5e7646d2SAndroid Build Coastguard Worker 	}
2075*5e7646d2SAndroid Build Coastguard Worker 
2076*5e7646d2SAndroid Build Coastguard Worker 	avahi_client_free(client);
2077*5e7646d2SAndroid Build Coastguard Worker       }
2078*5e7646d2SAndroid Build Coastguard Worker 
2079*5e7646d2SAndroid Build Coastguard Worker       avahi_simple_poll_free(uribuf.poll);
2080*5e7646d2SAndroid Build Coastguard Worker     }
2081*5e7646d2SAndroid Build Coastguard Worker #  endif /* HAVE_DNSSD */
2082*5e7646d2SAndroid Build Coastguard Worker 
2083*5e7646d2SAndroid Build Coastguard Worker     if (options & _HTTP_RESOLVE_STDERR)
2084*5e7646d2SAndroid Build Coastguard Worker     {
2085*5e7646d2SAndroid Build Coastguard Worker       if (uri)
2086*5e7646d2SAndroid Build Coastguard Worker       {
2087*5e7646d2SAndroid Build Coastguard Worker         fprintf(stderr, "DEBUG: Resolved as \"%s\"...\n", uri);
2088*5e7646d2SAndroid Build Coastguard Worker 	fputs("STATE: -connecting-to-device,offline-report\n", stderr);
2089*5e7646d2SAndroid Build Coastguard Worker       }
2090*5e7646d2SAndroid Build Coastguard Worker       else
2091*5e7646d2SAndroid Build Coastguard Worker       {
2092*5e7646d2SAndroid Build Coastguard Worker         fputs("DEBUG: Unable to resolve URI\n", stderr);
2093*5e7646d2SAndroid Build Coastguard Worker 	fputs("STATE: -connecting-to-device\n", stderr);
2094*5e7646d2SAndroid Build Coastguard Worker       }
2095*5e7646d2SAndroid Build Coastguard Worker     }
2096*5e7646d2SAndroid Build Coastguard Worker 
2097*5e7646d2SAndroid Build Coastguard Worker #else /* HAVE_DNSSD || HAVE_AVAHI */
2098*5e7646d2SAndroid Build Coastguard Worker    /*
2099*5e7646d2SAndroid Build Coastguard Worker     * No DNS-SD support...
2100*5e7646d2SAndroid Build Coastguard Worker     */
2101*5e7646d2SAndroid Build Coastguard Worker 
2102*5e7646d2SAndroid Build Coastguard Worker     uri = NULL;
2103*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_DNSSD || HAVE_AVAHI */
2104*5e7646d2SAndroid Build Coastguard Worker 
2105*5e7646d2SAndroid Build Coastguard Worker     if ((options & _HTTP_RESOLVE_STDERR) && !uri)
2106*5e7646d2SAndroid Build Coastguard Worker       _cupsLangPrintFilter(stderr, "INFO", _("Unable to find printer."));
2107*5e7646d2SAndroid Build Coastguard Worker   }
2108*5e7646d2SAndroid Build Coastguard Worker   else
2109*5e7646d2SAndroid Build Coastguard Worker   {
2110*5e7646d2SAndroid Build Coastguard Worker    /*
2111*5e7646d2SAndroid Build Coastguard Worker     * Nothing more to do...
2112*5e7646d2SAndroid Build Coastguard Worker     */
2113*5e7646d2SAndroid Build Coastguard Worker 
2114*5e7646d2SAndroid Build Coastguard Worker     strlcpy(resolved_uri, uri, resolved_size);
2115*5e7646d2SAndroid Build Coastguard Worker     uri = resolved_uri;
2116*5e7646d2SAndroid Build Coastguard Worker   }
2117*5e7646d2SAndroid Build Coastguard Worker 
2118*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2_httpResolveURI: Returning \"%s\"", uri));
2119*5e7646d2SAndroid Build Coastguard Worker 
2120*5e7646d2SAndroid Build Coastguard Worker   return (uri);
2121*5e7646d2SAndroid Build Coastguard Worker }
2122*5e7646d2SAndroid Build Coastguard Worker 
2123*5e7646d2SAndroid Build Coastguard Worker 
2124*5e7646d2SAndroid Build Coastguard Worker #ifdef HAVE_AVAHI
2125*5e7646d2SAndroid Build Coastguard Worker /*
2126*5e7646d2SAndroid Build Coastguard Worker  * 'http_client_cb()' - Client callback for resolving URI.
2127*5e7646d2SAndroid Build Coastguard Worker  */
2128*5e7646d2SAndroid Build Coastguard Worker 
2129*5e7646d2SAndroid Build Coastguard Worker static void
http_client_cb(AvahiClient * client,AvahiClientState state,void * context)2130*5e7646d2SAndroid Build Coastguard Worker http_client_cb(
2131*5e7646d2SAndroid Build Coastguard Worker     AvahiClient      *client,		/* I - Client information */
2132*5e7646d2SAndroid Build Coastguard Worker     AvahiClientState state,		/* I - Current state */
2133*5e7646d2SAndroid Build Coastguard Worker     void             *context)		/* I - Pointer to URI buffer */
2134*5e7646d2SAndroid Build Coastguard Worker {
2135*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("7http_client_cb(client=%p, state=%d, context=%p)", client,
2136*5e7646d2SAndroid Build Coastguard Worker                 state, context));
2137*5e7646d2SAndroid Build Coastguard Worker 
2138*5e7646d2SAndroid Build Coastguard Worker  /*
2139*5e7646d2SAndroid Build Coastguard Worker   * If the connection drops, quit.
2140*5e7646d2SAndroid Build Coastguard Worker   */
2141*5e7646d2SAndroid Build Coastguard Worker 
2142*5e7646d2SAndroid Build Coastguard Worker   if (state == AVAHI_CLIENT_FAILURE)
2143*5e7646d2SAndroid Build Coastguard Worker   {
2144*5e7646d2SAndroid Build Coastguard Worker     _http_uribuf_t *uribuf = (_http_uribuf_t *)context;
2145*5e7646d2SAndroid Build Coastguard Worker 					/* URI buffer */
2146*5e7646d2SAndroid Build Coastguard Worker 
2147*5e7646d2SAndroid Build Coastguard Worker     avahi_simple_poll_quit(uribuf->poll);
2148*5e7646d2SAndroid Build Coastguard Worker   }
2149*5e7646d2SAndroid Build Coastguard Worker }
2150*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_AVAHI */
2151*5e7646d2SAndroid Build Coastguard Worker 
2152*5e7646d2SAndroid Build Coastguard Worker 
2153*5e7646d2SAndroid Build Coastguard Worker /*
2154*5e7646d2SAndroid Build Coastguard Worker  * 'http_copy_decode()' - Copy and decode a URI.
2155*5e7646d2SAndroid Build Coastguard Worker  */
2156*5e7646d2SAndroid Build Coastguard Worker 
2157*5e7646d2SAndroid Build Coastguard Worker static const char *			/* O - New source pointer or NULL on error */
http_copy_decode(char * dst,const char * src,int dstsize,const char * term,int decode)2158*5e7646d2SAndroid Build Coastguard Worker http_copy_decode(char       *dst,	/* O - Destination buffer */
2159*5e7646d2SAndroid Build Coastguard Worker                  const char *src,	/* I - Source pointer */
2160*5e7646d2SAndroid Build Coastguard Worker 		 int        dstsize,	/* I - Destination size */
2161*5e7646d2SAndroid Build Coastguard Worker 		 const char *term,	/* I - Terminating characters */
2162*5e7646d2SAndroid Build Coastguard Worker 		 int        decode)	/* I - Decode %-encoded values */
2163*5e7646d2SAndroid Build Coastguard Worker {
2164*5e7646d2SAndroid Build Coastguard Worker   char	*ptr,				/* Pointer into buffer */
2165*5e7646d2SAndroid Build Coastguard Worker 	*end;				/* End of buffer */
2166*5e7646d2SAndroid Build Coastguard Worker   int	quoted;				/* Quoted character */
2167*5e7646d2SAndroid Build Coastguard Worker 
2168*5e7646d2SAndroid Build Coastguard Worker 
2169*5e7646d2SAndroid Build Coastguard Worker  /*
2170*5e7646d2SAndroid Build Coastguard Worker   * Copy the src to the destination until we hit a terminating character
2171*5e7646d2SAndroid Build Coastguard Worker   * or the end of the string.
2172*5e7646d2SAndroid Build Coastguard Worker   */
2173*5e7646d2SAndroid Build Coastguard Worker 
2174*5e7646d2SAndroid Build Coastguard Worker   for (ptr = dst, end = dst + dstsize - 1;
2175*5e7646d2SAndroid Build Coastguard Worker        *src && (!term || !strchr(term, *src));
2176*5e7646d2SAndroid Build Coastguard Worker        src ++)
2177*5e7646d2SAndroid Build Coastguard Worker     if (ptr < end)
2178*5e7646d2SAndroid Build Coastguard Worker     {
2179*5e7646d2SAndroid Build Coastguard Worker       if (*src == '%' && decode)
2180*5e7646d2SAndroid Build Coastguard Worker       {
2181*5e7646d2SAndroid Build Coastguard Worker         if (isxdigit(src[1] & 255) && isxdigit(src[2] & 255))
2182*5e7646d2SAndroid Build Coastguard Worker 	{
2183*5e7646d2SAndroid Build Coastguard Worker 	 /*
2184*5e7646d2SAndroid Build Coastguard Worker 	  * Grab a hex-encoded character...
2185*5e7646d2SAndroid Build Coastguard Worker 	  */
2186*5e7646d2SAndroid Build Coastguard Worker 
2187*5e7646d2SAndroid Build Coastguard Worker           src ++;
2188*5e7646d2SAndroid Build Coastguard Worker 	  if (isalpha(*src))
2189*5e7646d2SAndroid Build Coastguard Worker 	    quoted = (tolower(*src) - 'a' + 10) << 4;
2190*5e7646d2SAndroid Build Coastguard Worker 	  else
2191*5e7646d2SAndroid Build Coastguard Worker 	    quoted = (*src - '0') << 4;
2192*5e7646d2SAndroid Build Coastguard Worker 
2193*5e7646d2SAndroid Build Coastguard Worker           src ++;
2194*5e7646d2SAndroid Build Coastguard Worker 	  if (isalpha(*src))
2195*5e7646d2SAndroid Build Coastguard Worker 	    quoted |= tolower(*src) - 'a' + 10;
2196*5e7646d2SAndroid Build Coastguard Worker 	  else
2197*5e7646d2SAndroid Build Coastguard Worker 	    quoted |= *src - '0';
2198*5e7646d2SAndroid Build Coastguard Worker 
2199*5e7646d2SAndroid Build Coastguard Worker           *ptr++ = (char)quoted;
2200*5e7646d2SAndroid Build Coastguard Worker 	}
2201*5e7646d2SAndroid Build Coastguard Worker 	else
2202*5e7646d2SAndroid Build Coastguard Worker 	{
2203*5e7646d2SAndroid Build Coastguard Worker 	 /*
2204*5e7646d2SAndroid Build Coastguard Worker 	  * Bad hex-encoded character...
2205*5e7646d2SAndroid Build Coastguard Worker 	  */
2206*5e7646d2SAndroid Build Coastguard Worker 
2207*5e7646d2SAndroid Build Coastguard Worker 	  *ptr = '\0';
2208*5e7646d2SAndroid Build Coastguard Worker 	  return (NULL);
2209*5e7646d2SAndroid Build Coastguard Worker 	}
2210*5e7646d2SAndroid Build Coastguard Worker       }
2211*5e7646d2SAndroid Build Coastguard Worker       else if ((*src & 255) <= 0x20 || (*src & 255) >= 0x7f)
2212*5e7646d2SAndroid Build Coastguard Worker       {
2213*5e7646d2SAndroid Build Coastguard Worker         *ptr = '\0';
2214*5e7646d2SAndroid Build Coastguard Worker         return (NULL);
2215*5e7646d2SAndroid Build Coastguard Worker       }
2216*5e7646d2SAndroid Build Coastguard Worker       else
2217*5e7646d2SAndroid Build Coastguard Worker 	*ptr++ = *src;
2218*5e7646d2SAndroid Build Coastguard Worker     }
2219*5e7646d2SAndroid Build Coastguard Worker 
2220*5e7646d2SAndroid Build Coastguard Worker   *ptr = '\0';
2221*5e7646d2SAndroid Build Coastguard Worker 
2222*5e7646d2SAndroid Build Coastguard Worker   return (src);
2223*5e7646d2SAndroid Build Coastguard Worker }
2224*5e7646d2SAndroid Build Coastguard Worker 
2225*5e7646d2SAndroid Build Coastguard Worker 
2226*5e7646d2SAndroid Build Coastguard Worker /*
2227*5e7646d2SAndroid Build Coastguard Worker  * 'http_copy_encode()' - Copy and encode a URI.
2228*5e7646d2SAndroid Build Coastguard Worker  */
2229*5e7646d2SAndroid Build Coastguard Worker 
2230*5e7646d2SAndroid Build Coastguard Worker static char *				/* O - End of current URI */
http_copy_encode(char * dst,const char * src,char * dstend,const char * reserved,const char * term,int encode)2231*5e7646d2SAndroid Build Coastguard Worker http_copy_encode(char       *dst,	/* O - Destination buffer */
2232*5e7646d2SAndroid Build Coastguard Worker                  const char *src,	/* I - Source pointer */
2233*5e7646d2SAndroid Build Coastguard Worker 		 char       *dstend,	/* I - End of destination buffer */
2234*5e7646d2SAndroid Build Coastguard Worker                  const char *reserved,	/* I - Extra reserved characters */
2235*5e7646d2SAndroid Build Coastguard Worker 		 const char *term,	/* I - Terminating characters */
2236*5e7646d2SAndroid Build Coastguard Worker 		 int        encode)	/* I - %-encode reserved chars? */
2237*5e7646d2SAndroid Build Coastguard Worker {
2238*5e7646d2SAndroid Build Coastguard Worker   static const char hex[] = "0123456789ABCDEF";
2239*5e7646d2SAndroid Build Coastguard Worker 
2240*5e7646d2SAndroid Build Coastguard Worker 
2241*5e7646d2SAndroid Build Coastguard Worker   while (*src && dst < dstend)
2242*5e7646d2SAndroid Build Coastguard Worker   {
2243*5e7646d2SAndroid Build Coastguard Worker     if (term && *src == *term)
2244*5e7646d2SAndroid Build Coastguard Worker       return (dst);
2245*5e7646d2SAndroid Build Coastguard Worker 
2246*5e7646d2SAndroid Build Coastguard Worker     if (encode && (*src == '%' || *src <= ' ' || *src & 128 ||
2247*5e7646d2SAndroid Build Coastguard Worker                    (reserved && strchr(reserved, *src))))
2248*5e7646d2SAndroid Build Coastguard Worker     {
2249*5e7646d2SAndroid Build Coastguard Worker      /*
2250*5e7646d2SAndroid Build Coastguard Worker       * Hex encode reserved characters...
2251*5e7646d2SAndroid Build Coastguard Worker       */
2252*5e7646d2SAndroid Build Coastguard Worker 
2253*5e7646d2SAndroid Build Coastguard Worker       if ((dst + 2) >= dstend)
2254*5e7646d2SAndroid Build Coastguard Worker         break;
2255*5e7646d2SAndroid Build Coastguard Worker 
2256*5e7646d2SAndroid Build Coastguard Worker       *dst++ = '%';
2257*5e7646d2SAndroid Build Coastguard Worker       *dst++ = hex[(*src >> 4) & 15];
2258*5e7646d2SAndroid Build Coastguard Worker       *dst++ = hex[*src & 15];
2259*5e7646d2SAndroid Build Coastguard Worker 
2260*5e7646d2SAndroid Build Coastguard Worker       src ++;
2261*5e7646d2SAndroid Build Coastguard Worker     }
2262*5e7646d2SAndroid Build Coastguard Worker     else
2263*5e7646d2SAndroid Build Coastguard Worker       *dst++ = *src++;
2264*5e7646d2SAndroid Build Coastguard Worker   }
2265*5e7646d2SAndroid Build Coastguard Worker 
2266*5e7646d2SAndroid Build Coastguard Worker   *dst = '\0';
2267*5e7646d2SAndroid Build Coastguard Worker 
2268*5e7646d2SAndroid Build Coastguard Worker   if (*src)
2269*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
2270*5e7646d2SAndroid Build Coastguard Worker   else
2271*5e7646d2SAndroid Build Coastguard Worker     return (dst);
2272*5e7646d2SAndroid Build Coastguard Worker }
2273*5e7646d2SAndroid Build Coastguard Worker 
2274*5e7646d2SAndroid Build Coastguard Worker 
2275*5e7646d2SAndroid Build Coastguard Worker #ifdef HAVE_DNSSD
2276*5e7646d2SAndroid Build Coastguard Worker /*
2277*5e7646d2SAndroid Build Coastguard Worker  * 'http_resolve_cb()' - Build a device URI for the given service name.
2278*5e7646d2SAndroid Build Coastguard Worker  */
2279*5e7646d2SAndroid Build Coastguard Worker 
2280*5e7646d2SAndroid Build Coastguard Worker static void DNSSD_API
http_resolve_cb(DNSServiceRef sdRef,DNSServiceFlags flags,uint32_t interfaceIndex,DNSServiceErrorType errorCode,const char * fullName,const char * hostTarget,uint16_t port,uint16_t txtLen,const unsigned char * txtRecord,void * context)2281*5e7646d2SAndroid Build Coastguard Worker http_resolve_cb(
2282*5e7646d2SAndroid Build Coastguard Worker     DNSServiceRef       sdRef,		/* I - Service reference */
2283*5e7646d2SAndroid Build Coastguard Worker     DNSServiceFlags     flags,		/* I - Results flags */
2284*5e7646d2SAndroid Build Coastguard Worker     uint32_t            interfaceIndex,	/* I - Interface number */
2285*5e7646d2SAndroid Build Coastguard Worker     DNSServiceErrorType errorCode,	/* I - Error, if any */
2286*5e7646d2SAndroid Build Coastguard Worker     const char          *fullName,	/* I - Full service name */
2287*5e7646d2SAndroid Build Coastguard Worker     const char          *hostTarget,	/* I - Hostname */
2288*5e7646d2SAndroid Build Coastguard Worker     uint16_t            port,		/* I - Port number */
2289*5e7646d2SAndroid Build Coastguard Worker     uint16_t            txtLen,		/* I - Length of TXT record */
2290*5e7646d2SAndroid Build Coastguard Worker     const unsigned char *txtRecord,	/* I - TXT record data */
2291*5e7646d2SAndroid Build Coastguard Worker     void                *context)	/* I - Pointer to URI buffer */
2292*5e7646d2SAndroid Build Coastguard Worker {
2293*5e7646d2SAndroid Build Coastguard Worker   _http_uribuf_t	*uribuf = (_http_uribuf_t *)context;
2294*5e7646d2SAndroid Build Coastguard Worker 					/* URI buffer */
2295*5e7646d2SAndroid Build Coastguard Worker   const char		*scheme,	/* URI scheme */
2296*5e7646d2SAndroid Build Coastguard Worker 			*hostptr,	/* Pointer into hostTarget */
2297*5e7646d2SAndroid Build Coastguard Worker 			*reskey,	/* "rp" or "rfo" */
2298*5e7646d2SAndroid Build Coastguard Worker 			*resdefault;	/* Default path */
2299*5e7646d2SAndroid Build Coastguard Worker   char			resource[257],	/* Remote path */
2300*5e7646d2SAndroid Build Coastguard Worker 			fqdn[256];	/* FQDN of the .local name */
2301*5e7646d2SAndroid Build Coastguard Worker   const void		*value;		/* Value from TXT record */
2302*5e7646d2SAndroid Build Coastguard Worker   uint8_t		valueLen;	/* Length of value */
2303*5e7646d2SAndroid Build Coastguard Worker 
2304*5e7646d2SAndroid Build Coastguard Worker 
2305*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4http_resolve_cb(sdRef=%p, flags=%x, interfaceIndex=%u, errorCode=%d, fullName=\"%s\", hostTarget=\"%s\", port=%u, txtLen=%u, txtRecord=%p, context=%p)", (void *)sdRef, flags, interfaceIndex, errorCode, fullName, hostTarget, port, txtLen, (void *)txtRecord, context));
2306*5e7646d2SAndroid Build Coastguard Worker 
2307*5e7646d2SAndroid Build Coastguard Worker  /*
2308*5e7646d2SAndroid Build Coastguard Worker   * If we have a UUID, compare it...
2309*5e7646d2SAndroid Build Coastguard Worker   */
2310*5e7646d2SAndroid Build Coastguard Worker 
2311*5e7646d2SAndroid Build Coastguard Worker   if (uribuf->uuid &&
2312*5e7646d2SAndroid Build Coastguard Worker       (value = TXTRecordGetValuePtr(txtLen, txtRecord, "UUID",
2313*5e7646d2SAndroid Build Coastguard Worker                                     &valueLen)) != NULL)
2314*5e7646d2SAndroid Build Coastguard Worker   {
2315*5e7646d2SAndroid Build Coastguard Worker     char	uuid[256];		/* UUID value */
2316*5e7646d2SAndroid Build Coastguard Worker 
2317*5e7646d2SAndroid Build Coastguard Worker     memcpy(uuid, value, valueLen);
2318*5e7646d2SAndroid Build Coastguard Worker     uuid[valueLen] = '\0';
2319*5e7646d2SAndroid Build Coastguard Worker 
2320*5e7646d2SAndroid Build Coastguard Worker     if (_cups_strcasecmp(uuid, uribuf->uuid))
2321*5e7646d2SAndroid Build Coastguard Worker     {
2322*5e7646d2SAndroid Build Coastguard Worker       if (uribuf->options & _HTTP_RESOLVE_STDERR)
2323*5e7646d2SAndroid Build Coastguard Worker 	fprintf(stderr, "DEBUG: Found UUID %s, looking for %s.", uuid,
2324*5e7646d2SAndroid Build Coastguard Worker 		uribuf->uuid);
2325*5e7646d2SAndroid Build Coastguard Worker 
2326*5e7646d2SAndroid Build Coastguard Worker       DEBUG_printf(("5http_resolve_cb: Found UUID %s, looking for %s.", uuid,
2327*5e7646d2SAndroid Build Coastguard Worker                     uribuf->uuid));
2328*5e7646d2SAndroid Build Coastguard Worker       return;
2329*5e7646d2SAndroid Build Coastguard Worker     }
2330*5e7646d2SAndroid Build Coastguard Worker   }
2331*5e7646d2SAndroid Build Coastguard Worker 
2332*5e7646d2SAndroid Build Coastguard Worker  /*
2333*5e7646d2SAndroid Build Coastguard Worker   * Figure out the scheme from the full name...
2334*5e7646d2SAndroid Build Coastguard Worker   */
2335*5e7646d2SAndroid Build Coastguard Worker 
2336*5e7646d2SAndroid Build Coastguard Worker   if (strstr(fullName, "._ipps") || strstr(fullName, "._ipp-tls"))
2337*5e7646d2SAndroid Build Coastguard Worker     scheme = "ipps";
2338*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(fullName, "._ipp") || strstr(fullName, "._fax-ipp"))
2339*5e7646d2SAndroid Build Coastguard Worker     scheme = "ipp";
2340*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(fullName, "._http."))
2341*5e7646d2SAndroid Build Coastguard Worker     scheme = "http";
2342*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(fullName, "._https."))
2343*5e7646d2SAndroid Build Coastguard Worker     scheme = "https";
2344*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(fullName, "._printer."))
2345*5e7646d2SAndroid Build Coastguard Worker     scheme = "lpd";
2346*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(fullName, "._pdl-datastream."))
2347*5e7646d2SAndroid Build Coastguard Worker     scheme = "socket";
2348*5e7646d2SAndroid Build Coastguard Worker   else
2349*5e7646d2SAndroid Build Coastguard Worker     scheme = "riousbprint";
2350*5e7646d2SAndroid Build Coastguard Worker 
2351*5e7646d2SAndroid Build Coastguard Worker  /*
2352*5e7646d2SAndroid Build Coastguard Worker   * Extract the "remote printer" key from the TXT record...
2353*5e7646d2SAndroid Build Coastguard Worker   */
2354*5e7646d2SAndroid Build Coastguard Worker 
2355*5e7646d2SAndroid Build Coastguard Worker   if ((uribuf->options & _HTTP_RESOLVE_FAXOUT) &&
2356*5e7646d2SAndroid Build Coastguard Worker       (!strcmp(scheme, "ipp") || !strcmp(scheme, "ipps")) &&
2357*5e7646d2SAndroid Build Coastguard Worker       !TXTRecordGetValuePtr(txtLen, txtRecord, "printer-type", &valueLen))
2358*5e7646d2SAndroid Build Coastguard Worker   {
2359*5e7646d2SAndroid Build Coastguard Worker     reskey     = "rfo";
2360*5e7646d2SAndroid Build Coastguard Worker     resdefault = "/ipp/faxout";
2361*5e7646d2SAndroid Build Coastguard Worker   }
2362*5e7646d2SAndroid Build Coastguard Worker   else
2363*5e7646d2SAndroid Build Coastguard Worker   {
2364*5e7646d2SAndroid Build Coastguard Worker     reskey     = "rp";
2365*5e7646d2SAndroid Build Coastguard Worker     resdefault = "/";
2366*5e7646d2SAndroid Build Coastguard Worker   }
2367*5e7646d2SAndroid Build Coastguard Worker 
2368*5e7646d2SAndroid Build Coastguard Worker   if ((value = TXTRecordGetValuePtr(txtLen, txtRecord, reskey,
2369*5e7646d2SAndroid Build Coastguard Worker                                     &valueLen)) != NULL)
2370*5e7646d2SAndroid Build Coastguard Worker   {
2371*5e7646d2SAndroid Build Coastguard Worker     if (((char *)value)[0] == '/')
2372*5e7646d2SAndroid Build Coastguard Worker     {
2373*5e7646d2SAndroid Build Coastguard Worker      /*
2374*5e7646d2SAndroid Build Coastguard Worker       * Value (incorrectly) has a leading slash already...
2375*5e7646d2SAndroid Build Coastguard Worker       */
2376*5e7646d2SAndroid Build Coastguard Worker 
2377*5e7646d2SAndroid Build Coastguard Worker       memcpy(resource, value, valueLen);
2378*5e7646d2SAndroid Build Coastguard Worker       resource[valueLen] = '\0';
2379*5e7646d2SAndroid Build Coastguard Worker     }
2380*5e7646d2SAndroid Build Coastguard Worker     else
2381*5e7646d2SAndroid Build Coastguard Worker     {
2382*5e7646d2SAndroid Build Coastguard Worker      /*
2383*5e7646d2SAndroid Build Coastguard Worker       * Convert to resource by concatenating with a leading "/"...
2384*5e7646d2SAndroid Build Coastguard Worker       */
2385*5e7646d2SAndroid Build Coastguard Worker 
2386*5e7646d2SAndroid Build Coastguard Worker       resource[0] = '/';
2387*5e7646d2SAndroid Build Coastguard Worker       memcpy(resource + 1, value, valueLen);
2388*5e7646d2SAndroid Build Coastguard Worker       resource[valueLen + 1] = '\0';
2389*5e7646d2SAndroid Build Coastguard Worker     }
2390*5e7646d2SAndroid Build Coastguard Worker   }
2391*5e7646d2SAndroid Build Coastguard Worker   else
2392*5e7646d2SAndroid Build Coastguard Worker   {
2393*5e7646d2SAndroid Build Coastguard Worker    /*
2394*5e7646d2SAndroid Build Coastguard Worker     * Use the default value...
2395*5e7646d2SAndroid Build Coastguard Worker     */
2396*5e7646d2SAndroid Build Coastguard Worker 
2397*5e7646d2SAndroid Build Coastguard Worker     strlcpy(resource, resdefault, sizeof(resource));
2398*5e7646d2SAndroid Build Coastguard Worker   }
2399*5e7646d2SAndroid Build Coastguard Worker 
2400*5e7646d2SAndroid Build Coastguard Worker  /*
2401*5e7646d2SAndroid Build Coastguard Worker   * Lookup the FQDN if needed...
2402*5e7646d2SAndroid Build Coastguard Worker   */
2403*5e7646d2SAndroid Build Coastguard Worker 
2404*5e7646d2SAndroid Build Coastguard Worker   if ((uribuf->options & _HTTP_RESOLVE_FQDN) &&
2405*5e7646d2SAndroid Build Coastguard Worker       (hostptr = hostTarget + strlen(hostTarget) - 7) > hostTarget &&
2406*5e7646d2SAndroid Build Coastguard Worker       !_cups_strcasecmp(hostptr, ".local."))
2407*5e7646d2SAndroid Build Coastguard Worker   {
2408*5e7646d2SAndroid Build Coastguard Worker    /*
2409*5e7646d2SAndroid Build Coastguard Worker     * OK, we got a .local name but the caller needs a real domain.  Start by
2410*5e7646d2SAndroid Build Coastguard Worker     * getting the IP address of the .local name and then do reverse-lookups...
2411*5e7646d2SAndroid Build Coastguard Worker     */
2412*5e7646d2SAndroid Build Coastguard Worker 
2413*5e7646d2SAndroid Build Coastguard Worker     http_addrlist_t	*addrlist,	/* List of addresses */
2414*5e7646d2SAndroid Build Coastguard Worker 			*addr;		/* Current address */
2415*5e7646d2SAndroid Build Coastguard Worker 
2416*5e7646d2SAndroid Build Coastguard Worker     DEBUG_printf(("5http_resolve_cb: Looking up \"%s\".", hostTarget));
2417*5e7646d2SAndroid Build Coastguard Worker 
2418*5e7646d2SAndroid Build Coastguard Worker     snprintf(fqdn, sizeof(fqdn), "%d", ntohs(port));
2419*5e7646d2SAndroid Build Coastguard Worker     if ((addrlist = httpAddrGetList(hostTarget, AF_UNSPEC, fqdn)) != NULL)
2420*5e7646d2SAndroid Build Coastguard Worker     {
2421*5e7646d2SAndroid Build Coastguard Worker       for (addr = addrlist; addr; addr = addr->next)
2422*5e7646d2SAndroid Build Coastguard Worker       {
2423*5e7646d2SAndroid Build Coastguard Worker         int error = getnameinfo(&(addr->addr.addr), (socklen_t)httpAddrLength(&(addr->addr)), fqdn, sizeof(fqdn), NULL, 0, NI_NAMEREQD);
2424*5e7646d2SAndroid Build Coastguard Worker 
2425*5e7646d2SAndroid Build Coastguard Worker         if (!error)
2426*5e7646d2SAndroid Build Coastguard Worker 	{
2427*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_printf(("5http_resolve_cb: Found \"%s\".", fqdn));
2428*5e7646d2SAndroid Build Coastguard Worker 
2429*5e7646d2SAndroid Build Coastguard Worker 	  if ((hostptr = fqdn + strlen(fqdn) - 6) <= fqdn ||
2430*5e7646d2SAndroid Build Coastguard Worker 	      _cups_strcasecmp(hostptr, ".local"))
2431*5e7646d2SAndroid Build Coastguard Worker 	  {
2432*5e7646d2SAndroid Build Coastguard Worker 	    hostTarget = fqdn;
2433*5e7646d2SAndroid Build Coastguard Worker 	    break;
2434*5e7646d2SAndroid Build Coastguard Worker 	  }
2435*5e7646d2SAndroid Build Coastguard Worker 	}
2436*5e7646d2SAndroid Build Coastguard Worker #ifdef DEBUG
2437*5e7646d2SAndroid Build Coastguard Worker 	else
2438*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_printf(("5http_resolve_cb: \"%s\" did not resolve: %d",
2439*5e7646d2SAndroid Build Coastguard Worker 	                httpAddrString(&(addr->addr), fqdn, sizeof(fqdn)),
2440*5e7646d2SAndroid Build Coastguard Worker 			error));
2441*5e7646d2SAndroid Build Coastguard Worker #endif /* DEBUG */
2442*5e7646d2SAndroid Build Coastguard Worker       }
2443*5e7646d2SAndroid Build Coastguard Worker 
2444*5e7646d2SAndroid Build Coastguard Worker       httpAddrFreeList(addrlist);
2445*5e7646d2SAndroid Build Coastguard Worker     }
2446*5e7646d2SAndroid Build Coastguard Worker   }
2447*5e7646d2SAndroid Build Coastguard Worker 
2448*5e7646d2SAndroid Build Coastguard Worker  /*
2449*5e7646d2SAndroid Build Coastguard Worker   * Assemble the final device URI...
2450*5e7646d2SAndroid Build Coastguard Worker   */
2451*5e7646d2SAndroid Build Coastguard Worker 
2452*5e7646d2SAndroid Build Coastguard Worker   if ((!strcmp(scheme, "ipp") || !strcmp(scheme, "ipps")) &&
2453*5e7646d2SAndroid Build Coastguard Worker       !strcmp(uribuf->resource, "/cups"))
2454*5e7646d2SAndroid Build Coastguard Worker     httpAssembleURIf(HTTP_URI_CODING_ALL, uribuf->buffer, (int)uribuf->bufsize, scheme, NULL, hostTarget, ntohs(port), "%s?snmp=false", resource);
2455*5e7646d2SAndroid Build Coastguard Worker   else
2456*5e7646d2SAndroid Build Coastguard Worker     httpAssembleURI(HTTP_URI_CODING_ALL, uribuf->buffer, (int)uribuf->bufsize, scheme, NULL, hostTarget, ntohs(port), resource);
2457*5e7646d2SAndroid Build Coastguard Worker 
2458*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("5http_resolve_cb: Resolved URI is \"%s\"...", uribuf->buffer));
2459*5e7646d2SAndroid Build Coastguard Worker }
2460*5e7646d2SAndroid Build Coastguard Worker 
2461*5e7646d2SAndroid Build Coastguard Worker #elif defined(HAVE_AVAHI)
2462*5e7646d2SAndroid Build Coastguard Worker /*
2463*5e7646d2SAndroid Build Coastguard Worker  * 'http_poll_cb()' - Wait for input on the specified file descriptors.
2464*5e7646d2SAndroid Build Coastguard Worker  *
2465*5e7646d2SAndroid Build Coastguard Worker  * Note: This function is needed because avahi_simple_poll_iterate is broken
2466*5e7646d2SAndroid Build Coastguard Worker  *       and always uses a timeout of 0 (!) milliseconds.
2467*5e7646d2SAndroid Build Coastguard Worker  *       (Avahi Ticket #364)
2468*5e7646d2SAndroid Build Coastguard Worker  *
2469*5e7646d2SAndroid Build Coastguard Worker  * @private@
2470*5e7646d2SAndroid Build Coastguard Worker  */
2471*5e7646d2SAndroid Build Coastguard Worker 
2472*5e7646d2SAndroid Build Coastguard Worker static int				/* O - Number of file descriptors matching */
http_poll_cb(struct pollfd * pollfds,unsigned int num_pollfds,int timeout,void * context)2473*5e7646d2SAndroid Build Coastguard Worker http_poll_cb(
2474*5e7646d2SAndroid Build Coastguard Worker     struct pollfd *pollfds,		/* I - File descriptors */
2475*5e7646d2SAndroid Build Coastguard Worker     unsigned int  num_pollfds,		/* I - Number of file descriptors */
2476*5e7646d2SAndroid Build Coastguard Worker     int           timeout,		/* I - Timeout in milliseconds (used) */
2477*5e7646d2SAndroid Build Coastguard Worker     void          *context)		/* I - User data (unused) */
2478*5e7646d2SAndroid Build Coastguard Worker {
2479*5e7646d2SAndroid Build Coastguard Worker   (void)timeout;
2480*5e7646d2SAndroid Build Coastguard Worker   (void)context;
2481*5e7646d2SAndroid Build Coastguard Worker 
2482*5e7646d2SAndroid Build Coastguard Worker   return (poll(pollfds, num_pollfds, 2000));
2483*5e7646d2SAndroid Build Coastguard Worker }
2484*5e7646d2SAndroid Build Coastguard Worker 
2485*5e7646d2SAndroid Build Coastguard Worker 
2486*5e7646d2SAndroid Build Coastguard Worker /*
2487*5e7646d2SAndroid Build Coastguard Worker  * 'http_resolve_cb()' - Build a device URI for the given service name.
2488*5e7646d2SAndroid Build Coastguard Worker  */
2489*5e7646d2SAndroid Build Coastguard Worker 
2490*5e7646d2SAndroid Build Coastguard Worker static void
http_resolve_cb(AvahiServiceResolver * resolver,AvahiIfIndex interface,AvahiProtocol protocol,AvahiResolverEvent event,const char * name,const char * type,const char * domain,const char * hostTarget,const AvahiAddress * address,uint16_t port,AvahiStringList * txt,AvahiLookupResultFlags flags,void * context)2491*5e7646d2SAndroid Build Coastguard Worker http_resolve_cb(
2492*5e7646d2SAndroid Build Coastguard Worker     AvahiServiceResolver   *resolver,	/* I - Resolver (unused) */
2493*5e7646d2SAndroid Build Coastguard Worker     AvahiIfIndex           interface,	/* I - Interface index (unused) */
2494*5e7646d2SAndroid Build Coastguard Worker     AvahiProtocol          protocol,	/* I - Network protocol (unused) */
2495*5e7646d2SAndroid Build Coastguard Worker     AvahiResolverEvent     event,	/* I - Event (found, etc.) */
2496*5e7646d2SAndroid Build Coastguard Worker     const char             *name,	/* I - Service name */
2497*5e7646d2SAndroid Build Coastguard Worker     const char             *type,	/* I - Registration type */
2498*5e7646d2SAndroid Build Coastguard Worker     const char             *domain,	/* I - Domain (unused) */
2499*5e7646d2SAndroid Build Coastguard Worker     const char             *hostTarget,	/* I - Hostname */
2500*5e7646d2SAndroid Build Coastguard Worker     const AvahiAddress     *address,	/* I - Address (unused) */
2501*5e7646d2SAndroid Build Coastguard Worker     uint16_t               port,	/* I - Port number */
2502*5e7646d2SAndroid Build Coastguard Worker     AvahiStringList        *txt,	/* I - TXT record */
2503*5e7646d2SAndroid Build Coastguard Worker     AvahiLookupResultFlags flags,	/* I - Lookup flags (unused) */
2504*5e7646d2SAndroid Build Coastguard Worker     void                   *context)	/* I - Pointer to URI buffer */
2505*5e7646d2SAndroid Build Coastguard Worker {
2506*5e7646d2SAndroid Build Coastguard Worker   _http_uribuf_t	*uribuf = (_http_uribuf_t *)context;
2507*5e7646d2SAndroid Build Coastguard Worker 					/* URI buffer */
2508*5e7646d2SAndroid Build Coastguard Worker   const char		*scheme,	/* URI scheme */
2509*5e7646d2SAndroid Build Coastguard Worker 			*hostptr,	/* Pointer into hostTarget */
2510*5e7646d2SAndroid Build Coastguard Worker 			*reskey,	/* "rp" or "rfo" */
2511*5e7646d2SAndroid Build Coastguard Worker 			*resdefault;	/* Default path */
2512*5e7646d2SAndroid Build Coastguard Worker   char			resource[257],	/* Remote path */
2513*5e7646d2SAndroid Build Coastguard Worker 			fqdn[256];	/* FQDN of the .local name */
2514*5e7646d2SAndroid Build Coastguard Worker   AvahiStringList	*pair;		/* Current TXT record key/value pair */
2515*5e7646d2SAndroid Build Coastguard Worker   char			*value;		/* Value for "rp" key */
2516*5e7646d2SAndroid Build Coastguard Worker   size_t		valueLen = 0;	/* Length of "rp" key */
2517*5e7646d2SAndroid Build Coastguard Worker 
2518*5e7646d2SAndroid Build Coastguard Worker 
2519*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4http_resolve_cb(resolver=%p, "
2520*5e7646d2SAndroid Build Coastguard Worker 		"interface=%d, protocol=%d, event=%d, name=\"%s\", "
2521*5e7646d2SAndroid Build Coastguard Worker 		"type=\"%s\", domain=\"%s\", hostTarget=\"%s\", address=%p, "
2522*5e7646d2SAndroid Build Coastguard Worker 		"port=%d, txt=%p, flags=%d, context=%p)",
2523*5e7646d2SAndroid Build Coastguard Worker 		resolver, interface, protocol, event, name, type, domain,
2524*5e7646d2SAndroid Build Coastguard Worker 		hostTarget, address, port, txt, flags, context));
2525*5e7646d2SAndroid Build Coastguard Worker 
2526*5e7646d2SAndroid Build Coastguard Worker   if (event != AVAHI_RESOLVER_FOUND)
2527*5e7646d2SAndroid Build Coastguard Worker   {
2528*5e7646d2SAndroid Build Coastguard Worker     avahi_service_resolver_free(resolver);
2529*5e7646d2SAndroid Build Coastguard Worker     avahi_simple_poll_quit(uribuf->poll);
2530*5e7646d2SAndroid Build Coastguard Worker     return;
2531*5e7646d2SAndroid Build Coastguard Worker   }
2532*5e7646d2SAndroid Build Coastguard Worker 
2533*5e7646d2SAndroid Build Coastguard Worker  /*
2534*5e7646d2SAndroid Build Coastguard Worker   * If we have a UUID, compare it...
2535*5e7646d2SAndroid Build Coastguard Worker   */
2536*5e7646d2SAndroid Build Coastguard Worker 
2537*5e7646d2SAndroid Build Coastguard Worker   if (uribuf->uuid && (pair = avahi_string_list_find(txt, "UUID")) != NULL)
2538*5e7646d2SAndroid Build Coastguard Worker   {
2539*5e7646d2SAndroid Build Coastguard Worker     char	uuid[256];		/* UUID value */
2540*5e7646d2SAndroid Build Coastguard Worker 
2541*5e7646d2SAndroid Build Coastguard Worker     avahi_string_list_get_pair(pair, NULL, &value, &valueLen);
2542*5e7646d2SAndroid Build Coastguard Worker 
2543*5e7646d2SAndroid Build Coastguard Worker     memcpy(uuid, value, valueLen);
2544*5e7646d2SAndroid Build Coastguard Worker     uuid[valueLen] = '\0';
2545*5e7646d2SAndroid Build Coastguard Worker 
2546*5e7646d2SAndroid Build Coastguard Worker     if (_cups_strcasecmp(uuid, uribuf->uuid))
2547*5e7646d2SAndroid Build Coastguard Worker     {
2548*5e7646d2SAndroid Build Coastguard Worker       if (uribuf->options & _HTTP_RESOLVE_STDERR)
2549*5e7646d2SAndroid Build Coastguard Worker 	fprintf(stderr, "DEBUG: Found UUID %s, looking for %s.", uuid,
2550*5e7646d2SAndroid Build Coastguard Worker 		uribuf->uuid);
2551*5e7646d2SAndroid Build Coastguard Worker 
2552*5e7646d2SAndroid Build Coastguard Worker       DEBUG_printf(("5http_resolve_cb: Found UUID %s, looking for %s.", uuid,
2553*5e7646d2SAndroid Build Coastguard Worker                     uribuf->uuid));
2554*5e7646d2SAndroid Build Coastguard Worker       return;
2555*5e7646d2SAndroid Build Coastguard Worker     }
2556*5e7646d2SAndroid Build Coastguard Worker   }
2557*5e7646d2SAndroid Build Coastguard Worker 
2558*5e7646d2SAndroid Build Coastguard Worker  /*
2559*5e7646d2SAndroid Build Coastguard Worker   * Figure out the scheme from the full name...
2560*5e7646d2SAndroid Build Coastguard Worker   */
2561*5e7646d2SAndroid Build Coastguard Worker 
2562*5e7646d2SAndroid Build Coastguard Worker   if (strstr(type, "_ipp."))
2563*5e7646d2SAndroid Build Coastguard Worker     scheme = "ipp";
2564*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(type, "_printer."))
2565*5e7646d2SAndroid Build Coastguard Worker     scheme = "lpd";
2566*5e7646d2SAndroid Build Coastguard Worker   else if (strstr(type, "_pdl-datastream."))
2567*5e7646d2SAndroid Build Coastguard Worker     scheme = "socket";
2568*5e7646d2SAndroid Build Coastguard Worker   else
2569*5e7646d2SAndroid Build Coastguard Worker     scheme = "riousbprint";
2570*5e7646d2SAndroid Build Coastguard Worker 
2571*5e7646d2SAndroid Build Coastguard Worker   if (!strncmp(type, "_ipps.", 6) || !strncmp(type, "_ipp-tls.", 9))
2572*5e7646d2SAndroid Build Coastguard Worker     scheme = "ipps";
2573*5e7646d2SAndroid Build Coastguard Worker   else if (!strncmp(type, "_ipp.", 5) || !strncmp(type, "_fax-ipp.", 9))
2574*5e7646d2SAndroid Build Coastguard Worker     scheme = "ipp";
2575*5e7646d2SAndroid Build Coastguard Worker   else if (!strncmp(type, "_http.", 6))
2576*5e7646d2SAndroid Build Coastguard Worker     scheme = "http";
2577*5e7646d2SAndroid Build Coastguard Worker   else if (!strncmp(type, "_https.", 7))
2578*5e7646d2SAndroid Build Coastguard Worker     scheme = "https";
2579*5e7646d2SAndroid Build Coastguard Worker   else if (!strncmp(type, "_printer.", 9))
2580*5e7646d2SAndroid Build Coastguard Worker     scheme = "lpd";
2581*5e7646d2SAndroid Build Coastguard Worker   else if (!strncmp(type, "_pdl-datastream.", 16))
2582*5e7646d2SAndroid Build Coastguard Worker     scheme = "socket";
2583*5e7646d2SAndroid Build Coastguard Worker   else
2584*5e7646d2SAndroid Build Coastguard Worker   {
2585*5e7646d2SAndroid Build Coastguard Worker     avahi_service_resolver_free(resolver);
2586*5e7646d2SAndroid Build Coastguard Worker     avahi_simple_poll_quit(uribuf->poll);
2587*5e7646d2SAndroid Build Coastguard Worker     return;
2588*5e7646d2SAndroid Build Coastguard Worker   }
2589*5e7646d2SAndroid Build Coastguard Worker 
2590*5e7646d2SAndroid Build Coastguard Worker  /*
2591*5e7646d2SAndroid Build Coastguard Worker   * Extract the remote resource key from the TXT record...
2592*5e7646d2SAndroid Build Coastguard Worker   */
2593*5e7646d2SAndroid Build Coastguard Worker 
2594*5e7646d2SAndroid Build Coastguard Worker   if ((uribuf->options & _HTTP_RESOLVE_FAXOUT) &&
2595*5e7646d2SAndroid Build Coastguard Worker       (!strcmp(scheme, "ipp") || !strcmp(scheme, "ipps")) &&
2596*5e7646d2SAndroid Build Coastguard Worker       !avahi_string_list_find(txt, "printer-type"))
2597*5e7646d2SAndroid Build Coastguard Worker   {
2598*5e7646d2SAndroid Build Coastguard Worker     reskey     = "rfo";
2599*5e7646d2SAndroid Build Coastguard Worker     resdefault = "/ipp/faxout";
2600*5e7646d2SAndroid Build Coastguard Worker   }
2601*5e7646d2SAndroid Build Coastguard Worker   else
2602*5e7646d2SAndroid Build Coastguard Worker   {
2603*5e7646d2SAndroid Build Coastguard Worker     reskey     = "rp";
2604*5e7646d2SAndroid Build Coastguard Worker     resdefault = "/";
2605*5e7646d2SAndroid Build Coastguard Worker   }
2606*5e7646d2SAndroid Build Coastguard Worker 
2607*5e7646d2SAndroid Build Coastguard Worker   if ((pair = avahi_string_list_find(txt, reskey)) != NULL)
2608*5e7646d2SAndroid Build Coastguard Worker   {
2609*5e7646d2SAndroid Build Coastguard Worker     avahi_string_list_get_pair(pair, NULL, &value, &valueLen);
2610*5e7646d2SAndroid Build Coastguard Worker 
2611*5e7646d2SAndroid Build Coastguard Worker     if (value[0] == '/')
2612*5e7646d2SAndroid Build Coastguard Worker     {
2613*5e7646d2SAndroid Build Coastguard Worker      /*
2614*5e7646d2SAndroid Build Coastguard Worker       * Value (incorrectly) has a leading slash already...
2615*5e7646d2SAndroid Build Coastguard Worker       */
2616*5e7646d2SAndroid Build Coastguard Worker 
2617*5e7646d2SAndroid Build Coastguard Worker       memcpy(resource, value, valueLen);
2618*5e7646d2SAndroid Build Coastguard Worker       resource[valueLen] = '\0';
2619*5e7646d2SAndroid Build Coastguard Worker     }
2620*5e7646d2SAndroid Build Coastguard Worker     else
2621*5e7646d2SAndroid Build Coastguard Worker     {
2622*5e7646d2SAndroid Build Coastguard Worker      /*
2623*5e7646d2SAndroid Build Coastguard Worker       * Convert to resource by concatenating with a leading "/"...
2624*5e7646d2SAndroid Build Coastguard Worker       */
2625*5e7646d2SAndroid Build Coastguard Worker 
2626*5e7646d2SAndroid Build Coastguard Worker       resource[0] = '/';
2627*5e7646d2SAndroid Build Coastguard Worker       memcpy(resource + 1, value, valueLen);
2628*5e7646d2SAndroid Build Coastguard Worker       resource[valueLen + 1] = '\0';
2629*5e7646d2SAndroid Build Coastguard Worker     }
2630*5e7646d2SAndroid Build Coastguard Worker   }
2631*5e7646d2SAndroid Build Coastguard Worker   else
2632*5e7646d2SAndroid Build Coastguard Worker   {
2633*5e7646d2SAndroid Build Coastguard Worker    /*
2634*5e7646d2SAndroid Build Coastguard Worker     * Use the default value...
2635*5e7646d2SAndroid Build Coastguard Worker     */
2636*5e7646d2SAndroid Build Coastguard Worker 
2637*5e7646d2SAndroid Build Coastguard Worker     strlcpy(resource, resdefault, sizeof(resource));
2638*5e7646d2SAndroid Build Coastguard Worker   }
2639*5e7646d2SAndroid Build Coastguard Worker 
2640*5e7646d2SAndroid Build Coastguard Worker  /*
2641*5e7646d2SAndroid Build Coastguard Worker   * Lookup the FQDN if needed...
2642*5e7646d2SAndroid Build Coastguard Worker   */
2643*5e7646d2SAndroid Build Coastguard Worker 
2644*5e7646d2SAndroid Build Coastguard Worker   if ((uribuf->options & _HTTP_RESOLVE_FQDN) &&
2645*5e7646d2SAndroid Build Coastguard Worker       (hostptr = hostTarget + strlen(hostTarget) - 6) > hostTarget &&
2646*5e7646d2SAndroid Build Coastguard Worker       !_cups_strcasecmp(hostptr, ".local"))
2647*5e7646d2SAndroid Build Coastguard Worker   {
2648*5e7646d2SAndroid Build Coastguard Worker    /*
2649*5e7646d2SAndroid Build Coastguard Worker     * OK, we got a .local name but the caller needs a real domain.  Start by
2650*5e7646d2SAndroid Build Coastguard Worker     * getting the IP address of the .local name and then do reverse-lookups...
2651*5e7646d2SAndroid Build Coastguard Worker     */
2652*5e7646d2SAndroid Build Coastguard Worker 
2653*5e7646d2SAndroid Build Coastguard Worker     http_addrlist_t	*addrlist,	/* List of addresses */
2654*5e7646d2SAndroid Build Coastguard Worker 			*addr;		/* Current address */
2655*5e7646d2SAndroid Build Coastguard Worker 
2656*5e7646d2SAndroid Build Coastguard Worker     DEBUG_printf(("5http_resolve_cb: Looking up \"%s\".", hostTarget));
2657*5e7646d2SAndroid Build Coastguard Worker 
2658*5e7646d2SAndroid Build Coastguard Worker     snprintf(fqdn, sizeof(fqdn), "%d", ntohs(port));
2659*5e7646d2SAndroid Build Coastguard Worker     if ((addrlist = httpAddrGetList(hostTarget, AF_UNSPEC, fqdn)) != NULL)
2660*5e7646d2SAndroid Build Coastguard Worker     {
2661*5e7646d2SAndroid Build Coastguard Worker       for (addr = addrlist; addr; addr = addr->next)
2662*5e7646d2SAndroid Build Coastguard Worker       {
2663*5e7646d2SAndroid Build Coastguard Worker         int error = getnameinfo(&(addr->addr.addr), (socklen_t)httpAddrLength(&(addr->addr)), fqdn, sizeof(fqdn), NULL, 0, NI_NAMEREQD);
2664*5e7646d2SAndroid Build Coastguard Worker 
2665*5e7646d2SAndroid Build Coastguard Worker         if (!error)
2666*5e7646d2SAndroid Build Coastguard Worker 	{
2667*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_printf(("5http_resolve_cb: Found \"%s\".", fqdn));
2668*5e7646d2SAndroid Build Coastguard Worker 
2669*5e7646d2SAndroid Build Coastguard Worker 	  if ((hostptr = fqdn + strlen(fqdn) - 6) <= fqdn ||
2670*5e7646d2SAndroid Build Coastguard Worker 	      _cups_strcasecmp(hostptr, ".local"))
2671*5e7646d2SAndroid Build Coastguard Worker 	  {
2672*5e7646d2SAndroid Build Coastguard Worker 	    hostTarget = fqdn;
2673*5e7646d2SAndroid Build Coastguard Worker 	    break;
2674*5e7646d2SAndroid Build Coastguard Worker 	  }
2675*5e7646d2SAndroid Build Coastguard Worker 	}
2676*5e7646d2SAndroid Build Coastguard Worker #ifdef DEBUG
2677*5e7646d2SAndroid Build Coastguard Worker 	else
2678*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_printf(("5http_resolve_cb: \"%s\" did not resolve: %d",
2679*5e7646d2SAndroid Build Coastguard Worker 	                httpAddrString(&(addr->addr), fqdn, sizeof(fqdn)),
2680*5e7646d2SAndroid Build Coastguard Worker 			error));
2681*5e7646d2SAndroid Build Coastguard Worker #endif /* DEBUG */
2682*5e7646d2SAndroid Build Coastguard Worker       }
2683*5e7646d2SAndroid Build Coastguard Worker 
2684*5e7646d2SAndroid Build Coastguard Worker       httpAddrFreeList(addrlist);
2685*5e7646d2SAndroid Build Coastguard Worker     }
2686*5e7646d2SAndroid Build Coastguard Worker   }
2687*5e7646d2SAndroid Build Coastguard Worker 
2688*5e7646d2SAndroid Build Coastguard Worker  /*
2689*5e7646d2SAndroid Build Coastguard Worker   * Assemble the final device URI using the resolved hostname...
2690*5e7646d2SAndroid Build Coastguard Worker   */
2691*5e7646d2SAndroid Build Coastguard Worker 
2692*5e7646d2SAndroid Build Coastguard Worker   httpAssembleURI(HTTP_URI_CODING_ALL, uribuf->buffer, (int)uribuf->bufsize, scheme,
2693*5e7646d2SAndroid Build Coastguard Worker                   NULL, hostTarget, port, resource);
2694*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("5http_resolve_cb: Resolved URI is \"%s\".", uribuf->buffer));
2695*5e7646d2SAndroid Build Coastguard Worker 
2696*5e7646d2SAndroid Build Coastguard Worker   avahi_simple_poll_quit(uribuf->poll);
2697*5e7646d2SAndroid Build Coastguard Worker }
2698*5e7646d2SAndroid Build Coastguard Worker #endif /* HAVE_DNSSD */
2699