187 lines
4.3 KiB
Ruby
Executable File
187 lines
4.3 KiB
Ruby
Executable File
#!/usr/bin/env ruby
|
|
# frozen_string_literal: true
|
|
|
|
require 'fileutils'
|
|
require 'pathname'
|
|
require 'tempfile'
|
|
require 'securerandom'
|
|
require 'minitar'
|
|
require 'zlib'
|
|
require 'find'
|
|
require 'net/http'
|
|
require 'net/http/post/multipart'
|
|
require 'uri'
|
|
require 'listen'
|
|
require 'json'
|
|
|
|
# Work in progress theme watcher for Discourse
|
|
#
|
|
# Monitor a theme directory locally and automatically keep it in sync with Discourse
|
|
|
|
def usage
|
|
puts "Usage: theme-watcher DIR SITE"
|
|
exit 1
|
|
end
|
|
|
|
WATCHER_SETTINGS_FILE = File.expand_path("~/.discourse-theme-watcher")
|
|
|
|
$api_key = ENV['DISCOURSE_API_KEY']
|
|
$dir = ARGV[0]
|
|
$site = ARGV[1]
|
|
$theme_id = nil
|
|
|
|
if $site !~ /https?:\/\//i
|
|
$site = "http://#{$site}"
|
|
end
|
|
|
|
puts "Watching #{$dir} and uploading changes to #{$site}"
|
|
|
|
if !$api_key && File.exist?(WATCHER_SETTINGS_FILE)
|
|
$api_key = File.read(WATCHER_SETTINGS_FILE).strip
|
|
puts "Using previously stored api key in #{WATCHER_SETTINGS_FILE}"
|
|
end
|
|
|
|
if !$api_key
|
|
puts "No API key found in DISCOURSE_API_KEY env var enter your API key: "
|
|
$api_key = STDIN.gets.strip
|
|
puts "Would you like me to store this API key in #{WATCHER_SETTINGS_FILE}? (Yes|No)"
|
|
answer = STDIN.gets.strip
|
|
if answer =~ /y(es)?/i
|
|
File.write WATCHER_SETTINGS_FILE, $api_key
|
|
end
|
|
end
|
|
|
|
if !File.exist?("#{$dir}/about.json")
|
|
puts "No about.json file found in #{dir}!"
|
|
puts
|
|
usage
|
|
end
|
|
|
|
def compress_dir(gzip, dir)
|
|
sgz = Zlib::GzipWriter.new(File.open(gzip, 'wb'))
|
|
tar = Archive::Tar::Minitar::Output.new(sgz)
|
|
|
|
Dir.chdir(dir + "/../") do
|
|
Find.find(File.basename(dir)) do |x|
|
|
Find.prune if File.basename(x)[0] == ?.
|
|
next if File.directory?(x)
|
|
|
|
Minitar.pack_file(x, tar)
|
|
end
|
|
end
|
|
ensure
|
|
tar.close
|
|
sgz.close
|
|
end
|
|
|
|
def diagnose_errors(json)
|
|
count = 0
|
|
json["theme"]["theme_fields"].each do |row|
|
|
if (error = row["error"]) && error.length > 0
|
|
if count == 0
|
|
puts
|
|
end
|
|
count += 1
|
|
puts
|
|
puts "Error in #{row["target"]} #{row["name"]}: #{row["error"]}"
|
|
puts
|
|
end
|
|
end
|
|
count
|
|
end
|
|
|
|
def upload_theme_field(target: , name: , type_id: , value:)
|
|
args = {
|
|
theme: {
|
|
theme_fields: [{
|
|
name: name,
|
|
target: target,
|
|
type_id: type_id,
|
|
value: value
|
|
}]
|
|
}
|
|
}
|
|
|
|
uri = URI.parse($site + "/admin/themes/#{$theme_id}?api_key=#{$api_key}")
|
|
|
|
http = Net::HTTP.new(uri.host, uri.port)
|
|
request = Net::HTTP::Put.new(uri.request_uri, 'Content-Type' => 'application/json')
|
|
request.body = args.to_json
|
|
|
|
http.start do |h|
|
|
response = h.request(request)
|
|
if response.code.to_i == 200
|
|
json = JSON.parse(response.body)
|
|
if diagnose_errors(json) == 0
|
|
puts "(done)"
|
|
end
|
|
else
|
|
puts "Error importing field status: #{response.code}"
|
|
end
|
|
end
|
|
end
|
|
|
|
def upload_full_theme(dir, site)
|
|
filename = "#{Pathname.new(Dir.tmpdir).realpath}/bundle_#{SecureRandom.hex}.tar.gz"
|
|
compress_dir(filename, dir)
|
|
|
|
# new full upload endpoint
|
|
uri = URI.parse(site + "/admin/themes/import.json?api_key=#{$api_key}")
|
|
http = Net::HTTP.new(uri.host, uri.port)
|
|
File.open(filename) do |tgz|
|
|
|
|
request = Net::HTTP::Post::Multipart.new(
|
|
uri.request_uri,
|
|
"bundle" => UploadIO.new(tgz, "application/tar+gzip", "bundle.tar.gz"),
|
|
)
|
|
response = http.request(request)
|
|
if response.code.to_i == 201
|
|
json = JSON.parse(response.body)
|
|
$theme_id = json["theme"]["id"]
|
|
if diagnose_errors(json) == 0
|
|
puts "(done)"
|
|
end
|
|
else
|
|
puts "Error importing theme status: #{response.code}"
|
|
end
|
|
end
|
|
|
|
ensure
|
|
FileUtils.rm_f filename
|
|
end
|
|
|
|
print "Uploading theme: "
|
|
upload_full_theme($dir, $site)
|
|
|
|
def resolve_file(path)
|
|
dir_len = File.expand_path($dir).length
|
|
name = File.expand_path(path)[dir_len + 1..-1]
|
|
|
|
target, file = name.split("/")
|
|
|
|
if ["common", "desktop", "mobile"].include?(target)
|
|
if file = "#{target}.scss"
|
|
# a CSS file
|
|
return [target, "scss", 1]
|
|
end
|
|
end
|
|
|
|
nil
|
|
end
|
|
|
|
listener = Listen.to($dir) do |modified, added, removed|
|
|
if modified.length == 1 &&
|
|
added.length == 0 &&
|
|
removed.length == 0 &&
|
|
(target, name, type_id = resolve_file(modified[0]))
|
|
print "Updating #{target} #{name}: "
|
|
upload_theme_field(target: target, name: name, value: File.read(modified[0]), type_id: type_id)
|
|
else
|
|
print "Full re-sync is required, re-uploading theme: "
|
|
upload_full_theme($dir, $site)
|
|
end
|
|
end
|
|
|
|
listener.start
|
|
sleep
|