Как использовать Sprockets с Sinatra без файлового архива?

Я пишу библиотеку с встроенным приложением Sinatra, запущенным через Thor. Я хочу установить экземпляры Sprockets::Environment в /css и /js и отобразить главное приложение на /. Это было бы легко использовать Rack::URLMap в файле config.ru, но в этом случае его нет, потому что я запускаю приложение Sinatra программно с помощью Sinatra::Application.run!. Как я могу достичь этого?

Ответ 1

Я закончил это, написав специальное промежуточное программное обеспечение с некоторыми функциями из Rack::URLMap. Это выглядит примерно так:

require "sprockets"
require "sinatra/base"

class SprocketsMiddleware
  attr_reader :app, :prefix, :sprockets

  def initialize(app, prefix)
    @app = app
    @prefix = prefix
    @sprockets = Sprockets::Environment.new

    yield sprockets if block_given?
  end

  def call(env)
    path_info = env["PATH_INFO"]
    if path_info =~ prefix
      env["PATH_INFO"].sub!(prefix, "")
      sprockets.call(env)
    else
      app.call(env)
    end
  ensure
    env["PATH_INFO"] = path_info
  end
end

class App < Sinatra::Base
  use SprocketsMiddleware, %r{/assets} do |env|
    env.append_path "assets/css"
    env.append_path "assets/js"
  end
end

App.run!

Ответ 2

Собственно, это не так сложно. Все, что вам нужно сделать, это присвоить экземпляр Sprockets::Environment переменной конфигурации Sinatra и определить некоторые пути для поиска интересующих вас активов.

Вот базовый пример:

require "sass"
require "haml"
require "erubis"
require "sinatra"
require "sprockets"

set :assets, Sprockets::Environment.new

# Configure sprockets
settings.assets.append_path "app/javascripts"
settings.assets.append_path "app/stylesheets"

# For compressed JS and CSS output
require "yui/compressor"
settings.assets.js_compressor  = YUI::JavaScriptCompressor.new
settings.assets.css_compressor = YUI::CssCompressor.new

get "/" do
  haml :index
end

get "/javascripts/:file.js" do
  content_type "application/javascript"
  settings.assets["#{params[:file]}.js"]
end

get "/stylesheets/:file.css" do
  content_type "text/css"
  settings.assets["#{params[:file]}.css"]
end

Счастливые дробилки!

Ответ 3

Вот как я интегрировал Sprockets в Sinatra с Rails-подобным макетом каталогов, помощниками и минимизацией для JS и CSS.

Я решил написать расширение Sinatra. Это расширение инкапсулирует конфигурацию звездочек (пути, минимизацию, помощники) и может быть зарегистрировано приложением.

module Sinatra
  module Assets
    extend Sinatra::Extension

    configure do
      set :assets, Sprockets::Environment.new(root).tap { |assets|
        %w(assets vendor/assets).each do |base|
          %w(images javascripts stylesheets).each do |type|
            assets.append_path File.join(base, type)
          end
        end
        if production?
          assets.js_compressor = Closure::Compiler.new
          assets.css_compressor = YUI::CssCompressor.new
          uid = Digest::MD5.hexdigest(File.dirname(__FILE__))[0,8]
          assets.cache = Sprockets::Cache::FileStore.new("/tmp/sinatra-#{uid}")
        else
          assets.cache = nil
        end
      }
    end

    get "/assets/*" do
      env["PATH_INFO"].sub!(%r{^/assets}, "")
      expires Time.now + (365*24*60*60) if settings.production?
      settings.assets.call(env)
    end

    helpers do
      include Sprockets::Helpers

      Sprockets::Helpers.configure do |config|
        config.expand = development?
        config.digest = production?
      end

      def assets_environment
        settings.assets
      end
    end
  end
end

Использование расширения в вашем приложении прост:

class App < Sinatra::Base
  register Sinatra::Assets
  # ...
end

Активы можно поместить в assets или vendor/assets. Например, vendor/assets/jquery.js может ссылаться на логическое имя, т.е. http://localhost/assets/jquery.js.

В приведенном выше примере я использую sprockets-helpers, который предоставляет помощники, такие как javascript_tag. В приведенной выше конфигурации предполагается, что в процессе разработки вы хотите расширить активы, требуемые ссылочным активом (в результате получается несколько тегов для каждого актива).