electron_content_client.cc 8.2 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218
  1. // Copyright (c) 2014 GitHub, Inc.
  2. // Use of this source code is governed by the MIT license that can be
  3. // found in the LICENSE file.
  4. #include "shell/app/electron_content_client.h"
  5. #include <string>
  6. #include <string_view>
  7. #include <vector>
  8. #include "base/command_line.h"
  9. #include "base/containers/extend.h"
  10. #include "base/files/file_util.h"
  11. #include "base/strings/string_split.h"
  12. #include "content/public/common/content_constants.h"
  13. #include "electron/buildflags/buildflags.h"
  14. #include "electron/fuses.h"
  15. #include "extensions/common/constants.h"
  16. #include "pdf/buildflags.h"
  17. #include "ppapi/buildflags/buildflags.h"
  18. #include "shell/common/options_switches.h"
  19. #include "shell/common/process_util.h"
  20. #include "third_party/widevine/cdm/buildflags.h"
  21. #include "ui/base/l10n/l10n_util.h"
  22. #include "ui/base/resource/resource_bundle.h"
  23. #include "url/url_constants.h"
  24. #if BUILDFLAG(ENABLE_WIDEVINE)
  25. #include "base/native_library.h"
  26. #include "content/public/common/cdm_info.h"
  27. #include "media/base/video_codecs.h"
  28. #endif // BUILDFLAG(ENABLE_WIDEVINE)
  29. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  30. #include "components/pdf/common/constants.h" // nogncheck
  31. #include "shell/common/electron_constants.h"
  32. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  33. #if BUILDFLAG(ENABLE_PLUGINS)
  34. #include "content/public/common/content_plugin_info.h"
  35. #endif // BUILDFLAG(ENABLE_PLUGINS)
  36. namespace electron {
  37. namespace {
  38. enum class WidevineCdmFileCheck {
  39. kNotChecked,
  40. kFound,
  41. kNotFound,
  42. };
  43. #if BUILDFLAG(ENABLE_WIDEVINE)
  44. bool IsWidevineAvailable(
  45. base::FilePath* cdm_path,
  46. std::vector<media::VideoCodec>* codecs_supported,
  47. base::flat_set<media::CdmSessionType>* session_types_supported,
  48. base::flat_set<media::EncryptionMode>* modes_supported) {
  49. static WidevineCdmFileCheck widevine_cdm_file_check =
  50. WidevineCdmFileCheck::kNotChecked;
  51. if (widevine_cdm_file_check == WidevineCdmFileCheck::kNotChecked) {
  52. base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
  53. *cdm_path = command_line->GetSwitchValuePath(switches::kWidevineCdmPath);
  54. if (!cdm_path->empty()) {
  55. *cdm_path = cdm_path->AppendASCII(
  56. base::GetNativeLibraryName(kWidevineCdmLibraryName));
  57. widevine_cdm_file_check = base::PathExists(*cdm_path)
  58. ? WidevineCdmFileCheck::kFound
  59. : WidevineCdmFileCheck::kNotFound;
  60. }
  61. }
  62. if (widevine_cdm_file_check == WidevineCdmFileCheck::kFound) {
  63. // Add the supported codecs as if they came from the component manifest.
  64. // This list must match the CDM that is being bundled with Chrome.
  65. codecs_supported->push_back(media::VideoCodec::kCodecVP8);
  66. codecs_supported->push_back(media::VideoCodec::kCodecVP9);
  67. #if BUILDFLAG(USE_PROPRIETARY_CODECS)
  68. codecs_supported->push_back(media::VideoCodec::kCodecH264);
  69. #endif // BUILDFLAG(USE_PROPRIETARY_CODECS)
  70. // TODO(crbug.com/767941): Push persistent-license support info here once
  71. // we check in a new CDM that supports it on Linux.
  72. session_types_supported->insert(media::CdmSessionType::kTemporary);
  73. #if BUILDFLAG(IS_CHROMEOS)
  74. session_types_supported->insert(media::CdmSessionType::kPersistentLicense);
  75. #endif // BUILDFLAG(IS_CHROMEOS)
  76. modes_supported->insert(media::EncryptionMode::kCenc);
  77. return true;
  78. }
  79. return false;
  80. }
  81. #endif // BUILDFLAG(ENABLE_WIDEVINE)
  82. } // namespace
  83. ElectronContentClient::ElectronContentClient() = default;
  84. ElectronContentClient::~ElectronContentClient() = default;
  85. std::u16string ElectronContentClient::GetLocalizedString(int message_id) {
  86. return l10n_util::GetStringUTF16(message_id);
  87. }
  88. std::string_view ElectronContentClient::GetDataResource(
  89. int resource_id,
  90. ui::ResourceScaleFactor scale_factor) {
  91. return ui::ResourceBundle::GetSharedInstance().GetRawDataResourceForScale(
  92. resource_id, scale_factor);
  93. }
  94. gfx::Image& ElectronContentClient::GetNativeImageNamed(int resource_id) {
  95. return ui::ResourceBundle::GetSharedInstance().GetNativeImageNamed(
  96. resource_id);
  97. }
  98. base::RefCountedMemory* ElectronContentClient::GetDataResourceBytes(
  99. int resource_id) {
  100. return ui::ResourceBundle::GetSharedInstance().LoadDataResourceBytes(
  101. resource_id);
  102. }
  103. void ElectronContentClient::AddAdditionalSchemes(Schemes* schemes) {
  104. // Browser Process registration happens in
  105. // `api::Protocol::RegisterSchemesAsPrivileged`
  106. //
  107. // Renderer Process registration happens in `RendererClientBase`
  108. //
  109. // We use this for registration to network utility process
  110. if (IsUtilityProcess()) {
  111. const auto& cmd = *base::CommandLine::ForCurrentProcess();
  112. auto append_cli_schemes = [&cmd](auto& appendme, const auto key) {
  113. base::Extend(appendme, base::SplitString(cmd.GetSwitchValueASCII(key),
  114. ",", base::TRIM_WHITESPACE,
  115. base::SPLIT_WANT_NONEMPTY));
  116. };
  117. using namespace switches;
  118. append_cli_schemes(schemes->cors_enabled_schemes, kCORSSchemes);
  119. append_cli_schemes(schemes->csp_bypassing_schemes, kBypassCSPSchemes);
  120. append_cli_schemes(schemes->secure_schemes, kSecureSchemes);
  121. append_cli_schemes(schemes->service_worker_schemes, kServiceWorkerSchemes);
  122. append_cli_schemes(schemes->standard_schemes, kStandardSchemes);
  123. }
  124. if (electron::fuses::IsGrantFileProtocolExtraPrivilegesEnabled()) {
  125. schemes->service_worker_schemes.emplace_back(url::kFileScheme);
  126. }
  127. #if BUILDFLAG(ENABLE_ELECTRON_EXTENSIONS)
  128. schemes->standard_schemes.push_back(extensions::kExtensionScheme);
  129. schemes->savable_schemes.push_back(extensions::kExtensionScheme);
  130. schemes->secure_schemes.push_back(extensions::kExtensionScheme);
  131. schemes->service_worker_schemes.push_back(extensions::kExtensionScheme);
  132. schemes->cors_enabled_schemes.push_back(extensions::kExtensionScheme);
  133. schemes->csp_bypassing_schemes.push_back(extensions::kExtensionScheme);
  134. #endif
  135. }
  136. void ElectronContentClient::AddPlugins(
  137. std::vector<content::ContentPluginInfo>* plugins) {
  138. #if BUILDFLAG(ENABLE_PDF_VIEWER)
  139. static constexpr char kPDFPluginExtension[] = "pdf";
  140. static constexpr char kPDFPluginDescription[] = "Portable Document Format";
  141. content::ContentPluginInfo pdf_info;
  142. pdf_info.is_internal = true;
  143. pdf_info.is_out_of_process = true;
  144. pdf_info.name = kPDFInternalPluginName;
  145. pdf_info.description = kPDFPluginDescription;
  146. // This isn't a real file path; it's just used as a unique identifier.
  147. static constexpr std::string_view kPdfPluginPath = "internal-pdf-viewer";
  148. pdf_info.path = base::FilePath::FromASCII(kPdfPluginPath);
  149. content::WebPluginMimeType pdf_mime_type(
  150. pdf::kInternalPluginMimeType, kPDFPluginExtension, kPDFPluginDescription);
  151. pdf_info.mime_types.push_back(pdf_mime_type);
  152. plugins->push_back(pdf_info);
  153. #endif // BUILDFLAG(ENABLE_PDF_VIEWER)
  154. }
  155. void ElectronContentClient::AddContentDecryptionModules(
  156. std::vector<content::CdmInfo>* cdms,
  157. std::vector<media::CdmHostFilePath>* cdm_host_file_paths) {
  158. if (cdms) {
  159. #if BUILDFLAG(ENABLE_WIDEVINE)
  160. base::FilePath cdm_path;
  161. std::vector<media::VideoCodec> video_codecs_supported;
  162. base::flat_set<media::CdmSessionType> session_types_supported;
  163. base::flat_set<media::EncryptionMode> encryption_modes_supported;
  164. if (IsWidevineAvailable(&cdm_path, &video_codecs_supported,
  165. &session_types_supported,
  166. &encryption_modes_supported)) {
  167. base::CommandLine* command_line = base::CommandLine::ForCurrentProcess();
  168. auto cdm_version_string =
  169. command_line->GetSwitchValueASCII(switches::kWidevineCdmVersion);
  170. // CdmInfo needs |path| to be the actual Widevine library,
  171. // not the adapter, so adjust as necessary. It will be in the
  172. // same directory as the installed adapter.
  173. const base::Version version(cdm_version_string);
  174. DCHECK(version.IsValid());
  175. content::CdmCapability capability(
  176. video_codecs_supported, encryption_modes_supported,
  177. session_types_supported, base::flat_set<media::CdmProxy::Protocol>());
  178. cdms->push_back(content::CdmInfo(
  179. kWidevineCdmDisplayName, kWidevineCdmGuid, version, cdm_path,
  180. kWidevineCdmFileSystemId, capability, kWidevineKeySystem, false));
  181. }
  182. #endif // BUILDFLAG(ENABLE_WIDEVINE)
  183. }
  184. }
  185. } // namespace electron