Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 1 | // Copyright 2019 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/services/cups_proxy/proxy_manager.h" |
| 6 | |
| 7 | #include <cups/ipp.h> |
| 8 | |
| 9 | #include <memory> |
| 10 | #include <string> |
| 11 | #include <utility> |
| 12 | #include <vector> |
| 13 | |
| 14 | #include "base/bind.h" |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 15 | #include "base/containers/ring_buffer.h" |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 16 | #include "base/feature_list.h" |
| 17 | #include "base/logging.h" |
| 18 | #include "base/task/post_task.h" |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 19 | #include "chrome/services/cups_proxy/cups_proxy_service_delegate.h" |
| 20 | #include "chrome/services/cups_proxy/ipp_validator.h" |
| 21 | #include "chrome/services/cups_proxy/printer_installer.h" |
| 22 | #include "chrome/services/cups_proxy/public/cpp/cups_util.h" |
| 23 | #include "chrome/services/cups_proxy/public/cpp/ipp_messages.h" |
| 24 | #include "chrome/services/cups_proxy/public/cpp/type_conversions.h" |
| 25 | #include "chrome/services/cups_proxy/socket_manager.h" |
Robbie McElrath | 8b72844 | 2021-01-14 00:20:01 | [diff] [blame] | 26 | #include "chrome/services/ipp_parser/public/cpp/browser/ipp_parser_launcher.h" |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 27 | #include "chrome/services/ipp_parser/public/cpp/ipp_converter.h" |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 28 | #include "mojo/public/cpp/bindings/receiver.h" |
| 29 | #include "mojo/public/cpp/bindings/remote.h" |
| 30 | #include "net/http/http_response_headers.h" |
| 31 | #include "net/http/http_util.h" |
Daniel Hosseinian | 9c72599 | 2020-02-20 23:54:37 | [diff] [blame] | 32 | #include "printing/backend/cups_ipp_helper.h" |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 33 | |
| 34 | namespace cups_proxy { |
| 35 | namespace { |
| 36 | |
| 37 | struct InFlightRequest { |
| 38 | IppRequest request; |
| 39 | ProxyManager::ProxyRequestCallback cb; |
| 40 | }; |
| 41 | |
| 42 | class ProxyManagerImpl : public ProxyManager { |
| 43 | public: |
| 44 | ProxyManagerImpl(mojo::PendingReceiver<mojom::CupsProxier> request, |
| 45 | std::unique_ptr<CupsProxyServiceDelegate> delegate, |
| 46 | std::unique_ptr<IppValidator> ipp_validator, |
| 47 | std::unique_ptr<PrinterInstaller> printer_installer, |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 48 | std::unique_ptr<SocketManager> socket_manager) |
| 49 | : delegate_(std::move(delegate)), |
| 50 | ipp_validator_(std::move(ipp_validator)), |
| 51 | printer_installer_(std::move(printer_installer)), |
| 52 | socket_manager_(std::move(socket_manager)), |
| 53 | receiver_(this, std::move(request)) { |
| 54 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
Pranav Batra | e992b11 | 2020-11-20 02:21:44 | [diff] [blame] | 55 | receiver_.set_disconnect_handler( |
| 56 | base::BindOnce([] { LOG(ERROR) << "CupsProxy mojo connection lost"; })); |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 57 | } |
| 58 | |
Peter Boström | 53c6c595 | 2021-09-17 09:41:26 | [diff] [blame] | 59 | ProxyManagerImpl(const ProxyManagerImpl&) = delete; |
| 60 | ProxyManagerImpl& operator=(const ProxyManagerImpl&) = delete; |
| 61 | |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 62 | ~ProxyManagerImpl() override = default; |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 63 | |
| 64 | void ProxyRequest(const std::string& method, |
| 65 | const std::string& url, |
| 66 | const std::string& version, |
| 67 | const std::vector<ipp_converter::HttpHeader>& headers, |
| 68 | const std::vector<uint8_t>& body, |
| 69 | ProxyRequestCallback cb) override; |
| 70 | |
| 71 | private: |
| 72 | // These methods interface with |ipp_parser_| to parse the syntax of the |
| 73 | // inflight IPP request. |
| 74 | void ParseIpp(); |
| 75 | void OnParseIpp(ipp_parser::mojom::IppRequestPtr parsed_request); |
| 76 | |
| 77 | // For CUPS-Get-Printers requests, we spoof the response. |
| 78 | void SpoofGetPrinters(); |
| 79 | |
| 80 | // The callback for interfacing with |printer_installer_|; used to |
| 81 | // pre-install any printers referenced by the inflight request into CUPS. |
| 82 | void OnInstallPrinter(InstallPrinterResult res); |
| 83 | |
| 84 | // These methods interface with |socket_manager_| to actually proxy the |
| 85 | // inflight request to CUPS and propagate back its IPP response. |
| 86 | void ProxyToCups(); |
| 87 | void OnProxyToCups(std::unique_ptr<std::vector<uint8_t>> response); |
| 88 | |
| 89 | // Proxy the IPP response back to the caller. |
| 90 | void ProxyResponseToCaller(const std::vector<uint8_t>& response); |
| 91 | |
| 92 | // Fail in-flight request. |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 93 | void Fail(const std::string& error_message, int http_status_code); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 94 | |
| 95 | // Delegate providing necessary Profile dependencies. |
| 96 | std::unique_ptr<CupsProxyServiceDelegate> delegate_; |
| 97 | |
| 98 | // Current in-flight request. |
| 99 | std::unique_ptr<InFlightRequest> in_flight_; |
| 100 | |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 101 | // Timestamp ring buffer. |
| 102 | base::RingBuffer<base::TimeTicks, kRateLimit> timestamp_; |
| 103 | |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 104 | // CupsIppParser Service handle. |
| 105 | mojo::Remote<ipp_parser::mojom::IppParser> ipp_parser_; |
| 106 | |
| 107 | // Runs in current sequence. |
| 108 | std::unique_ptr<IppValidator> ipp_validator_; |
| 109 | std::unique_ptr<PrinterInstaller> printer_installer_; |
| 110 | std::unique_ptr<SocketManager> socket_manager_; |
| 111 | |
| 112 | SEQUENCE_CHECKER(sequence_checker_); |
| 113 | |
| 114 | mojo::Receiver<mojom::CupsProxier> receiver_; |
| 115 | base::WeakPtrFactory<ProxyManagerImpl> weak_factory_{this}; |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 116 | }; |
| 117 | |
Anton Bikineev | 46bbb97 | 2021-05-15 17:53:53 | [diff] [blame] | 118 | absl::optional<std::vector<uint8_t>> RebuildIppRequest( |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 119 | const std::string& method, |
| 120 | const std::string& url, |
| 121 | const std::string& version, |
| 122 | const std::vector<ipp_converter::HttpHeader>& headers, |
| 123 | const std::vector<uint8_t>& body) { |
| 124 | auto request_line_buffer = |
| 125 | ipp_converter::BuildRequestLine(method, url, version); |
| 126 | if (!request_line_buffer.has_value()) { |
Anton Bikineev | 46bbb97 | 2021-05-15 17:53:53 | [diff] [blame] | 127 | return absl::nullopt; |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 128 | } |
| 129 | |
| 130 | auto headers_buffer = ipp_converter::BuildHeaders(headers); |
| 131 | if (!headers_buffer.has_value()) { |
Anton Bikineev | 46bbb97 | 2021-05-15 17:53:53 | [diff] [blame] | 132 | return absl::nullopt; |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 133 | } |
| 134 | |
| 135 | std::vector<uint8_t> ret; |
| 136 | ret.insert(ret.end(), request_line_buffer->begin(), |
| 137 | request_line_buffer->end()); |
| 138 | ret.insert(ret.end(), headers_buffer->begin(), headers_buffer->end()); |
| 139 | ret.insert(ret.end(), body.begin(), body.end()); |
| 140 | return ret; |
| 141 | } |
| 142 | |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 143 | void ProxyManagerImpl::ProxyRequest( |
| 144 | const std::string& method, |
| 145 | const std::string& url, |
| 146 | const std::string& version, |
| 147 | const std::vector<ipp_converter::HttpHeader>& headers, |
| 148 | const std::vector<uint8_t>& body, |
| 149 | ProxyRequestCallback cb) { |
| 150 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 151 | |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 152 | // Limit the rate of requests to kRateLimit per second. |
| 153 | // The way the algorithm works is if the number of times this method was |
| 154 | // called between a second ago and now, including the current call, exceeds |
| 155 | // kRateLimit, the request is blocked and the HTTP 429 Too Many Requests |
| 156 | // response status code is returned. |
| 157 | base::TimeTicks time = base::TimeTicks::Now(); |
Peter Kasting | e5a38ed | 2021-10-02 03:06:35 | [diff] [blame] | 158 | bool block_request = timestamp_.CurrentIndex() >= timestamp_.BufferSize() && |
| 159 | time - timestamp_.ReadBuffer(0) < base::Seconds(1); |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 160 | timestamp_.SaveToBuffer(time); |
| 161 | if (block_request) { |
Pranav Batra | a280e3a | 2020-12-11 19:20:12 | [diff] [blame] | 162 | LOG(WARNING) << "CupsPrintService: Rate limit (" << kRateLimit |
| 163 | << ") exceeded"; |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 164 | std::move(cb).Run({}, {}, 429); // HTTP_STATUS_TOO_MANY_REQUESTS |
| 165 | return; |
| 166 | } |
| 167 | |
Timothy Loh | 2058fef8e | 2020-04-02 23:03:27 | [diff] [blame] | 168 | if (!delegate_->IsPrinterAccessAllowed()) { |
| 169 | DVLOG(1) << "Printer access not allowed"; |
| 170 | std::move(cb).Run(/*headers=*/{}, /*ipp_message=*/{}, |
| 171 | HTTP_STATUS_FORBIDDEN); |
| 172 | return; |
| 173 | } |
| 174 | |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 175 | // If we already have an in-flight request, we fail this incoming one |
| 176 | // directly. |
| 177 | if (in_flight_) { |
| 178 | DVLOG(1) << "CupsPrintService Error: Already have an in-flight request"; |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 179 | std::move(cb).Run({}, {}, HTTP_STATUS_SERVICE_UNAVAILABLE); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 180 | return; |
| 181 | } |
| 182 | |
| 183 | in_flight_ = std::make_unique<InFlightRequest>(); |
| 184 | in_flight_->cb = std::move(cb); |
| 185 | |
| 186 | // TODO(crbug.com/945409): Rename in both proxy.mojom's: url -> endpoint. |
| 187 | auto request_buffer = RebuildIppRequest(method, url, version, headers, body); |
| 188 | if (!request_buffer) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 189 | return Fail("Failed to rebuild incoming IPP request", |
| 190 | HTTP_STATUS_SERVER_ERROR); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 191 | } |
| 192 | |
| 193 | // Save request. |
| 194 | in_flight_->request.buffer = *request_buffer; |
| 195 | ParseIpp(); |
| 196 | return; |
| 197 | } |
| 198 | |
| 199 | void ProxyManagerImpl::ParseIpp() { |
| 200 | // Launch CupsIppParser service, if needed. |
| 201 | if (!ipp_parser_) { |
| 202 | ipp_parser_.Bind(ipp_parser::LaunchIppParser()); |
| 203 | ipp_parser_.reset_on_disconnect(); |
| 204 | } |
| 205 | |
| 206 | // Run out-of-process IPP parsing. |
| 207 | ipp_parser_->ParseIpp(in_flight_->request.buffer, |
| 208 | base::BindOnce(&ProxyManagerImpl::OnParseIpp, |
| 209 | weak_factory_.GetWeakPtr())); |
| 210 | } |
| 211 | |
| 212 | void ProxyManagerImpl::OnParseIpp( |
| 213 | ipp_parser::mojom::IppRequestPtr parsed_request) { |
| 214 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 215 | DCHECK(in_flight_); |
| 216 | |
| 217 | if (!parsed_request) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 218 | return Fail("Failed to parse IPP request", HTTP_STATUS_BAD_REQUEST); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 219 | } |
| 220 | |
| 221 | // Validate parsed request. |
| 222 | auto valid_request = |
| 223 | ipp_validator_->ValidateIppRequest(std::move(parsed_request)); |
| 224 | if (!valid_request) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 225 | return Fail("Failed to validate IPP request", HTTP_STATUS_BAD_REQUEST); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 226 | } |
| 227 | |
| 228 | // Save newly validated request. |
| 229 | in_flight_->request = std::move(*valid_request); |
| 230 | |
| 231 | auto opcode = ippGetOperation(in_flight_->request.ipp.get()); |
| 232 | if (opcode == IPP_OP_CUPS_NONE) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 233 | return Fail("Failed to parse IPP operation ID", HTTP_STATUS_BAD_REQUEST); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 234 | } |
| 235 | |
| 236 | // Since Chrome is the source-of-truth for printers on ChromeOS, for |
| 237 | // CUPS-Get-Printers requests, we spoof the response rather than proxying to |
| 238 | // CUPS. |
| 239 | if (opcode == IPP_OP_CUPS_GET_PRINTERS) { |
| 240 | SpoofGetPrinters(); |
| 241 | return; |
| 242 | } |
| 243 | |
| 244 | // If this request references a printer, pre-install it into CUPS. |
| 245 | auto printer_uuid = GetPrinterId(in_flight_->request.ipp.get()); |
| 246 | if (printer_uuid.has_value()) { |
| 247 | printer_installer_->InstallPrinter( |
| 248 | *printer_uuid, base::BindOnce(&ProxyManagerImpl::OnInstallPrinter, |
| 249 | weak_factory_.GetWeakPtr())); |
| 250 | return; |
| 251 | } |
| 252 | |
| 253 | // Nothing left to do, skip straight to proxying. |
| 254 | ProxyToCups(); |
| 255 | return; |
| 256 | } |
| 257 | |
| 258 | void ProxyManagerImpl::SpoofGetPrinters() { |
Timothy Loh | 2058fef8e | 2020-04-02 23:03:27 | [diff] [blame] | 259 | std::vector<chromeos::Printer> printers = FilterPrintersForPluginVm( |
Luum Habtemariam | 1c157b3 | 2019-09-20 01:12:02 | [diff] [blame] | 260 | delegate_->GetPrinters(chromeos::PrinterClass::kSaved), |
Pranav Batra | abf9ca8 | 2020-09-18 04:32:33 | [diff] [blame] | 261 | delegate_->GetPrinters(chromeos::PrinterClass::kEnterprise), |
| 262 | delegate_->GetRecentlyUsedPrinters()); |
Anton Bikineev | 46bbb97 | 2021-05-15 17:53:53 | [diff] [blame] | 263 | absl::optional<IppResponse> response = |
Timothy Loh | 2058fef8e | 2020-04-02 23:03:27 | [diff] [blame] | 264 | BuildGetDestsResponse(in_flight_->request, printers); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 265 | if (!response.has_value()) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 266 | return Fail("Failed to spoof CUPS-Get-Printers response", |
| 267 | HTTP_STATUS_SERVER_ERROR); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 268 | } |
| 269 | |
| 270 | ProxyResponseToCaller(response->buffer); |
| 271 | return; |
| 272 | } |
| 273 | |
| 274 | void ProxyManagerImpl::OnInstallPrinter(InstallPrinterResult res) { |
| 275 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 276 | DCHECK(in_flight_); |
| 277 | |
| 278 | if (res != InstallPrinterResult::kSuccess) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 279 | return Fail("Failed to pre-install printer", HTTP_STATUS_SERVER_ERROR); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 280 | } |
| 281 | |
| 282 | ProxyToCups(); |
| 283 | return; |
| 284 | } |
| 285 | |
| 286 | void ProxyManagerImpl::ProxyToCups() { |
| 287 | // Queue request with socket_manager_ |
| 288 | socket_manager_->ProxyToCups(std::move(in_flight_->request.buffer), |
| 289 | base::BindOnce(&ProxyManagerImpl::OnProxyToCups, |
| 290 | weak_factory_.GetWeakPtr())); |
| 291 | } |
| 292 | |
| 293 | void ProxyManagerImpl::OnProxyToCups( |
| 294 | std::unique_ptr<std::vector<uint8_t>> response) { |
| 295 | DCHECK_CALLED_ON_VALID_SEQUENCE(sequence_checker_); |
| 296 | DCHECK(in_flight_); |
| 297 | |
| 298 | if (!response) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 299 | return Fail("Failed to proxy request", HTTP_STATUS_SERVER_ERROR); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 300 | } |
| 301 | |
| 302 | ProxyResponseToCaller(*response); |
| 303 | return; |
| 304 | } |
| 305 | |
| 306 | void ProxyManagerImpl::ProxyResponseToCaller( |
| 307 | const std::vector<uint8_t>& response) { |
| 308 | // Convert to string for parsing HTTP headers. |
| 309 | std::string response_str = ipp_converter::ConvertToString(response); |
| 310 | auto end_of_headers = net::HttpUtil::LocateEndOfHeaders(response_str.data(), |
| 311 | response_str.size()); |
| 312 | if (end_of_headers < 0) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 313 | return Fail("IPP response missing end of headers", |
| 314 | HTTP_STATUS_SERVER_ERROR); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 315 | } |
| 316 | |
| 317 | base::StringPiece headers_slice(response_str.data(), end_of_headers); |
| 318 | scoped_refptr<net::HttpResponseHeaders> response_headers = |
| 319 | net::HttpResponseHeaders::TryToCreate(headers_slice); |
| 320 | if (!response_headers) { |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 321 | return Fail("Failed to parse HTTP response headers", |
| 322 | HTTP_STATUS_SERVER_ERROR); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 323 | } |
| 324 | |
| 325 | std::vector<ipp_converter::HttpHeader> parsed_headers; |
| 326 | size_t iter = 0; |
| 327 | std::string name, value; |
| 328 | while (response_headers->EnumerateHeaderLines(&iter, &name, &value)) { |
| 329 | parsed_headers.push_back({name, value}); |
| 330 | } |
| 331 | |
| 332 | // Slice off the ipp_message as is. |
| 333 | std::vector<uint8_t> ipp_message(response.begin() + end_of_headers, |
| 334 | response.end()); |
| 335 | |
| 336 | // Send parsed response back to caller. |
| 337 | std::move(in_flight_->cb) |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 338 | .Run(std::move(parsed_headers), std::move(ipp_message), HTTP_STATUS_OK); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 339 | in_flight_.reset(); |
| 340 | } |
| 341 | |
| 342 | // TODO(crbug.com/945409): Fail with comprehensive HTTP Error response. |
| 343 | // Fails current request by running its callback with an empty response and |
| 344 | // clearing in_flight_. |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 345 | void ProxyManagerImpl::Fail(const std::string& error_message, |
| 346 | int http_status_code) { |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 347 | DCHECK(in_flight_); |
| 348 | |
| 349 | DVLOG(1) << "CupsPrintService Error: " << error_message; |
| 350 | |
Luum Habtemariam | b9a4f7f6 | 2019-10-03 19:18:58 | [diff] [blame] | 351 | std::move(in_flight_->cb).Run({}, {}, http_status_code); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 352 | in_flight_.reset(); |
| 353 | } |
| 354 | |
| 355 | } // namespace |
| 356 | |
| 357 | // static |
| 358 | std::unique_ptr<ProxyManager> ProxyManager::Create( |
| 359 | mojo::PendingReceiver<mojom::CupsProxier> request, |
| 360 | std::unique_ptr<CupsProxyServiceDelegate> delegate) { |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 361 | auto* delegate_ptr = delegate.get(); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 362 | return std::make_unique<ProxyManagerImpl>( |
Pranav Batra | ffbed60 | 2020-09-23 02:38:35 | [diff] [blame] | 363 | std::move(request), std::move(delegate), |
| 364 | std::make_unique<IppValidator>(delegate_ptr), |
| 365 | std::make_unique<PrinterInstaller>(delegate_ptr), |
| 366 | SocketManager::Create(delegate_ptr)); |
Luum Habtemariam | c2ab5bc8 | 2019-09-06 21:10:05 | [diff] [blame] | 367 | } |
| 368 | |
| 369 | std::unique_ptr<ProxyManager> ProxyManager::CreateForTesting( |
| 370 | mojo::PendingReceiver<mojom::CupsProxier> request, |
| 371 | std::unique_ptr<CupsProxyServiceDelegate> delegate, |
| 372 | std::unique_ptr<IppValidator> ipp_validator, |
| 373 | std::unique_ptr<PrinterInstaller> printer_installer, |
| 374 | std::unique_ptr<SocketManager> socket_manager) { |
| 375 | return std::make_unique<ProxyManagerImpl>( |
| 376 | std::move(request), std::move(delegate), std::move(ipp_validator), |
| 377 | std::move(printer_installer), std::move(socket_manager)); |
| 378 | } |
| 379 | |
| 380 | } // namespace cups_proxy |