| OLD | NEW |
| 1 // Copyright (c) 2012 The Chromium Authors. All rights reserved. | 1 // Copyright (c) 2012 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/print_system.h" | 5 #include "chrome/service/cloud_print/print_system.h" |
| 6 | 6 |
| 7 #include <cups/cups.h> | 7 #include <cups/cups.h> |
| 8 #include <dlfcn.h> | 8 #include <dlfcn.h> |
| 9 #include <errno.h> | 9 #include <errno.h> |
| 10 #include <pthread.h> | 10 #include <pthread.h> |
| 11 | 11 |
| 12 #include <algorithm> | 12 #include <algorithm> |
| 13 #include <list> | 13 #include <list> |
| 14 #include <map> | 14 #include <map> |
| 15 | 15 |
| 16 #include "base/bind.h" | 16 #include "base/bind.h" |
| 17 #include "base/file_path.h" | 17 #include "base/file_path.h" |
| 18 #include "base/json/json_reader.h" | 18 #include "base/json/json_reader.h" |
| 19 #include "base/logging.h" | 19 #include "base/logging.h" |
| 20 #include "base/md5.h" | 20 #include "base/md5.h" |
| 21 #include "base/memory/scoped_ptr.h" | 21 #include "base/memory/scoped_ptr.h" |
| 22 #include "base/message_loop.h" | 22 #include "base/message_loop.h" |
| 23 #include "base/rand_util.h" | 23 #include "base/rand_util.h" |
| 24 #include "base/string_number_conversions.h" | 24 #include "base/string_number_conversions.h" |
| 25 #include "base/string_util.h" | 25 #include "base/string_util.h" |
| 26 #include "base/utf_string_conversions.h" | 26 #include "base/utf_string_conversions.h" |
| 27 #include "base/values.h" | 27 #include "base/values.h" |
| 28 #include "chrome/common/child_process_logging.h" |
| 28 #include "chrome/service/cloud_print/cloud_print_consts.h" | 29 #include "chrome/service/cloud_print/cloud_print_consts.h" |
| 29 #include "chrome/service/cloud_print/cloud_print_helpers.h" | 30 #include "chrome/service/cloud_print/cloud_print_helpers.h" |
| 30 #include "googleurl/src/gurl.h" | 31 #include "googleurl/src/gurl.h" |
| 31 #include "grit/generated_resources.h" | 32 #include "grit/generated_resources.h" |
| 32 #include "printing/backend/cups_helper.h" | 33 #include "printing/backend/cups_helper.h" |
| 33 #include "printing/backend/print_backend.h" | 34 #include "printing/backend/print_backend.h" |
| 34 #include "printing/backend/print_backend_consts.h" | 35 #include "printing/backend/print_backend_consts.h" |
| 35 #include "ui/base/l10n/l10n_util.h" | 36 #include "ui/base/l10n/l10n_util.h" |
| 36 | 37 |
| 37 namespace { | 38 namespace { |
| (...skipping 195 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 233 | 234 |
| 234 DISALLOW_COPY_AND_ASSIGN(PrintServerWatcherCUPS); | 235 DISALLOW_COPY_AND_ASSIGN(PrintServerWatcherCUPS); |
| 235 }; | 236 }; |
| 236 | 237 |
| 237 class PrinterWatcherCUPS | 238 class PrinterWatcherCUPS |
| 238 : public PrintSystem::PrinterWatcher { | 239 : public PrintSystem::PrinterWatcher { |
| 239 public: | 240 public: |
| 240 PrinterWatcherCUPS(PrintSystemCUPS* print_system, | 241 PrinterWatcherCUPS(PrintSystemCUPS* print_system, |
| 241 const std::string& printer_name) | 242 const std::string& printer_name) |
| 242 : printer_name_(printer_name), | 243 : printer_name_(printer_name), |
| 243 delegate_(NULL), | 244 delegate_(NULL), |
| 244 print_system_(print_system) { | 245 print_system_(print_system) { |
| 245 } | 246 } |
| 246 | 247 |
| 247 ~PrinterWatcherCUPS() { | 248 ~PrinterWatcherCUPS() { |
| 248 StopWatching(); | 249 StopWatching(); |
| 249 } | 250 } |
| 250 | 251 |
| 251 // PrintSystem::PrinterWatcher implementation. | 252 // PrintSystem::PrinterWatcher implementation. |
| 252 virtual bool StartWatching( | 253 virtual bool StartWatching( |
| 253 PrintSystem::PrinterWatcher::Delegate* delegate) OVERRIDE{ | 254 PrintSystem::PrinterWatcher::Delegate* delegate) OVERRIDE{ |
| 255 scoped_refptr<printing::PrintBackend> print_backend( |
| 256 printing::PrintBackend::CreateInstance(NULL)); |
| 257 child_process_logging::ScopedPrinterInfoSetter prn_info( |
| 258 print_backend->GetPrinterDriverInfo(printer_name_)); |
| 254 if (delegate_ != NULL) | 259 if (delegate_ != NULL) |
| 255 StopWatching(); | 260 StopWatching(); |
| 256 delegate_ = delegate; | 261 delegate_ = delegate; |
| 257 settings_hash_ = GetSettingsHash(); | 262 settings_hash_ = GetSettingsHash(); |
| 258 // Schedule next job status update. | 263 // Schedule next job status update. |
| 259 MessageLoop::current()->PostDelayedTask( | 264 MessageLoop::current()->PostDelayedTask( |
| 260 FROM_HERE, | 265 FROM_HERE, |
| 261 base::Bind(&PrinterWatcherCUPS::JobStatusUpdate, this), | 266 base::Bind(&PrinterWatcherCUPS::JobStatusUpdate, this), |
| 262 base::TimeDelta::FromSeconds(kJobUpdateTimeoutSeconds)); | 267 base::TimeDelta::FromSeconds(kJobUpdateTimeoutSeconds)); |
| 263 // Schedule next printer check. | 268 // Schedule next printer check. |
| (...skipping 69 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 333 to_hash += it->second; | 338 to_hash += it->second; |
| 334 } | 339 } |
| 335 | 340 |
| 336 to_hash += caps.printer_capabilities; | 341 to_hash += caps.printer_capabilities; |
| 337 to_hash += caps.caps_mime_type; | 342 to_hash += caps.caps_mime_type; |
| 338 to_hash += caps.printer_defaults; | 343 to_hash += caps.printer_defaults; |
| 339 to_hash += caps.defaults_mime_type; | 344 to_hash += caps.defaults_mime_type; |
| 340 | 345 |
| 341 return base::MD5String(to_hash); | 346 return base::MD5String(to_hash); |
| 342 } | 347 } |
| 343 | |
| 344 std::string printer_name_; | 348 std::string printer_name_; |
| 345 PrintSystem::PrinterWatcher::Delegate* delegate_; | 349 PrintSystem::PrinterWatcher::Delegate* delegate_; |
| 346 scoped_refptr<PrintSystemCUPS> print_system_; | 350 scoped_refptr<PrintSystemCUPS> print_system_; |
| 347 std::string settings_hash_; | 351 std::string settings_hash_; |
| 348 | 352 |
| 349 DISALLOW_COPY_AND_ASSIGN(PrinterWatcherCUPS); | 353 DISALLOW_COPY_AND_ASSIGN(PrinterWatcherCUPS); |
| 350 }; | 354 }; |
| 351 | 355 |
| 352 class JobSpoolerCUPS : public PrintSystem::JobSpooler { | 356 class JobSpoolerCUPS : public PrintSystem::JobSpooler { |
| 353 public: | 357 public: |
| (...skipping 151 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 505 | 509 |
| 506 bool PrintSystemCUPS::ValidatePrintTicket(const std::string& printer_name, | 510 bool PrintSystemCUPS::ValidatePrintTicket(const std::string& printer_name, |
| 507 const std::string& print_ticket_data) { | 511 const std::string& print_ticket_data) { |
| 508 DCHECK(initialized_); | 512 DCHECK(initialized_); |
| 509 scoped_ptr<Value> ticket_value(base::JSONReader::Read(print_ticket_data, | 513 scoped_ptr<Value> ticket_value(base::JSONReader::Read(print_ticket_data, |
| 510 false)); | 514 false)); |
| 511 return ticket_value != NULL && ticket_value->IsType(Value::TYPE_DICTIONARY); | 515 return ticket_value != NULL && ticket_value->IsType(Value::TYPE_DICTIONARY); |
| 512 } | 516 } |
| 513 | 517 |
| 514 // Print ticket on linux is a JSON string containing only one dictionary. | 518 // Print ticket on linux is a JSON string containing only one dictionary. |
| 515 bool PrintSystemCUPS::ParsePrintTicket(const std::string& print_ticket, | 519 bool PrintSystemCUPS::ParsePrintTicket( |
| 516 std::map<std::string, std::string>* options) { | 520 const std::string& print_ticket, |
| 521 std::map<std::string, std::string>* options) { |
| 517 DCHECK(options); | 522 DCHECK(options); |
| 518 scoped_ptr<Value> ticket_value(base::JSONReader::Read(print_ticket, false)); | 523 scoped_ptr<Value> ticket_value(base::JSONReader::Read(print_ticket, false)); |
| 519 if (ticket_value == NULL || !ticket_value->IsType(Value::TYPE_DICTIONARY)) | 524 if (ticket_value == NULL || !ticket_value->IsType(Value::TYPE_DICTIONARY)) |
| 520 return false; | 525 return false; |
| 521 | 526 |
| 522 options->clear(); | 527 options->clear(); |
| 523 DictionaryValue* ticket_dict = | 528 DictionaryValue* ticket_dict = |
| 524 static_cast<DictionaryValue*>(ticket_value.get()); | 529 static_cast<DictionaryValue*>(ticket_value.get()); |
| 525 DictionaryValue::key_iterator it(ticket_dict->begin_keys()); | 530 DictionaryValue::key_iterator it(ticket_dict->begin_keys()); |
| 526 for (; it != ticket_dict->end_keys(); ++it) { | 531 for (; it != ticket_dict->end_keys(); ++it) { |
| (...skipping 18 matching lines...) Expand all Loading... |
| 545 return false; | 550 return false; |
| 546 | 551 |
| 547 PrintServerInfoCUPS::CapsMap::iterator caps_it = | 552 PrintServerInfoCUPS::CapsMap::iterator caps_it = |
| 548 server_info->caps_cache.find(printer_name); | 553 server_info->caps_cache.find(printer_name); |
| 549 if (caps_it != server_info->caps_cache.end()) { | 554 if (caps_it != server_info->caps_cache.end()) { |
| 550 *printer_info = caps_it->second; | 555 *printer_info = caps_it->second; |
| 551 return true; | 556 return true; |
| 552 } | 557 } |
| 553 | 558 |
| 554 // TODO(gene): Retry multiple times in case of error. | 559 // TODO(gene): Retry multiple times in case of error. |
| 560 child_process_logging::ScopedPrinterInfoSetter prn_info( |
| 561 server_info->backend->GetPrinterDriverInfo(short_printer_name)); |
| 555 if (!server_info->backend->GetPrinterCapsAndDefaults(short_printer_name, | 562 if (!server_info->backend->GetPrinterCapsAndDefaults(short_printer_name, |
| 556 printer_info) ) { | 563 printer_info) ) { |
| 557 return false; | 564 return false; |
| 558 } | 565 } |
| 559 | 566 |
| 560 server_info->caps_cache[printer_name] = *printer_info; | 567 server_info->caps_cache[printer_name] = *printer_info; |
| 561 return true; | 568 return true; |
| 562 } | 569 } |
| 563 | 570 |
| 564 bool PrintSystemCUPS::GetJobDetails(const std::string& printer_name, | 571 bool PrintSystemCUPS::GetJobDetails(const std::string& printer_name, |
| 565 PlatformJobId job_id, | 572 PlatformJobId job_id, |
| 566 PrintJobDetails *job_details) { | 573 PrintJobDetails *job_details) { |
| 567 DCHECK(initialized_); | 574 DCHECK(initialized_); |
| 568 DCHECK(job_details); | 575 DCHECK(job_details); |
| 569 | 576 |
| 570 std::string short_printer_name; | 577 std::string short_printer_name; |
| 571 PrintServerInfoCUPS* server_info = | 578 PrintServerInfoCUPS* server_info = |
| 572 FindServerByFullName(printer_name, &short_printer_name); | 579 FindServerByFullName(printer_name, &short_printer_name); |
| 573 if (!server_info) | 580 if (!server_info) |
| 574 return false; | 581 return false; |
| 575 | 582 |
| 583 child_process_logging::ScopedPrinterInfoSetter prn_info( |
| 584 server_info->backend->GetPrinterDriverInfo(short_printer_name)); |
| 576 cups_job_t* jobs = NULL; | 585 cups_job_t* jobs = NULL; |
| 577 int num_jobs = GetJobs(&jobs, server_info->url, | 586 int num_jobs = GetJobs(&jobs, server_info->url, |
| 578 short_printer_name.c_str(), 1, -1); | 587 short_printer_name.c_str(), 1, -1); |
| 579 bool error = (num_jobs == 0) && (cupsLastError() > IPP_OK_EVENTS_COMPLETE); | 588 bool error = (num_jobs == 0) && (cupsLastError() > IPP_OK_EVENTS_COMPLETE); |
| 580 if (error) { | 589 if (error) { |
| 581 VLOG(1) << "CP_CUPS: Error getting jobs from CUPS server. Printer:" | 590 VLOG(1) << "CP_CUPS: Error getting jobs from CUPS server. Printer:" |
| 582 << printer_name | 591 << printer_name |
| 583 << " Error: " | 592 << " Error: " |
| 584 << static_cast<int>(cupsLastError()); | 593 << static_cast<int>(cupsLastError()); |
| 585 return false; | 594 return false; |
| (...skipping 145 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 731 bool* dry_run) { | 740 bool* dry_run) { |
| 732 DCHECK(initialized_); | 741 DCHECK(initialized_); |
| 733 VLOG(1) << "CP_CUPS: Spooling print job for: " << printer_name; | 742 VLOG(1) << "CP_CUPS: Spooling print job for: " << printer_name; |
| 734 | 743 |
| 735 std::string short_printer_name; | 744 std::string short_printer_name; |
| 736 PrintServerInfoCUPS* server_info = | 745 PrintServerInfoCUPS* server_info = |
| 737 FindServerByFullName(printer_name, &short_printer_name); | 746 FindServerByFullName(printer_name, &short_printer_name); |
| 738 if (!server_info) | 747 if (!server_info) |
| 739 return false; | 748 return false; |
| 740 | 749 |
| 750 child_process_logging::ScopedPrinterInfoSetter prn_info( |
| 751 server_info->backend->GetPrinterDriverInfo(printer_name)); |
| 752 |
| 741 // We need to store options as char* string for the duration of the | 753 // We need to store options as char* string for the duration of the |
| 742 // cupsPrintFile2 call. We'll use map here to store options, since | 754 // cupsPrintFile2 call. We'll use map here to store options, since |
| 743 // Dictionary value from JSON parser returns wchat_t. | 755 // Dictionary value from JSON parser returns wchat_t. |
| 744 std::map<std::string, std::string> options; | 756 std::map<std::string, std::string> options; |
| 745 bool res = ParsePrintTicket(print_ticket, &options); | 757 bool res = ParsePrintTicket(print_ticket, &options); |
| 746 DCHECK(res); // If print ticket is invalid we still print using defaults. | 758 DCHECK(res); // If print ticket is invalid we still print using defaults. |
| 747 | 759 |
| 748 // Check if this is a dry run (test) job. | 760 // Check if this is a dry run (test) job. |
| 749 *dry_run = CloudPrintHelpers::IsDryRunJob(tags); | 761 *dry_run = CloudPrintHelpers::IsDryRunJob(tags); |
| 750 if (*dry_run) { | 762 if (*dry_run) { |
| (...skipping 67 matching lines...) Expand 10 before | Expand all | Expand 10 after Loading... |
| 818 | 830 |
| 819 void PrintSystemCUPS::RunCapsCallback( | 831 void PrintSystemCUPS::RunCapsCallback( |
| 820 const PrinterCapsAndDefaultsCallback& callback, | 832 const PrinterCapsAndDefaultsCallback& callback, |
| 821 bool succeeded, | 833 bool succeeded, |
| 822 const std::string& printer_name, | 834 const std::string& printer_name, |
| 823 const printing::PrinterCapsAndDefaults& printer_info) { | 835 const printing::PrinterCapsAndDefaults& printer_info) { |
| 824 callback.Run(succeeded, printer_name, printer_info); | 836 callback.Run(succeeded, printer_name, printer_info); |
| 825 } | 837 } |
| 826 | 838 |
| 827 } // namespace cloud_print | 839 } // namespace cloud_print |
| OLD | NEW |