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 3 // found in the LICENSE file. 4 5 #ifndef CHROME_SERVICE_CLOUD_PRINT_PRINTER_JOB_HANDLER_H_ 6 #define CHROME_SERVICE_CLOUD_PRINT_PRINTER_JOB_HANDLER_H_ 7 8 #include <list> 9 #include <string> 10 #include <vector> 11 12 #include "base/files/file_path.h" 13 #include "base/memory/ref_counted.h" 14 #include "base/memory/weak_ptr.h" 15 #include "base/message_loop/message_loop_proxy.h" 16 #include "base/threading/thread.h" 17 #include "base/time/time.h" 18 #include "chrome/service/cloud_print/cloud_print_url_fetcher.h" 19 #include "chrome/service/cloud_print/job_status_updater.h" 20 #include "chrome/service/cloud_print/printer_job_queue_handler.h" 21 #include "net/url_request/url_request_status.h" 22 #include "printing/backend/print_backend.h" 23 #include "url/gurl.h" 24 25 class URLFetcher; 26 // A class that handles cloud print jobs for a particular printer. This class 27 // imlements a state machine that transitions from Start to various states. The 28 // various states are shown in the below diagram. 29 // the status on the server. 30 31 // Start --> No pending tasks --> Done 32 // | 33 // | 34 // | Have Pending tasks 35 // | 36 // | 37 // | ---Update Pending-----> 38 // | | 39 // | | 40 // | | 41 // | Update Printer info on server 42 // | Go to Stop 43 // | 44 // | Job Available 45 // | 46 // | 47 // Fetch Next Job Metadata 48 // Fetch Print Ticket 49 // Fetch Print Data 50 // Spool Print Job 51 // Create Job StatusUpdater for job 52 // Mark job as "in progress" on server 53 // (On any unrecoverable error in any of the above steps go to Stop) 54 // Go to Stop 55 // | 56 // | 57 // | 58 // | 59 // | 60 // | 61 // | 62 // Stop 63 // (If there are pending tasks go back to Start) 64 65 namespace cloud_print { 66 67 class PrinterJobHandler : public base::RefCountedThreadSafe<PrinterJobHandler>, 68 public CloudPrintURLFetcherDelegate, 69 public JobStatusUpdaterDelegate, 70 public PrintSystem::PrinterWatcher::Delegate, 71 public PrintSystem::JobSpooler::Delegate { 72 public: 73 class Delegate { 74 public: 75 // Notify delegate about authentication error. 76 virtual void OnAuthError() = 0; 77 // Notify delegate that printer has been deleted. 78 virtual void OnPrinterDeleted(const std::string& printer_name) = 0; 79 80 protected: 81 virtual ~Delegate() {} 82 }; 83 84 struct PrinterInfoFromCloud { 85 std::string printer_id; 86 std::string caps_hash; 87 std::string tags_hash; 88 int current_xmpp_timeout; 89 int pending_xmpp_timeout; 90 91 PrinterInfoFromCloud(); 92 }; 93 94 // Begin public interface 95 PrinterJobHandler(const printing::PrinterBasicInfo& printer_info, 96 const PrinterInfoFromCloud& printer_info_from_server, 97 const GURL& cloud_print_server_url, 98 PrintSystem* print_system, 99 Delegate* delegate); 100 101 bool Initialize(); 102 103 std::string GetPrinterName() const; 104 105 // Requests a job check. |reason| is the reason for fetching the job. Used 106 // for logging and diagnostc purposes. 107 void CheckForJobs(const std::string& reason); 108 109 // Shutdown everything (the process is exiting). 110 void Shutdown(); 111 112 base::TimeTicks last_job_fetch_time() const { return last_job_fetch_time_; } 113 // End public interface 114 115 // Begin Delegate implementations 116 117 // CloudPrintURLFetcher::Delegate implementation. 118 virtual CloudPrintURLFetcher::ResponseAction HandleRawResponse( 119 const net::URLFetcher* source, 120 const GURL& url, 121 const net::URLRequestStatus& status, 122 int response_code, 123 const net::ResponseCookies& cookies, 124 const std::string& data) OVERRIDE; 125 virtual CloudPrintURLFetcher::ResponseAction HandleRawData( 126 const net::URLFetcher* source, 127 const GURL& url, 128 const std::string& data) OVERRIDE; 129 virtual CloudPrintURLFetcher::ResponseAction HandleJSONData( 130 const net::URLFetcher* source, 131 const GURL& url, 132 base::DictionaryValue* json_data, 133 bool succeeded) OVERRIDE; 134 virtual void OnRequestGiveUp() OVERRIDE; 135 virtual CloudPrintURLFetcher::ResponseAction OnRequestAuthError() OVERRIDE; 136 virtual std::string GetAuthHeader() OVERRIDE; 137 138 // JobStatusUpdater::Delegate implementation 139 virtual bool OnJobCompleted(JobStatusUpdater* updater) OVERRIDE; 140 virtual void OnAuthError() OVERRIDE; 141 142 // PrinterWatcherDelegate implementation 143 virtual void OnPrinterDeleted() OVERRIDE; 144 virtual void OnPrinterChanged() OVERRIDE; 145 virtual void OnJobChanged() OVERRIDE; 146 147 // JobSpoolerDelegate implementation. 148 // Called on print_thread_. 149 virtual void OnJobSpoolSucceeded(const PlatformJobId& job_id) OVERRIDE; 150 virtual void OnJobSpoolFailed() OVERRIDE; 151 152 // End Delegate implementations 153 154 static void ReportsStats(); 155 156 private: 157 friend class base::RefCountedThreadSafe<PrinterJobHandler>; 158 159 enum PrintJobError { 160 JOB_SUCCESS, 161 JOB_DOWNLOAD_FAILED, 162 JOB_VALIDATE_TICKET_FAILED, 163 JOB_FAILED, 164 JOB_MAX, 165 }; 166 167 // Prototype for a JSON data handler. 168 typedef CloudPrintURLFetcher::ResponseAction 169 (PrinterJobHandler::*JSONDataHandler)(const net::URLFetcher* source, 170 const GURL& url, 171 base::DictionaryValue* json_data, 172 bool succeeded); 173 // Prototype for a data handler. 174 typedef CloudPrintURLFetcher::ResponseAction 175 (PrinterJobHandler::*DataHandler)(const net::URLFetcher* source, 176 const GURL& url, 177 const std::string& data); 178 179 virtual ~PrinterJobHandler(); 180 181 // Begin request handlers for each state in the state machine 182 CloudPrintURLFetcher::ResponseAction HandlePrinterUpdateResponse( 183 const net::URLFetcher* source, 184 const GURL& url, 185 base::DictionaryValue* json_data, 186 bool succeeded); 187 188 CloudPrintURLFetcher::ResponseAction HandleJobMetadataResponse( 189 const net::URLFetcher* source, 190 const GURL& url, 191 base::DictionaryValue* json_data, 192 bool succeeded); 193 194 CloudPrintURLFetcher::ResponseAction HandlePrintTicketResponse( 195 const net::URLFetcher* source, 196 const GURL& url, 197 const std::string& data); 198 199 CloudPrintURLFetcher::ResponseAction HandlePrintDataResponse( 200 const net::URLFetcher* source, 201 const GURL& url, 202 const std::string& data); 203 204 CloudPrintURLFetcher::ResponseAction HandleInProgressStatusUpdateResponse( 205 const net::URLFetcher* source, 206 const GURL& url, 207 base::DictionaryValue* json_data, 208 bool succeeded); 209 210 CloudPrintURLFetcher::ResponseAction HandleFailureStatusUpdateResponse( 211 const net::URLFetcher* source, 212 const GURL& url, 213 base::DictionaryValue* json_data, 214 bool succeeded); 215 // End request handlers for each state in the state machine 216 217 // Start the state machine. Based on the flags set this could mean updating 218 // printer information, deleting the printer from the server or looking for 219 // new print jobs 220 void Start(); 221 222 // End the state machine. If there are pending tasks, we will post a Start 223 // again. 224 void Stop(); 225 226 void StartPrinting(); 227 void Reset(); 228 void UpdateJobStatus(PrintJobStatus status, PrintJobError error); 229 230 // Run a job check as the result of a scheduled check 231 void RunScheduledJobCheck(); 232 233 // Sets the next response handler to the specified JSON data handler. 234 void SetNextJSONHandler(JSONDataHandler handler); 235 // Sets the next response handler to the specified data handler. 236 void SetNextDataHandler(DataHandler handler); 237 238 void JobFailed(PrintJobError error); 239 void JobSpooled(PlatformJobId local_job_id); 240 // Returns false if printer info is up to date and no updating is needed. 241 bool UpdatePrinterInfo(); 242 bool HavePendingTasks(); 243 void ValidatePrintTicketFailed(); 244 245 // Callback that asynchronously receives printer caps and defaults. 246 void OnReceivePrinterCaps( 247 bool succeeded, 248 const std::string& printer_name, 249 const printing::PrinterCapsAndDefaults& caps_and_defaults); 250 251 // Called on print_thread_. 252 void DoPrint(const JobDetails& job_details, 253 const std::string& printer_name); 254 255 scoped_refptr<CloudPrintURLFetcher> request_; 256 scoped_refptr<PrintSystem> print_system_; 257 printing::PrinterBasicInfo printer_info_; 258 PrinterInfoFromCloud printer_info_cloud_; 259 GURL cloud_print_server_url_; 260 std::string print_data_url_; 261 JobDetails job_details_; 262 Delegate* delegate_; 263 // Once the job has been spooled to the local spooler, this specifies the 264 // job id of the job on the local spooler. 265 PlatformJobId local_job_id_; 266 267 // The next response handler can either be a JSONDataHandler or a 268 // DataHandler (depending on the current request being made). 269 JSONDataHandler next_json_data_handler_; 270 DataHandler next_data_handler_; 271 // The number of consecutive times that connecting to the server failed. 272 int server_error_count_; 273 // The thread on which the actual print operation happens 274 base::Thread print_thread_; 275 // The Job spooler object. This is only non-NULL during a print operation. 276 // It lives and dies on |print_thread_| 277 scoped_refptr<PrintSystem::JobSpooler> job_spooler_; 278 // The message loop proxy representing the thread on which this object 279 // was created. Used by the print thread. 280 scoped_refptr<base::MessageLoopProxy> job_handler_message_loop_proxy_; 281 282 // There may be pending tasks in the message queue when Shutdown is called. 283 // We set this flag so as to do nothing in those tasks. 284 bool shutting_down_; 285 286 // A string indicating the reason we are fetching jobs from the server 287 // (used to specify the reason in the fetch URL). 288 std::string job_fetch_reason_; 289 // Flags that specify various pending server updates 290 bool job_check_pending_; 291 bool printer_update_pending_; 292 293 // Number of seconds between XMPP pings (for server registration) 294 int xmpp_ping_interval_; 295 296 // Some task in the state machine is in progress. 297 bool task_in_progress_; 298 scoped_refptr<PrintSystem::PrinterWatcher> printer_watcher_; 299 typedef std::list< scoped_refptr<JobStatusUpdater> > JobStatusUpdaterList; 300 JobStatusUpdaterList job_status_updater_list_; 301 302 // Manages parsing the job queue 303 PrinterJobQueueHandler job_queue_handler_; 304 305 base::TimeTicks last_job_fetch_time_; 306 307 base::Time job_start_time_; 308 base::Time spooling_start_time_; 309 base::Time last_caps_update_time_; 310 311 base::WeakPtrFactory<PrinterJobHandler> weak_ptr_factory_; 312 313 DISALLOW_COPY_AND_ASSIGN(PrinterJobHandler); 314 }; 315 316 // This typedef is to workaround the issue with certain versions of 317 // Visual Studio where it gets confused between multiple Delegate 318 // classes and gives a C2500 error. (I saw this error on the try bots - 319 // the workaround was not needed for my machine). 320 typedef PrinterJobHandler::Delegate PrinterJobHandlerDelegate; 321 322 } // namespace cloud_print 323 324 #endif // CHROME_SERVICE_CLOUD_PRINT_PRINTER_JOB_HANDLER_H_ 325