mirror of
https://github.com/discourse/discourse.git
synced 2025-01-28 00:22:00 +08:00
3157d5ee1b
* DEV: ensures stylesheet watcher isn't crashing with gems plugins This bug has been exhibited since discourse_dev is now including an auth plugin which was loaded as a relative path instead of an absolute path, eg: `Users/bob/.gem/ruby/2.6.6/gems/discourse_dev-0.1.0/auth/plugin.rb` Instead of `/Users/bob/.gem/ruby/2.6.6/gems/discourse_dev-0.1.0/auth/plugin.rb`
152 lines
4.0 KiB
Ruby
152 lines
4.0 KiB
Ruby
# frozen_string_literal: true
|
||
|
||
require 'listen'
|
||
|
||
module Stylesheet
|
||
class Watcher
|
||
|
||
def self.watch(paths = nil)
|
||
watcher = new(paths)
|
||
watcher.start
|
||
watcher
|
||
end
|
||
|
||
def initialize(paths)
|
||
@paths = paths || Watcher.default_paths
|
||
@queue = Queue.new
|
||
end
|
||
|
||
def self.default_paths
|
||
return @default_paths if @default_paths
|
||
|
||
@default_paths = ["app/assets/stylesheets"]
|
||
Discourse.plugins.each do |plugin|
|
||
if plugin.path.to_s.include?(Rails.root.to_s)
|
||
@default_paths << File.dirname(plugin.path).sub(Rails.root.to_s, '').sub(/^\//, '')
|
||
else
|
||
# if plugin doesn’t seem to be in our app, consider it as outside of the app
|
||
# and ignore it
|
||
warn("[stylesheet watcher] Ignoring outside of rails root plugin: #{plugin.path.to_s}")
|
||
end
|
||
end
|
||
@default_paths
|
||
end
|
||
|
||
def start
|
||
Thread.new do
|
||
begin
|
||
while true
|
||
worker_loop
|
||
end
|
||
rescue => e
|
||
STDERR.puts "CSS change notifier crashed \n#{e}"
|
||
start
|
||
end
|
||
end
|
||
|
||
listener_opts = { ignore: /xxxx/, only: /\.(css|scss)$/ }
|
||
listener_opts[:force_polling] = true if ENV['FORCE_POLLING']
|
||
|
||
Thread.new do
|
||
begin
|
||
plugins_paths = Dir.glob("#{Rails.root}/plugins/*").map do |file|
|
||
if File.symlink?(file)
|
||
File.expand_path(File.readlink(file), "#{Rails.root}/plugins")
|
||
else
|
||
file
|
||
end
|
||
end.compact
|
||
|
||
listener = Listen.to(*@paths, listener_opts) do |modified, added, _|
|
||
paths = [modified, added].flatten
|
||
paths.compact!
|
||
paths.map! do |long|
|
||
plugin_name = nil
|
||
plugins_paths.each do |plugin_path|
|
||
if long.include?("#{plugin_path}/")
|
||
plugin_name = File.basename(plugin_path)
|
||
break
|
||
end
|
||
end
|
||
|
||
target = nil
|
||
target_match = long.match(/admin|desktop|mobile|publish/)
|
||
if target_match&.length
|
||
target = target_match[0]
|
||
end
|
||
|
||
{
|
||
basename: File.basename(long),
|
||
target: target,
|
||
plugin_name: plugin_name
|
||
}
|
||
end
|
||
|
||
process_change(paths)
|
||
end
|
||
rescue => e
|
||
STDERR.puts "Failed to listen for CSS changes: \n#{e}"
|
||
end
|
||
listener.start
|
||
sleep
|
||
end
|
||
end
|
||
|
||
def core_assets_refresh(target)
|
||
targets = target ? [target] : ["desktop", "mobile", "admin"]
|
||
Stylesheet::Manager.clear_core_cache!(targets)
|
||
message = targets.map! do |name|
|
||
msgs = []
|
||
active_themes.each do |theme_id|
|
||
msgs << Stylesheet::Manager.stylesheet_data(name.to_sym, theme_id)
|
||
end
|
||
msgs
|
||
end.flatten!
|
||
MessageBus.publish '/file-change', message
|
||
end
|
||
|
||
def plugin_assets_refresh(plugin_name, target)
|
||
Stylesheet::Manager.clear_plugin_cache!(plugin_name)
|
||
targets = []
|
||
if target.present?
|
||
targets.push("#{plugin_name}_#{target.to_s}") if DiscoursePluginRegistry.stylesheets_exists?(plugin_name, target.to_sym)
|
||
else
|
||
targets.push(plugin_name)
|
||
end
|
||
message = targets.map! do |name|
|
||
msgs = []
|
||
active_themes.each do |theme_id|
|
||
msgs << Stylesheet::Manager.stylesheet_data(name.to_sym, theme_id)
|
||
end
|
||
msgs
|
||
end.flatten!
|
||
MessageBus.publish '/file-change', message
|
||
end
|
||
|
||
def worker_loop
|
||
path = @queue.pop
|
||
|
||
while @queue.length > 0
|
||
@queue.pop
|
||
end
|
||
|
||
if path[:plugin_name]
|
||
plugin_assets_refresh(path[:plugin_name], path[:target])
|
||
else
|
||
core_assets_refresh(path[:target])
|
||
end
|
||
end
|
||
|
||
def process_change(paths)
|
||
paths.each do |path|
|
||
@queue.push path
|
||
end
|
||
end
|
||
|
||
def active_themes
|
||
@active_themes ||= Theme.user_selectable.pluck(:id)
|
||
end
|
||
|
||
end
|
||
end
|