OLD | NEW |
---|---|
(Empty) | |
1 // Copyright 2016 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 "printing/backend/cups_connection.h" | |
6 | |
7 #include <string> | |
8 | |
9 #include "base/logging.h" | |
10 #include "base/strings/stringprintf.h" | |
11 | |
12 namespace { | |
13 | |
14 static const int kTimeoutMs = 3000; | |
15 | |
16 class DestinationEnumerator { | |
17 public: | |
18 ~DestinationEnumerator() { | |
19 for (cups_dest_t* dest : dests_) { | |
20 cupsFreeDests(1, dest); | |
Lei Zhang
2016/07/13 01:08:42
Can |dests_| be: std::vector<std::unique_ptr<cups_
skau
2016/07/14 20:43:05
I didn't think of that. Thanks!
| |
21 } | |
22 | |
23 dests_.clear(); | |
24 } | |
25 | |
26 static int cups_callback(void* user_data, unsigned flags, cups_dest_t* dest) { | |
27 cups_dest_t* copied_dest; | |
28 cupsCopyDest(dest, 0, &copied_dest); | |
29 reinterpret_cast<DestinationEnumerator*>(user_data)->store_dest( | |
30 copied_dest); | |
31 | |
32 // keep going | |
33 return 1; | |
34 } | |
35 | |
36 void store_dest(cups_dest_t* dest) { dests_.push_back(dest); } | |
37 | |
38 // Returns the collected destinations. Remove desired destinations from the | |
39 // vector or they will be cleaned up when this object is destroyed. | |
40 std::vector<cups_dest_t*>& get_dests() { return dests_; } | |
41 | |
42 private: | |
43 std::vector<cups_dest_t*> dests_; | |
44 }; | |
45 | |
46 } // namespace | |
47 | |
48 namespace printing { | |
49 | |
50 void HttpDeleter::operator()(http_t* http) const { | |
51 httpClose(http); | |
52 } | |
53 | |
54 CupsConnection::Impl::Impl(http_t* http) | |
55 : cups_http_(http), http_factory_(http) {} | |
56 | |
57 CupsConnection::Impl::~Impl() { | |
58 http_factory_.InvalidateWeakPtrs(); | |
59 } | |
60 | |
61 base::WeakPtr<http_t> CupsConnection::Impl::GetHttp() { | |
62 return http_factory_.GetWeakPtr(); | |
63 } | |
64 | |
65 CupsConnection::CupsConnection(const GURL& print_server_url, | |
66 http_encryption_t encryption, | |
67 bool blocking) | |
68 : print_server_url_(print_server_url), | |
69 cups_encryption_(encryption), | |
70 blocking_(blocking), | |
71 impl_(nullptr), | |
72 cups_http_(nullptr) {} | |
73 | |
74 CupsConnection::CupsConnection(CupsConnection&& connection) | |
75 : print_server_url_(connection.print_server_url_), | |
76 cups_encryption_(connection.cups_encryption_), | |
77 blocking_(connection.blocking_), | |
78 impl_(std::move(connection.impl_)), | |
79 cups_http_(std::move(connection.cups_http_)) {} | |
80 | |
81 CupsConnection::~CupsConnection() {} | |
82 | |
83 bool CupsConnection::Connect() { | |
84 if (cups_http_) | |
85 return true; // we're already connected | |
86 | |
87 std::string host_string; | |
Lei Zhang
2016/07/13 01:08:42
Only used inside the if block below - move it ther
skau
2016/07/14 20:43:05
I needed the c_str buffer to stay in scope. I swi
| |
88 const char* host; | |
89 int port; | |
90 | |
91 if (!print_server_url_.is_empty()) { | |
92 host_string = print_server_url_.host(); | |
93 host = host_string.c_str(); | |
94 port = print_server_url_.IntPort(); | |
95 } else { | |
96 host = cupsServer(); | |
97 port = ippPort(); | |
98 } | |
99 | |
100 http_t* connection = | |
101 httpConnect2(host, port, NULL, AF_UNSPEC, cups_encryption_, | |
Lei Zhang
2016/07/13 01:08:42
nullptr all around
skau
2016/07/14 20:43:05
Done.
| |
102 blocking_ ? 1 : 0, kTimeoutMs, NULL); | |
103 | |
104 if (!connection) | |
105 return false; | |
106 | |
107 impl_ = std::unique_ptr<Impl>(new Impl(connection)); | |
Lei Zhang
2016/07/13 01:08:42
Use base::MakeUnique<Impl>(connection) ?
skau
2016/07/14 20:43:05
Done.
| |
108 cups_http_ = impl_->GetHttp(); | |
109 return true; | |
110 } | |
111 | |
112 std::vector<CupsPrinter> CupsConnection::GetDests() { | |
113 if (!Connect()) { | |
114 LOG(WARNING) << "CUPS connection failed"; | |
115 return std::vector<CupsPrinter>(); | |
116 } | |
117 | |
118 DestinationEnumerator enumerator; | |
119 int success = | |
120 cupsEnumDests(CUPS_DEST_FLAGS_NONE, kTimeoutMs, | |
121 NULL, // no cancel signal | |
122 0, // all the printers | |
123 CUPS_PRINTER_SCANNER, // except the scanners | |
124 &DestinationEnumerator::cups_callback, &enumerator); | |
125 | |
126 if (!success) { | |
127 LOG(WARNING) << "Enumerating printers failed"; | |
128 return std::vector<CupsPrinter>(); | |
129 } | |
130 | |
131 std::vector<CupsPrinter> printers; | |
132 std::vector<cups_dest_t*>& dests = enumerator.get_dests(); | |
133 for (cups_dest_t* dest : dests) { | |
134 CupsPrinter printer(impl_->GetHttp(), dest, nullptr); | |
135 printers.push_back(std::move(printer)); | |
136 } | |
137 | |
138 dests.clear(); // CupsPrinter takes ownership of all the cups_dest_t objects | |
139 | |
140 return printers; | |
141 } | |
142 | |
143 CupsPrinter* CupsConnection::GetPrinter(const std::string& name) { | |
144 if (!Connect()) | |
145 return nullptr; | |
146 | |
147 cups_dest_t* dest = cupsGetNamedDest(cups_http_.get(), name.c_str(), NULL); | |
148 if (!dest) | |
149 return nullptr; | |
150 | |
151 cups_dinfo_t* info = cupsCopyDestInfo(cups_http_.get(), dest); | |
152 return new CupsPrinter(impl_->GetHttp(), dest, info); | |
153 } | |
154 | |
155 std::string CupsConnection::server_name() const { | |
156 return print_server_url_.host(); | |
157 } | |
158 | |
159 int CupsConnection::last_error() const { | |
160 return cupsLastError(); | |
161 } | |
162 | |
163 } // namespace printing | |
OLD | NEW |