blob: 833a0498582e35c9d543194ea084f9206f49f63b [file] [log] [blame]
// Copyright 2018 The Chromium Authors. All rights reserved.
// Use of this source code is governed by a BSD-style license that can be
// found in the LICENSE file.
#include "components/printing/browser/printer_capabilities.h"
#include <memory>
#include <string>
#include <utility>
#include <vector>
#include "base/logging.h"
#include "base/stl_util.h"
#include "base/strings/string_piece.h"
#include "base/threading/scoped_blocking_call.h"
#include "base/values.h"
#include "build/build_config.h"
#include "build/buildflag.h"
#include "components/crash/core/common/crash_keys.h"
#include "components/printing/browser/printing_buildflags.h"
#include "components/printing/common/cloud_print_cdd_conversion.h"
#include "printing/backend/print_backend.h"
#include "printing/backend/print_backend_consts.h"
#if defined(OS_WIN)
#include "base/strings/string_split.h"
#include "base/strings/string_util.h"
#include "base/strings/utf_string_conversions.h"
#include "components/strings/grit/components_strings.h"
#include "ui/base/l10n/l10n_util.h"
#endif
#if BUILDFLAG(PRINT_MEDIA_L10N_ENABLED)
#include "components/printing/browser/print_media_l10n.h"
#endif
namespace printing {
const char kPrinter[] = "printer";
namespace {
#if BUILDFLAG(PRINT_MEDIA_L10N_ENABLED)
// Iterate on the Papers of a given printer |info| and set the
// display_name members, localizing where possible.
void PopulateAllPaperDisplayNames(PrinterSemanticCapsAndDefaults* info) {
info->default_paper.display_name =
LocalizePaperDisplayName(info->default_paper.vendor_id);
for (PrinterSemanticCapsAndDefaults::Paper& paper : info->papers) {
paper.display_name = LocalizePaperDisplayName(paper.vendor_id);
}
}
#endif // BUILDFLAG(PRINT_MEDIA_L10N_ENABLED)
// Returns a dictionary representing printer capabilities as CDD. Returns
// an empty dictionary if a dictionary could not be generated.
base::Value GetPrinterCapabilitiesOnBlockingPoolThread(
const std::string& device_name,
const PrinterSemanticCapsAndDefaults::Papers& additional_papers,
scoped_refptr<PrintBackend> print_backend) {
base::ScopedBlockingCall scoped_blocking_call(FROM_HERE,
base::BlockingType::MAY_BLOCK);
DCHECK(!device_name.empty());
scoped_refptr<PrintBackend> backend =
print_backend ? print_backend
: printing::PrintBackend::CreateInstance(nullptr);
VLOG(1) << "Get printer capabilities start for " << device_name;
crash_keys::ScopedPrinterInfo crash_key(
backend->GetPrinterDriverInfo(device_name));
if (!backend->IsValidPrinter(device_name)) {
LOG(WARNING) << "Invalid printer " << device_name;
return base::Value(base::Value::Type::DICTIONARY);
}
PrinterSemanticCapsAndDefaults info;
if (!backend->GetPrinterSemanticCapsAndDefaults(device_name, &info)) {
LOG(WARNING) << "Failed to get capabilities for " << device_name;
return base::Value(base::Value::Type::DICTIONARY);
}
#if BUILDFLAG(PRINT_MEDIA_L10N_ENABLED)
PopulateAllPaperDisplayNames(&info);
#endif
info.papers.insert(info.papers.end(), additional_papers.begin(),
additional_papers.end());
return cloud_print::PrinterSemanticCapsAndDefaultsToCdd(info);
}
#if defined(OS_WIN)
std::string GetUserFriendlyName(const std::string& printer_name) {
// |printer_name| may be a UNC path like \\printserver\printername.
if (!base::StartsWith(printer_name, "\\\\",
base::CompareCase::INSENSITIVE_ASCII)) {
return printer_name;
}
// If it is a UNC path, split the "printserver\printername" portion and
// generate a friendly name, like Windows does.
std::string printer_name_trimmed = printer_name.substr(2);
std::vector<std::string> tokens = base::SplitString(
printer_name_trimmed, "\\", base::KEEP_WHITESPACE, base::SPLIT_WANT_ALL);
if (tokens.size() != 2 || tokens[0].empty() || tokens[1].empty())
return printer_name;
return l10n_util::GetStringFUTF8(
IDS_PRINT_PREVIEW_FRIENDLY_WIN_NETWORK_PRINTER_NAME,
base::UTF8ToUTF16(tokens[1]), base::UTF8ToUTF16(tokens[0]));
}
#endif
} // namespace
std::pair<std::string, std::string> GetPrinterNameAndDescription(
const PrinterBasicInfo& printer) {
#if defined(OS_MACOSX) || defined(OS_CHROMEOS)
// On Mac, |printer.printer_description| specifies the printer name and
// |printer.printer_name| specifies the device name / printer queue name.
// Chrome OS emulates the Mac behavior.
const std::string& real_name = printer.printer_description;
std::string real_description;
const auto it = printer.options.find(kDriverNameTagName);
if (it != printer.options.end())
real_description = it->second;
return std::make_pair(real_name, real_description);
#elif defined(OS_WIN)
return std::make_pair(GetUserFriendlyName(printer.printer_name),
printer.printer_description);
#else
return std::make_pair(printer.printer_name, printer.printer_description);
#endif
}
base::Value GetSettingsOnBlockingPool(
const std::string& device_name,
const PrinterBasicInfo& basic_info,
const PrinterSemanticCapsAndDefaults::Papers& additional_papers,
scoped_refptr<PrintBackend> print_backend) {
base::ScopedBlockingCall scoped_blocking_call(FROM_HERE,
base::BlockingType::MAY_BLOCK);
const auto printer_name_description =
GetPrinterNameAndDescription(basic_info);
base::Value printer_info(base::Value::Type::DICTIONARY);
printer_info.SetKey(kSettingDeviceName, base::Value(device_name));
printer_info.SetKey(kSettingPrinterName,
base::Value(printer_name_description.first));
printer_info.SetKey(kSettingPrinterDescription,
base::Value(printer_name_description.second));
printer_info.SetKey(
kCUPSEnterprisePrinter,
base::Value(
base::ContainsKey(basic_info.options, kCUPSEnterprisePrinter) &&
basic_info.options.at(kCUPSEnterprisePrinter) == kValueTrue));
base::Value printer_info_capabilities(base::Value::Type::DICTIONARY);
printer_info_capabilities.SetKey(kPrinter, std::move(printer_info));
printer_info_capabilities.SetKey(
kSettingCapabilities, GetPrinterCapabilitiesOnBlockingPoolThread(
device_name, additional_papers, print_backend));
return printer_info_capabilities;
}
} // namespace printing