OLD | NEW |
1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2010 The Chromium Authors. All rights reserved. |
2 // Use of this source code is governed by a BSD-style license that can be | 2 // Use of this source code is governed by a BSD-style license that can be |
3 // found in the LICENSE file. | 3 // found in the LICENSE file. |
4 | 4 |
5 #include "chrome/service/cloud_print/printer_info.h" | 5 #include "chrome/service/cloud_print/printer_info.h" |
6 | 6 |
7 #include <cups/cups.h> | 7 #include "base/logging.h" |
8 #include <list> | |
9 #include <map> | |
10 | 8 |
11 #include "base/json/json_reader.h" | 9 // TODO(sanjeevr): Implement the Linux interfaces. |
12 #include "base/file_path.h" | |
13 #include "base/file_util.h" | |
14 #include "base/logging.h" | |
15 #include "base/lock.h" | |
16 #include "base/message_loop.h" | |
17 #include "base/rand_util.h" | |
18 #include "base/string_util.h" | |
19 #include "base/task.h" | |
20 #include "base/values.h" | |
21 #include "base/utf_string_conversions.h" | |
22 | |
23 namespace cloud_print { | 10 namespace cloud_print { |
24 | 11 |
25 static const char kCUPSPrinterInfoOpt[] = "printer-info"; | |
26 static const char kCUPSPrinterStateOpt[] = "printer-state"; | |
27 | |
28 void EnumeratePrinters(PrinterList* printer_list) { | 12 void EnumeratePrinters(PrinterList* printer_list) { |
29 DCHECK(printer_list); | 13 DCHECK(printer_list); |
30 printer_list->clear(); | 14 NOTIMPLEMENTED(); |
31 | |
32 cups_dest_t* destinations = NULL; | |
33 int num_dests = cupsGetDests(&destinations); | |
34 | |
35 for (int i = 0; i < num_dests; i++) { | |
36 PrinterBasicInfo printer_info; | |
37 printer_info.printer_name = destinations[i].name; | |
38 | |
39 const char* info = cupsGetOption(kCUPSPrinterInfoOpt, | |
40 destinations[i].num_options, destinations[i].options); | |
41 if (info != NULL) | |
42 printer_info.printer_description = info; | |
43 | |
44 const char* state = cupsGetOption(kCUPSPrinterStateOpt, | |
45 destinations[i].num_options, destinations[i].options); | |
46 if (state != NULL) | |
47 StringToInt(state, &printer_info.printer_status); | |
48 | |
49 printer_list->push_back(printer_info); | |
50 } | |
51 | |
52 cupsFreeDests(num_dests, destinations); | |
53 } | 15 } |
54 | 16 |
55 bool GetPrinterCapsAndDefaults(const std::string& printer_name, | 17 bool GetPrinterCapsAndDefaults(const std::string& printer_name, |
56 PrinterCapsAndDefaults* printer_info) { | 18 PrinterCapsAndDefaults* printer_info) { |
57 DCHECK(printer_info); | 19 NOTIMPLEMENTED(); |
58 | 20 return false; |
59 static Lock ppd_lock; | |
60 // cupsGetPPD returns a filename stored in a static buffer in CUPS. | |
61 // Protect this code with lock. | |
62 ppd_lock.Acquire(); | |
63 FilePath ppd_path(cupsGetPPD(printer_name.c_str())); | |
64 ppd_lock.Release(); | |
65 | |
66 std::string content; | |
67 bool res = file_util::ReadFileToString(ppd_path, &content); | |
68 | |
69 file_util::Delete(ppd_path, false); | |
70 | |
71 if (res) { | |
72 printer_info->printer_capabilities.swap(content); | |
73 printer_info->caps_mime_type = "application/pagemaker"; | |
74 // In CUPS, printer defaults is a part of PPD file. Nothing to upload here. | |
75 printer_info->printer_defaults.clear(); | |
76 printer_info->defaults_mime_type.clear(); | |
77 } | |
78 | |
79 return res; | |
80 } | 21 } |
81 | 22 |
82 bool ValidatePrintTicket(const std::string& printer_name, | 23 bool ValidatePrintTicket(const std::string& printer_name, |
83 const std::string& print_ticket_data) { | 24 const std::string& print_ticket_data) { |
84 scoped_ptr<Value> ticket_value(base::JSONReader::Read(print_ticket_data, | 25 NOTIMPLEMENTED(); |
85 false)); | 26 return false; |
86 return ticket_value != NULL && ticket_value->IsType(Value::TYPE_DICTIONARY); | |
87 } | 27 } |
88 | 28 |
89 std::string GenerateProxyId() { | 29 std::string GenerateProxyId() { |
90 // TODO(gene): This code should generate a unique id for proxy. ID should be | 30 NOTIMPLEMENTED(); |
91 // unique for this user. Rand may return the same number. We'll need to change | 31 return std::string(); |
92 // this in the future. | |
93 std::string id("CP_PROXY_"); | |
94 id += Uint64ToString(base::RandUint64()); | |
95 return id; | |
96 } | |
97 | |
98 // Print ticket on linux is a JSON string containing only one dictionary. | |
99 bool ParsePrintTicket(const std::string& print_ticket, | |
100 std::map<std::string, std::string>* options) { | |
101 DCHECK(options); | |
102 scoped_ptr<Value> ticket_value(base::JSONReader::Read(print_ticket, false)); | |
103 if (ticket_value == NULL || !ticket_value->IsType(Value::TYPE_DICTIONARY)) | |
104 return false; | |
105 | |
106 options->clear(); | |
107 DictionaryValue* ticket_dict = | |
108 static_cast<DictionaryValue*>(ticket_value.get()); | |
109 DictionaryValue::key_iterator it(ticket_dict->begin_keys()); | |
110 for (; it != ticket_dict->end_keys(); ++it) { | |
111 std::wstring key = *it; | |
112 std::string value; | |
113 if (ticket_dict->GetString(key, &value)) { | |
114 (*options)[WideToUTF8(key.c_str())] = value; | |
115 } | |
116 } | |
117 | |
118 return true; | |
119 } | 32 } |
120 | 33 |
121 bool SpoolPrintJob(const std::string& print_ticket, | 34 bool SpoolPrintJob(const std::string& print_ticket, |
122 const FilePath& print_data_file_path, | 35 const FilePath& print_data_file_path, |
123 const std::string& print_data_mime_type, | 36 const std::string& print_data_mime_type, |
124 const std::string& printer_name, | 37 const std::string& printer_name, |
125 const std::string& job_title, | 38 const std::string& job_title, |
126 PlatformJobId* job_id_ret) { | 39 PlatformJobId* job_id_ret) { |
127 DCHECK(job_id_ret); | 40 NOTIMPLEMENTED(); |
128 | 41 return false; |
129 // We need to store options as char* string for the duration of the | |
130 // cupsPrintFile call. We'll use map here to store options, since | |
131 // Dictionary value from JSON parser returns wchat_t. | |
132 std::map<std::string, std::string> options; | |
133 bool res = ParsePrintTicket(print_ticket, &options); | |
134 DCHECK(res); // If print ticket is invalid we still print using defaults. | |
135 | |
136 std::vector<cups_option_t> cups_options; | |
137 std::map<std::string, std::string>::iterator it; | |
138 for (it = options.begin(); it != options.end(); ++it) { | |
139 cups_option_t opt; | |
140 opt.name = const_cast<char*>(it->first.c_str()); | |
141 opt.value = const_cast<char*>(it->second.c_str()); | |
142 cups_options.push_back(opt); | |
143 } | |
144 | |
145 int job_id = cupsPrintFile(printer_name.c_str(), | |
146 print_data_file_path.value().c_str(), | |
147 job_title.c_str(), | |
148 cups_options.size(), | |
149 &(cups_options[0])); | |
150 | |
151 if (job_id == 0) | |
152 return false; | |
153 | |
154 *job_id_ret = job_id; | |
155 return true; | |
156 } | 42 } |
157 | 43 |
158 bool GetJobDetails(const std::string& printer_name, | 44 bool GetJobDetails(const std::string& printer_name, |
159 PlatformJobId job_id, | 45 PlatformJobId job_id, |
160 PrintJobDetails *job_details) { | 46 PrintJobDetails *job_details) { |
161 DCHECK(job_details); | 47 NOTIMPLEMENTED(); |
162 cups_job_t* jobs = NULL; | |
163 int num_jobs = cupsGetJobs(&jobs, printer_name.c_str(), 1, -1); | |
164 | |
165 bool found = false; | |
166 for (int i = 0; i < num_jobs; i++) { | |
167 if (jobs[i].id == job_id) { | |
168 found = true; | |
169 switch (jobs[i].state) { | |
170 case IPP_JOB_PENDING : | |
171 case IPP_JOB_HELD : | |
172 case IPP_JOB_PROCESSING : | |
173 job_details->status = PRINT_JOB_STATUS_IN_PROGRESS; | |
174 break; | |
175 case IPP_JOB_STOPPED : | |
176 case IPP_JOB_CANCELLED : | |
177 case IPP_JOB_ABORTED : | |
178 job_details->status = PRINT_JOB_STATUS_ERROR; | |
179 break; | |
180 case IPP_JOB_COMPLETED : | |
181 job_details->status = PRINT_JOB_STATUS_COMPLETED; | |
182 break; | |
183 default: | |
184 job_details->status = PRINT_JOB_STATUS_INVALID; | |
185 } | |
186 job_details->platform_status_flags = jobs[i].state; | |
187 | |
188 // We don't have any details on the number of processed pages here. | |
189 break; | |
190 } | |
191 } | |
192 | |
193 cupsFreeJobs(num_jobs, jobs); | |
194 return found; | |
195 } | |
196 | |
197 bool GetPrinterInfo(const std::string& printer_name, PrinterBasicInfo* info) { | |
198 DCHECK(info); | |
199 | |
200 // This is not very efficient way to get specific printer info. CUPS 1.4 | |
201 // supports cupsGetNamedDest() function. However, CUPS 1.4 is not available | |
202 // everywhere (for example, it supported from Mac OS 10.6 only). | |
203 PrinterList printer_list; | |
204 EnumeratePrinters(&printer_list); | |
205 | |
206 PrinterList::iterator it; | |
207 for (it = printer_list.begin(); it != printer_list.end(); ++it) { | |
208 if (it->printer_name == printer_name) { | |
209 *info = *it; | |
210 return true; | |
211 } | |
212 } | |
213 return false; | 48 return false; |
214 } | 49 } |
215 | 50 |
216 bool IsValidPrinter(const std::string& printer_name) { | 51 bool IsValidPrinter(const std::string& printer_name) { |
217 PrinterBasicInfo info; | 52 NOTIMPLEMENTED(); |
218 return GetPrinterInfo(printer_name, &info); | 53 return false; |
219 } | 54 } |
220 | 55 |
221 class PrinterChangeNotifier::NotificationState | |
222 : public base::RefCountedThreadSafe< | |
223 PrinterChangeNotifier::NotificationState> { | |
224 public: | |
225 NotificationState() : delegate_(NULL) {} | |
226 bool Start(const std::string& printer_name, | |
227 PrinterChangeNotifier::Delegate* delegate) { | |
228 if (delegate_ != NULL) | |
229 Stop(); | |
230 | |
231 printer_name_ = printer_name; | |
232 delegate_ = delegate; | |
233 | |
234 MessageLoop::current()->PostDelayedTask(FROM_HERE, | |
235 NewRunnableMethod(this, | |
236 &PrinterChangeNotifier::NotificationState::Update), 5000); | |
237 return true; | |
238 } | |
239 bool Stop() { | |
240 delegate_ = NULL; | |
241 return true; | |
242 } | |
243 void Update() { | |
244 if (delegate_ == NULL) | |
245 return; // Orphan call. We have been stopped already. | |
246 // For CUPS proxy, we are going to fire OnJobChanged notification | |
247 // periodically. Higher level will check if there are any outstanding | |
248 // jobs for this printer and check their status. If printer has no | |
249 // outstanding jobs, OnJobChanged() will do nothing. | |
250 delegate_->OnJobChanged(); | |
251 MessageLoop::current()->PostDelayedTask(FROM_HERE, | |
252 NewRunnableMethod(this, | |
253 &PrinterChangeNotifier::NotificationState::Update), | |
254 kNotificationTimeout); | |
255 } | |
256 std::string printer_name() const { | |
257 return printer_name_; | |
258 } | |
259 private: | |
260 friend class base::RefCountedThreadSafe< | |
261 PrinterChangeNotifier::NotificationState>; | |
262 ~NotificationState() { | |
263 Stop(); | |
264 } | |
265 static const int kNotificationTimeout = 5000; // in ms | |
266 std::string printer_name_; // The printer being watched | |
267 PrinterChangeNotifier::Delegate* delegate_; // Delegate to notify | |
268 }; | |
269 | |
270 PrinterChangeNotifier::PrinterChangeNotifier() : state_(NULL) { | 56 PrinterChangeNotifier::PrinterChangeNotifier() : state_(NULL) { |
271 } | 57 } |
272 | 58 |
273 PrinterChangeNotifier::~PrinterChangeNotifier() { | 59 PrinterChangeNotifier::~PrinterChangeNotifier() { |
274 StopWatching(); | 60 StopWatching(); |
275 } | 61 } |
276 | 62 |
277 bool PrinterChangeNotifier::StartWatching(const std::string& printer_name, | 63 bool PrinterChangeNotifier::StartWatching(const std::string& printer_name, |
278 Delegate* delegate) { | 64 Delegate* delegate) { |
279 if (state_) { | 65 NOTIMPLEMENTED(); |
280 NOTREACHED(); | 66 return false; |
281 return false; | |
282 } | |
283 state_ = new NotificationState; | |
284 state_->AddRef(); | |
285 if (!state_->Start(printer_name, delegate)) { | |
286 StopWatching(); | |
287 return false; | |
288 } | |
289 return true; | |
290 } | 67 } |
291 | 68 |
292 bool PrinterChangeNotifier::StopWatching() { | 69 bool PrinterChangeNotifier::StopWatching() { |
293 if (!state_) { | 70 NOTIMPLEMENTED(); |
294 return false; | 71 return false; |
295 } | |
296 state_->Stop(); | |
297 state_->Release(); | |
298 state_ = NULL; | |
299 return true; | |
300 } | 72 } |
301 | 73 |
302 bool PrinterChangeNotifier::GetCurrentPrinterInfo( | 74 bool PrinterChangeNotifier::GetCurrentPrinterInfo( |
303 PrinterBasicInfo* printer_info) { | 75 PrinterBasicInfo* printer_info) { |
304 if (!state_) { | 76 NOTIMPLEMENTED(); |
305 return false; | 77 return false; |
306 } | |
307 DCHECK(printer_info); | |
308 return GetPrinterInfo(state_->printer_name(), printer_info); | |
309 } | 78 } |
310 } // namespace cloud_print | 79 } // namespace cloud_print |
311 | 80 |
OLD | NEW |