2014-04-01 07:42:14 -04:00
|
|
|
require_dependency 'discourse_sass_importer'
|
|
|
|
|
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'
|
2013-02-18 19:02:00 -05:00
|
|
|
# placing this in uploads to ease deployment rules
|
|
|
|
CACHE_PATH = 'uploads/stylesheet-cache'
|
2013-02-07 10:45:24 -05:00
|
|
|
@lock = Mutex.new
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
before_create do
|
2013-02-05 14:16:51 -05:00
|
|
|
self.position ||= (SiteCustomization.maximum(:position) || 0) + 1
|
|
|
|
self.enabled ||= false
|
|
|
|
self.key ||= SecureRandom.uuid
|
|
|
|
true
|
|
|
|
end
|
|
|
|
|
2014-03-08 08:25:03 -05:00
|
|
|
def compile_stylesheet(scss)
|
2014-04-01 07:42:14 -04:00
|
|
|
::Sass::Engine.new(scss, {
|
|
|
|
syntax: :scss,
|
|
|
|
cache: false,
|
|
|
|
read_cache: false,
|
|
|
|
style: :compressed,
|
|
|
|
filesystem_importer: DiscourseSassImporter
|
|
|
|
}).render
|
2014-03-08 08:25:03 -05:00
|
|
|
end
|
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
before_save do
|
2013-09-16 12:21:49 -04:00
|
|
|
['stylesheet', 'mobile_stylesheet'].each do |stylesheet_attr|
|
|
|
|
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
|
|
|
|
error = e.sass_backtrace_str("custom stylesheet")
|
|
|
|
error.gsub!("\n", '\A ')
|
|
|
|
error.gsub!("'", '\27 ')
|
|
|
|
|
|
|
|
self.send("#{stylesheet_attr}_baked=",
|
2013-11-12 12:13:17 -05:00
|
|
|
"footer { white-space: pre; }
|
|
|
|
footer:after { content: '#{error}' }")
|
2013-09-16 12:21:49 -04:00
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2014-04-02 01:24:36 -04:00
|
|
|
after_save do
|
2013-02-28 13:54:12 -05:00
|
|
|
if stylesheet_changed?
|
2013-09-16 12:21:49 -04:00
|
|
|
File.delete(stylesheet_fullpath) if File.exists?(stylesheet_fullpath)
|
|
|
|
end
|
|
|
|
if mobile_stylesheet_changed?
|
|
|
|
File.delete(stylesheet_fullpath(:mobile)) if File.exists?(stylesheet_fullpath(:mobile))
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-02-28 13:54:12 -05:00
|
|
|
remove_from_cache!
|
2013-09-16 12:21:49 -04:00
|
|
|
if stylesheet_changed? or mobile_stylesheet_changed?
|
|
|
|
ensure_stylesheets_on_disk!
|
|
|
|
# TODO: this is broken now because there's mobile stuff too
|
2013-02-28 13:54:12 -05:00
|
|
|
MessageBus.publish "/file-change/#{key}", stylesheet_hash
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-02-28 13:54:12 -05:00
|
|
|
MessageBus.publish "/header-change/#{key}", header if header_changed?
|
2014-04-02 01:24:36 -04:00
|
|
|
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2014-04-02 01:24:36 -04:00
|
|
|
after_destroy do
|
2013-02-28 13:54:12 -05:00
|
|
|
if File.exists?(stylesheet_fullpath)
|
|
|
|
File.delete stylesheet_fullpath
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-09-16 12:21:49 -04:00
|
|
|
if File.exists?(stylesheet_fullpath(:mobile))
|
|
|
|
File.delete stylesheet_fullpath(:mobile)
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
self.remove_from_cache!
|
|
|
|
end
|
|
|
|
|
2013-02-07 08:55:04 -05:00
|
|
|
def self.enabled_key
|
|
|
|
ENABLED_KEY.dup << RailsMultisite::ConnectionManagement.current_db
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.enabled_style_key
|
2013-02-07 02:11:56 -05:00
|
|
|
@cache ||= {}
|
2013-02-28 13:54:12 -05:00
|
|
|
preview_style = @cache[enabled_key]
|
|
|
|
return if preview_style == :none
|
2013-02-07 02:11:56 -05:00
|
|
|
return preview_style if preview_style
|
|
|
|
|
2013-02-07 10:45:24 -05:00
|
|
|
@lock.synchronize do
|
2013-02-28 13:54:12 -05:00
|
|
|
style = where(enabled: true).first
|
2013-02-07 02:11:56 -05:00
|
|
|
if style
|
2013-02-28 13:54:12 -05:00
|
|
|
@cache[enabled_key] = style.key
|
2013-02-07 02:11:56 -05:00
|
|
|
else
|
2013-02-28 13:54:12 -05:00
|
|
|
@cache[enabled_key] = :none
|
|
|
|
nil
|
2013-02-07 02:11:56 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def self.custom_stylesheet(preview_style, target=:desktop)
|
2013-02-07 08:55:04 -05:00
|
|
|
preview_style ||= enabled_style_key
|
2013-02-05 14:16:51 -05:00
|
|
|
style = lookup_style(preview_style)
|
2013-09-16 12:21:49 -04:00
|
|
|
style.stylesheet_link_tag(target).html_safe if style
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def self.custom_header(preview_style, target=:desktop)
|
2013-02-07 08:55:04 -05:00
|
|
|
preview_style ||= enabled_style_key
|
2013-02-05 14:16:51 -05:00
|
|
|
style = lookup_style(preview_style)
|
2013-09-16 12:55:44 -04:00
|
|
|
if style && ((target != :mobile && style.header) || (target == :mobile && style.mobile_header))
|
2013-09-16 12:21:49 -04:00
|
|
|
target == :mobile ? style.mobile_header.html_safe : style.header.html_safe
|
2013-02-07 02:25:18 -05:00
|
|
|
else
|
|
|
|
""
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
def self.override_default_style(preview_style)
|
2013-02-07 08:55:04 -05:00
|
|
|
preview_style ||= enabled_style_key
|
2013-02-05 14:16:51 -05:00
|
|
|
style = lookup_style(preview_style)
|
|
|
|
style.override_default_style if style
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.lookup_style(key)
|
|
|
|
return if key.blank?
|
2013-02-07 10:45:24 -05:00
|
|
|
|
|
|
|
# cache is cross site resiliant cause key is secure random
|
2013-02-05 14:16:51 -05:00
|
|
|
@cache ||= {}
|
|
|
|
ensure_cache_listener
|
|
|
|
style = @cache[key]
|
|
|
|
return style if style
|
2013-02-07 10:45:24 -05:00
|
|
|
|
|
|
|
@lock.synchronize do
|
2013-02-28 13:54:12 -05:00
|
|
|
style = where(key: key).first
|
2013-09-16 12:21:49 -04:00
|
|
|
style.ensure_stylesheets_on_disk! if style
|
2013-02-05 14:16:51 -05:00
|
|
|
@cache[key] = style
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
def self.ensure_cache_listener
|
|
|
|
unless @subscribed
|
|
|
|
klass = self
|
|
|
|
MessageBus.subscribe("/site_customization") do |msg|
|
|
|
|
message = msg.data
|
2013-02-07 10:45:24 -05:00
|
|
|
klass.remove_from_cache!(message["key"], false)
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
|
|
|
@subscribed = true
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-02-28 13:54:12 -05:00
|
|
|
def self.remove_from_cache!(key, broadcast = true)
|
|
|
|
MessageBus.publish('/site_customization', key: key) if broadcast
|
2013-02-05 14:16:51 -05:00
|
|
|
if @cache
|
2013-02-07 10:45:24 -05:00
|
|
|
@lock.synchronize do
|
2013-02-05 14:16:51 -05:00
|
|
|
@cache[key] = nil
|
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
|
|
|
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
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def stylesheet_hash(target=:desktop)
|
|
|
|
Digest::MD5.hexdigest( target == :mobile ? mobile_stylesheet : stylesheet )
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-02-10 07:05:11 -05:00
|
|
|
def cache_fullpath
|
|
|
|
"#{Rails.root}/public/#{CACHE_PATH}"
|
|
|
|
end
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def ensure_stylesheets_on_disk!
|
|
|
|
[[:desktop, 'stylesheet_baked'], [:mobile, 'mobile_stylesheet_baked']].each do |target, baked_attr|
|
|
|
|
path = stylesheet_fullpath(target)
|
|
|
|
dir = cache_fullpath
|
2014-03-07 03:22:40 -05:00
|
|
|
FileUtils.mkdir_p(dir)
|
2013-09-16 12:21:49 -04:00
|
|
|
unless File.exists?(path)
|
|
|
|
File.open(path, "w") do |f|
|
|
|
|
f.puts self.send(baked_attr)
|
|
|
|
end
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
end
|
|
|
|
end
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def stylesheet_filename(target=:desktop)
|
|
|
|
target == :desktop ? "/#{self.key}.css" : "/#{target}_#{self.key}.css"
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def stylesheet_fullpath(target=:desktop)
|
|
|
|
"#{cache_fullpath}#{stylesheet_filename(target)}"
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
|
|
|
|
2013-09-16 12:21:49 -04:00
|
|
|
def stylesheet_link_tag(target=:desktop)
|
|
|
|
return mobile_stylesheet_link_tag if target == :mobile
|
2013-02-28 13:54:12 -05:00
|
|
|
return "" unless stylesheet.present?
|
2013-02-05 14:16:51 -05:00
|
|
|
return @stylesheet_link_tag if @stylesheet_link_tag
|
2013-09-16 12:21:49 -04:00
|
|
|
ensure_stylesheets_on_disk!
|
2013-12-18 14:47:22 -05:00
|
|
|
@stylesheet_link_tag = "<link class=\"custom-css\" rel=\"stylesheet\" href=\"/#{CACHE_PATH}#{stylesheet_filename}?#{stylesheet_hash}\" type=\"text/css\" media=\"screen\">"
|
2013-02-05 14:16:51 -05:00
|
|
|
end
|
2013-09-16 12:21:49 -04:00
|
|
|
|
|
|
|
def mobile_stylesheet_link_tag
|
|
|
|
return "" unless mobile_stylesheet.present?
|
|
|
|
return @mobile_stylesheet_link_tag if @mobile_stylesheet_link_tag
|
|
|
|
ensure_stylesheets_on_disk!
|
2013-12-18 14:47:22 -05:00
|
|
|
@mobile_stylesheet_link_tag = "<link class=\"custom-css\" rel=\"stylesheet\" href=\"/#{CACHE_PATH}#{stylesheet_filename(:mobile)}?#{stylesheet_hash(:mobile)}\" type=\"text/css\" media=\"screen\">"
|
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
|
|
|
|
# name :string(255) not null
|
|
|
|
# stylesheet :text
|
|
|
|
# header :text
|
|
|
|
# position :integer not null
|
|
|
|
# user_id :integer not null
|
|
|
|
# enabled :boolean not null
|
|
|
|
# key :string(255) not null
|
|
|
|
# created_at :datetime not null
|
|
|
|
# updated_at :datetime not null
|
|
|
|
# override_default_style :boolean default(FALSE), not null
|
|
|
|
# stylesheet_baked :text default(""), not null
|
|
|
|
# mobile_stylesheet :text
|
|
|
|
# mobile_header :text
|
|
|
|
# mobile_stylesheet_baked :text
|
2013-05-23 22:48:32 -04:00
|
|
|
#
|
|
|
|
# Indexes
|
|
|
|
#
|
|
|
|
# index_site_customizations_on_key (key)
|
|
|
|
#
|
|
|
|
|