xref: /aosp_15_r20/external/libcups/cups/ppd-util.c (revision 5e7646d21f1134fb0638875d812ef646c12ab91e)
1*5e7646d2SAndroid Build Coastguard Worker /*
2*5e7646d2SAndroid Build Coastguard Worker  * PPD utilities for CUPS.
3*5e7646d2SAndroid Build Coastguard Worker  *
4*5e7646d2SAndroid Build Coastguard Worker  * Copyright © 2007-2018 by Apple Inc.
5*5e7646d2SAndroid Build Coastguard Worker  * Copyright © 1997-2006 by Easy Software Products.
6*5e7646d2SAndroid Build Coastguard Worker  *
7*5e7646d2SAndroid Build Coastguard Worker  * Licensed under Apache License v2.0.  See the file "LICENSE" for more
8*5e7646d2SAndroid Build Coastguard Worker  * information.
9*5e7646d2SAndroid Build Coastguard Worker  */
10*5e7646d2SAndroid Build Coastguard Worker 
11*5e7646d2SAndroid Build Coastguard Worker /*
12*5e7646d2SAndroid Build Coastguard Worker  * Include necessary headers...
13*5e7646d2SAndroid Build Coastguard Worker  */
14*5e7646d2SAndroid Build Coastguard Worker 
15*5e7646d2SAndroid Build Coastguard Worker #include "cups-private.h"
16*5e7646d2SAndroid Build Coastguard Worker #include "ppd-private.h"
17*5e7646d2SAndroid Build Coastguard Worker #include "debug-internal.h"
18*5e7646d2SAndroid Build Coastguard Worker #include <fcntl.h>
19*5e7646d2SAndroid Build Coastguard Worker #include <sys/stat.h>
20*5e7646d2SAndroid Build Coastguard Worker #if defined(_WIN32) || defined(__EMX__)
21*5e7646d2SAndroid Build Coastguard Worker #  include <io.h>
22*5e7646d2SAndroid Build Coastguard Worker #else
23*5e7646d2SAndroid Build Coastguard Worker #  include <unistd.h>
24*5e7646d2SAndroid Build Coastguard Worker #endif /* _WIN32 || __EMX__ */
25*5e7646d2SAndroid Build Coastguard Worker 
26*5e7646d2SAndroid Build Coastguard Worker 
27*5e7646d2SAndroid Build Coastguard Worker /*
28*5e7646d2SAndroid Build Coastguard Worker  * Local functions...
29*5e7646d2SAndroid Build Coastguard Worker  */
30*5e7646d2SAndroid Build Coastguard Worker 
31*5e7646d2SAndroid Build Coastguard Worker static int	cups_get_printer_uri(http_t *http, const char *name,
32*5e7646d2SAndroid Build Coastguard Worker 		                     char *host, int hostsize, int *port,
33*5e7646d2SAndroid Build Coastguard Worker 				     char *resource, int resourcesize,
34*5e7646d2SAndroid Build Coastguard Worker 				     int depth);
35*5e7646d2SAndroid Build Coastguard Worker 
36*5e7646d2SAndroid Build Coastguard Worker 
37*5e7646d2SAndroid Build Coastguard Worker /*
38*5e7646d2SAndroid Build Coastguard Worker  * 'cupsGetPPD()' - Get the PPD file for a printer on the default server.
39*5e7646d2SAndroid Build Coastguard Worker  *
40*5e7646d2SAndroid Build Coastguard Worker  * For classes, @code cupsGetPPD@ returns the PPD file for the first printer
41*5e7646d2SAndroid Build Coastguard Worker  * in the class.
42*5e7646d2SAndroid Build Coastguard Worker  *
43*5e7646d2SAndroid Build Coastguard Worker  * The returned filename is stored in a static buffer and is overwritten with
44*5e7646d2SAndroid Build Coastguard Worker  * each call to @code cupsGetPPD@ or @link cupsGetPPD2@.  The caller "owns" the
45*5e7646d2SAndroid Build Coastguard Worker  * file that is created and must @code unlink@ the returned filename.
46*5e7646d2SAndroid Build Coastguard Worker  */
47*5e7646d2SAndroid Build Coastguard Worker 
48*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Filename for PPD file */
cupsGetPPD(const char * name)49*5e7646d2SAndroid Build Coastguard Worker cupsGetPPD(const char *name)		/* I - Destination name */
50*5e7646d2SAndroid Build Coastguard Worker {
51*5e7646d2SAndroid Build Coastguard Worker   _ppd_globals_t *pg = _ppdGlobals();	/* Pointer to library globals */
52*5e7646d2SAndroid Build Coastguard Worker   time_t	modtime = 0;		/* Modification time */
53*5e7646d2SAndroid Build Coastguard Worker 
54*5e7646d2SAndroid Build Coastguard Worker 
55*5e7646d2SAndroid Build Coastguard Worker  /*
56*5e7646d2SAndroid Build Coastguard Worker   * Return the PPD file...
57*5e7646d2SAndroid Build Coastguard Worker   */
58*5e7646d2SAndroid Build Coastguard Worker 
59*5e7646d2SAndroid Build Coastguard Worker   pg->ppd_filename[0] = '\0';
60*5e7646d2SAndroid Build Coastguard Worker 
61*5e7646d2SAndroid Build Coastguard Worker   if (cupsGetPPD3(CUPS_HTTP_DEFAULT, name, &modtime, pg->ppd_filename,
62*5e7646d2SAndroid Build Coastguard Worker                   sizeof(pg->ppd_filename)) == HTTP_STATUS_OK)
63*5e7646d2SAndroid Build Coastguard Worker     return (pg->ppd_filename);
64*5e7646d2SAndroid Build Coastguard Worker   else
65*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
66*5e7646d2SAndroid Build Coastguard Worker }
67*5e7646d2SAndroid Build Coastguard Worker 
68*5e7646d2SAndroid Build Coastguard Worker 
69*5e7646d2SAndroid Build Coastguard Worker /*
70*5e7646d2SAndroid Build Coastguard Worker  * 'cupsGetPPD2()' - Get the PPD file for a printer from the specified server.
71*5e7646d2SAndroid Build Coastguard Worker  *
72*5e7646d2SAndroid Build Coastguard Worker  * For classes, @code cupsGetPPD2@ returns the PPD file for the first printer
73*5e7646d2SAndroid Build Coastguard Worker  * in the class.
74*5e7646d2SAndroid Build Coastguard Worker  *
75*5e7646d2SAndroid Build Coastguard Worker  * The returned filename is stored in a static buffer and is overwritten with
76*5e7646d2SAndroid Build Coastguard Worker  * each call to @link cupsGetPPD@ or @code cupsGetPPD2@.  The caller "owns" the
77*5e7646d2SAndroid Build Coastguard Worker  * file that is created and must @code unlink@ the returned filename.
78*5e7646d2SAndroid Build Coastguard Worker  *
79*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.1.21/macOS 10.4@
80*5e7646d2SAndroid Build Coastguard Worker  */
81*5e7646d2SAndroid Build Coastguard Worker 
82*5e7646d2SAndroid Build Coastguard Worker const char *				/* O - Filename for PPD file */
cupsGetPPD2(http_t * http,const char * name)83*5e7646d2SAndroid Build Coastguard Worker cupsGetPPD2(http_t     *http,		/* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */
84*5e7646d2SAndroid Build Coastguard Worker             const char *name)		/* I - Destination name */
85*5e7646d2SAndroid Build Coastguard Worker {
86*5e7646d2SAndroid Build Coastguard Worker   _ppd_globals_t *pg = _ppdGlobals();	/* Pointer to library globals */
87*5e7646d2SAndroid Build Coastguard Worker   time_t	modtime = 0;		/* Modification time */
88*5e7646d2SAndroid Build Coastguard Worker 
89*5e7646d2SAndroid Build Coastguard Worker 
90*5e7646d2SAndroid Build Coastguard Worker   pg->ppd_filename[0] = '\0';
91*5e7646d2SAndroid Build Coastguard Worker 
92*5e7646d2SAndroid Build Coastguard Worker   if (cupsGetPPD3(http, name, &modtime, pg->ppd_filename,
93*5e7646d2SAndroid Build Coastguard Worker                   sizeof(pg->ppd_filename)) == HTTP_STATUS_OK)
94*5e7646d2SAndroid Build Coastguard Worker     return (pg->ppd_filename);
95*5e7646d2SAndroid Build Coastguard Worker   else
96*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
97*5e7646d2SAndroid Build Coastguard Worker }
98*5e7646d2SAndroid Build Coastguard Worker 
99*5e7646d2SAndroid Build Coastguard Worker 
100*5e7646d2SAndroid Build Coastguard Worker /*
101*5e7646d2SAndroid Build Coastguard Worker  * 'cupsGetPPD3()' - Get the PPD file for a printer on the specified
102*5e7646d2SAndroid Build Coastguard Worker  *                   server if it has changed.
103*5e7646d2SAndroid Build Coastguard Worker  *
104*5e7646d2SAndroid Build Coastguard Worker  * The "modtime" parameter contains the modification time of any
105*5e7646d2SAndroid Build Coastguard Worker  * locally-cached content and is updated with the time from the PPD file on
106*5e7646d2SAndroid Build Coastguard Worker  * the server.
107*5e7646d2SAndroid Build Coastguard Worker  *
108*5e7646d2SAndroid Build Coastguard Worker  * The "buffer" parameter contains the local PPD filename.  If it contains
109*5e7646d2SAndroid Build Coastguard Worker  * the empty string, a new temporary file is created, otherwise the existing
110*5e7646d2SAndroid Build Coastguard Worker  * file will be overwritten as needed.  The caller "owns" the file that is
111*5e7646d2SAndroid Build Coastguard Worker  * created and must @code unlink@ the returned filename.
112*5e7646d2SAndroid Build Coastguard Worker  *
113*5e7646d2SAndroid Build Coastguard Worker  * On success, @code HTTP_STATUS_OK@ is returned for a new PPD file and
114*5e7646d2SAndroid Build Coastguard Worker  * @code HTTP_STATUS_NOT_MODIFIED@ if the existing PPD file is up-to-date.  Any other
115*5e7646d2SAndroid Build Coastguard Worker  * status is an error.
116*5e7646d2SAndroid Build Coastguard Worker  *
117*5e7646d2SAndroid Build Coastguard Worker  * For classes, @code cupsGetPPD3@ returns the PPD file for the first printer
118*5e7646d2SAndroid Build Coastguard Worker  * in the class.
119*5e7646d2SAndroid Build Coastguard Worker  *
120*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.4/macOS 10.6@
121*5e7646d2SAndroid Build Coastguard Worker  */
122*5e7646d2SAndroid Build Coastguard Worker 
123*5e7646d2SAndroid Build Coastguard Worker http_status_t				/* O  - HTTP status */
cupsGetPPD3(http_t * http,const char * name,time_t * modtime,char * buffer,size_t bufsize)124*5e7646d2SAndroid Build Coastguard Worker cupsGetPPD3(http_t     *http,		/* I  - HTTP connection or @code CUPS_HTTP_DEFAULT@ */
125*5e7646d2SAndroid Build Coastguard Worker             const char *name,		/* I  - Destination name */
126*5e7646d2SAndroid Build Coastguard Worker 	    time_t     *modtime,	/* IO - Modification time */
127*5e7646d2SAndroid Build Coastguard Worker 	    char       *buffer,		/* I  - Filename buffer */
128*5e7646d2SAndroid Build Coastguard Worker 	    size_t     bufsize)		/* I  - Size of filename buffer */
129*5e7646d2SAndroid Build Coastguard Worker {
130*5e7646d2SAndroid Build Coastguard Worker   int		http_port;		/* Port number */
131*5e7646d2SAndroid Build Coastguard Worker   char		http_hostname[HTTP_MAX_HOST];
132*5e7646d2SAndroid Build Coastguard Worker 					/* Hostname associated with connection */
133*5e7646d2SAndroid Build Coastguard Worker   http_t	*http2;			/* Alternate HTTP connection */
134*5e7646d2SAndroid Build Coastguard Worker   int		fd;			/* PPD file */
135*5e7646d2SAndroid Build Coastguard Worker   char		localhost[HTTP_MAX_URI],/* Local hostname */
136*5e7646d2SAndroid Build Coastguard Worker 		hostname[HTTP_MAX_URI],	/* Hostname */
137*5e7646d2SAndroid Build Coastguard Worker 		resource[HTTP_MAX_URI];	/* Resource name */
138*5e7646d2SAndroid Build Coastguard Worker   int		port;			/* Port number */
139*5e7646d2SAndroid Build Coastguard Worker   http_status_t	status;			/* HTTP status from server */
140*5e7646d2SAndroid Build Coastguard Worker   char		tempfile[1024] = "";	/* Temporary filename */
141*5e7646d2SAndroid Build Coastguard Worker   _cups_globals_t *cg = _cupsGlobals();	/* Pointer to library globals */
142*5e7646d2SAndroid Build Coastguard Worker 
143*5e7646d2SAndroid Build Coastguard Worker 
144*5e7646d2SAndroid Build Coastguard Worker  /*
145*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
146*5e7646d2SAndroid Build Coastguard Worker   */
147*5e7646d2SAndroid Build Coastguard Worker 
148*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("cupsGetPPD3(http=%p, name=\"%s\", modtime=%p(%d), buffer=%p, "
149*5e7646d2SAndroid Build Coastguard Worker                 "bufsize=%d)", http, name, modtime,
150*5e7646d2SAndroid Build Coastguard Worker 		modtime ? (int)*modtime : 0, buffer, (int)bufsize));
151*5e7646d2SAndroid Build Coastguard Worker 
152*5e7646d2SAndroid Build Coastguard Worker   if (!name)
153*5e7646d2SAndroid Build Coastguard Worker   {
154*5e7646d2SAndroid Build Coastguard Worker     DEBUG_puts("2cupsGetPPD3: No printer name, returning NULL.");
155*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("No printer name"), 1);
156*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_STATUS_NOT_ACCEPTABLE);
157*5e7646d2SAndroid Build Coastguard Worker   }
158*5e7646d2SAndroid Build Coastguard Worker 
159*5e7646d2SAndroid Build Coastguard Worker   if (!modtime)
160*5e7646d2SAndroid Build Coastguard Worker   {
161*5e7646d2SAndroid Build Coastguard Worker     DEBUG_puts("2cupsGetPPD3: No modtime, returning NULL.");
162*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("No modification time"), 1);
163*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_STATUS_NOT_ACCEPTABLE);
164*5e7646d2SAndroid Build Coastguard Worker   }
165*5e7646d2SAndroid Build Coastguard Worker 
166*5e7646d2SAndroid Build Coastguard Worker   if (!buffer || bufsize <= 1)
167*5e7646d2SAndroid Build Coastguard Worker   {
168*5e7646d2SAndroid Build Coastguard Worker     DEBUG_puts("2cupsGetPPD3: No filename buffer, returning NULL.");
169*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("Bad filename buffer"), 1);
170*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_STATUS_NOT_ACCEPTABLE);
171*5e7646d2SAndroid Build Coastguard Worker   }
172*5e7646d2SAndroid Build Coastguard Worker 
173*5e7646d2SAndroid Build Coastguard Worker #ifndef _WIN32
174*5e7646d2SAndroid Build Coastguard Worker  /*
175*5e7646d2SAndroid Build Coastguard Worker   * See if the PPD file is available locally...
176*5e7646d2SAndroid Build Coastguard Worker   */
177*5e7646d2SAndroid Build Coastguard Worker 
178*5e7646d2SAndroid Build Coastguard Worker   if (http)
179*5e7646d2SAndroid Build Coastguard Worker     httpGetHostname(http, hostname, sizeof(hostname));
180*5e7646d2SAndroid Build Coastguard Worker   else
181*5e7646d2SAndroid Build Coastguard Worker   {
182*5e7646d2SAndroid Build Coastguard Worker     strlcpy(hostname, cupsServer(), sizeof(hostname));
183*5e7646d2SAndroid Build Coastguard Worker     if (hostname[0] == '/')
184*5e7646d2SAndroid Build Coastguard Worker       strlcpy(hostname, "localhost", sizeof(hostname));
185*5e7646d2SAndroid Build Coastguard Worker   }
186*5e7646d2SAndroid Build Coastguard Worker 
187*5e7646d2SAndroid Build Coastguard Worker   if (!_cups_strcasecmp(hostname, "localhost"))
188*5e7646d2SAndroid Build Coastguard Worker   {
189*5e7646d2SAndroid Build Coastguard Worker     char	ppdname[1024];		/* PPD filename */
190*5e7646d2SAndroid Build Coastguard Worker     struct stat	ppdinfo;		/* PPD file information */
191*5e7646d2SAndroid Build Coastguard Worker 
192*5e7646d2SAndroid Build Coastguard Worker 
193*5e7646d2SAndroid Build Coastguard Worker     snprintf(ppdname, sizeof(ppdname), "%s/ppd/%s.ppd", cg->cups_serverroot,
194*5e7646d2SAndroid Build Coastguard Worker              name);
195*5e7646d2SAndroid Build Coastguard Worker     if (!stat(ppdname, &ppdinfo) && !access(ppdname, R_OK))
196*5e7646d2SAndroid Build Coastguard Worker     {
197*5e7646d2SAndroid Build Coastguard Worker      /*
198*5e7646d2SAndroid Build Coastguard Worker       * OK, the file exists and is readable, use it!
199*5e7646d2SAndroid Build Coastguard Worker       */
200*5e7646d2SAndroid Build Coastguard Worker 
201*5e7646d2SAndroid Build Coastguard Worker       if (buffer[0])
202*5e7646d2SAndroid Build Coastguard Worker       {
203*5e7646d2SAndroid Build Coastguard Worker         DEBUG_printf(("2cupsGetPPD3: Using filename \"%s\".", buffer));
204*5e7646d2SAndroid Build Coastguard Worker 
205*5e7646d2SAndroid Build Coastguard Worker         unlink(buffer);
206*5e7646d2SAndroid Build Coastguard Worker 
207*5e7646d2SAndroid Build Coastguard Worker 	if (symlink(ppdname, buffer) && errno != EEXIST)
208*5e7646d2SAndroid Build Coastguard Worker         {
209*5e7646d2SAndroid Build Coastguard Worker           _cupsSetError(IPP_STATUS_ERROR_INTERNAL, NULL, 0);
210*5e7646d2SAndroid Build Coastguard Worker 
211*5e7646d2SAndroid Build Coastguard Worker 	  return (HTTP_STATUS_SERVER_ERROR);
212*5e7646d2SAndroid Build Coastguard Worker 	}
213*5e7646d2SAndroid Build Coastguard Worker       }
214*5e7646d2SAndroid Build Coastguard Worker       else
215*5e7646d2SAndroid Build Coastguard Worker       {
216*5e7646d2SAndroid Build Coastguard Worker         int		tries;		/* Number of tries */
217*5e7646d2SAndroid Build Coastguard Worker         const char	*tmpdir;	/* TMPDIR environment variable */
218*5e7646d2SAndroid Build Coastguard Worker 	struct timeval	curtime;	/* Current time */
219*5e7646d2SAndroid Build Coastguard Worker 
220*5e7646d2SAndroid Build Coastguard Worker 
221*5e7646d2SAndroid Build Coastguard Worker #ifdef __APPLE__
222*5e7646d2SAndroid Build Coastguard Worker        /*
223*5e7646d2SAndroid Build Coastguard Worker 	* On macOS and iOS, the TMPDIR environment variable is not always the
224*5e7646d2SAndroid Build Coastguard Worker 	* best location to place temporary files due to sandboxing.  Instead,
225*5e7646d2SAndroid Build Coastguard Worker 	* the confstr function should be called to get the proper per-user,
226*5e7646d2SAndroid Build Coastguard Worker 	* per-process TMPDIR value.
227*5e7646d2SAndroid Build Coastguard Worker 	*/
228*5e7646d2SAndroid Build Coastguard Worker 
229*5e7646d2SAndroid Build Coastguard Worker         char		tmppath[1024];	/* Temporary directory */
230*5e7646d2SAndroid Build Coastguard Worker 
231*5e7646d2SAndroid Build Coastguard Worker 	if ((tmpdir = getenv("TMPDIR")) != NULL && access(tmpdir, W_OK))
232*5e7646d2SAndroid Build Coastguard Worker 	  tmpdir = NULL;
233*5e7646d2SAndroid Build Coastguard Worker 
234*5e7646d2SAndroid Build Coastguard Worker 	if (!tmpdir)
235*5e7646d2SAndroid Build Coastguard Worker 	{
236*5e7646d2SAndroid Build Coastguard Worker 	  if (confstr(_CS_DARWIN_USER_TEMP_DIR, tmppath, sizeof(tmppath)))
237*5e7646d2SAndroid Build Coastguard Worker 	    tmpdir = tmppath;
238*5e7646d2SAndroid Build Coastguard Worker 	  else
239*5e7646d2SAndroid Build Coastguard Worker 	    tmpdir = "/private/tmp";		/* This should never happen */
240*5e7646d2SAndroid Build Coastguard Worker 	}
241*5e7646d2SAndroid Build Coastguard Worker #else
242*5e7646d2SAndroid Build Coastguard Worker        /*
243*5e7646d2SAndroid Build Coastguard Worker 	* Previously we put root temporary files in the default CUPS temporary
244*5e7646d2SAndroid Build Coastguard Worker 	* directory under /var/spool/cups.  However, since the scheduler cleans
245*5e7646d2SAndroid Build Coastguard Worker 	* out temporary files there and runs independently of the user apps, we
246*5e7646d2SAndroid Build Coastguard Worker 	* don't want to use it unless specifically told to by cupsd.
247*5e7646d2SAndroid Build Coastguard Worker 	*/
248*5e7646d2SAndroid Build Coastguard Worker 
249*5e7646d2SAndroid Build Coastguard Worker 	if ((tmpdir = getenv("TMPDIR")) == NULL)
250*5e7646d2SAndroid Build Coastguard Worker 	  tmpdir = "/tmp";
251*5e7646d2SAndroid Build Coastguard Worker #endif /* __APPLE__ */
252*5e7646d2SAndroid Build Coastguard Worker 
253*5e7646d2SAndroid Build Coastguard Worker         DEBUG_printf(("2cupsGetPPD3: tmpdir=\"%s\".", tmpdir));
254*5e7646d2SAndroid Build Coastguard Worker 
255*5e7646d2SAndroid Build Coastguard Worker        /*
256*5e7646d2SAndroid Build Coastguard Worker 	* Make the temporary name using the specified directory...
257*5e7646d2SAndroid Build Coastguard Worker 	*/
258*5e7646d2SAndroid Build Coastguard Worker 
259*5e7646d2SAndroid Build Coastguard Worker 	tries = 0;
260*5e7646d2SAndroid Build Coastguard Worker 
261*5e7646d2SAndroid Build Coastguard Worker 	do
262*5e7646d2SAndroid Build Coastguard Worker 	{
263*5e7646d2SAndroid Build Coastguard Worker 	 /*
264*5e7646d2SAndroid Build Coastguard Worker 	  * Get the current time of day...
265*5e7646d2SAndroid Build Coastguard Worker 	  */
266*5e7646d2SAndroid Build Coastguard Worker 
267*5e7646d2SAndroid Build Coastguard Worker 	  gettimeofday(&curtime, NULL);
268*5e7646d2SAndroid Build Coastguard Worker 
269*5e7646d2SAndroid Build Coastguard Worker 	 /*
270*5e7646d2SAndroid Build Coastguard Worker 	  * Format a string using the hex time values...
271*5e7646d2SAndroid Build Coastguard Worker 	  */
272*5e7646d2SAndroid Build Coastguard Worker 
273*5e7646d2SAndroid Build Coastguard Worker 	  snprintf(buffer, bufsize, "%s/%08lx%05lx", tmpdir,
274*5e7646d2SAndroid Build Coastguard Worker 		   (unsigned long)curtime.tv_sec,
275*5e7646d2SAndroid Build Coastguard Worker 		   (unsigned long)curtime.tv_usec);
276*5e7646d2SAndroid Build Coastguard Worker 
277*5e7646d2SAndroid Build Coastguard Worker 	 /*
278*5e7646d2SAndroid Build Coastguard Worker 	  * Try to make a symlink...
279*5e7646d2SAndroid Build Coastguard Worker 	  */
280*5e7646d2SAndroid Build Coastguard Worker 
281*5e7646d2SAndroid Build Coastguard Worker 	  if (!symlink(ppdname, buffer))
282*5e7646d2SAndroid Build Coastguard Worker 	    break;
283*5e7646d2SAndroid Build Coastguard Worker 
284*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_printf(("2cupsGetPPD3: Symlink \"%s\" to \"%s\" failed: %s", ppdname, buffer, strerror(errno)));
285*5e7646d2SAndroid Build Coastguard Worker 
286*5e7646d2SAndroid Build Coastguard Worker 	  tries ++;
287*5e7646d2SAndroid Build Coastguard Worker 	}
288*5e7646d2SAndroid Build Coastguard Worker 	while (tries < 1000);
289*5e7646d2SAndroid Build Coastguard Worker 
290*5e7646d2SAndroid Build Coastguard Worker         if (tries >= 1000)
291*5e7646d2SAndroid Build Coastguard Worker 	{
292*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_puts("2cupsGetPPD3: Unable to symlink after 1000 tries, returning error.");
293*5e7646d2SAndroid Build Coastguard Worker 
294*5e7646d2SAndroid Build Coastguard Worker           _cupsSetError(IPP_STATUS_ERROR_INTERNAL, NULL, 0);
295*5e7646d2SAndroid Build Coastguard Worker 
296*5e7646d2SAndroid Build Coastguard Worker 	  return (HTTP_STATUS_SERVER_ERROR);
297*5e7646d2SAndroid Build Coastguard Worker 	}
298*5e7646d2SAndroid Build Coastguard Worker       }
299*5e7646d2SAndroid Build Coastguard Worker 
300*5e7646d2SAndroid Build Coastguard Worker       if (*modtime >= ppdinfo.st_mtime)
301*5e7646d2SAndroid Build Coastguard Worker       {
302*5e7646d2SAndroid Build Coastguard Worker         DEBUG_printf(("2cupsGetPPD3: Returning not-modified, filename=\"%s\".", buffer));
303*5e7646d2SAndroid Build Coastguard Worker         return (HTTP_STATUS_NOT_MODIFIED);
304*5e7646d2SAndroid Build Coastguard Worker       }
305*5e7646d2SAndroid Build Coastguard Worker       else
306*5e7646d2SAndroid Build Coastguard Worker       {
307*5e7646d2SAndroid Build Coastguard Worker         DEBUG_printf(("2cupsGetPPD3: Returning ok, filename=\"%s\", modtime=%ld.", buffer, (long)ppdinfo.st_mtime));
308*5e7646d2SAndroid Build Coastguard Worker         *modtime = ppdinfo.st_mtime;
309*5e7646d2SAndroid Build Coastguard Worker 	return (HTTP_STATUS_OK);
310*5e7646d2SAndroid Build Coastguard Worker       }
311*5e7646d2SAndroid Build Coastguard Worker     }
312*5e7646d2SAndroid Build Coastguard Worker   }
313*5e7646d2SAndroid Build Coastguard Worker #endif /* !_WIN32 */
314*5e7646d2SAndroid Build Coastguard Worker 
315*5e7646d2SAndroid Build Coastguard Worker  /*
316*5e7646d2SAndroid Build Coastguard Worker   * Try finding a printer URI for this printer...
317*5e7646d2SAndroid Build Coastguard Worker   */
318*5e7646d2SAndroid Build Coastguard Worker 
319*5e7646d2SAndroid Build Coastguard Worker   DEBUG_puts("2cupsGetPPD3: Unable to access local file, copying...");
320*5e7646d2SAndroid Build Coastguard Worker 
321*5e7646d2SAndroid Build Coastguard Worker   if (!http)
322*5e7646d2SAndroid Build Coastguard Worker   {
323*5e7646d2SAndroid Build Coastguard Worker     if ((http = _cupsConnect()) == NULL)
324*5e7646d2SAndroid Build Coastguard Worker     {
325*5e7646d2SAndroid Build Coastguard Worker       DEBUG_puts("2cupsGetPPD3: Unable to connect to scheduler.");
326*5e7646d2SAndroid Build Coastguard Worker       return (HTTP_STATUS_SERVICE_UNAVAILABLE);
327*5e7646d2SAndroid Build Coastguard Worker     }
328*5e7646d2SAndroid Build Coastguard Worker   }
329*5e7646d2SAndroid Build Coastguard Worker 
330*5e7646d2SAndroid Build Coastguard Worker   if (!cups_get_printer_uri(http, name, hostname, sizeof(hostname), &port, resource, sizeof(resource), 0))
331*5e7646d2SAndroid Build Coastguard Worker   {
332*5e7646d2SAndroid Build Coastguard Worker     DEBUG_puts("2cupsGetPPD3: Unable to get printer URI.");
333*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_STATUS_NOT_FOUND);
334*5e7646d2SAndroid Build Coastguard Worker   }
335*5e7646d2SAndroid Build Coastguard Worker 
336*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2cupsGetPPD3: Printer hostname=\"%s\", port=%d", hostname, port));
337*5e7646d2SAndroid Build Coastguard Worker 
338*5e7646d2SAndroid Build Coastguard Worker   if (cupsServer()[0] == '/' && !_cups_strcasecmp(hostname, "localhost") && port == ippPort())
339*5e7646d2SAndroid Build Coastguard Worker   {
340*5e7646d2SAndroid Build Coastguard Worker    /*
341*5e7646d2SAndroid Build Coastguard Worker     * Redirect localhost to domain socket...
342*5e7646d2SAndroid Build Coastguard Worker     */
343*5e7646d2SAndroid Build Coastguard Worker 
344*5e7646d2SAndroid Build Coastguard Worker     strlcpy(hostname, cupsServer(), sizeof(hostname));
345*5e7646d2SAndroid Build Coastguard Worker     port = 0;
346*5e7646d2SAndroid Build Coastguard Worker 
347*5e7646d2SAndroid Build Coastguard Worker     DEBUG_printf(("2cupsGetPPD3: Redirecting to \"%s\".", hostname));
348*5e7646d2SAndroid Build Coastguard Worker   }
349*5e7646d2SAndroid Build Coastguard Worker 
350*5e7646d2SAndroid Build Coastguard Worker  /*
351*5e7646d2SAndroid Build Coastguard Worker   * Remap local hostname to localhost...
352*5e7646d2SAndroid Build Coastguard Worker   */
353*5e7646d2SAndroid Build Coastguard Worker 
354*5e7646d2SAndroid Build Coastguard Worker   httpGetHostname(NULL, localhost, sizeof(localhost));
355*5e7646d2SAndroid Build Coastguard Worker 
356*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2cupsGetPPD3: Local hostname=\"%s\"", localhost));
357*5e7646d2SAndroid Build Coastguard Worker 
358*5e7646d2SAndroid Build Coastguard Worker   if (!_cups_strcasecmp(localhost, hostname))
359*5e7646d2SAndroid Build Coastguard Worker     strlcpy(hostname, "localhost", sizeof(hostname));
360*5e7646d2SAndroid Build Coastguard Worker 
361*5e7646d2SAndroid Build Coastguard Worker  /*
362*5e7646d2SAndroid Build Coastguard Worker   * Get the hostname and port number we are connected to...
363*5e7646d2SAndroid Build Coastguard Worker   */
364*5e7646d2SAndroid Build Coastguard Worker 
365*5e7646d2SAndroid Build Coastguard Worker   httpGetHostname(http, http_hostname, sizeof(http_hostname));
366*5e7646d2SAndroid Build Coastguard Worker   http_port = httpAddrPort(http->hostaddr);
367*5e7646d2SAndroid Build Coastguard Worker 
368*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2cupsGetPPD3: Connection hostname=\"%s\", port=%d",
369*5e7646d2SAndroid Build Coastguard Worker                 http_hostname, http_port));
370*5e7646d2SAndroid Build Coastguard Worker 
371*5e7646d2SAndroid Build Coastguard Worker  /*
372*5e7646d2SAndroid Build Coastguard Worker   * Reconnect to the correct server as needed...
373*5e7646d2SAndroid Build Coastguard Worker   */
374*5e7646d2SAndroid Build Coastguard Worker 
375*5e7646d2SAndroid Build Coastguard Worker   if (!_cups_strcasecmp(http_hostname, hostname) && port == http_port)
376*5e7646d2SAndroid Build Coastguard Worker     http2 = http;
377*5e7646d2SAndroid Build Coastguard Worker   else if ((http2 = httpConnect2(hostname, port, NULL, AF_UNSPEC,
378*5e7646d2SAndroid Build Coastguard Worker 				 cupsEncryption(), 1, 30000, NULL)) == NULL)
379*5e7646d2SAndroid Build Coastguard Worker   {
380*5e7646d2SAndroid Build Coastguard Worker     DEBUG_puts("2cupsGetPPD3: Unable to connect to server");
381*5e7646d2SAndroid Build Coastguard Worker 
382*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_STATUS_SERVICE_UNAVAILABLE);
383*5e7646d2SAndroid Build Coastguard Worker   }
384*5e7646d2SAndroid Build Coastguard Worker 
385*5e7646d2SAndroid Build Coastguard Worker  /*
386*5e7646d2SAndroid Build Coastguard Worker   * Get a temp file...
387*5e7646d2SAndroid Build Coastguard Worker   */
388*5e7646d2SAndroid Build Coastguard Worker 
389*5e7646d2SAndroid Build Coastguard Worker   if (buffer[0])
390*5e7646d2SAndroid Build Coastguard Worker     fd = open(buffer, O_CREAT | O_TRUNC | O_WRONLY, 0600);
391*5e7646d2SAndroid Build Coastguard Worker   else
392*5e7646d2SAndroid Build Coastguard Worker     fd = cupsTempFd(tempfile, sizeof(tempfile));
393*5e7646d2SAndroid Build Coastguard Worker 
394*5e7646d2SAndroid Build Coastguard Worker   if (fd < 0)
395*5e7646d2SAndroid Build Coastguard Worker   {
396*5e7646d2SAndroid Build Coastguard Worker    /*
397*5e7646d2SAndroid Build Coastguard Worker     * Can't open file; close the server connection and return NULL...
398*5e7646d2SAndroid Build Coastguard Worker     */
399*5e7646d2SAndroid Build Coastguard Worker 
400*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, NULL, 0);
401*5e7646d2SAndroid Build Coastguard Worker 
402*5e7646d2SAndroid Build Coastguard Worker     if (http2 != http)
403*5e7646d2SAndroid Build Coastguard Worker       httpClose(http2);
404*5e7646d2SAndroid Build Coastguard Worker 
405*5e7646d2SAndroid Build Coastguard Worker     return (HTTP_STATUS_SERVER_ERROR);
406*5e7646d2SAndroid Build Coastguard Worker   }
407*5e7646d2SAndroid Build Coastguard Worker 
408*5e7646d2SAndroid Build Coastguard Worker  /*
409*5e7646d2SAndroid Build Coastguard Worker   * And send a request to the HTTP server...
410*5e7646d2SAndroid Build Coastguard Worker   */
411*5e7646d2SAndroid Build Coastguard Worker 
412*5e7646d2SAndroid Build Coastguard Worker   strlcat(resource, ".ppd", sizeof(resource));
413*5e7646d2SAndroid Build Coastguard Worker 
414*5e7646d2SAndroid Build Coastguard Worker   if (*modtime > 0)
415*5e7646d2SAndroid Build Coastguard Worker     httpSetField(http2, HTTP_FIELD_IF_MODIFIED_SINCE,
416*5e7646d2SAndroid Build Coastguard Worker                  httpGetDateString(*modtime));
417*5e7646d2SAndroid Build Coastguard Worker 
418*5e7646d2SAndroid Build Coastguard Worker   status = cupsGetFd(http2, resource, fd);
419*5e7646d2SAndroid Build Coastguard Worker 
420*5e7646d2SAndroid Build Coastguard Worker   close(fd);
421*5e7646d2SAndroid Build Coastguard Worker 
422*5e7646d2SAndroid Build Coastguard Worker  /*
423*5e7646d2SAndroid Build Coastguard Worker   * See if we actually got the file or an error...
424*5e7646d2SAndroid Build Coastguard Worker   */
425*5e7646d2SAndroid Build Coastguard Worker 
426*5e7646d2SAndroid Build Coastguard Worker   if (status == HTTP_STATUS_OK)
427*5e7646d2SAndroid Build Coastguard Worker   {
428*5e7646d2SAndroid Build Coastguard Worker     *modtime = httpGetDateTime(httpGetField(http2, HTTP_FIELD_DATE));
429*5e7646d2SAndroid Build Coastguard Worker 
430*5e7646d2SAndroid Build Coastguard Worker     if (tempfile[0])
431*5e7646d2SAndroid Build Coastguard Worker       strlcpy(buffer, tempfile, bufsize);
432*5e7646d2SAndroid Build Coastguard Worker   }
433*5e7646d2SAndroid Build Coastguard Worker   else if (status != HTTP_STATUS_NOT_MODIFIED)
434*5e7646d2SAndroid Build Coastguard Worker   {
435*5e7646d2SAndroid Build Coastguard Worker     _cupsSetHTTPError(status);
436*5e7646d2SAndroid Build Coastguard Worker 
437*5e7646d2SAndroid Build Coastguard Worker     if (buffer[0])
438*5e7646d2SAndroid Build Coastguard Worker       unlink(buffer);
439*5e7646d2SAndroid Build Coastguard Worker     else if (tempfile[0])
440*5e7646d2SAndroid Build Coastguard Worker       unlink(tempfile);
441*5e7646d2SAndroid Build Coastguard Worker   }
442*5e7646d2SAndroid Build Coastguard Worker   else if (tempfile[0])
443*5e7646d2SAndroid Build Coastguard Worker     unlink(tempfile);
444*5e7646d2SAndroid Build Coastguard Worker 
445*5e7646d2SAndroid Build Coastguard Worker   if (http2 != http)
446*5e7646d2SAndroid Build Coastguard Worker     httpClose(http2);
447*5e7646d2SAndroid Build Coastguard Worker 
448*5e7646d2SAndroid Build Coastguard Worker  /*
449*5e7646d2SAndroid Build Coastguard Worker   * Return the PPD file...
450*5e7646d2SAndroid Build Coastguard Worker   */
451*5e7646d2SAndroid Build Coastguard Worker 
452*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("2cupsGetPPD3: Returning status %d", status));
453*5e7646d2SAndroid Build Coastguard Worker 
454*5e7646d2SAndroid Build Coastguard Worker   return (status);
455*5e7646d2SAndroid Build Coastguard Worker }
456*5e7646d2SAndroid Build Coastguard Worker 
457*5e7646d2SAndroid Build Coastguard Worker 
458*5e7646d2SAndroid Build Coastguard Worker /*
459*5e7646d2SAndroid Build Coastguard Worker  * 'cupsGetServerPPD()' - Get an available PPD file from the server.
460*5e7646d2SAndroid Build Coastguard Worker  *
461*5e7646d2SAndroid Build Coastguard Worker  * This function returns the named PPD file from the server.  The
462*5e7646d2SAndroid Build Coastguard Worker  * list of available PPDs is provided by the IPP @code CUPS_GET_PPDS@
463*5e7646d2SAndroid Build Coastguard Worker  * operation.
464*5e7646d2SAndroid Build Coastguard Worker  *
465*5e7646d2SAndroid Build Coastguard Worker  * You must remove (unlink) the PPD file when you are finished with
466*5e7646d2SAndroid Build Coastguard Worker  * it. The PPD filename is stored in a static location that will be
467*5e7646d2SAndroid Build Coastguard Worker  * overwritten on the next call to @link cupsGetPPD@, @link cupsGetPPD2@,
468*5e7646d2SAndroid Build Coastguard Worker  * or @link cupsGetServerPPD@.
469*5e7646d2SAndroid Build Coastguard Worker  *
470*5e7646d2SAndroid Build Coastguard Worker  * @since CUPS 1.3/macOS 10.5@
471*5e7646d2SAndroid Build Coastguard Worker  */
472*5e7646d2SAndroid Build Coastguard Worker 
473*5e7646d2SAndroid Build Coastguard Worker char *					/* O - Name of PPD file or @code NULL@ on error */
cupsGetServerPPD(http_t * http,const char * name)474*5e7646d2SAndroid Build Coastguard Worker cupsGetServerPPD(http_t     *http,	/* I - Connection to server or @code CUPS_HTTP_DEFAULT@ */
475*5e7646d2SAndroid Build Coastguard Worker                  const char *name)	/* I - Name of PPD file ("ppd-name") */
476*5e7646d2SAndroid Build Coastguard Worker {
477*5e7646d2SAndroid Build Coastguard Worker   int			fd;		/* PPD file descriptor */
478*5e7646d2SAndroid Build Coastguard Worker   ipp_t			*request;	/* IPP request */
479*5e7646d2SAndroid Build Coastguard Worker   _ppd_globals_t	*pg = _ppdGlobals();
480*5e7646d2SAndroid Build Coastguard Worker 					/* Pointer to library globals */
481*5e7646d2SAndroid Build Coastguard Worker 
482*5e7646d2SAndroid Build Coastguard Worker 
483*5e7646d2SAndroid Build Coastguard Worker  /*
484*5e7646d2SAndroid Build Coastguard Worker   * Range check input...
485*5e7646d2SAndroid Build Coastguard Worker   */
486*5e7646d2SAndroid Build Coastguard Worker 
487*5e7646d2SAndroid Build Coastguard Worker   if (!name)
488*5e7646d2SAndroid Build Coastguard Worker   {
489*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("No PPD name"), 1);
490*5e7646d2SAndroid Build Coastguard Worker 
491*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
492*5e7646d2SAndroid Build Coastguard Worker   }
493*5e7646d2SAndroid Build Coastguard Worker 
494*5e7646d2SAndroid Build Coastguard Worker   if (!http)
495*5e7646d2SAndroid Build Coastguard Worker     if ((http = _cupsConnect()) == NULL)
496*5e7646d2SAndroid Build Coastguard Worker       return (NULL);
497*5e7646d2SAndroid Build Coastguard Worker 
498*5e7646d2SAndroid Build Coastguard Worker  /*
499*5e7646d2SAndroid Build Coastguard Worker   * Get a temp file...
500*5e7646d2SAndroid Build Coastguard Worker   */
501*5e7646d2SAndroid Build Coastguard Worker 
502*5e7646d2SAndroid Build Coastguard Worker   if ((fd = cupsTempFd(pg->ppd_filename, sizeof(pg->ppd_filename))) < 0)
503*5e7646d2SAndroid Build Coastguard Worker   {
504*5e7646d2SAndroid Build Coastguard Worker    /*
505*5e7646d2SAndroid Build Coastguard Worker     * Can't open file; close the server connection and return NULL...
506*5e7646d2SAndroid Build Coastguard Worker     */
507*5e7646d2SAndroid Build Coastguard Worker 
508*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, NULL, 0);
509*5e7646d2SAndroid Build Coastguard Worker 
510*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
511*5e7646d2SAndroid Build Coastguard Worker   }
512*5e7646d2SAndroid Build Coastguard Worker 
513*5e7646d2SAndroid Build Coastguard Worker  /*
514*5e7646d2SAndroid Build Coastguard Worker   * Get the PPD file...
515*5e7646d2SAndroid Build Coastguard Worker   */
516*5e7646d2SAndroid Build Coastguard Worker 
517*5e7646d2SAndroid Build Coastguard Worker   request = ippNewRequest(IPP_OP_CUPS_GET_PPD);
518*5e7646d2SAndroid Build Coastguard Worker   ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "ppd-name", NULL,
519*5e7646d2SAndroid Build Coastguard Worker                name);
520*5e7646d2SAndroid Build Coastguard Worker 
521*5e7646d2SAndroid Build Coastguard Worker   ippDelete(cupsDoIORequest(http, request, "/", -1, fd));
522*5e7646d2SAndroid Build Coastguard Worker 
523*5e7646d2SAndroid Build Coastguard Worker   close(fd);
524*5e7646d2SAndroid Build Coastguard Worker 
525*5e7646d2SAndroid Build Coastguard Worker   if (cupsLastError() != IPP_STATUS_OK)
526*5e7646d2SAndroid Build Coastguard Worker   {
527*5e7646d2SAndroid Build Coastguard Worker     unlink(pg->ppd_filename);
528*5e7646d2SAndroid Build Coastguard Worker     return (NULL);
529*5e7646d2SAndroid Build Coastguard Worker   }
530*5e7646d2SAndroid Build Coastguard Worker   else
531*5e7646d2SAndroid Build Coastguard Worker     return (pg->ppd_filename);
532*5e7646d2SAndroid Build Coastguard Worker }
533*5e7646d2SAndroid Build Coastguard Worker 
534*5e7646d2SAndroid Build Coastguard Worker 
535*5e7646d2SAndroid Build Coastguard Worker /*
536*5e7646d2SAndroid Build Coastguard Worker  * 'cups_get_printer_uri()' - Get the printer-uri-supported attribute for the
537*5e7646d2SAndroid Build Coastguard Worker  *                            first printer in a class.
538*5e7646d2SAndroid Build Coastguard Worker  */
539*5e7646d2SAndroid Build Coastguard Worker 
540*5e7646d2SAndroid Build Coastguard Worker static int				/* O - 1 on success, 0 on failure */
cups_get_printer_uri(http_t * http,const char * name,char * host,int hostsize,int * port,char * resource,int resourcesize,int depth)541*5e7646d2SAndroid Build Coastguard Worker cups_get_printer_uri(
542*5e7646d2SAndroid Build Coastguard Worker     http_t     *http,			/* I - Connection to server */
543*5e7646d2SAndroid Build Coastguard Worker     const char *name,			/* I - Name of printer or class */
544*5e7646d2SAndroid Build Coastguard Worker     char       *host,			/* I - Hostname buffer */
545*5e7646d2SAndroid Build Coastguard Worker     int        hostsize,		/* I - Size of hostname buffer */
546*5e7646d2SAndroid Build Coastguard Worker     int        *port,			/* O - Port number */
547*5e7646d2SAndroid Build Coastguard Worker     char       *resource,		/* I - Resource buffer */
548*5e7646d2SAndroid Build Coastguard Worker     int        resourcesize,		/* I - Size of resource buffer */
549*5e7646d2SAndroid Build Coastguard Worker     int        depth)			/* I - Depth of query */
550*5e7646d2SAndroid Build Coastguard Worker {
551*5e7646d2SAndroid Build Coastguard Worker   int		i;			/* Looping var */
552*5e7646d2SAndroid Build Coastguard Worker   ipp_t		*request,		/* IPP request */
553*5e7646d2SAndroid Build Coastguard Worker 		*response;		/* IPP response */
554*5e7646d2SAndroid Build Coastguard Worker   ipp_attribute_t *attr;		/* Current attribute */
555*5e7646d2SAndroid Build Coastguard Worker   char		uri[HTTP_MAX_URI],	/* printer-uri attribute */
556*5e7646d2SAndroid Build Coastguard Worker 		scheme[HTTP_MAX_URI],	/* Scheme name */
557*5e7646d2SAndroid Build Coastguard Worker 		username[HTTP_MAX_URI];	/* Username:password */
558*5e7646d2SAndroid Build Coastguard Worker   static const char * const requested_attrs[] =
559*5e7646d2SAndroid Build Coastguard Worker 		{			/* Requested attributes */
560*5e7646d2SAndroid Build Coastguard Worker 		  "member-uris",
561*5e7646d2SAndroid Build Coastguard Worker 		  "printer-uri-supported"
562*5e7646d2SAndroid Build Coastguard Worker 		};
563*5e7646d2SAndroid Build Coastguard Worker 
564*5e7646d2SAndroid Build Coastguard Worker 
565*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("4cups_get_printer_uri(http=%p, name=\"%s\", host=%p, hostsize=%d, resource=%p, resourcesize=%d, depth=%d)", http, name, host, hostsize, resource, resourcesize, depth));
566*5e7646d2SAndroid Build Coastguard Worker 
567*5e7646d2SAndroid Build Coastguard Worker  /*
568*5e7646d2SAndroid Build Coastguard Worker   * Setup the printer URI...
569*5e7646d2SAndroid Build Coastguard Worker   */
570*5e7646d2SAndroid Build Coastguard Worker 
571*5e7646d2SAndroid Build Coastguard Worker   if (httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, "localhost", 0, "/printers/%s", name) < HTTP_URI_STATUS_OK)
572*5e7646d2SAndroid Build Coastguard Worker   {
573*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("Unable to create printer-uri"), 1);
574*5e7646d2SAndroid Build Coastguard Worker 
575*5e7646d2SAndroid Build Coastguard Worker     *host     = '\0';
576*5e7646d2SAndroid Build Coastguard Worker     *resource = '\0';
577*5e7646d2SAndroid Build Coastguard Worker 
578*5e7646d2SAndroid Build Coastguard Worker     return (0);
579*5e7646d2SAndroid Build Coastguard Worker   }
580*5e7646d2SAndroid Build Coastguard Worker 
581*5e7646d2SAndroid Build Coastguard Worker   DEBUG_printf(("5cups_get_printer_uri: printer-uri=\"%s\"", uri));
582*5e7646d2SAndroid Build Coastguard Worker 
583*5e7646d2SAndroid Build Coastguard Worker  /*
584*5e7646d2SAndroid Build Coastguard Worker   * Build an IPP_GET_PRINTER_ATTRIBUTES request, which requires the following
585*5e7646d2SAndroid Build Coastguard Worker   * attributes:
586*5e7646d2SAndroid Build Coastguard Worker   *
587*5e7646d2SAndroid Build Coastguard Worker   *    attributes-charset
588*5e7646d2SAndroid Build Coastguard Worker   *    attributes-natural-language
589*5e7646d2SAndroid Build Coastguard Worker   *    printer-uri
590*5e7646d2SAndroid Build Coastguard Worker   *    requested-attributes
591*5e7646d2SAndroid Build Coastguard Worker   */
592*5e7646d2SAndroid Build Coastguard Worker 
593*5e7646d2SAndroid Build Coastguard Worker   request = ippNewRequest(IPP_OP_GET_PRINTER_ATTRIBUTES);
594*5e7646d2SAndroid Build Coastguard Worker 
595*5e7646d2SAndroid Build Coastguard Worker   ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", NULL, uri);
596*5e7646d2SAndroid Build Coastguard Worker 
597*5e7646d2SAndroid Build Coastguard Worker   ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, "requested-attributes", sizeof(requested_attrs) / sizeof(requested_attrs[0]), NULL, requested_attrs);
598*5e7646d2SAndroid Build Coastguard Worker 
599*5e7646d2SAndroid Build Coastguard Worker  /*
600*5e7646d2SAndroid Build Coastguard Worker   * Do the request and get back a response...
601*5e7646d2SAndroid Build Coastguard Worker   */
602*5e7646d2SAndroid Build Coastguard Worker 
603*5e7646d2SAndroid Build Coastguard Worker   snprintf(resource, (size_t)resourcesize, "/printers/%s", name);
604*5e7646d2SAndroid Build Coastguard Worker 
605*5e7646d2SAndroid Build Coastguard Worker   if ((response = cupsDoRequest(http, request, resource)) != NULL)
606*5e7646d2SAndroid Build Coastguard Worker   {
607*5e7646d2SAndroid Build Coastguard Worker     if ((attr = ippFindAttribute(response, "member-uris", IPP_TAG_URI)) != NULL)
608*5e7646d2SAndroid Build Coastguard Worker     {
609*5e7646d2SAndroid Build Coastguard Worker      /*
610*5e7646d2SAndroid Build Coastguard Worker       * Get the first actual printer name in the class...
611*5e7646d2SAndroid Build Coastguard Worker       */
612*5e7646d2SAndroid Build Coastguard Worker 
613*5e7646d2SAndroid Build Coastguard Worker       DEBUG_printf(("5cups_get_printer_uri: Got member-uris with %d values.", ippGetCount(attr)));
614*5e7646d2SAndroid Build Coastguard Worker 
615*5e7646d2SAndroid Build Coastguard Worker       for (i = 0; i < attr->num_values; i ++)
616*5e7646d2SAndroid Build Coastguard Worker       {
617*5e7646d2SAndroid Build Coastguard Worker         DEBUG_printf(("5cups_get_printer_uri: member-uris[%d]=\"%s\"", i, ippGetString(attr, i, NULL)));
618*5e7646d2SAndroid Build Coastguard Worker 
619*5e7646d2SAndroid Build Coastguard Worker 	httpSeparateURI(HTTP_URI_CODING_ALL, attr->values[i].string.text, scheme, sizeof(scheme), username, sizeof(username), host, hostsize, port, resource, resourcesize);
620*5e7646d2SAndroid Build Coastguard Worker 	if (!strncmp(resource, "/printers/", 10))
621*5e7646d2SAndroid Build Coastguard Worker 	{
622*5e7646d2SAndroid Build Coastguard Worker 	 /*
623*5e7646d2SAndroid Build Coastguard Worker 	  * Found a printer!
624*5e7646d2SAndroid Build Coastguard Worker 	  */
625*5e7646d2SAndroid Build Coastguard Worker 
626*5e7646d2SAndroid Build Coastguard Worker           ippDelete(response);
627*5e7646d2SAndroid Build Coastguard Worker 
628*5e7646d2SAndroid Build Coastguard Worker 	  DEBUG_printf(("5cups_get_printer_uri: Found printer member with host=\"%s\", port=%d, resource=\"%s\"", host, *port, resource));
629*5e7646d2SAndroid Build Coastguard Worker 	  return (1);
630*5e7646d2SAndroid Build Coastguard Worker 	}
631*5e7646d2SAndroid Build Coastguard Worker       }
632*5e7646d2SAndroid Build Coastguard Worker     }
633*5e7646d2SAndroid Build Coastguard Worker     else if ((attr = ippFindAttribute(response, "printer-uri-supported", IPP_TAG_URI)) != NULL)
634*5e7646d2SAndroid Build Coastguard Worker     {
635*5e7646d2SAndroid Build Coastguard Worker       httpSeparateURI(HTTP_URI_CODING_ALL, _httpResolveURI(attr->values[0].string.text, uri, sizeof(uri), _HTTP_RESOLVE_DEFAULT, NULL, NULL), scheme, sizeof(scheme), username, sizeof(username), host, hostsize, port, resource, resourcesize);
636*5e7646d2SAndroid Build Coastguard Worker       ippDelete(response);
637*5e7646d2SAndroid Build Coastguard Worker 
638*5e7646d2SAndroid Build Coastguard Worker       DEBUG_printf(("5cups_get_printer_uri: Resolved to host=\"%s\", port=%d, resource=\"%s\"", host, *port, resource));
639*5e7646d2SAndroid Build Coastguard Worker 
640*5e7646d2SAndroid Build Coastguard Worker       if (!strncmp(resource, "/classes/", 9))
641*5e7646d2SAndroid Build Coastguard Worker       {
642*5e7646d2SAndroid Build Coastguard Worker         _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("No printer-uri found for class"), 1);
643*5e7646d2SAndroid Build Coastguard Worker 
644*5e7646d2SAndroid Build Coastguard Worker 	*host     = '\0';
645*5e7646d2SAndroid Build Coastguard Worker 	*resource = '\0';
646*5e7646d2SAndroid Build Coastguard Worker 
647*5e7646d2SAndroid Build Coastguard Worker         DEBUG_puts("5cups_get_printer_uri: Not returning class.");
648*5e7646d2SAndroid Build Coastguard Worker 	return (0);
649*5e7646d2SAndroid Build Coastguard Worker       }
650*5e7646d2SAndroid Build Coastguard Worker 
651*5e7646d2SAndroid Build Coastguard Worker       return (1);
652*5e7646d2SAndroid Build Coastguard Worker     }
653*5e7646d2SAndroid Build Coastguard Worker 
654*5e7646d2SAndroid Build Coastguard Worker     ippDelete(response);
655*5e7646d2SAndroid Build Coastguard Worker   }
656*5e7646d2SAndroid Build Coastguard Worker 
657*5e7646d2SAndroid Build Coastguard Worker   if (cupsLastError() != IPP_STATUS_ERROR_NOT_FOUND)
658*5e7646d2SAndroid Build Coastguard Worker     _cupsSetError(IPP_STATUS_ERROR_INTERNAL, _("No printer-uri found"), 1);
659*5e7646d2SAndroid Build Coastguard Worker 
660*5e7646d2SAndroid Build Coastguard Worker   *host     = '\0';
661*5e7646d2SAndroid Build Coastguard Worker   *resource = '\0';
662*5e7646d2SAndroid Build Coastguard Worker 
663*5e7646d2SAndroid Build Coastguard Worker   DEBUG_puts("5cups_get_printer_uri: Printer URI not found.");
664*5e7646d2SAndroid Build Coastguard Worker   return (0);
665*5e7646d2SAndroid Build Coastguard Worker }
666