]> git.ipfire.org Git - thirdparty/cups.git/blame - cups/testcups.c
The EPL2 and ZPL sample drivers did not properly support the CutMedia option.
[thirdparty/cups.git] / cups / testcups.c
CommitLineData
09a101d6 1/*
2 * "$Id$"
3 *
7e86f2f6 4 * CUPS API test program for CUPS.
09a101d6 5 *
7e86f2f6
MS
6 * Copyright 2007-2014 by Apple Inc.
7 * Copyright 2007 by Easy Software Products.
09a101d6 8 *
7e86f2f6
MS
9 * These coded instructions, statements, and computer programs are the
10 * property of Apple Inc. and are protected by Federal copyright
11 * law. Distribution and use rights are outlined in the file "LICENSE.txt"
12 * which should have been included with this file. If this file is
13 * file is missing or damaged, see the license at "http://www.cups.org/".
09a101d6 14 *
7e86f2f6 15 * This file is subject to the Apple OS-Developed Software exception.
09a101d6 16 */
17
18/*
19 * Include necessary headers...
20 */
21
5489ad43 22#undef _CUPS_NO_DEPRECATED
71e16022 23#include "string-private.h"
09a101d6 24#include "cups.h"
aaf19ab0 25#include "ppd.h"
71e16022 26#include <stdlib.h>
09a101d6 27
28
a4924f6c
MS
29/*
30 * Local functions...
31 */
32
33static int dests_equal(cups_dest_t *a, cups_dest_t *b);
dcb445bc 34static int enum_cb(void *user_data, unsigned flags, cups_dest_t *dest);
a4924f6c
MS
35static void show_diffs(cups_dest_t *a, cups_dest_t *b);
36
37
09a101d6 38/*
39 * 'main()' - Main entry.
40 */
41
42int /* O - Exit status */
43main(int argc, /* I - Number of command-line arguments */
44 char *argv[]) /* I - Command-line arguments */
45{
46 int status = 0, /* Exit status */
a4924f6c 47 i, /* Looping var */
09a101d6 48 num_dests; /* Number of destinations */
49 cups_dest_t *dests, /* Destinations */
a4924f6c
MS
50 *dest, /* Current destination */
51 *named_dest; /* Current named destination */
09a101d6 52 const char *ppdfile; /* PPD file */
53 ppd_file_t *ppd; /* PPD file data */
54 int num_jobs; /* Number of jobs for queue */
55 cups_job_t *jobs; /* Jobs for queue */
56
a29fd7dd 57
dfd5680b
MS
58 if (argc > 1)
59 {
dcb445bc
MS
60 if (!strcmp(argv[1], "enum"))
61 {
a29fd7dd
MS
62 cups_ptype_t mask = CUPS_PRINTER_LOCAL,
63 /* Printer type mask */
64 type = CUPS_PRINTER_LOCAL;
65 /* Printer type */
66 int msec = 0; /* Timeout in milliseconds */
67
68
69 for (i = 2; i < argc; i ++)
70 if (isdigit(argv[i][0] & 255) || argv[i][0] == '.')
71 msec = (int)(atof(argv[i]) * 1000);
72 else if (!_cups_strcasecmp(argv[i], "bw"))
73 {
74 mask |= CUPS_PRINTER_BW;
75 type |= CUPS_PRINTER_BW;
76 }
77 else if (!_cups_strcasecmp(argv[i], "color"))
78 {
79 mask |= CUPS_PRINTER_COLOR;
80 type |= CUPS_PRINTER_COLOR;
81 }
82 else if (!_cups_strcasecmp(argv[i], "mono"))
83 {
84 mask |= CUPS_PRINTER_COLOR;
85 }
86 else if (!_cups_strcasecmp(argv[i], "duplex"))
87 {
88 mask |= CUPS_PRINTER_DUPLEX;
89 type |= CUPS_PRINTER_DUPLEX;
90 }
91 else if (!_cups_strcasecmp(argv[i], "simplex"))
92 {
93 mask |= CUPS_PRINTER_DUPLEX;
94 }
95 else if (!_cups_strcasecmp(argv[i], "staple"))
96 {
97 mask |= CUPS_PRINTER_STAPLE;
98 type |= CUPS_PRINTER_STAPLE;
99 }
100 else if (!_cups_strcasecmp(argv[i], "copies"))
101 {
102 mask |= CUPS_PRINTER_COPIES;
103 type |= CUPS_PRINTER_COPIES;
104 }
105 else if (!_cups_strcasecmp(argv[i], "collate"))
106 {
107 mask |= CUPS_PRINTER_COLLATE;
108 type |= CUPS_PRINTER_COLLATE;
109 }
110 else if (!_cups_strcasecmp(argv[i], "punch"))
111 {
112 mask |= CUPS_PRINTER_PUNCH;
113 type |= CUPS_PRINTER_PUNCH;
114 }
115 else if (!_cups_strcasecmp(argv[i], "cover"))
116 {
117 mask |= CUPS_PRINTER_COVER;
118 type |= CUPS_PRINTER_COVER;
119 }
120 else if (!_cups_strcasecmp(argv[i], "bind"))
121 {
122 mask |= CUPS_PRINTER_BIND;
123 type |= CUPS_PRINTER_BIND;
124 }
125 else if (!_cups_strcasecmp(argv[i], "sort"))
126 {
127 mask |= CUPS_PRINTER_SORT;
128 type |= CUPS_PRINTER_SORT;
129 }
130 else if (!_cups_strcasecmp(argv[i], "mfp"))
131 {
132 mask |= CUPS_PRINTER_MFP;
133 type |= CUPS_PRINTER_MFP;
134 }
135 else if (!_cups_strcasecmp(argv[i], "printer"))
136 {
137 mask |= CUPS_PRINTER_MFP;
138 }
139 else if (!_cups_strcasecmp(argv[i], "large"))
140 {
141 mask |= CUPS_PRINTER_LARGE;
142 type |= CUPS_PRINTER_LARGE;
143 }
144 else if (!_cups_strcasecmp(argv[i], "medium"))
145 {
146 mask |= CUPS_PRINTER_MEDIUM;
147 type |= CUPS_PRINTER_MEDIUM;
148 }
149 else if (!_cups_strcasecmp(argv[i], "small"))
150 {
151 mask |= CUPS_PRINTER_SMALL;
152 type |= CUPS_PRINTER_SMALL;
153 }
154 else
155 fprintf(stderr, "Unknown argument \"%s\" ignored...\n", argv[i]);
156
157 cupsEnumDests(CUPS_DEST_FLAGS_NONE, msec, NULL, type, mask, enum_cb, NULL);
dfd5680b 158 }
dcb445bc 159 else if (!strcmp(argv[1], "password"))
dfd5680b 160 {
dcb445bc
MS
161 const char *pass = cupsGetPassword("Password:");
162 /* Password string */
dfd5680b 163
dcb445bc
MS
164 if (pass)
165 printf("Password entered: %s\n", pass);
166 else
167 puts("No password entered.");
dfd5680b 168 }
4972f4fe
MS
169 else if (!strcmp(argv[1], "ppd") && argc == 3)
170 {
171 /*
172 * ./testcups ppd printer
173 */
174
175 http_status_t http_status; /* Status */
176 char buffer[1024]; /* PPD filename */
177 time_t modtime = 0; /* Last modified */
178
179 if ((http_status = cupsGetPPD3(CUPS_HTTP_DEFAULT, argv[2], &modtime,
180 buffer, sizeof(buffer))) != HTTP_STATUS_OK)
181 printf("Unable to get PPD: %d (%s)\n", (int)http_status,
182 cupsLastErrorString());
183 else
184 puts(buffer);
185 }
dcb445bc
MS
186 else if (!strcmp(argv[1], "print") && argc == 5)
187 {
188 /*
4972f4fe 189 * ./testcups print printer file interval
dcb445bc 190 */
aaf19ab0 191
dcb445bc
MS
192 int interval, /* Interval between writes */
193 job_id; /* Job ID */
194 cups_file_t *fp; /* Print file */
195 char buffer[16384]; /* Read/write buffer */
196 ssize_t bytes; /* Bytes read/written */
dfd5680b 197
dcb445bc
MS
198 if ((fp = cupsFileOpen(argv[3], "r")) == NULL)
199 {
200 printf("Unable to open \"%s\": %s\n", argv[2], strerror(errno));
201 return (1);
202 }
dfd5680b 203
dcb445bc
MS
204 if ((job_id = cupsCreateJob(CUPS_HTTP_DEFAULT, argv[2], "testcups", 0,
205 NULL)) <= 0)
206 {
207 printf("Unable to create print job on %s: %s\n", argv[1],
208 cupsLastErrorString());
209 return (1);
210 }
211
212 interval = atoi(argv[4]);
dfd5680b 213
dcb445bc 214 if (cupsStartDocument(CUPS_HTTP_DEFAULT, argv[1], job_id, argv[2],
cb7f98ee 215 CUPS_FORMAT_AUTO, 1) != HTTP_STATUS_CONTINUE)
dfd5680b 216 {
dcb445bc 217 puts("Unable to start document!");
dfd5680b
MS
218 return (1);
219 }
220
dcb445bc
MS
221 while ((bytes = cupsFileRead(fp, buffer, sizeof(buffer))) > 0)
222 {
223 printf("Writing %d bytes...\n", (int)bytes);
224
7e86f2f6 225 if (cupsWriteRequestData(CUPS_HTTP_DEFAULT, buffer, (size_t)bytes) != HTTP_STATUS_CONTINUE)
dcb445bc
MS
226 {
227 puts("Unable to write bytes!");
228 return (1);
229 }
dfd5680b 230
dcb445bc 231 if (interval > 0)
7e86f2f6 232 sleep((unsigned)interval);
dcb445bc
MS
233 }
234
235 cupsFileClose(fp);
dfd5680b 236
cb7f98ee
MS
237 if (cupsFinishDocument(CUPS_HTTP_DEFAULT,
238 argv[1]) > IPP_STATUS_OK_IGNORED_OR_SUBSTITUTED)
dcb445bc
MS
239 {
240 puts("Unable to finish document!");
241 return (1);
242 }
243 }
244 else
dfd5680b 245 {
dcb445bc
MS
246 puts("Usage:");
247 puts("");
248 puts("Run basic unit tests:");
249 puts("");
250 puts(" ./testcups");
251 puts("");
252 puts("Enumerate printers (for N seconds, -1 for indefinitely):");
253 puts("");
254 puts(" ./testcups enum [seconds]");
255 puts("");
256 puts("Ask for a password:");
257 puts("");
258 puts(" ./testcups password");
259 puts("");
4972f4fe
MS
260 puts("Get the PPD file:");
261 puts("");
262 puts(" ./testcups ppd printer");
263 puts("");
dcb445bc
MS
264 puts("Print a file (interval controls delay between buffers in seconds):");
265 puts("");
266 puts(" ./testcups print printer file interval");
dfd5680b
MS
267 return (1);
268 }
269
270 return (0);
271 }
272
09a101d6 273 /*
274 * cupsGetDests()
275 */
276
277 fputs("cupsGetDests: ", stdout);
278 fflush(stdout);
279
280 num_dests = cupsGetDests(&dests);
281
282 if (num_dests == 0)
283 {
284 puts("FAIL");
285 return (1);
286 }
287 else
a4924f6c
MS
288 {
289 printf("PASS (%d dests)\n", num_dests);
290
291 for (i = num_dests, dest = dests; i > 0; i --, dest ++)
292 {
293 printf(" %s", dest->name);
294
295 if (dest->instance)
296 printf(" /%s", dest->instance);
297
298 if (dest->is_default)
299 puts(" ***DEFAULT***");
300 else
301 putchar('\n');
302 }
303 }
304
305 /*
306 * cupsGetDest(NULL)
307 */
308
309 fputs("cupsGetDest(NULL): ", stdout);
310 fflush(stdout);
311
312 if ((dest = cupsGetDest(NULL, NULL, num_dests, dests)) == NULL)
313 {
314 for (i = num_dests, dest = dests; i > 0; i --, dest ++)
315 if (dest->is_default)
316 break;
317
318 if (i)
319 {
320 status = 1;
321 puts("FAIL");
322 }
323 else
324 puts("PASS (no default)");
325
326 dest = NULL;
327 }
328 else
329 printf("PASS (%s)\n", dest->name);
330
331 /*
332 * cupsGetNamedDest(NULL, NULL, NULL)
333 */
334
335 fputs("cupsGetNamedDest(NULL, NULL, NULL): ", stdout);
336 fflush(stdout);
337
338 if ((named_dest = cupsGetNamedDest(NULL, NULL, NULL)) == NULL ||
339 !dests_equal(dest, named_dest))
340 {
341 if (!dest)
342 puts("PASS (no default)");
343 else if (named_dest)
344 {
345 puts("FAIL (different values)");
346 show_diffs(dest, named_dest);
347 status = 1;
348 }
349 else
350 {
351 puts("FAIL (no default)");
352 status = 1;
353 }
354 }
355 else
356 printf("PASS (%s)\n", named_dest->name);
357
358 if (named_dest)
359 cupsFreeDests(1, named_dest);
09a101d6 360
361 /*
362 * cupsGetDest(printer)
363 */
364
365 printf("cupsGetDest(\"%s\"): ", dests[num_dests / 2].name);
366 fflush(stdout);
367
368 if ((dest = cupsGetDest(dests[num_dests / 2].name, NULL, num_dests,
369 dests)) == NULL)
370 {
09a101d6 371 puts("FAIL");
91c84a35 372 return (1);
09a101d6 373 }
374 else
375 puts("PASS");
376
377 /*
a4924f6c 378 * cupsGetNamedDest(NULL, printer, instance)
09a101d6 379 */
380
a4924f6c
MS
381 printf("cupsGetNamedDest(NULL, \"%s\", \"%s\"): ", dest->name,
382 dest->instance ? dest->instance : "(null)");
09a101d6 383 fflush(stdout);
384
a4924f6c
MS
385 if ((named_dest = cupsGetNamedDest(NULL, dest->name,
386 dest->instance)) == NULL ||
387 !dests_equal(dest, named_dest))
09a101d6 388 {
a4924f6c
MS
389 if (named_dest)
390 {
391 puts("FAIL (different values)");
392 show_diffs(dest, named_dest);
393 }
394 else
395 puts("FAIL (no destination)");
396
397
09a101d6 398 status = 1;
09a101d6 399 }
400 else
401 puts("PASS");
402
a4924f6c
MS
403 if (named_dest)
404 cupsFreeDests(1, named_dest);
405
09a101d6 406 /*
407 * cupsPrintFile()
408 */
409
410 fputs("cupsPrintFile: ", stdout);
411 fflush(stdout);
412
426c6a59 413 if (cupsPrintFile(dest->name, "../data/testprint", "Test Page",
09a101d6 414 dest->num_options, dest->options) <= 0)
415 {
426c6a59 416 printf("FAIL (%s)\n", cupsLastErrorString());
1f0275e3 417 return (1);
09a101d6 418 }
419 else
420 puts("PASS");
421
422 /*
423 * cupsGetPPD(printer)
424 */
425
426 fputs("cupsGetPPD(): ", stdout);
427 fflush(stdout);
428
429 if ((ppdfile = cupsGetPPD(dest->name)) == NULL)
430 {
09a101d6 431 puts("FAIL");
432 }
433 else
434 {
435 puts("PASS");
436
437 /*
438 * ppdOpenFile()
439 */
440
441 fputs("ppdOpenFile(): ", stdout);
442 fflush(stdout);
443
444 if ((ppd = ppdOpenFile(ppdfile)) == NULL)
445 {
446 puts("FAIL");
447 return (1);
448 }
449 else
450 puts("PASS");
451
452 ppdClose(ppd);
453 unlink(ppdfile);
454 }
455
456 /*
457 * cupsGetJobs()
458 */
459
460 fputs("cupsGetJobs: ", stdout);
461 fflush(stdout);
462
463 num_jobs = cupsGetJobs(&jobs, NULL, 0, -1);
464
465 if (num_jobs == 0)
466 {
467 puts("FAIL");
468 return (1);
469 }
470 else
471 puts("PASS");
472
473 cupsFreeJobs(num_jobs, jobs);
474 cupsFreeDests(num_dests, dests);
475
476 return (status);
477}
478
479
a4924f6c
MS
480/*
481 * 'dests_equal()' - Determine whether two destinations are equal.
482 */
483
484static int /* O - 1 if equal, 0 if not equal */
485dests_equal(cups_dest_t *a, /* I - First destination */
486 cups_dest_t *b) /* I - Second destination */
487{
488 int i; /* Looping var */
489 cups_option_t *aoption; /* Current option */
490 const char *bval; /* Option value */
491
492
493 if (a == b)
494 return (1);
495
1f0275e3 496 if (!a || !b)
a4924f6c
MS
497 return (0);
498
88f9aafc 499 if (_cups_strcasecmp(a->name, b->name) ||
a4924f6c
MS
500 (a->instance && !b->instance) ||
501 (!a->instance && b->instance) ||
88f9aafc 502 (a->instance && _cups_strcasecmp(a->instance, b->instance)) ||
a4924f6c
MS
503 a->num_options != b->num_options)
504 return (0);
505
506 for (i = a->num_options, aoption = a->options; i > 0; i --, aoption ++)
507 if ((bval = cupsGetOption(aoption->name, b->num_options,
508 b->options)) == NULL ||
509 strcmp(aoption->value, bval))
510 return (0);
511
512 return (1);
513}
514
515
dcb445bc
MS
516/*
517 * 'enum_cb()' - Report additions and removals.
518 */
519
520static int /* O - 1 to continue, 0 to stop */
521enum_cb(void *user_data, /* I - User data (unused) */
522 unsigned flags, /* I - Destination flags */
523 cups_dest_t *dest) /* I - Destination */
524{
a29fd7dd
MS
525 int i; /* Looping var */
526 cups_option_t *option; /* Current option */
527
528
7e86f2f6
MS
529 (void)user_data;
530
dcb445bc 531 if (flags & CUPS_DEST_FLAGS_REMOVED)
a29fd7dd 532 printf("Removed '%s':\n", dest->name);
dcb445bc 533 else
a29fd7dd
MS
534 printf("Added '%s':\n", dest->name);
535
536 for (i = dest->num_options, option = dest->options; i > 0; i --, option ++)
537 printf(" %s=\"%s\"\n", option->name, option->value);
538
539 putchar('\n');
dcb445bc
MS
540
541 return (1);
542}
543
544
a4924f6c
MS
545/*
546 * 'show_diffs()' - Show differences between two destinations.
547 */
548
549static void
550show_diffs(cups_dest_t *a, /* I - First destination */
551 cups_dest_t *b) /* I - Second destination */
552{
553 int i; /* Looping var */
554 cups_option_t *aoption; /* Current option */
555 const char *bval; /* Option value */
556
557
558 if (!a || !b)
559 return;
560
561 puts(" Item cupsGetDest cupsGetNamedDest");
562 puts(" -------------------- -------------------- --------------------");
563
88f9aafc 564 if (_cups_strcasecmp(a->name, b->name))
a4924f6c
MS
565 printf(" name %-20.20s %-20.20s\n", a->name, b->name);
566
567 if ((a->instance && !b->instance) ||
568 (!a->instance && b->instance) ||
88f9aafc 569 (a->instance && _cups_strcasecmp(a->instance, b->instance)))
a4924f6c
MS
570 printf(" instance %-20.20s %-20.20s\n",
571 a->instance ? a->instance : "(null)",
572 b->instance ? b->instance : "(null)");
573
574 if (a->num_options != b->num_options)
575 printf(" num_options %-20d %-20d\n", a->num_options,
576 b->num_options);
577
578 for (i = a->num_options, aoption = a->options; i > 0; i --, aoption ++)
579 if ((bval = cupsGetOption(aoption->name, b->num_options,
580 b->options)) == NULL ||
581 strcmp(aoption->value, bval))
582 printf(" %-20.20s %-20.20s %-20.20s\n", aoption->name,
583 aoption->value, bval ? bval : "(null)");
584}
585
586
09a101d6 587/*
1f0275e3 588 * End of "$Id$".
09a101d6 589 */