7a733b1c3cbbe6e0afd16cbc09aab21b711633ed
[obnox/samba/samba-obnox.git] / source3 / printing / pcap.c
1 /* 
2    Unix SMB/CIFS implementation.
3    printcap parsing
4    Copyright (C) Karl Auer 1993-1998
5
6    Re-working by Martin Kiff, 1994
7    
8    Re-written again by Andrew Tridgell
9
10    Modified for SVID support by Norm Jacobs, 1997
11
12    Modified for CUPS support by Michael Sweet, 1999
13    
14    This program is free software; you can redistribute it and/or modify
15    it under the terms of the GNU General Public License as published by
16    the Free Software Foundation; either version 3 of the License, or
17    (at your option) any later version.
18    
19    This program is distributed in the hope that it will be useful,
20    but WITHOUT ANY WARRANTY; without even the implied warranty of
21    MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the
22    GNU General Public License for more details.
23    
24    You should have received a copy of the GNU General Public License
25    along with this program.  If not, see <http://www.gnu.org/licenses/>.
26 */
27
28 /*
29  *  Modified to call SVID/XPG4 support if printcap name is set to "lpstat"
30  *  in smb.conf under Solaris.
31  *
32  *  Modified to call CUPS support if printcap name is set to "cups"
33  *  in smb.conf.
34  *
35  *  Modified to call iPrint support if printcap name is set to "iprint"
36  *  in smb.conf.
37  */
38
39 #include "includes.h"
40 #include "printing/pcap.h"
41 #include "printer_list.h"
42
43 struct pcap_cache {
44         char *name;
45         char *comment;
46         struct pcap_cache *next;
47 };
48
49 bool pcap_cache_add_specific(struct pcap_cache **ppcache, const char *name, const char *comment)
50 {
51         struct pcap_cache *p;
52
53         if (name == NULL || ((p = SMB_MALLOC_P(struct pcap_cache)) == NULL))
54                 return false;
55
56         p->name = SMB_STRDUP(name);
57         p->comment = (comment && *comment) ? SMB_STRDUP(comment) : NULL;
58
59         DEBUG(11,("pcap_cache_add_specific: Adding name %s info %s\n",
60                 p->name, p->comment ? p->comment : ""));
61
62         p->next = *ppcache;
63         *ppcache = p;
64
65         return true;
66 }
67
68 void pcap_cache_destroy_specific(struct pcap_cache **pp_cache)
69 {
70         struct pcap_cache *p, *next;
71
72         for (p = *pp_cache; p != NULL; p = next) {
73                 next = p->next;
74
75                 SAFE_FREE(p->name);
76                 SAFE_FREE(p->comment);
77                 SAFE_FREE(p);
78         }
79         *pp_cache = NULL;
80 }
81
82 bool pcap_cache_add(const char *name, const char *comment)
83 {
84         NTSTATUS status;
85         time_t t = time_mono(NULL);
86
87         status = printer_list_set_printer(talloc_tos(), name, comment, t);
88         return NT_STATUS_IS_OK(status);
89 }
90
91 bool pcap_cache_loaded(void)
92 {
93         NTSTATUS status;
94         time_t last;
95
96         status = printer_list_get_last_refresh(&last);
97         return NT_STATUS_IS_OK(status);
98 }
99
100 bool pcap_cache_replace(const struct pcap_cache *pcache)
101 {
102         const struct pcap_cache *p;
103         NTSTATUS status;
104
105         status = printer_list_mark_reload();
106         if (!NT_STATUS_IS_OK(status)) {
107                 DEBUG(0, ("Failed to mark printer list for reload!\n"));
108                 return false;
109         }
110
111         for (p = pcache; p; p = p->next) {
112                 pcap_cache_add(p->name, p->comment);
113         }
114
115         status = printer_list_clean_old();
116         if (!NT_STATUS_IS_OK(status)) {
117                 DEBUG(0, ("Failed to cleanup printer list!\n"));
118                 return false;
119         }
120
121         return true;
122 }
123
124 void pcap_cache_reload(struct tevent_context *ev,
125                        struct messaging_context *msg_ctx,
126                        void (*post_cache_fill_fn)(struct tevent_context *,
127                                                   struct messaging_context *))
128 {
129         const char *pcap_name = lp_printcapname();
130         bool pcap_reloaded = False;
131         NTSTATUS status;
132         bool post_cache_fill_fn_handled = false;
133
134         DEBUG(3, ("reloading printcap cache\n"));
135
136         /* only go looking if no printcap name supplied */
137         if (pcap_name == NULL || *pcap_name == 0) {
138                 DEBUG(0, ("No printcap file name configured!\n"));
139                 return;
140         }
141
142         if (!printer_list_need_refresh()) {
143                 /* has been just refeshed, skip */
144                 DEBUG(5, ("Refresh just happend, skipping.\n"));
145                 return;
146         }
147
148         status = printer_list_mark_reload();
149         if (!NT_STATUS_IS_OK(status)) {
150                 DEBUG(0, ("Failed to mark printer list for reload!\n"));
151                 return;
152         }
153
154 #ifdef HAVE_CUPS
155         if (strequal(pcap_name, "cups")) {
156                 pcap_reloaded = cups_cache_reload(ev, msg_ctx,
157                                                   post_cache_fill_fn);
158                 /*
159                  * cups_cache_reload() is async and calls post_cache_fill_fn()
160                  * on successful completion
161                  */
162                 post_cache_fill_fn_handled = true;
163                 goto done;
164         }
165 #endif
166
167 #ifdef HAVE_IPRINT
168         if (strequal(pcap_name, "iprint")) {
169                 pcap_reloaded = iprint_cache_reload();
170                 goto done;
171         }
172 #endif
173
174 #if defined(SYSV) || defined(HPUX)
175         if (strequal(pcap_name, "lpstat")) {
176                 pcap_reloaded = sysv_cache_reload();
177                 goto done;
178         }
179 #endif
180
181 #ifdef AIX
182         if (strstr_m(pcap_name, "/qconfig") != NULL) {
183                 pcap_reloaded = aix_cache_reload();
184                 goto done;
185         }
186 #endif
187
188         pcap_reloaded = std_pcap_cache_reload(pcap_name);
189
190 done:
191         DEBUG(3, ("reload status: %s\n", (pcap_reloaded) ? "ok" : "error"));
192
193         if ((pcap_reloaded) && (post_cache_fill_fn_handled == false)) {
194                 /* cleanup old entries only if the operation was successful,
195                  * otherwise keep around the old entries until we can
196                  * successfuly reaload */
197                 status = printer_list_clean_old();
198                 if (!NT_STATUS_IS_OK(status)) {
199                         DEBUG(0, ("Failed to cleanup printer list!\n"));
200                 }
201                 if (post_cache_fill_fn != NULL) {
202                         post_cache_fill_fn(ev, msg_ctx);
203                 }
204         }
205
206         return;
207 }
208
209
210 bool pcap_printername_ok(const char *printername)
211 {
212         NTSTATUS status;
213
214         status = printer_list_get_printer(talloc_tos(), printername, NULL, 0);
215         return NT_STATUS_IS_OK(status);
216 }
217
218 /***************************************************************************
219 run a function on each printer name in the printcap file.
220 ***************************************************************************/
221
222 void pcap_printer_fn_specific(const struct pcap_cache *pc,
223                         void (*fn)(const char *, const char *, void *),
224                         void *pdata)
225 {
226         const struct pcap_cache *p;
227
228         for (p = pc; p != NULL; p = p->next)
229                 fn(p->name, p->comment, pdata);
230
231         return;
232 }
233
234 void pcap_printer_fn(void (*fn)(const char *, const char *, void *), void *pdata)
235 {
236         NTSTATUS status;
237
238         status = printer_list_run_fn(fn, pdata);
239         if (!NT_STATUS_IS_OK(status)) {
240                 DEBUG(3, ("Failed to run fn for all printers!\n"));
241         }
242         return;
243 }