2014-05-02 17:46:03 -04:00
|
|
|
require_dependency 'sass/discourse_sass_compiler'
|
2014-11-20 16:32:16 -05:00
|
|
|
require_dependency 'sass/discourse_stylesheets'
|
2014-12-22 20:46:10 -05:00
|
|
|
require_dependency 'distributed_cache'
|
2014-04-01 07:42:14 -04:00
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
class SiteCustomization < ActiveRecord::Base
|
2013-02-07 02:11:56 -05:00
|
|
|
ENABLED_KEY = '7e202ef2-56d7-47d5-98d8-a9c8d15e57dd'
|
2014-12-22 20:46:10 -05:00
|
|
|
@cache = DistributedCache.new('site_customization')
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2015-08-06 17:18:00 -04:00
|
|
|
def self.css_fields
|
|
|
|
%w(stylesheet mobile_stylesheet embedded_css)
|
|
|
|
end
|
|
|
|
|
2015-11-26 19:58:46 -05:00
|
|
|
def self.html_fields
|
2015-11-26 20:22:54 -05:00
|
|
|
%w(body_tag head_tag header mobile_header footer mobile_footer)
|
2015-11-26 19:58:46 -05:00
|
|
|
end
|
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
before_create do
|
2013-02-05 14:16:51 -05:00
|
|
|
self.enabled ||= false
|
|
|
|
self.key ||= SecureRandom.uuid
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2014-03-08 08:25:03 -05:00
|
|
|
def compile_stylesheet(scss)
|
2015-08-16 20:59:46 -04:00
|
|
|
DiscourseSassCompiler.compile("@import \"theme_variables\";\n" << scss, 'custom')
|
2014-04-06 21:34:35 -04:00
|
|
|
rescue => e
|
|
|
|
puts e.backtrace.join("\n") unless Sass::SyntaxError === e
|
|
|
|
raise e
|
2014-03-08 08:25:03 -05:00
|
|
|
end
|
|
|
|
|
2016-03-18 14:41:27 -04:00
|
|
|
def transpile(es6_source, version)
|
|
|
|
template = Tilt::ES6ModuleTranspilerTemplate.new {}
|
|
|
|
wrapped = <<PLUGIN_API_JS
|
|
|
|
Discourse._registerPluginCode('#{version}', api => {
|
|
|
|
#{es6_source}
|
|
|
|
});
|
|
|
|
PLUGIN_API_JS
|
|
|
|
|
|
|
|
template.babel_transpile(wrapped)
|
|
|
|
end
|
|
|
|
|
2015-11-26 19:58:46 -05:00
|
|
|
def process_html(html)
|
|
|
|
doc = Nokogiri::HTML.fragment(html)
|
|
|
|
doc.css('script[type="text/x-handlebars"]').each do |node|
|
|
|
|
name = node["name"] || node["data-template-name"] || "broken"
|
|
|
|
precompiled =
|
|
|
|
if name =~ /\.raw$/
|
2016-04-17 22:47:52 -04:00
|
|
|
"Discourse.EmberCompatHandlebars.template(#{Barber::Precompiler.compile(node.inner_html)})"
|
2015-11-26 19:58:46 -05:00
|
|
|
else
|
|
|
|
"Ember.HTMLBars.template(#{Barber::Ember::Precompiler.compile(node.inner_html)})"
|
|
|
|
end
|
|
|
|
compiled = <<SCRIPT
|
|
|
|
Ember.TEMPLATES[#{name.inspect}] = #{precompiled};
|
|
|
|
SCRIPT
|
|
|
|
node.replace("<script>#{compiled}</script>")
|
|
|
|
end
|
|
|
|
|
2016-03-18 14:41:27 -04:00
|
|
|
doc.css('script[type="text/discourse-plugin"]').each do |node|
|
|
|
|
if node['version'].present?
|
|
|
|
begin
|
|
|
|
code = transpile(node.inner_html, node['version'])
|
|
|
|
node.replace("<script>#{code}</script>")
|
2016-05-19 08:25:08 -04:00
|
|
|
rescue MiniRacer::RuntimeError => ex
|
2016-03-18 14:41:27 -04:00
|
|
|
node.replace("<script type='text/discourse-js-error'>#{ex.message}</script>")
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-11-26 19:58:46 -05:00
|
|
|
doc.to_s
|
|
|
|
end
|
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
before_save do
|
2015-11-26 19:58:46 -05:00
|
|
|
SiteCustomization.html_fields.each do |html_attr|
|
|
|
|
if self.send("#{html_attr}_changed?")
|
|
|
|
self.send("#{html_attr}_baked=", process_html(self.send(html_attr)))
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-08-06 17:18:00 -04:00
|
|
|
SiteCustomization.css_fields.each do |stylesheet_attr|
|
2013-09-16 12:21:49 -04:00
|
|
|
if self.send("#{stylesheet_attr}_changed?")
|
|
|
|
begin
|
2014-03-08 08:25:03 -05:00
|
|
|
self.send("#{stylesheet_attr}_baked=", compile_stylesheet(self.send(stylesheet_attr)))
|
2013-09-16 12:21:49 -04:00
|
|
|
rescue Sass::SyntaxError => e
|
2014-05-02 17:46:03 -04:00
|
|
|
self.send("#{stylesheet_attr}_baked=", DiscourseSassCompiler.error_as_css(e, "custom stylesheet"))
|
2013-09-16 12:21:49 -04:00
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-08-06 17:18:00 -04:00
|
|
|
def any_stylesheet_changed?
|
|
|
|
SiteCustomization.css_fields.each do |fieldname|
|
|
|
|
return true if self.send("#{fieldname}_changed?")
|
|
|
|
end
|
|
|
|
false
|
|
|
|
end
|
|
|
|
|
2014-04-02 01:24:36 -04:00
|
|
|
after_save do
|
2013-02-28 13:54:12 -05:00
|
|
|
remove_from_cache!
|
2015-08-06 17:18:00 -04:00
|
|
|
if any_stylesheet_changed?
|
2015-05-03 22:21:00 -04:00
|
|
|
MessageBus.publish "/file-change/#{key}", SecureRandom.hex
|
|
|
|
MessageBus.publish "/file-change/#{SiteCustomization::ENABLED_KEY}", SecureRandom.hex
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2015-05-03 22:21:00 -04:00
|
|
|
MessageBus.publish "/header-change/#{key}", header if header_changed?
|
|
|
|
MessageBus.publish "/footer-change/#{key}", footer if footer_changed?
|
2014-11-20 16:32:16 -05:00
|
|
|
DiscourseStylesheets.cache.clear
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2014-04-02 01:24:36 -04:00
|
|
|
after_destroy do
|
2014-12-22 20:46:10 -05:00
|
|
|
remove_from_cache!
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-02-07 08:55:04 -05:00
|
|
|
def self.enabled_key
|
|
|
|
ENABLED_KEY.dup << RailsMultisite::ConnectionManagement.current_db
|
|
|
|
end
|
|
|
|
|
2015-08-06 17:18:00 -04:00
|
|
|
def self.field_for_target(target=nil)
|
|
|
|
target ||= :desktop
|
|
|
|
|
|
|
|
case target.to_sym
|
|
|
|
when :mobile then :mobile_stylesheet
|
|
|
|
when :desktop then :stylesheet
|
|
|
|
when :embedded then :embedded_css
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.baked_for_target(target=nil)
|
|
|
|
"#{field_for_target(target)}_baked".to_sym
|
|
|
|
end
|
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
def self.enabled_stylesheet_contents(target=:desktop)
|
|
|
|
@cache["enabled_stylesheet_#{target}"] ||= where(enabled: true)
|
|
|
|
.order(:name)
|
2015-08-06 17:18:00 -04:00
|
|
|
.pluck(baked_for_target(target))
|
2014-12-22 20:46:10 -05:00
|
|
|
.compact
|
|
|
|
.join("\n")
|
2013-02-07 02:11:56 -05:00
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
def self.stylesheet_contents(key, target)
|
|
|
|
if key == ENABLED_KEY
|
|
|
|
enabled_stylesheet_contents(target)
|
2013-02-07 02:25:18 -05:00
|
|
|
else
|
2014-12-22 20:46:10 -05:00
|
|
|
where(key: key)
|
2015-08-06 17:18:00 -04:00
|
|
|
.pluck(baked_for_target(target))
|
2014-12-22 20:46:10 -05:00
|
|
|
.first
|
2013-02-07 02:25:18 -05:00
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
def self.custom_stylesheet(preview_style=nil, target=:desktop)
|
|
|
|
preview_style ||= ENABLED_KEY
|
|
|
|
if preview_style == ENABLED_KEY
|
|
|
|
stylesheet_link_tag(ENABLED_KEY, target, enabled_stylesheet_contents(target))
|
2014-11-10 15:51:55 -05:00
|
|
|
else
|
2014-12-22 20:46:10 -05:00
|
|
|
lookup_field(preview_style, target, :stylesheet_link_tag)
|
2014-11-10 15:51:55 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2015-01-14 05:52:42 -05:00
|
|
|
%i{header top footer head_tag body_tag}.each do |name|
|
|
|
|
define_singleton_method("custom_#{name}") do |preview_style=nil, target=:desktop|
|
|
|
|
preview_style ||= ENABLED_KEY
|
|
|
|
lookup_field(preview_style, target, name)
|
|
|
|
end
|
2014-12-22 20:46:10 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.lookup_field(key, target, field)
|
2013-02-05 14:16:51 -05:00
|
|
|
return if key.blank?
|
2013-02-07 10:45:24 -05:00
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
cache_key = key + target.to_s + field.to_s;
|
2013-02-07 10:45:24 -05:00
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
lookup = @cache[cache_key]
|
|
|
|
return lookup.html_safe if lookup
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2015-01-14 05:52:42 -05:00
|
|
|
styles = if key == ENABLED_KEY
|
|
|
|
order(:name).where(enabled:true).to_a
|
|
|
|
else
|
|
|
|
[find_by(key: key)].compact
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2015-01-14 05:52:42 -05:00
|
|
|
val = if styles.present?
|
|
|
|
styles.map do |style|
|
|
|
|
lookup = target == :mobile ? "mobile_#{field}" : field
|
2015-11-26 19:58:46 -05:00
|
|
|
if html_fields.include?(lookup.to_s)
|
|
|
|
style.ensure_baked!(lookup)
|
|
|
|
style.send("#{lookup}_baked")
|
|
|
|
else
|
|
|
|
style.send(lookup)
|
|
|
|
end
|
2015-01-14 05:52:42 -05:00
|
|
|
end.compact.join("\n")
|
|
|
|
end
|
2014-12-22 20:46:10 -05:00
|
|
|
|
|
|
|
(@cache[cache_key] = val || "").html_safe
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-02-28 13:54:12 -05:00
|
|
|
def self.remove_from_cache!(key, broadcast = true)
|
2015-05-03 22:21:00 -04:00
|
|
|
MessageBus.publish('/site_customization', key: key) if broadcast
|
2014-12-22 20:46:10 -05:00
|
|
|
clear_cache!
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.clear_cache!
|
|
|
|
@cache.clear
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2015-11-26 19:58:46 -05:00
|
|
|
def ensure_baked!(field)
|
|
|
|
unless self.send("#{field}_baked")
|
|
|
|
if val = self.send(field)
|
|
|
|
val = process_html(val) rescue ""
|
|
|
|
self.update_columns("#{field}_baked" => val)
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
def remove_from_cache!
|
2013-02-07 08:55:04 -05:00
|
|
|
self.class.remove_from_cache!(self.class.enabled_key)
|
2013-02-28 13:54:12 -05:00
|
|
|
self.class.remove_from_cache!(key)
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
def mobile_stylesheet_link_tag
|
|
|
|
stylesheet_link_tag(:mobile)
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def stylesheet_link_tag(target=:desktop)
|
2015-08-06 17:18:00 -04:00
|
|
|
content = self.send(SiteCustomization.field_for_target(target))
|
2014-12-22 20:46:10 -05:00
|
|
|
SiteCustomization.stylesheet_link_tag(key, target, content)
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-09-16 12:21:49 -04:00
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
def self.stylesheet_link_tag(key, target, content)
|
|
|
|
return "" unless content.present?
|
|
|
|
|
|
|
|
hash = Digest::MD5.hexdigest(content)
|
|
|
|
link_css_tag "/site_customizations/#{key}.css?target=#{target}&v=#{hash}"
|
2014-07-29 00:55:48 -04:00
|
|
|
end
|
|
|
|
|
2014-12-22 20:46:10 -05:00
|
|
|
def self.link_css_tag(href)
|
2015-03-09 16:38:25 -04:00
|
|
|
href = (GlobalSetting.cdn_url || "") + "#{GlobalSetting.relative_url_root}#{href}&__ws=#{Discourse.current_hostname}"
|
2014-12-22 20:46:10 -05:00
|
|
|
%Q{<link class="custom-css" rel="stylesheet" href="#{href}" type="text/css" media="all">}.html_safe
|
2013-09-16 12:21:49 -04:00
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-05-23 22:48:32 -04:00
|
|
|
|
|
|
|
# == Schema Information
|
|
|
|
#
|
|
|
|
# Table name: site_customizations
|
|
|
|
#
|
2013-10-03 23:28:49 -04:00
|
|
|
# id :integer not null, primary key
|
2016-02-22 18:33:53 -05:00
|
|
|
# name :string not null
|
2013-10-03 23:28:49 -04:00
|
|
|
# stylesheet :text
|
|
|
|
# header :text
|
|
|
|
# user_id :integer not null
|
|
|
|
# enabled :boolean not null
|
2016-02-22 18:33:53 -05:00
|
|
|
# key :string not null
|
2014-08-27 01:19:25 -04:00
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
2013-10-03 23:28:49 -04:00
|
|
|
# stylesheet_baked :text default(""), not null
|
|
|
|
# mobile_stylesheet :text
|
2016-02-22 18:33:53 -05:00
|
|
|
# mobile_header :text
|
2013-10-03 23:28:49 -04:00
|
|
|
# mobile_stylesheet_baked :text
|
2014-11-19 22:53:15 -05:00
|
|
|
# footer :text
|
|
|
|
# mobile_footer :text
|
2015-02-03 23:09:00 -05:00
|
|
|
# head_tag :text
|
|
|
|
# body_tag :text
|
|
|
|
# top :text
|
|
|
|
# mobile_top :text
|
2015-09-17 20:41:10 -04:00
|
|
|
# embedded_css :text
|
|
|
|
# embedded_css_baked :text
|
2016-02-22 18:33:53 -05:00
|
|
|
# head_tag_baked :text
|
|
|
|
# body_tag_baked :text
|
|
|
|
# header_baked :text
|
|
|
|
# mobile_header_baked :text
|
|
|
|
# footer_baked :text
|
|
|
|
# mobile_footer_baked :text
|
2013-05-23 22:48:32 -04:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
|
|
|
# index_site_customizations_on_key (key)
|
|
|
|
#
|