$LOAD_PATH.push File.expand_path(File.dirname(__FILE__)) # @Jammit@ is the central namespace for all Jammit classes, and provides access # to all of the configuration options. module Jammit VERSION = "0.4.3.1" ROOT = File.expand_path(File.dirname(__FILE__) + '/..') ASSET_ROOT = File.expand_path(defined?(Rails) ? Rails.root : ".") unless defined?(ASSET_ROOT) PUBLIC_ROOT = File.join(ASSET_ROOT, 'public') DEFAULT_CONFIG_PATH = File.join(ASSET_ROOT, 'config', 'assets.yml') DEFAULT_PACKAGE_PATH = "assets" DEFAULT_JST_SCRIPT = File.join(ROOT, 'lib/jammit/jst.js') DEFAULT_JST_COMPILER = "template" DEFAULT_JST_NAMESPACE = "window.JST" AVAILABLE_COMPRESSORS = [:yui, :closure] DEFAULT_COMPRESSOR = :yui # Jammit raises a @PackageNotFound@ exception when a non-existent package is # requested by a browser -- rendering a 404. class PackageNotFound < NameError; end # Jammit raises a ConfigurationNotFound exception when you try to load the # configuration of an assets.yml file that doesn't exist. class ConfigurationNotFound < NameError; end # Jammit raises an OutputNotWritable exception if the output directory for # cached packages is locked. class OutputNotWritable < StandardError; end class << self attr_reader :configuration, :template_function, :template_namespace, :embed_assets, :package_assets, :compress_assets, :gzip_assets, :package_path, :mhtml_enabled, :include_jst_script, :javascript_compressor, :compressor_options, :css_compressor_options end # The minimal required configuration. @configuration = {} @package_path = DEFAULT_PACKAGE_PATH # Load the complete asset configuration from the specified @config_path@. def self.load_configuration(config_path) exists = config_path && File.exists?(config_path) raise ConfigurationNotFound, "could not find the \"#{config_path}\" configuration file" unless exists conf = YAML.load(ERB.new(File.read(config_path)).result) @config_path = config_path @configuration = conf = conf.symbolize_keys @package_path = conf[:package_path] || DEFAULT_PACKAGE_PATH @embed_assets = conf[:embed_assets] || conf[:embed_images] @compress_assets = !(conf[:compress_assets] == false) @gzip_assets = !(conf[:gzip_assets] == false) @mhtml_enabled = @embed_assets && @embed_assets != "datauri" @compressor_options = (conf[:compressor_options] || {}).symbolize_keys @css_compressor_options = (conf[:css_compressor_options] || {}).symbolize_keys set_javascript_compressor(conf[:javascript_compressor]) set_package_assets(conf[:package_assets]) set_template_function(conf[:template_function]) set_template_namespace(conf[:template_namespace]) check_java_version self end # Force a reload by resetting the Packager and reloading the configuration. # In development, this will be called as a before_filter before every request. def self.reload! Thread.current[:jammit_packager] = nil load_configuration(@config_path) end # Keep a global (thread-local) reference to a @Jammit::Packager@, to avoid # recomputing asset lists unnecessarily. def self.packager Thread.current[:jammit_packager] ||= Packager.new end # Generate the base filename for a version of a given package. def self.filename(package, extension, suffix=nil) suffix_part = suffix ? "-#{suffix}" : '' "#{package}#{suffix_part}.#{extension}" end # Generates the server-absolute URL to an asset package. def self.asset_url(package, extension, suffix=nil, mtime=nil) timestamp = mtime ? "?#{mtime.to_i}" : '' "/#{package_path}/#{filename(package, extension, suffix)}#{timestamp}" end private # Ensure that the JavaScript compressor is a valid choice. def self.set_javascript_compressor(value) value = value && value.to_sym @javascript_compressor = AVAILABLE_COMPRESSORS.include?(value) ? value : DEFAULT_COMPRESSOR end # Turn asset packaging on or off, depending on configuration and environment. def self.set_package_assets(value) package_env = !defined?(Rails) || !Rails.env.development? @package_assets = value == true || value.nil? ? package_env : value == 'always' ? true : false end # Assign the JST template function, unless explicitly turned off. def self.set_template_function(value) @template_function = value == true || value.nil? ? DEFAULT_JST_COMPILER : value == false ? '' : value @include_jst_script = @template_function == DEFAULT_JST_COMPILER end # Set the root JS object in which to stash all compiled JST. def self.set_template_namespace(value) @template_namespace = value == true || value.nil? ? DEFAULT_JST_NAMESPACE : value.to_s end # The YUI Compressor requires Java > 1.4, and Closure requires Java > 1.6. def self.check_java_version java = @compressor_options[:java] || 'java' @css_compressor_options[:java] ||= java if @compressor_options[:java] version = (`#{java} -version 2>&1`)[/\d+\.\d+/] disable_compression if !version || (@javascript_compressor == :closure && version < '1.6') || (@javascript_compressor == :yui && version < '1.4') end # If we don't have a working Java VM, then disable asset compression and # complain loudly. def self.disable_compression @compress_assets = false warn("Asset compression disabled -- Java unavailable.") end def self.warn(message) message = "Jammit Warning: #{message}" @logger ||= (defined?(Rails) && Rails.logger ? Rails.logger : defined?(RAILS_DEFAULT_LOGGER) ? RAILS_DEFAULT_LOGGER : nil) @logger ? @logger.warn(message) : STDERR.puts(message) end end require 'jammit/dependencies'