]>
Commit | Line | Data |
---|---|---|
1 | /* | |
2 | * "$Id: ipp-var.c 5838 2006-08-17 14:41:42Z mike $" | |
3 | * | |
4 | * CGI <-> IPP variable routines for the Common UNIX Printing System (CUPS). | |
5 | * | |
6 | * Copyright 1997-2006 by Easy Software Products. | |
7 | * | |
8 | * These coded instructions, statements, and computer programs are the | |
9 | * property of Easy Software Products and are protected by Federal | |
10 | * copyright law. Distribution and use rights are outlined in the file | |
11 | * "LICENSE.txt" which should have been included with this file. If this | |
12 | * file is missing or damaged please contact Easy Software Products | |
13 | * at: | |
14 | * | |
15 | * Attn: CUPS Licensing Information | |
16 | * Easy Software Products | |
17 | * 44141 Airport View Drive, Suite 204 | |
18 | * Hollywood, Maryland 20636 USA | |
19 | * | |
20 | * Voice: (301) 373-9600 | |
21 | * EMail: cups-info@cups.org | |
22 | * WWW: http://www.cups.org | |
23 | * | |
24 | * Contents: | |
25 | * | |
26 | * cgiGetAttributes() - Get the list of attributes that are needed | |
27 | * by the template file. | |
28 | * cgiGetIPPObjects() - Get the objects in an IPP response. | |
29 | * cgiMoveJobs() - Move one or more jobs. | |
30 | * cgiPrintTestPage() - Print a test page. | |
31 | * cgiRewriteURL() - Rewrite a printer URI into a web browser URL... | |
32 | * cgiSetIPPObjectVars() - Set CGI variables from an IPP object. | |
33 | * cgiSetIPPVars() - Set CGI variables from an IPP response. | |
34 | * cgiShowIPPError() - Show the last IPP error message. | |
35 | * cgiShowJobs() - Show print jobs. | |
36 | * cgiText() - Return localized text. | |
37 | */ | |
38 | ||
39 | /* | |
40 | * Include necessary headers... | |
41 | */ | |
42 | ||
43 | #include "cgi-private.h" | |
44 | ||
45 | ||
46 | /* | |
47 | * 'cgiGetAttributes()' - Get the list of attributes that are needed | |
48 | * by the template file. | |
49 | */ | |
50 | ||
51 | void | |
52 | cgiGetAttributes(ipp_t *request, /* I - IPP request */ | |
53 | const char *tmpl) /* I - Base filename */ | |
54 | { | |
55 | int num_attrs; /* Number of attributes */ | |
56 | char *attrs[1000]; /* Attributes */ | |
57 | int i; /* Looping var */ | |
58 | char filename[1024], /* Filename */ | |
59 | locale[16]; /* Locale name */ | |
60 | const char *directory, /* Directory */ | |
61 | *lang; /* Language */ | |
62 | FILE *in; /* Input file */ | |
63 | int ch; /* Character from file */ | |
64 | char name[255], /* Name of variable */ | |
65 | *nameptr; /* Pointer into name */ | |
66 | ||
67 | ||
68 | /* | |
69 | * Convert the language to a locale name... | |
70 | */ | |
71 | ||
72 | if ((lang = getenv("LANG")) != NULL) | |
73 | { | |
74 | for (i = 0; lang[i] && i < 15; i ++) | |
75 | if (isalnum(lang[i] & 255)) | |
76 | locale[i] = tolower(lang[i]); | |
77 | else | |
78 | locale[i] = '_'; | |
79 | ||
80 | locale[i] = '\0'; | |
81 | } | |
82 | else | |
83 | locale[0] = '\0'; | |
84 | ||
85 | /* | |
86 | * See if we have a template file for this language... | |
87 | */ | |
88 | ||
89 | directory = cgiGetTemplateDir(); | |
90 | ||
91 | snprintf(filename, sizeof(filename), "%s/%s/%s", directory, locale, tmpl); | |
92 | if (access(filename, 0)) | |
93 | { | |
94 | locale[2] = '\0'; | |
95 | ||
96 | snprintf(filename, sizeof(filename), "%s/%s/%s", directory, locale, tmpl); | |
97 | if (access(filename, 0)) | |
98 | snprintf(filename, sizeof(filename), "%s/%s", directory, tmpl); | |
99 | } | |
100 | ||
101 | /* | |
102 | * Open the template file... | |
103 | */ | |
104 | ||
105 | if ((in = fopen(filename, "r")) == NULL) | |
106 | return; | |
107 | ||
108 | /* | |
109 | * Loop through the file adding attribute names as needed... | |
110 | */ | |
111 | ||
112 | num_attrs = 0; | |
113 | attrs[0] = NULL; /* Eliminate compiler warning */ | |
114 | ||
115 | while ((ch = getc(in)) != EOF) | |
116 | if (ch == '\\') | |
117 | getc(in); | |
118 | else if (ch == '{' && num_attrs < (sizeof(attrs) / sizeof(attrs[0]))) | |
119 | { | |
120 | /* | |
121 | * Grab the name... | |
122 | */ | |
123 | ||
124 | for (nameptr = name; (ch = getc(in)) != EOF;) | |
125 | if (strchr("}]<>=! \t\n", ch)) | |
126 | break; | |
127 | else if (nameptr > name && ch == '?') | |
128 | break; | |
129 | else if (nameptr < (name + sizeof(name) - 1)) | |
130 | { | |
131 | if (ch == '_') | |
132 | *nameptr++ = '-'; | |
133 | else | |
134 | *nameptr++ = ch; | |
135 | } | |
136 | ||
137 | *nameptr = '\0'; | |
138 | ||
139 | if (!strncmp(name, "printer_state_history", 21)) | |
140 | strcpy(name, "printer_state_history"); | |
141 | ||
142 | /* | |
143 | * Possibly add it to the list of attributes... | |
144 | */ | |
145 | ||
146 | for (i = 0; i < num_attrs; i ++) | |
147 | if (!strcmp(attrs[i], name)) | |
148 | break; | |
149 | ||
150 | if (i >= num_attrs) | |
151 | { | |
152 | attrs[num_attrs] = strdup(name); | |
153 | num_attrs ++; | |
154 | } | |
155 | } | |
156 | ||
157 | /* | |
158 | * If we have attributes, add a requested-attributes attribute to the | |
159 | * request... | |
160 | */ | |
161 | ||
162 | if (num_attrs > 0) | |
163 | { | |
164 | ippAddStrings(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, | |
165 | "requested-attributes", num_attrs, NULL, (const char **)attrs); | |
166 | ||
167 | for (i = 0; i < num_attrs; i ++) | |
168 | free(attrs[i]); | |
169 | } | |
170 | } | |
171 | ||
172 | ||
173 | /* | |
174 | * 'cgiGetIPPObjects()' - Get the objects in an IPP response. | |
175 | */ | |
176 | ||
177 | cups_array_t * /* O - Array of objects */ | |
178 | cgiGetIPPObjects(ipp_t *response, /* I - IPP response */ | |
179 | void *search) /* I - Search filter */ | |
180 | { | |
181 | int i; /* Looping var */ | |
182 | cups_array_t *objs; /* Array of objects */ | |
183 | ipp_attribute_t *attr, /* Current attribute */ | |
184 | *first; /* First attribute for object */ | |
185 | ipp_tag_t group; /* Current group tag */ | |
186 | int add; /* Add this object to the array? */ | |
187 | ||
188 | ||
189 | if (!response) | |
190 | return (0); | |
191 | ||
192 | for (add = 0, first = NULL, objs = cupsArrayNew(NULL, NULL), | |
193 | group = IPP_TAG_ZERO, attr = response->attrs; | |
194 | attr; | |
195 | attr = attr->next) | |
196 | { | |
197 | if (attr->group_tag != group) | |
198 | { | |
199 | group = attr->group_tag; | |
200 | ||
201 | if (group != IPP_TAG_ZERO && group != IPP_TAG_OPERATION) | |
202 | { | |
203 | first = attr; | |
204 | add = 0; | |
205 | } | |
206 | else if (add && first) | |
207 | { | |
208 | cupsArrayAdd(objs, first); | |
209 | ||
210 | add = 0; | |
211 | first = NULL; | |
212 | } | |
213 | } | |
214 | ||
215 | if (attr->name && attr->group_tag != IPP_TAG_OPERATION && !add) | |
216 | { | |
217 | if (!search) | |
218 | { | |
219 | /* | |
220 | * Add all objects if there is no search... | |
221 | */ | |
222 | ||
223 | add = 1; | |
224 | } | |
225 | else | |
226 | { | |
227 | /* | |
228 | * Check the search string against the string and integer values. | |
229 | */ | |
230 | ||
231 | switch (attr->value_tag) | |
232 | { | |
233 | case IPP_TAG_TEXTLANG : | |
234 | case IPP_TAG_NAMELANG : | |
235 | case IPP_TAG_TEXT : | |
236 | case IPP_TAG_NAME : | |
237 | case IPP_TAG_KEYWORD : | |
238 | case IPP_TAG_URI : | |
239 | case IPP_TAG_MIMETYPE : | |
240 | for (i = 0; !add && i < attr->num_values; i ++) | |
241 | if (cgiDoSearch(search, attr->values[i].string.text)) | |
242 | add = 1; | |
243 | break; | |
244 | ||
245 | case IPP_TAG_INTEGER : | |
246 | for (i = 0; !add && i < attr->num_values; i ++) | |
247 | { | |
248 | char buf[255]; /* Number buffer */ | |
249 | ||
250 | ||
251 | sprintf(buf, "%d", attr->values[i].integer); | |
252 | ||
253 | if (cgiDoSearch(search, buf)) | |
254 | add = 1; | |
255 | } | |
256 | break; | |
257 | ||
258 | default : | |
259 | break; | |
260 | } | |
261 | } | |
262 | } | |
263 | } | |
264 | ||
265 | if (add && first) | |
266 | cupsArrayAdd(objs, first); | |
267 | ||
268 | return (objs); | |
269 | } | |
270 | ||
271 | ||
272 | /* | |
273 | * 'cgiMoveJobs()' - Move one or more jobs. | |
274 | * | |
275 | * At least one of dest or job_id must be non-zero/NULL. | |
276 | */ | |
277 | ||
278 | void | |
279 | cgiMoveJobs(http_t *http, /* I - Connection to server */ | |
280 | const char *dest, /* I - Destination or NULL */ | |
281 | int job_id) /* I - Job ID or 0 for all */ | |
282 | { | |
283 | int i; /* Looping var */ | |
284 | const char *user; /* Username */ | |
285 | ipp_t *request, /* IPP request */ | |
286 | *response; /* IPP response */ | |
287 | ipp_attribute_t *attr; /* Current attribute */ | |
288 | const char *name; /* Destination name */ | |
289 | const char *job_printer_uri; /* JOB_PRINTER_URI form variable */ | |
290 | char current_dest[1024]; /* Current destination */ | |
291 | ||
292 | ||
293 | /* | |
294 | * See who is logged in... | |
295 | */ | |
296 | ||
297 | if ((user = getenv("REMOTE_USER")) == NULL) | |
298 | user = "guest"; | |
299 | ||
300 | /* | |
301 | * See if the user has already selected a new destination... | |
302 | */ | |
303 | ||
304 | if ((job_printer_uri = cgiGetVariable("JOB_PRINTER_URI")) == NULL) | |
305 | { | |
306 | /* | |
307 | * Make sure necessary form variables are set... | |
308 | */ | |
309 | ||
310 | if (job_id) | |
311 | { | |
312 | char temp[255]; /* Temporary string */ | |
313 | ||
314 | ||
315 | sprintf(temp, "%d", job_id); | |
316 | cgiSetVariable("JOB_ID", temp); | |
317 | } | |
318 | ||
319 | if (dest) | |
320 | cgiSetVariable("PRINTER_NAME", dest); | |
321 | ||
322 | /* | |
323 | * No new destination specified, show the user what the available | |
324 | * printers/classes are... | |
325 | */ | |
326 | ||
327 | if (!dest) | |
328 | { | |
329 | /* | |
330 | * Get the current destination for job N... | |
331 | */ | |
332 | ||
333 | char job_uri[1024]; /* Job URI */ | |
334 | ||
335 | ||
336 | request = ippNewRequest(IPP_GET_JOB_ATTRIBUTES); | |
337 | ||
338 | snprintf(job_uri, sizeof(job_uri), "ipp://localhost/jobs/%d", job_id); | |
339 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", | |
340 | NULL, job_uri); | |
341 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, | |
342 | "requested-attributes", NULL, "job-printer-uri"); | |
343 | ||
344 | if ((response = cupsDoRequest(http, request, "/")) != NULL) | |
345 | { | |
346 | if ((attr = ippFindAttribute(response, "job-printer-uri", | |
347 | IPP_TAG_URI)) != NULL) | |
348 | { | |
349 | /* | |
350 | * Pull the name from the URI... | |
351 | */ | |
352 | ||
353 | strlcpy(current_dest, strrchr(attr->values[0].string.text, '/') + 1, | |
354 | sizeof(current_dest)); | |
355 | dest = current_dest; | |
356 | } | |
357 | ||
358 | ippDelete(response); | |
359 | } | |
360 | ||
361 | if (!dest) | |
362 | { | |
363 | /* | |
364 | * Couldn't get the current destination... | |
365 | */ | |
366 | ||
367 | cgiStartHTML(cgiText(_("Move Job"))); | |
368 | cgiShowIPPError(_("Unable to find destination for job!")); | |
369 | cgiEndHTML(); | |
370 | return; | |
371 | } | |
372 | } | |
373 | ||
374 | /* | |
375 | * Get the list of available destinations... | |
376 | */ | |
377 | ||
378 | request = ippNewRequest(CUPS_GET_PRINTERS); | |
379 | ||
380 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, | |
381 | "requested-attributes", NULL, "printer-uri-supported"); | |
382 | ||
383 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, | |
384 | "requesting-user-name", NULL, user); | |
385 | ||
386 | if ((response = cupsDoRequest(http, request, "/")) != NULL) | |
387 | { | |
388 | for (i = 0, attr = ippFindAttribute(response, "printer-uri-supported", | |
389 | IPP_TAG_URI); | |
390 | attr; | |
391 | attr = ippFindNextAttribute(response, "printer-uri-supported", | |
392 | IPP_TAG_URI)) | |
393 | { | |
394 | /* | |
395 | * Pull the name from the URI... | |
396 | */ | |
397 | ||
398 | name = strrchr(attr->values[0].string.text, '/') + 1; | |
399 | ||
400 | /* | |
401 | * If the name is not the same as the current destination, add it! | |
402 | */ | |
403 | ||
404 | if (strcasecmp(name, dest)) | |
405 | { | |
406 | cgiSetArray("JOB_PRINTER_URI", i, attr->values[0].string.text); | |
407 | cgiSetArray("JOB_PRINTER_NAME", i, name); | |
408 | i ++; | |
409 | } | |
410 | } | |
411 | ||
412 | ippDelete(response); | |
413 | } | |
414 | ||
415 | /* | |
416 | * Show the form... | |
417 | */ | |
418 | ||
419 | if (job_id) | |
420 | cgiStartHTML(cgiText(_("Move Job"))); | |
421 | else | |
422 | cgiStartHTML(cgiText(_("Move All Jobs"))); | |
423 | ||
424 | cgiCopyTemplateLang("job-move.tmpl"); | |
425 | } | |
426 | else | |
427 | { | |
428 | /* | |
429 | * Try moving the job or jobs... | |
430 | */ | |
431 | ||
432 | char uri[1024], /* Job/printer URI */ | |
433 | resource[1024], /* Post resource */ | |
434 | refresh[1024]; /* Refresh URL */ | |
435 | const char *job_printer_name; /* New printer name */ | |
436 | ||
437 | ||
438 | request = ippNewRequest(CUPS_MOVE_JOB); | |
439 | ||
440 | if (job_id) | |
441 | { | |
442 | /* | |
443 | * Move 1 job... | |
444 | */ | |
445 | ||
446 | snprintf(resource, sizeof(resource), "/jobs/%d", job_id); | |
447 | ||
448 | snprintf(uri, sizeof(uri), "ipp://localhost/jobs/%d", job_id); | |
449 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", | |
450 | NULL, uri); | |
451 | } | |
452 | else | |
453 | { | |
454 | /* | |
455 | * Move all active jobs on a destination... | |
456 | */ | |
457 | ||
458 | snprintf(resource, sizeof(resource), "/%s/%s", | |
459 | cgiGetVariable("SECTION"), dest); | |
460 | ||
461 | httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, | |
462 | "localhost", ippPort(), "/%s/%s", | |
463 | cgiGetVariable("SECTION"), dest); | |
464 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", | |
465 | NULL, uri); | |
466 | } | |
467 | ||
468 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-printer-uri", | |
469 | NULL, job_printer_uri); | |
470 | ||
471 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, | |
472 | "requesting-user-name", NULL, user); | |
473 | ||
474 | ippDelete(cupsDoRequest(http, request, resource)); | |
475 | ||
476 | /* | |
477 | * Show the results... | |
478 | */ | |
479 | ||
480 | job_printer_name = strrchr(job_printer_uri, '/') + 1; | |
481 | ||
482 | if (cupsLastError() <= IPP_OK_CONFLICT) | |
483 | { | |
484 | cgiRewriteURL(job_printer_uri, resource, sizeof(resource), NULL); | |
485 | cgiFormEncode(uri, resource, sizeof(uri)); | |
486 | snprintf(refresh, sizeof(refresh), "2;URL=%s", uri); | |
487 | cgiSetVariable("refresh_page", refresh); | |
488 | } | |
489 | ||
490 | if (job_id) | |
491 | cgiStartHTML(cgiText(_("Move Job"))); | |
492 | else | |
493 | cgiStartHTML(cgiText(_("Move All Jobs"))); | |
494 | ||
495 | if (cupsLastError() > IPP_OK_CONFLICT) | |
496 | { | |
497 | if (job_id) | |
498 | cgiShowIPPError(_("Unable to move job")); | |
499 | else | |
500 | cgiShowIPPError(_("Unable to move jobs")); | |
501 | } | |
502 | else | |
503 | { | |
504 | cgiSetVariable("JOB_PRINTER_NAME", job_printer_name); | |
505 | cgiCopyTemplateLang("job-moved.tmpl"); | |
506 | } | |
507 | } | |
508 | ||
509 | cgiEndHTML(); | |
510 | } | |
511 | ||
512 | ||
513 | /* | |
514 | * 'cgiPrintTestPage()' - Print a test page. | |
515 | */ | |
516 | ||
517 | void | |
518 | cgiPrintTestPage(http_t *http, /* I - Connection to server */ | |
519 | const char *dest) /* I - Destination printer/class */ | |
520 | { | |
521 | ipp_t *request, /* IPP request */ | |
522 | *response; /* IPP response */ | |
523 | char uri[HTTP_MAX_URI], /* Printer URI */ | |
524 | resource[1024], /* POST resource path */ | |
525 | refresh[1024], /* Refresh URL */ | |
526 | filename[1024]; /* Test page filename */ | |
527 | const char *datadir; /* CUPS_DATADIR env var */ | |
528 | const char *user; /* Username */ | |
529 | ||
530 | ||
531 | /* | |
532 | * See who is logged in... | |
533 | */ | |
534 | ||
535 | if ((user = getenv("REMOTE_USER")) == NULL) | |
536 | user = "guest"; | |
537 | ||
538 | /* | |
539 | * Locate the test page file... | |
540 | */ | |
541 | ||
542 | if ((datadir = getenv("CUPS_DATADIR")) == NULL) | |
543 | datadir = CUPS_DATADIR; | |
544 | ||
545 | snprintf(filename, sizeof(filename), "%s/data/testprint.ps", datadir); | |
546 | ||
547 | /* | |
548 | * Point to the printer/class... | |
549 | */ | |
550 | ||
551 | snprintf(resource, sizeof(resource), "/%s/%s", cgiGetVariable("SECTION"), | |
552 | dest); | |
553 | ||
554 | httpAssembleURIf(HTTP_URI_CODING_ALL, uri, sizeof(uri), "ipp", NULL, | |
555 | "localhost", ippPort(), "/%s/%s", cgiGetVariable("SECTION"), | |
556 | dest); | |
557 | ||
558 | /* | |
559 | * Build an IPP_PRINT_JOB request, which requires the following | |
560 | * attributes: | |
561 | * | |
562 | * attributes-charset | |
563 | * attributes-natural-language | |
564 | * printer-uri | |
565 | * requesting-user-name | |
566 | * document-format | |
567 | */ | |
568 | ||
569 | request = ippNewRequest(IPP_PRINT_JOB); | |
570 | ||
571 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", | |
572 | NULL, uri); | |
573 | ||
574 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, | |
575 | "requesting-user-name", NULL, user); | |
576 | ||
577 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_NAME, "job-name", | |
578 | NULL, "Test Page"); | |
579 | ||
580 | ippAddString(request, IPP_TAG_JOB, IPP_TAG_MIMETYPE, "document-format", | |
581 | NULL, "application/postscript"); | |
582 | ||
583 | /* | |
584 | * Do the request and get back a response... | |
585 | */ | |
586 | ||
587 | if ((response = cupsDoFileRequest(http, request, resource, | |
588 | filename)) != NULL) | |
589 | { | |
590 | cgiSetIPPVars(response, NULL, NULL, NULL, 0); | |
591 | ||
592 | ippDelete(response); | |
593 | } | |
594 | ||
595 | if (cupsLastError() <= IPP_OK_CONFLICT) | |
596 | { | |
597 | /* | |
598 | * Automatically reload the printer status page... | |
599 | */ | |
600 | ||
601 | cgiFormEncode(uri, resource, sizeof(uri)); | |
602 | snprintf(refresh, sizeof(refresh), "2;URL=%s", uri); | |
603 | cgiSetVariable("refresh_page", refresh); | |
604 | } | |
605 | ||
606 | cgiStartHTML(cgiText(_("Print Test Page"))); | |
607 | ||
608 | if (cupsLastError() > IPP_OK_CONFLICT) | |
609 | cgiShowIPPError(_("Unable to print test page:")); | |
610 | else | |
611 | { | |
612 | cgiSetVariable("PRINTER_NAME", dest); | |
613 | ||
614 | cgiCopyTemplateLang("test-page.tmpl"); | |
615 | } | |
616 | ||
617 | cgiEndHTML(); | |
618 | } | |
619 | ||
620 | ||
621 | /* | |
622 | * 'cgiRewriteURL()' - Rewrite a printer URI into a web browser URL... | |
623 | */ | |
624 | ||
625 | char * /* O - New URL */ | |
626 | cgiRewriteURL(const char *uri, /* I - Current URI */ | |
627 | char *url, /* O - New URL */ | |
628 | int urlsize, /* I - Size of URL buffer */ | |
629 | const char *newresource) /* I - Replacement resource */ | |
630 | { | |
631 | char method[HTTP_MAX_URI], | |
632 | userpass[HTTP_MAX_URI], | |
633 | hostname[HTTP_MAX_URI], | |
634 | rawresource[HTTP_MAX_URI], | |
635 | resource[HTTP_MAX_URI], | |
636 | /* URI components... */ | |
637 | *rawptr, /* Pointer into rawresource */ | |
638 | *resptr; /* Pointer into resource */ | |
639 | int port; /* Port number */ | |
640 | static int ishttps = -1; /* Using encryption? */ | |
641 | static const char *server; /* Name of server */ | |
642 | static char servername[1024]; | |
643 | /* Local server name */ | |
644 | static const char hexchars[] = "0123456789ABCDEF"; | |
645 | /* Hexadecimal conversion characters */ | |
646 | ||
647 | ||
648 | /* | |
649 | * Check if we have been called before... | |
650 | */ | |
651 | ||
652 | if (ishttps < 0) | |
653 | { | |
654 | /* | |
655 | * No, initialize static vars for the conversion... | |
656 | * | |
657 | * First get the server name associated with the client interface as | |
658 | * well as the locally configured hostname. We'll check *both* of | |
659 | * these to see if the printer URL is local... | |
660 | */ | |
661 | ||
662 | if ((server = getenv("SERVER_NAME")) == NULL) | |
663 | server = ""; | |
664 | ||
665 | httpGetHostname(NULL, servername, sizeof(servername)); | |
666 | ||
667 | /* | |
668 | * Then flag whether we are using SSL on this connection... | |
669 | */ | |
670 | ||
671 | ishttps = getenv("HTTPS") != NULL; | |
672 | } | |
673 | ||
674 | /* | |
675 | * Convert the URI to a URL... | |
676 | */ | |
677 | ||
678 | httpSeparateURI(HTTP_URI_CODING_ALL, uri, method, sizeof(method), userpass, | |
679 | sizeof(userpass), hostname, sizeof(hostname), &port, | |
680 | rawresource, sizeof(rawresource)); | |
681 | ||
682 | if (!strcmp(method, "ipp") || | |
683 | !strcmp(method, "http") || | |
684 | !strcmp(method, "https")) | |
685 | { | |
686 | if (newresource) | |
687 | { | |
688 | /* | |
689 | * Force the specified resource name instead of the one in the URL... | |
690 | */ | |
691 | ||
692 | strlcpy(resource, newresource, sizeof(resource)); | |
693 | } | |
694 | else | |
695 | { | |
696 | /* | |
697 | * Rewrite the resource string so it doesn't contain any | |
698 | * illegal chars... | |
699 | */ | |
700 | ||
701 | for (rawptr = rawresource, resptr = resource; *rawptr; rawptr ++) | |
702 | if ((*rawptr & 128) || *rawptr == '%' || *rawptr == ' ' || | |
703 | *rawptr == '#' || *rawptr == '?' || | |
704 | *rawptr == '.') /* For MSIE */ | |
705 | { | |
706 | if (resptr < (resource + sizeof(resource) - 3)) | |
707 | { | |
708 | *resptr++ = '%'; | |
709 | *resptr++ = hexchars[(*rawptr >> 4) & 15]; | |
710 | *resptr++ = hexchars[*rawptr & 15]; | |
711 | } | |
712 | } | |
713 | else if (resptr < (resource + sizeof(resource) - 1)) | |
714 | *resptr++ = *rawptr; | |
715 | ||
716 | *resptr = '\0'; | |
717 | } | |
718 | ||
719 | /* | |
720 | * Map local access to a local URI... | |
721 | */ | |
722 | ||
723 | if (!strcasecmp(hostname, "localhost") || | |
724 | !strncasecmp(hostname, "localhost.", 10) || | |
725 | !strcasecmp(hostname, server) || | |
726 | !strcasecmp(hostname, servername)) | |
727 | { | |
728 | /* | |
729 | * Make URI relative to the current server... | |
730 | */ | |
731 | ||
732 | strlcpy(url, resource, urlsize); | |
733 | } | |
734 | else | |
735 | { | |
736 | /* | |
737 | * Rewrite URI with HTTP/HTTPS scheme... | |
738 | */ | |
739 | ||
740 | if (userpass[0]) | |
741 | snprintf(url, urlsize, "%s://%s@%s:%d%s", | |
742 | ishttps ? "https" : "http", | |
743 | userpass, hostname, port, resource); | |
744 | else | |
745 | snprintf(url, urlsize, "%s://%s:%d%s", | |
746 | ishttps ? "https" : "http", | |
747 | hostname, port, resource); | |
748 | } | |
749 | } | |
750 | else | |
751 | strlcpy(url, uri, urlsize); | |
752 | ||
753 | return (url); | |
754 | } | |
755 | ||
756 | ||
757 | /* | |
758 | * 'cgiSetIPPObjectVars()' - Set CGI variables from an IPP object. | |
759 | */ | |
760 | ||
761 | ipp_attribute_t * /* O - Next object */ | |
762 | cgiSetIPPObjectVars( | |
763 | ipp_attribute_t *obj, /* I - Response data to be copied... */ | |
764 | const char *prefix, /* I - Prefix for name or NULL */ | |
765 | int element) /* I - Parent element number */ | |
766 | { | |
767 | ipp_attribute_t *attr; /* Attribute in response... */ | |
768 | int i; /* Looping var */ | |
769 | char name[1024], /* Name of attribute */ | |
770 | *nameptr, /* Pointer into name */ | |
771 | value[16384], /* Value(s) */ | |
772 | *valptr; /* Pointer into value */ | |
773 | struct tm *date; /* Date information */ | |
774 | ||
775 | ||
776 | fprintf(stderr, "DEBUG2: cgiSetIPPObjectVars(obj=%p, prefix=\"%s\", " | |
777 | "element=%d)\n", | |
778 | obj, prefix ? prefix : "(null)", element); | |
779 | ||
780 | /* | |
781 | * Set common CGI template variables... | |
782 | */ | |
783 | ||
784 | if (!prefix) | |
785 | cgiSetServerVersion(); | |
786 | ||
787 | /* | |
788 | * Loop through the attributes and set them for the template... | |
789 | */ | |
790 | ||
791 | for (attr = obj; attr && attr->group_tag != IPP_TAG_ZERO; attr = attr->next) | |
792 | { | |
793 | /* | |
794 | * Copy the attribute name, substituting "_" for "-"... | |
795 | */ | |
796 | ||
797 | if (!attr->name) | |
798 | continue; | |
799 | ||
800 | if (prefix) | |
801 | { | |
802 | snprintf(name, sizeof(name), "%s.", prefix); | |
803 | nameptr = name + strlen(name); | |
804 | } | |
805 | else | |
806 | nameptr = name; | |
807 | ||
808 | for (i = 0; attr->name[i] && nameptr < (name + sizeof(name) - 1); i ++) | |
809 | if (attr->name[i] == '-') | |
810 | *nameptr++ = '_'; | |
811 | else | |
812 | *nameptr++ = attr->name[i]; | |
813 | ||
814 | *nameptr = '\0'; | |
815 | ||
816 | /* | |
817 | * Add "job_printer_name" variable if we have a "job_printer_uri" | |
818 | * attribute... | |
819 | */ | |
820 | ||
821 | if (!strcmp(name, "job_printer_uri")) | |
822 | { | |
823 | if ((valptr = strrchr(attr->values[0].string.text, '/')) == NULL) | |
824 | valptr = "unknown"; | |
825 | else | |
826 | valptr ++; | |
827 | ||
828 | cgiSetArray("job_printer_name", element, valptr); | |
829 | } | |
830 | ||
831 | /* | |
832 | * Add "admin_uri" variable if we have a "printer_uri_supported" | |
833 | * attribute... | |
834 | */ | |
835 | ||
836 | if (!strcmp(name, "printer_uri_supported")) | |
837 | { | |
838 | cgiRewriteURL(attr->values[0].string.text, value, sizeof(value), | |
839 | "/admin/"); | |
840 | ||
841 | cgiSetArray("admin_uri", element, value); | |
842 | } | |
843 | ||
844 | /* | |
845 | * Copy values... | |
846 | */ | |
847 | ||
848 | value[0] = '\0'; /* Initially an empty string */ | |
849 | valptr = value; /* Start at the beginning */ | |
850 | ||
851 | for (i = 0; i < attr->num_values; i ++) | |
852 | { | |
853 | if (i) | |
854 | strlcat(valptr, ",", sizeof(value) - (valptr - value)); | |
855 | ||
856 | valptr += strlen(valptr); | |
857 | ||
858 | switch (attr->value_tag) | |
859 | { | |
860 | case IPP_TAG_INTEGER : | |
861 | case IPP_TAG_ENUM : | |
862 | if (strncmp(name, "time_at_", 8) == 0) | |
863 | { | |
864 | time_t t; /* Temporary time value */ | |
865 | ||
866 | t = (time_t)attr->values[i].integer; | |
867 | date = localtime(&t); | |
868 | ||
869 | strftime(valptr, sizeof(value) - (valptr - value), "%c", date); | |
870 | } | |
871 | else | |
872 | snprintf(valptr, sizeof(value) - (valptr - value), | |
873 | "%d", attr->values[i].integer); | |
874 | break; | |
875 | ||
876 | case IPP_TAG_BOOLEAN : | |
877 | snprintf(valptr, sizeof(value) - (valptr - value), | |
878 | "%d", attr->values[i].boolean); | |
879 | break; | |
880 | ||
881 | case IPP_TAG_NOVALUE : | |
882 | strlcat(valptr, "novalue", sizeof(value) - (valptr - value)); | |
883 | break; | |
884 | ||
885 | case IPP_TAG_RANGE : | |
886 | snprintf(valptr, sizeof(value) - (valptr - value), | |
887 | "%d-%d", attr->values[i].range.lower, | |
888 | attr->values[i].range.upper); | |
889 | break; | |
890 | ||
891 | case IPP_TAG_RESOLUTION : | |
892 | snprintf(valptr, sizeof(value) - (valptr - value), | |
893 | "%dx%d%s", attr->values[i].resolution.xres, | |
894 | attr->values[i].resolution.yres, | |
895 | attr->values[i].resolution.units == IPP_RES_PER_INCH ? | |
896 | "dpi" : "dpc"); | |
897 | break; | |
898 | ||
899 | case IPP_TAG_URI : | |
900 | if (strchr(attr->values[i].string.text, ':') && | |
901 | strcmp(name, "device_uri")) | |
902 | { | |
903 | /* | |
904 | * Rewrite URIs... | |
905 | */ | |
906 | ||
907 | if (!strcmp(name, "member_uris")) | |
908 | { | |
909 | char url[1024]; /* URL for class member... */ | |
910 | ||
911 | ||
912 | cgiRewriteURL(attr->values[i].string.text, url, | |
913 | sizeof(url), NULL); | |
914 | ||
915 | snprintf(valptr, sizeof(value) - (valptr - value), | |
916 | "<A HREF=\"%s\">%s</A>", url, | |
917 | strrchr(url, '/') + 1); | |
918 | } | |
919 | else | |
920 | cgiRewriteURL(attr->values[i].string.text, valptr, | |
921 | sizeof(value) - (valptr - value), NULL); | |
922 | break; | |
923 | } | |
924 | ||
925 | case IPP_TAG_STRING : | |
926 | case IPP_TAG_TEXT : | |
927 | case IPP_TAG_NAME : | |
928 | case IPP_TAG_KEYWORD : | |
929 | case IPP_TAG_CHARSET : | |
930 | case IPP_TAG_LANGUAGE : | |
931 | case IPP_TAG_MIMETYPE : | |
932 | strlcat(valptr, attr->values[i].string.text, | |
933 | sizeof(value) - (valptr - value)); | |
934 | break; | |
935 | ||
936 | case IPP_TAG_BEGIN_COLLECTION : | |
937 | snprintf(value, sizeof(value), "%s%d", name, i + 1); | |
938 | cgiSetIPPVars(attr->values[i].collection, NULL, NULL, value, | |
939 | element); | |
940 | break; | |
941 | ||
942 | default : | |
943 | break; /* anti-compiler-warning-code */ | |
944 | } | |
945 | } | |
946 | ||
947 | /* | |
948 | * Add the element... | |
949 | */ | |
950 | ||
951 | if (attr->value_tag != IPP_TAG_BEGIN_COLLECTION) | |
952 | { | |
953 | cgiSetArray(name, element, value); | |
954 | ||
955 | fprintf(stderr, "DEBUG2: %s[%d]=\"%s\"\n", name, element, value); | |
956 | } | |
957 | } | |
958 | ||
959 | return (attr ? attr->next : NULL); | |
960 | } | |
961 | ||
962 | ||
963 | /* | |
964 | * 'cgiSetIPPVars()' - Set CGI variables from an IPP response. | |
965 | */ | |
966 | ||
967 | int /* O - Maximum number of elements */ | |
968 | cgiSetIPPVars(ipp_t *response, /* I - Response data to be copied... */ | |
969 | const char *filter_name, /* I - Filter name */ | |
970 | const char *filter_value, /* I - Filter value */ | |
971 | const char *prefix, /* I - Prefix for name or NULL */ | |
972 | int parent_el) /* I - Parent element number */ | |
973 | { | |
974 | int element; /* Element in CGI array */ | |
975 | ipp_attribute_t *attr, /* Attribute in response... */ | |
976 | *filter; /* Filtering attribute */ | |
977 | ||
978 | ||
979 | fprintf(stderr, "DEBUG2: cgiSetIPPVars(response=%p, filter_name=\"%s\", " | |
980 | "filter_value=\"%s\", prefix=\"%s\", parent_el=%d)\n", | |
981 | response, filter_name ? filter_name : "(null)", | |
982 | filter_value ? filter_value : "(null)", | |
983 | prefix ? prefix : "(null)", parent_el); | |
984 | ||
985 | /* | |
986 | * Set common CGI template variables... | |
987 | */ | |
988 | ||
989 | if (!prefix) | |
990 | cgiSetServerVersion(); | |
991 | ||
992 | /* | |
993 | * Loop through the attributes and set them for the template... | |
994 | */ | |
995 | ||
996 | attr = response->attrs; | |
997 | ||
998 | if (!prefix) | |
999 | while (attr && attr->group_tag == IPP_TAG_OPERATION) | |
1000 | attr = attr->next; | |
1001 | ||
1002 | for (element = parent_el; attr; element ++) | |
1003 | { | |
1004 | /* | |
1005 | * Copy attributes to a separator... | |
1006 | */ | |
1007 | ||
1008 | while (attr && attr->group_tag == IPP_TAG_ZERO) | |
1009 | attr= attr->next; | |
1010 | ||
1011 | if (!attr) | |
1012 | break; | |
1013 | ||
1014 | if (filter_name) | |
1015 | { | |
1016 | for (filter = attr; | |
1017 | filter != NULL && filter->group_tag != IPP_TAG_ZERO; | |
1018 | filter = filter->next) | |
1019 | if (filter->name && !strcmp(filter->name, filter_name) && | |
1020 | (filter->value_tag == IPP_TAG_STRING || | |
1021 | (filter->value_tag >= IPP_TAG_TEXTLANG && | |
1022 | filter->value_tag <= IPP_TAG_MIMETYPE)) && | |
1023 | filter->values[0].string.text != NULL && | |
1024 | !strcasecmp(filter->values[0].string.text, filter_value)) | |
1025 | break; | |
1026 | ||
1027 | if (!filter) | |
1028 | return (element + 1); | |
1029 | ||
1030 | if (filter->group_tag == IPP_TAG_ZERO) | |
1031 | { | |
1032 | attr = filter; | |
1033 | element --; | |
1034 | continue; | |
1035 | } | |
1036 | } | |
1037 | ||
1038 | attr = cgiSetIPPObjectVars(attr, prefix, element); | |
1039 | } | |
1040 | ||
1041 | fprintf(stderr, "DEBUG2: Returing %d from cgiSetIPPVars()...\n", element); | |
1042 | ||
1043 | return (element); | |
1044 | } | |
1045 | ||
1046 | ||
1047 | /* | |
1048 | * 'cgiShowIPPError()' - Show the last IPP error message. | |
1049 | * | |
1050 | * The caller must still call cgiStartHTML() and cgiEndHTML(). | |
1051 | */ | |
1052 | ||
1053 | void | |
1054 | cgiShowIPPError(const char *message) /* I - Contextual message */ | |
1055 | { | |
1056 | cgiSetVariable("MESSAGE", cgiText(message)); | |
1057 | cgiSetVariable("ERROR", cupsLastErrorString()); | |
1058 | cgiCopyTemplateLang("error.tmpl"); | |
1059 | } | |
1060 | ||
1061 | ||
1062 | /* | |
1063 | * 'cgiShowJobs()' - Show print jobs. | |
1064 | */ | |
1065 | ||
1066 | void | |
1067 | cgiShowJobs(http_t *http, /* I - Connection to server */ | |
1068 | const char *dest) /* I - Destination name or NULL */ | |
1069 | { | |
1070 | int i; /* Looping var */ | |
1071 | const char *which_jobs; /* Which jobs to show */ | |
1072 | ipp_t *request, /* IPP request */ | |
1073 | *response; /* IPP response */ | |
1074 | cups_array_t *jobs; /* Array of job objects */ | |
1075 | ipp_attribute_t *job; /* Job object */ | |
1076 | int ascending, /* Order of jobs (0 = descending) */ | |
1077 | first, /* First job to show */ | |
1078 | count; /* Number of jobs */ | |
1079 | const char *var; /* Form variable */ | |
1080 | void *search; /* Search data */ | |
1081 | char url[1024], /* URL for prev/next/this */ | |
1082 | *urlptr, /* Position in URL */ | |
1083 | *urlend; /* End of URL */ | |
1084 | ||
1085 | ||
1086 | /* | |
1087 | * Build an IPP_GET_JOBS request, which requires the following | |
1088 | * attributes: | |
1089 | * | |
1090 | * attributes-charset | |
1091 | * attributes-natural-language | |
1092 | * printer-uri | |
1093 | */ | |
1094 | ||
1095 | request = ippNewRequest(IPP_GET_JOBS); | |
1096 | ||
1097 | if (dest) | |
1098 | { | |
1099 | httpAssembleURIf(HTTP_URI_CODING_ALL, url, sizeof(url), "ipp", NULL, | |
1100 | "localhost", ippPort(), "/printers/%s", dest); | |
1101 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "printer-uri", | |
1102 | NULL, url); | |
1103 | } | |
1104 | else | |
1105 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_URI, "job-uri", NULL, | |
1106 | "ipp://localhost/jobs"); | |
1107 | ||
1108 | if ((which_jobs = cgiGetVariable("which_jobs")) != NULL) | |
1109 | ippAddString(request, IPP_TAG_OPERATION, IPP_TAG_KEYWORD, "which-jobs", | |
1110 | NULL, which_jobs); | |
1111 | ||
1112 | cgiGetAttributes(request, "jobs.tmpl"); | |
1113 | ||
1114 | /* | |
1115 | * Do the request and get back a response... | |
1116 | */ | |
1117 | ||
1118 | if ((response = cupsDoRequest(http, request, "/")) != NULL) | |
1119 | { | |
1120 | /* | |
1121 | * Get a list of matching job objects. | |
1122 | */ | |
1123 | ||
1124 | if ((var = cgiGetVariable("QUERY")) != NULL) | |
1125 | search = cgiCompileSearch(var); | |
1126 | else | |
1127 | search = NULL; | |
1128 | ||
1129 | jobs = cgiGetIPPObjects(response, search); | |
1130 | count = cupsArrayCount(jobs); | |
1131 | ||
1132 | if (search) | |
1133 | cgiFreeSearch(search); | |
1134 | ||
1135 | /* | |
1136 | * Figure out which jobs to display... | |
1137 | */ | |
1138 | ||
1139 | if ((var = cgiGetVariable("FIRST")) != NULL) | |
1140 | first = atoi(var); | |
1141 | else | |
1142 | first = 0; | |
1143 | ||
1144 | if (first >= count) | |
1145 | first = count - CUPS_PAGE_MAX; | |
1146 | ||
1147 | first = (first / CUPS_PAGE_MAX) * CUPS_PAGE_MAX; | |
1148 | ||
1149 | if (first < 0) | |
1150 | first = 0; | |
1151 | ||
1152 | sprintf(url, "%d", count); | |
1153 | cgiSetVariable("TOTAL", url); | |
1154 | ||
1155 | if ((var = cgiGetVariable("ORDER")) != NULL) | |
1156 | ascending = !strcasecmp(var, "asc"); | |
1157 | else | |
1158 | { | |
1159 | ascending = !which_jobs || !strcasecmp(which_jobs, "not-completed"); | |
1160 | cgiSetVariable("ORDER", ascending ? "asc" : "dec"); | |
1161 | } | |
1162 | ||
1163 | if (ascending) | |
1164 | { | |
1165 | for (i = 0, job = (ipp_attribute_t *)cupsArrayIndex(jobs, first); | |
1166 | i < CUPS_PAGE_MAX && job; | |
1167 | i ++, job = (ipp_attribute_t *)cupsArrayNext(jobs)) | |
1168 | cgiSetIPPObjectVars(job, NULL, i); | |
1169 | } | |
1170 | else | |
1171 | { | |
1172 | for (i = 0, job = (ipp_attribute_t *)cupsArrayIndex(jobs, count - first - 1); | |
1173 | i < CUPS_PAGE_MAX && job; | |
1174 | i ++, job = (ipp_attribute_t *)cupsArrayPrev(jobs)) | |
1175 | cgiSetIPPObjectVars(job, NULL, i); | |
1176 | } | |
1177 | ||
1178 | /* | |
1179 | * Save navigation URLs... | |
1180 | */ | |
1181 | ||
1182 | urlend = url + sizeof(url); | |
1183 | ||
1184 | if ((var = cgiGetVariable("QUERY")) != NULL) | |
1185 | { | |
1186 | if (dest) | |
1187 | snprintf(url, sizeof(url), "/%s/%s?QUERY=", cgiGetVariable("SECTION"), | |
1188 | dest); | |
1189 | else | |
1190 | strlcpy(url, "/jobs/?QUERY=", sizeof(url)); | |
1191 | ||
1192 | urlptr = url + strlen(url); | |
1193 | ||
1194 | cgiFormEncode(urlptr, var, urlend - urlptr); | |
1195 | urlptr += strlen(urlptr); | |
1196 | ||
1197 | strlcpy(urlptr, "&", urlend - urlptr); | |
1198 | urlptr += strlen(urlptr); | |
1199 | } | |
1200 | else | |
1201 | { | |
1202 | if (dest) | |
1203 | snprintf(url, sizeof(url), "/%s/%s?", cgiGetVariable("SECTION"), dest); | |
1204 | else | |
1205 | strlcpy(url, "/jobs/?", sizeof(url)); | |
1206 | ||
1207 | urlptr = url + strlen(url); | |
1208 | } | |
1209 | ||
1210 | if (which_jobs) | |
1211 | { | |
1212 | strlcpy(urlptr, "WHICH_JOBS=", urlend - urlptr); | |
1213 | urlptr += strlen(urlptr); | |
1214 | ||
1215 | cgiFormEncode(urlptr, which_jobs, urlend - urlptr); | |
1216 | urlptr += strlen(urlptr); | |
1217 | ||
1218 | strlcpy(urlptr, "&", urlend - urlptr); | |
1219 | urlptr += strlen(urlptr); | |
1220 | } | |
1221 | ||
1222 | snprintf(urlptr, urlend - urlptr, "FIRST=%d", first); | |
1223 | cgiSetVariable("THISURL", url); | |
1224 | ||
1225 | if (first > 0) | |
1226 | { | |
1227 | snprintf(urlptr, urlend - urlptr, "FIRST=%d&ORDER=%s", | |
1228 | first - CUPS_PAGE_MAX, ascending ? "asc" : "dec"); | |
1229 | cgiSetVariable("PREVURL", url); | |
1230 | } | |
1231 | ||
1232 | if ((first + CUPS_PAGE_MAX) < count) | |
1233 | { | |
1234 | snprintf(urlptr, urlend - urlptr, "FIRST=%d&ORDER=%s", | |
1235 | first + CUPS_PAGE_MAX, ascending ? "asc" : "dec"); | |
1236 | cgiSetVariable("NEXTURL", url); | |
1237 | } | |
1238 | ||
1239 | /* | |
1240 | * Then show everything... | |
1241 | */ | |
1242 | ||
1243 | if (dest) | |
1244 | cgiSetVariable("SEARCH_DEST", dest); | |
1245 | ||
1246 | cgiCopyTemplateLang("search.tmpl"); | |
1247 | ||
1248 | cgiCopyTemplateLang("jobs-header.tmpl"); | |
1249 | ||
1250 | if (count > 0) | |
1251 | cgiCopyTemplateLang("pager.tmpl"); | |
1252 | ||
1253 | cgiCopyTemplateLang("jobs.tmpl"); | |
1254 | ||
1255 | if (count > 0) | |
1256 | cgiCopyTemplateLang("pager.tmpl"); | |
1257 | ||
1258 | cupsArrayDelete(jobs); | |
1259 | ippDelete(response); | |
1260 | } | |
1261 | } | |
1262 | ||
1263 | ||
1264 | /* | |
1265 | * 'cgiText()' - Return localized text. | |
1266 | */ | |
1267 | ||
1268 | const char * /* O - Localized message */ | |
1269 | cgiText(const char *message) /* I - Message */ | |
1270 | { | |
1271 | static cups_lang_t *language = NULL; | |
1272 | /* Language */ | |
1273 | ||
1274 | ||
1275 | if (!language) | |
1276 | language = cupsLangDefault(); | |
1277 | ||
1278 | return (_cupsLangString(language, message)); | |
1279 | } | |
1280 | ||
1281 | ||
1282 | /* | |
1283 | * End of "$Id: ipp-var.c 5838 2006-08-17 14:41:42Z mike $". | |
1284 | */ |