electron_content_client.cc 8.2 KB

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