2018-11-30 22:51:45 +08:00
|
|
|
# frozen_string_literal: true
|
|
|
|
require_dependency 'content_security_policy'
|
|
|
|
|
|
|
|
class ContentSecurityPolicy
|
|
|
|
class Default
|
|
|
|
attr_reader :directives
|
|
|
|
|
2020-03-20 03:54:42 +08:00
|
|
|
def initialize(base_url:)
|
|
|
|
@base_url = base_url
|
2018-11-30 22:51:45 +08:00
|
|
|
@directives = {}.tap do |directives|
|
2019-01-10 04:04:50 +08:00
|
|
|
directives[:base_uri] = [:none]
|
|
|
|
directives[:object_src] = [:none]
|
2018-11-30 22:51:45 +08:00
|
|
|
directives[:script_src] = script_src
|
|
|
|
directives[:worker_src] = worker_src
|
|
|
|
directives[:report_uri] = report_uri if SiteSetting.content_security_policy_collect_reports
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
private
|
|
|
|
|
2020-03-20 03:54:42 +08:00
|
|
|
def base_url
|
|
|
|
@base_url
|
|
|
|
end
|
2018-11-30 22:51:45 +08:00
|
|
|
|
|
|
|
SCRIPT_ASSET_DIRECTORIES = [
|
2020-08-14 19:30:39 +08:00
|
|
|
# [dir, can_use_s3_cdn, can_use_cdn, for_worker]
|
|
|
|
['/assets/', true, true, true],
|
|
|
|
['/brotli_asset/', true, true, true],
|
|
|
|
['/extra-locales/', false, false, false],
|
|
|
|
['/highlight-js/', false, true, false],
|
|
|
|
['/javascripts/', false, true, true],
|
|
|
|
['/plugins/', false, true, true],
|
|
|
|
['/theme-javascripts/', false, true, false],
|
|
|
|
['/svg-sprite/', false, true, false],
|
2018-11-30 22:51:45 +08:00
|
|
|
]
|
|
|
|
|
2020-08-14 19:30:39 +08:00
|
|
|
def script_assets(base = base_url, s3_cdn = GlobalSetting.s3_cdn_url, cdn = GlobalSetting.cdn_url, worker: false)
|
|
|
|
SCRIPT_ASSET_DIRECTORIES.map do |dir, can_use_s3_cdn, can_use_cdn, for_worker|
|
|
|
|
next if worker && !for_worker
|
2018-11-30 22:51:45 +08:00
|
|
|
if can_use_s3_cdn && s3_cdn
|
|
|
|
s3_cdn + dir
|
|
|
|
elsif can_use_cdn && cdn
|
2020-10-09 19:51:24 +08:00
|
|
|
cdn + Discourse.base_path + dir
|
2018-11-30 22:51:45 +08:00
|
|
|
else
|
|
|
|
base + dir
|
|
|
|
end
|
2020-08-14 19:30:39 +08:00
|
|
|
end.compact
|
2018-11-30 22:51:45 +08:00
|
|
|
end
|
|
|
|
|
|
|
|
def script_src
|
|
|
|
[
|
|
|
|
"#{base_url}/logs/",
|
|
|
|
"#{base_url}/sidekiq/",
|
|
|
|
"#{base_url}/mini-profiler-resources/",
|
|
|
|
*script_assets
|
|
|
|
].tap do |sources|
|
2020-04-02 23:16:38 +08:00
|
|
|
sources << :report_sample if SiteSetting.content_security_policy_collect_reports
|
2020-01-25 02:18:03 +08:00
|
|
|
sources << :unsafe_eval if Rails.env.development? # TODO remove this once we have proper source maps in dev
|
2020-11-07 04:15:36 +08:00
|
|
|
# we need analytics.js still as gtag/js is a script wrapper for it
|
2020-10-14 01:20:41 +08:00
|
|
|
sources << 'https://www.google-analytics.com/analytics.js' if SiteSetting.ga_universal_tracking_code.present?
|
2020-11-07 04:15:36 +08:00
|
|
|
sources << 'https://www.googletagmanager.com/gtag/js' if SiteSetting.ga_universal_tracking_code.present? && SiteSetting.ga_version == "v4_gtag"
|
2018-12-14 21:17:31 +08:00
|
|
|
sources << 'https://www.googletagmanager.com/gtm.js' if SiteSetting.gtm_container_id.present?
|
2018-11-30 22:51:45 +08:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def worker_src
|
|
|
|
[
|
2020-08-15 01:15:30 +08:00
|
|
|
"'self'", # For service worker
|
2020-08-14 19:30:39 +08:00
|
|
|
*script_assets(worker: true)
|
2018-11-30 22:51:45 +08:00
|
|
|
]
|
|
|
|
end
|
|
|
|
|
|
|
|
def report_uri
|
|
|
|
"#{base_url}/csp_reports"
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|