class ShopifyTheme::Cli

Constants

DEFAULT_WHITELIST
IGNORE
TIMEFORMAT

Public Instance Methods

bootstrap(api_key=nil, password=nil, store=nil, theme_name=nil, master=nil) click to toggle source
# File lib/shopify_theme/cli.rb, line 55
def bootstrap(api_key=nil, password=nil, store=nil, theme_name=nil, master=nil)
  ShopifyTheme.config = {:api_key => api_key, :password => password, :store => store}

  theme_name ||= 'Timber'
  say("Registering #{theme_name} theme on #{store}", :green)
  theme = ShopifyTheme.upload_timber(theme_name, master || false)

  say("Creating directory named #{theme_name}", :green)
  empty_directory(theme_name)

  say("Saving configuration to #{theme_name}", :green)
  ShopifyTheme.config.merge!(theme_id: theme['id'])
  create_file("#{theme_name}/config.yml", ShopifyTheme.config.to_yaml)

  say("Downloading #{theme_name} assets from Shopify")
  Dir.chdir(theme_name)
  download()
end
check() click to toggle source
# File lib/shopify_theme/cli.rb, line 39
def check
  if ShopifyTheme.check_config
    say("Configuration [OK]", :green)
  else
    say("Configuration [FAIL]", :red)
  end
end
configure(api_key=nil, password=nil, store=nil, theme_id=nil) click to toggle source
# File lib/shopify_theme/cli.rb, line 48
def configure(api_key=nil, password=nil, store=nil, theme_id=nil)
  config = {:api_key => api_key, :password => password, :store => store, :theme_id => theme_id}
  create_file('config.yml', config.to_yaml)
end
download(*keys) click to toggle source
# File lib/shopify_theme/cli.rb, line 77
def download(*keys)
  assets = keys.empty? ? ShopifyTheme.asset_list : keys

  if options['exclude']
    assets = assets.delete_if { |asset| asset =~ Regexp.new(options['exclude']) }
  end

  assets.each do |asset|
    download_asset(asset)
    say("#{ShopifyTheme.api_usage} Downloaded: #{asset}", :green) unless options['quiet']
  end
  say("Done.", :green) unless options['quiet']
end
open(*keys) click to toggle source
# File lib/shopify_theme/cli.rb, line 92
def open(*keys)
  if Launchy.open shop_theme_url
    say("Done.", :green)
  end
end
remove(*keys) click to toggle source
# File lib/shopify_theme/cli.rb, line 129
def remove(*keys)
  keys.each do |key|
    delete_asset(key, options['quiet'])
  end
  say("Done.", :green) unless options['quiet']
end
replace(*keys) click to toggle source
# File lib/shopify_theme/cli.rb, line 110
def replace(*keys)
  say("Are you sure you want to completely replace your shop theme assets? This is not undoable.", :yellow)
  if ask("Continue? (Y/N): ") == "Y"
    # only delete files on remote that are not present locally
    # files present on remote and present locally get overridden anyway
    remote_assets = keys.empty? ? (ShopifyTheme.asset_list - local_assets_list) : keys
    remote_assets.each do |asset|
      delete_asset(asset, options['quiet']) unless ShopifyTheme.ignore_files.any? { |regex| regex =~ asset }
    end
    local_assets = keys.empty? ? local_assets_list : keys
    local_assets.each do |asset|
      send_asset(asset, options['quiet'])
    end
    say("Done.", :green) unless options['quiet']
  end
end
systeminfo() click to toggle source
# File lib/shopify_theme/cli.rb, line 158
def systeminfo
  ruby_version = "#{RUBY_VERSION}"
  ruby_version += "-p#{RUBY_PATCHLEVEL}" if RUBY_PATCHLEVEL
  puts "Ruby: v#{ruby_version}"
  puts "Operating System: #{RUBY_PLATFORM}"
  %w(Thor Listen HTTParty Launchy).each do |lib|
    require "#{lib.downcase}/version"
    puts "#{lib}: v" +  Kernel.const_get("#{lib}::VERSION")
  end
end
upload(*keys) click to toggle source
# File lib/shopify_theme/cli.rb, line 100
def upload(*keys)
  assets = keys.empty? ? local_assets_list : keys
  assets.each do |asset|
    send_asset(asset, options['quiet'])
  end
  say("Done.", :green) unless options['quiet']
end
watch() click to toggle source
# File lib/shopify_theme/cli.rb, line 139
def watch
  puts "Watching current folder: #{Dir.pwd}"
  watcher do |filename, event|
    filename = filename.gsub("#{Dir.pwd}/", '')

    next unless local_assets_list.include?(filename)
    action = if [:changed, :new].include?(event)
      :send_asset
    elsif event == :delete
      :delete_asset
    else
      raise NotImplementedError, "Unknown event -- #{event} -- #{filename}"
    end

    send(action, filename, options['quiet'])
  end
end

Protected Instance Methods

config() click to toggle source
# File lib/shopify_theme/cli.rb, line 171
def config
  @config ||= YAML.load_file 'config.yml'
end
shop_theme_url() click to toggle source
# File lib/shopify_theme/cli.rb, line 175
def shop_theme_url
  url = config[:store]
  url += "?preview_theme_id=#{config[:theme_id]}" if config[:theme_id] && config[:theme_id].to_i > 0
  url
end

Private Instance Methods

binary_file?(path) click to toggle source
# File lib/shopify_theme/cli.rb, line 264
def binary_file?(path)
  mime = MimeMagic.by_path(path)
  say("'#{path}' is an unknown file-type, uploading asset as binary", :yellow) if mime.nil? && ENV['TEST'] != 'true'
  mime.nil? || !mime.text?
end
delete_asset(key, quiet=false) click to toggle source
# File lib/shopify_theme/cli.rb, line 240
def delete_asset(key, quiet=false)
  return unless valid?(key)
  response = show_during("[#{timestamp}] Removing: #{key}", quiet) do
    ShopifyTheme.delete_asset(key)
  end
  if response.success?
    say("[#{timestamp}] Removed: #{key}", :green) unless quiet
  else
    report_error(Time.now, "Could not remove #{key}", response)
  end
end
download_asset(key) click to toggle source
# File lib/shopify_theme/cli.rb, line 202
def download_asset(key)
  return unless valid?(key)
  notify_and_sleep("Approaching limit of API permits. Naptime until more permits become available!") if ShopifyTheme.needs_sleep?
  asset = ShopifyTheme.get_asset(key)
  if asset['value']
    # For CRLF line endings
    content = asset['value'].gsub("\r", "")
    format = "w"
  elsif asset['attachment']
    content = Base64.decode64(asset['attachment'])
    format = "w+b"
  end

  FileUtils.mkdir_p(File.dirname(key))
  File.open(key, format) {|f| f.write content} if content
end
errors_from_response(response) click to toggle source
# File lib/shopify_theme/cli.rb, line 275
def errors_from_response(response)
  object = {status: response.headers['status'], request_id: response.headers['x-request-id']}

  errors = response.parsed_response ? response.parsed_response["errors"] : response.body

  object[:errors] = case errors
                    when NilClass
                      ''
                    when String
                      errors.strip
                    else
                      errors.values.join(", ")
                    end
  object.delete(:errors) if object[:errors].length <= 0
  object
end
local_assets_list() click to toggle source
# File lib/shopify_theme/cli.rb, line 189
def local_assets_list
  local_files.reject do |p|
    @permitted_files ||= (DEFAULT_WHITELIST | ShopifyTheme.whitelist_files).map{|pattern| Regexp.new(pattern)}
    @permitted_files.none? { |regex| regex =~ p } || ShopifyTheme.ignore_files.any? { |regex| regex =~ p }
  end
end
local_files() click to toggle source
# File lib/shopify_theme/cli.rb, line 196
def local_files
  Dir.glob(File.join('**', '*')).reject do |f|
    File.directory?(f)
  end
end
notify_and_sleep(message) click to toggle source
# File lib/shopify_theme/cli.rb, line 252
def notify_and_sleep(message)
  say(message, :red)
  ShopifyTheme.sleep
end
report_error(time, message, response) click to toggle source
# File lib/shopify_theme/cli.rb, line 270
def report_error(time, message, response)
  say("[#{timestamp(time)}] Error: #{message}", :red)
  say("Error Details: #{errors_from_response(response)}", :yellow)
end
send_asset(asset, quiet=false) click to toggle source
# File lib/shopify_theme/cli.rb, line 219
def send_asset(asset, quiet=false)
  return unless valid?(asset)
  data = {:key => asset}
  content = File.read(asset)
  if binary_file?(asset) || ShopifyTheme.is_binary_data?(content)
    content = File.open(asset, "rb") { |io| io.read }
    data.merge!(:attachment => Base64.encode64(content))
  else
    data.merge!(:value => content)
  end

  response = show_during("[#{timestamp}] Uploading: #{asset}", quiet) do
    ShopifyTheme.send_asset(data)
  end
  if response.success?
    say("[#{timestamp}] Uploaded: #{asset}", :green) unless quiet
  else
    report_error(Time.now, "Could not upload #{asset}", response)
  end
end
show_during(message = '', quiet = false) { || ... } click to toggle source
# File lib/shopify_theme/cli.rb, line 292
def show_during(message = '', quiet = false, &block)
  print(message) unless quiet
  result = yield
  print("\r#{' ' * message.length}\r") unless quiet
  result
end
timestamp(time = Time.now) click to toggle source
# File lib/shopify_theme/cli.rb, line 299
def timestamp(time = Time.now)
  time.strftime(TIMEFORMAT)
end
valid?(key) click to toggle source
# File lib/shopify_theme/cli.rb, line 257
def valid?(key)
  return true if DEFAULT_WHITELIST.include?(key.split('/').first + "/")
  say("'#{key}' is not in a valid file for theme uploads", :yellow)
  say("Files need to be in one of the following subdirectories: #{DEFAULT_WHITELIST.join(' ')}", :yellow)
  false
end
watcher() { |filename, event| ... } click to toggle source
# File lib/shopify_theme/cli.rb, line 183
def watcher
  FileWatcher.new(Dir.pwd).watch() do |filename, event|
    yield(filename, event)
  end
end