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 #include "chrome/service/cloud_print/job_status_updater.h"
6
7 #include "base/bind.h"
8 #include "base/json/json_reader.h"
9 #include "base/metrics/histogram.h"
10 #include "base/strings/string_util.h"
11 #include "base/strings/utf_string_conversions.h"
12 #include "base/values.h"
13 #include "chrome/common/cloud_print/cloud_print_constants.h"
14 #include "chrome/service/cloud_print/cloud_print_service_helpers.h"
15 #include "url/gurl.h"
16
17 namespace cloud_print {
18
19 namespace {
20
IsTerminalJobState(PrintJobStatus status)21 bool IsTerminalJobState(PrintJobStatus status) {
22 return status == PRINT_JOB_STATUS_ERROR ||
23 status == PRINT_JOB_STATUS_COMPLETED;
24 }
25
26 } // namespace
27
JobStatusUpdater(const std::string & printer_name,const std::string & job_id,PlatformJobId & local_job_id,const GURL & cloud_print_server_url,PrintSystem * print_system,Delegate * delegate)28 JobStatusUpdater::JobStatusUpdater(const std::string& printer_name,
29 const std::string& job_id,
30 PlatformJobId& local_job_id,
31 const GURL& cloud_print_server_url,
32 PrintSystem* print_system,
33 Delegate* delegate)
34 : start_time_(base::Time::Now()),
35 printer_name_(printer_name),
36 job_id_(job_id),
37 local_job_id_(local_job_id),
38 cloud_print_server_url_(cloud_print_server_url),
39 print_system_(print_system),
40 delegate_(delegate),
41 stopped_(false) {
42 DCHECK(delegate_);
43 }
44
45 // Start checking the status of the local print job.
UpdateStatus()46 void JobStatusUpdater::UpdateStatus() {
47 // It does not matter if we had already sent out an update and are waiting for
48 // a response. This is a new update and we will simply cancel the old request
49 // and send a new one.
50 if (!stopped_) {
51 bool need_update = false;
52 // If the job has already been completed, we just need to update the server
53 // with that status. The *only* reason we would come back here in that case
54 // is if our last server update attempt failed.
55 if (IsTerminalJobState(last_job_details_.status)) {
56 need_update = true;
57 } else {
58 PrintJobDetails details;
59 if (print_system_->GetJobDetails(printer_name_, local_job_id_,
60 &details)) {
61 if (details != last_job_details_) {
62 last_job_details_ = details;
63 need_update = true;
64 }
65 } else {
66 // If GetJobDetails failed, the most likely case is that the job no
67 // longer exists in the OS queue. We are going to assume it is done in
68 // this case.
69 last_job_details_.Clear();
70 last_job_details_.status = PRINT_JOB_STATUS_COMPLETED;
71 need_update = true;
72 }
73 UMA_HISTOGRAM_ENUMERATION("CloudPrint.NativeJobStatus",
74 last_job_details_.status, PRINT_JOB_STATUS_MAX);
75 }
76 if (need_update) {
77 request_ = CloudPrintURLFetcher::Create();
78 request_->StartGetRequest(
79 CloudPrintURLFetcher::REQUEST_UPDATE_JOB,
80 GetUrlForJobStatusUpdate(
81 cloud_print_server_url_, job_id_, last_job_details_),
82 this,
83 kCloudPrintAPIMaxRetryCount,
84 std::string());
85 }
86 }
87 }
88
Stop()89 void JobStatusUpdater::Stop() {
90 request_ = NULL;
91 DCHECK(delegate_);
92 stopped_ = true;
93 delegate_->OnJobCompleted(this);
94 }
95
96 // CloudPrintURLFetcher::Delegate implementation.
HandleJSONData(const net::URLFetcher * source,const GURL & url,base::DictionaryValue * json_data,bool succeeded)97 CloudPrintURLFetcher::ResponseAction JobStatusUpdater::HandleJSONData(
98 const net::URLFetcher* source,
99 const GURL& url,
100 base::DictionaryValue* json_data,
101 bool succeeded) {
102 if (IsTerminalJobState(last_job_details_.status)) {
103 base::MessageLoop::current()->PostTask(
104 FROM_HERE, base::Bind(&JobStatusUpdater::Stop, this));
105 }
106 return CloudPrintURLFetcher::STOP_PROCESSING;
107 }
108
OnRequestAuthError()109 CloudPrintURLFetcher::ResponseAction JobStatusUpdater::OnRequestAuthError() {
110 // We got an Auth error and have no idea how long it will take to refresh
111 // auth information (may take forever). We'll drop current request and
112 // propagate this error to the upper level. After auth issues will be
113 // resolved, GCP connector will restart.
114 if (delegate_)
115 delegate_->OnAuthError();
116 return CloudPrintURLFetcher::STOP_PROCESSING;
117 }
118
GetAuthHeader()119 std::string JobStatusUpdater::GetAuthHeader() {
120 return GetCloudPrintAuthHeaderFromStore();
121 }
122
~JobStatusUpdater()123 JobStatusUpdater::~JobStatusUpdater() {}
124
125 } // namespace cloud_print
126