# -*- coding: utf-8 -*-
require 'pathname'
require 'digest/md5'
require 'matrix'
require 'set'
module Slinky
# extensions of non-compiled files that can contain build directives
DIRECTIVE_FILES = %w{js css html}
DEPENDS_DIRECTIVE = /^[^\n\w]*(slinky_depends)\((".*"|'.+'|)\)[^\n\w]*$/
EXTERNAL_DEPENDS_DIRECTIVE = /^[^\n\w]*(slinky_depends_external)\((".*"|'.+'|)\)[^\n\w]*$/
REQUIRE_DIRECTIVE = /^[^\n\w]*(slinky_require)\((".*"|'.+'|)\)[^\n\w]*$/
SCRIPTS_DIRECTIVE = /^[^\n\w]*(slinky_scripts)[^\n\w]*$/
STYLES_DIRECTIVE = /^[^\n\w]*(slinky_styles)[^\n\w]*$/
PRODUCT_DIRECTIVE = /^[^\n\w]*(slinky_product)\((".*"|'.+'|)\)[^\n\w]*$/
BUILD_DIRECTIVES = Regexp.union(DEPENDS_DIRECTIVE,
EXTERNAL_DEPENDS_DIRECTIVE,
REQUIRE_DIRECTIVE,
SCRIPTS_DIRECTIVE,
STYLES_DIRECTIVE,
PRODUCT_DIRECTIVE)
CSS_URL_MATCHER = /url\(['"]?([^'"\/][^\s)]+\.[a-z]+)(\?\d+)?['"]?\)/
class Manifest
attr_accessor :manifest_dir, :dir, :config
def initialize dir, config, options = {}
@dir = dir
@build_to = if d = options[:build_to]
File.expand_path(d)
else
dir
end
@manifest_dir = ManifestDir.new dir, self, @build_to, self
@devel = (options[:devel].nil?) ? true : options[:devel]
@config = config
@no_minify = options[:no_minify] || config.dont_minify
end
# Returns a list of all files contained in this manifest
#
# @return [ManifestFile] a list of manifest files
def files include_ignores = true
unless @files
@files = []
files_rec @manifest_dir
end
if include_ignores
@files
else
@files.reject{|f| @config.ignore.any?{|p| f.in_tree? p}}
end
end
# Adds a file to the manifest, updating the dependency graph
def add_all_by_path paths
manifest_update paths do |path|
md = find_by_path(File.dirname(path)).first
mf = md.add_file(File.basename(path))
end
end
# Notifies of an update to a file in the manifest
def update_all_by_path paths
manifest_update paths
end
# Removes a file from the manifest
def remove_all_by_path paths
manifest_update paths do |path|
mf = find_by_path(path).first()
if mf
mf.parent.remove_file(mf)
end
end
end
# Finds the file at the given path in the manifest if one exists,
# otherwise nil.
#
# @param String path the path of the file relative to the manifest
#
# @return ManifestFile the manifest file at that path if one exists
def find_by_path path, allow_multiple = false
@manifest_dir.find_by_path path, allow_multiple
end
# Finds all files that match the given pattern. The match rules
# are similar to those for .gitignore and given by
#
# 1. If the pattern ends with a slash, it will only match directories;
# e.g. `foo/` would match a directory `foo/` but not a file `foo`. In
# a file context, matching a directory is equivalent to matching all
# files under that directory, recursively.
# 2. If the pattern does not contain a slash, slinky treats it as a
# relative pathname which can match files in any directory. For
# example, the rule `test.js` will matching `/test.js` and
# `/component/test.js`.
# 3. If the pattern begins with a slash, it will be treated as an
# absolute path starting at the root of the source directory.
# 4. If the pattern does not begin with a slash, but does contain one or
# more slashes, it will be treated as a path relative to any
# directory. For example, `test/*.js` will match `/test/main.js`, and
# /component/test/component.js`, but not `main.js`.
# 5. A single star `*` in a pattern will match any number of characters within a
# single path component. For example, `/test/*.js` will match
# `/test/main_test.js` but not `/test/component/test.js`.
# 6. A double star `**` will match any number of characters including
# path separators. For example `/scripts/**/main.js` will match any
# file named `main.js` under the `/scripts` directory, including
# `/scripts/main.js` and `/scripts/component/main.js`.
def find_by_pattern pattern
# The strategy here is to convert the pattern into an equivalent
# regex and run that against the pathnames of all the files in
# the manifest.
regex_str = Regexp.escape(pattern)
.gsub('\*\*/', ".*")
.gsub('\*\*', ".*")
.gsub('\*', "[^/]*")
if regex_str[0] != '/'
regex_str = '.*/' + regex_str
end
if regex_str[-1] == '/'
regex_str += '.*'
end
regex_str = "^#{regex_str}$"
regex = Regexp.new(regex_str)
files(false).reject{|f|
!regex.match('/' + f.relative_source_path.to_s) &&
!regex.match('/' + f.relative_output_path.to_s)
}
end
# Finds all the matching manifest files for a particular product.
# This does not take into account dependencies.
def files_for_product product
if !p = @config.produce[product]
SlinkyError.raise NoSuchProductError,
"Product '#{product}' has not been configured"
end
type = type_for_product product
if type != ".js" && type != ".css"
SlinkyError.raise InvalidConfigError, "Only .js and .css products are supported"
end
g = dependency_graph.transitive_closure
# Topological indices for each file
indices = {}
dependency_list.each_with_index{|f, i| indices[f] = i}
# Compute the set of excluded files
excludes = Set.new((p["exclude"] || []).map{|p|
find_by_pattern(p)
}.flatten.uniq)
SlinkyError.batch_errors do
# First find the list of files that have been explictly
# included/excluded
p["include"].map{|f|
mfs = find_by_pattern(f)
.map{|mf| [mf] + g[f]}
.flatten
.reject{|f| f.output_path.extname != type}
if mfs.empty?
SlinkyError.raise FileNotFoundError,
"No files matched by include #{f} in product #{product}"
end
mfs.flatten
}.flatten.reject{|f|
excludes.include?(f)
# Then add all the files these require
}.map{|f|
# Find all of the downstream files
# check that we're not excluding any required files
g[f].each{|rf|
if p["exclude"] && r = p["exclude"].find{|ex| rf.matches_path?(ex, true)}
SlinkyError.raise DependencyError,
"File #{f} requires #{rf} which is excluded by exclusion rule #{r}"
end
}
[f] + g[f]
}.flatten.uniq.sort_by{|f|
# Sort by topological order
indices[f]
}
end
end
def files_for_all_products
return @files_for_all_products if @files_for_all_products
SlinkyError.batch_errors do
@files_for_all_products = @config.produce.keys.map{|product|
files_for_product(product)
}.flatten.uniq
end
end
def compress_product product
compressor = compressor_for_product product
post_processor = post_processor_for_product product
s = files_for_product(product).map{|mf|
f = File.open(mf.build_to.to_s, 'rb'){|f| f.read}
post_processor ? (post_processor.call(mf, f)) : f
}.join("\n")
# Make the directory the product is in
FileUtils.mkdir_p("#{@build_to}/#{Pathname.new(product).dirname}")
File.open("#{@build_to}/#{product}", "w+"){|f|
unless @no_minify
f.write(compressor[s])
else
f.write(s)
end
}
end
# These are special cases for simplicity and backwards
# compatability. If no products are defined, we have two default
# products, one which includes are .js files in the repo and one
# that includes all .css files. This method produces an HTML include
# string for all of the .js files.
def scripts_string
product_string ConfigReader::DEFAULT_SCRIPT_PRODUCT
end
# These are special cases for simplicity and backwards
# compatability. If no products are defined, we have two default
# products, one which includes are .js files in the repo and one
# that includes all .css files. This method produces an HTML include
# string for all of the .css files.
def styles_string
product_string ConfigReader::DEFAULT_STYLE_PRODUCT
end
# Produces a string of HTML that includes all of the files for the
# given product.
def product_string product
if @devel
files_for_product(product).map{|f|
html_for_path("/#{f.relative_output_path}")
}.join("\n")
else
html_for_path("#{product}?#{rand(999999999)}")
end
end
# Builds the directed graph representing the dependencies of all
# files in the manifest that contain a slinky_require
# declaration. The graph is represented as a list of pairs
# (required, by), each of which describes an edge.
#
# @return [[ManifestFile, ManifestFile]] the graph
def dependency_graph
return @dependency_graph if @dependency_graph
graph = []
files(false).each{|mf|
mf.dependencies.each{|d|
graph << [d, mf]
}
}
@dependency_graph = Graph.new(files(false), graph)
end
def dependency_list
dependency_graph.dependency_list
end
def build
@manifest_dir.build
unless @devel
@config.produce.keys.each{|product|
compress_product(product)
}
# clean up the files that have been processed
files_for_all_products.each{|mf| FileUtils.rm(mf.build_to, :force => true)}
end
end
# Returns a md5 encompassing the current state of the manifest.
# Any change to the manifest should produce a different hash.
# This can be used to determine if the manifest has changed.
def md5
if @md5
@md5
else
@md5 = Digest::MD5.hexdigest(files.map{|f| [f.source, f.md5]}
.sort.flatten.join(":"))
end
end
private
def files_rec md
@files += md.files
md.children.each do |c|
files_rec c
end
end
def compressor_for_product product
case type_for_product(product)
when ".js"
# Use UglifyJS
lambda{|s| Uglifier.compile(s.force_encoding("UTF-8"),
mangle: false, output: {ascii_only: false})}
when ".css"
# Use SASS's compressed output
lambda{|s| Sass::Engine.new(s, :syntax => :scss, :style => :compressed).render}
end
end
def post_processor_for_product product
case type_for_product(product)
when ".css"
lambda{|s, css| css.gsub(CSS_URL_MATCHER){|url|
p = s.relative_output_path.dirname.to_s + "/#{$1}"
"url('/#{p}')"
}}
end
end
def invalidate_cache
@files = nil
@dependency_graph = nil
@md5 = nil
@files_for_all_products = nil
end
def html_for_path path
ext = path.split("?").first.split(".").last
case ext
when "css"
%Q||
when "js"
%Q||
else
raise InvalidConfigError.new("Unsupported file extension #{ext}")
end
end
def type_for_product product
"." + product.split(".")[-1]
end
def manifest_update paths
paths.each{|path|
if path[0] == '/'
path = Pathname.new(path).relative_path_from(Pathname.new(@dir).expand_path).to_s
end
yield path if block_given?
}
invalidate_cache
files.each{|f|
if f.directives.include?(:slinky_scripts) ||
f.directives.include?(:slinky_styles) ||
f.directives.include?(:slinky_product)
f.invalidate
f.find_directives
end
}
end
end
class ManifestDir
attr_accessor :dir, :parent, :files, :children
def initialize dir, parent, build_dir, manifest
@dir = dir
@parent = parent
@files = []
@children = []
@build_dir = Pathname.new(build_dir)
@manifest = manifest
Dir.glob("#{dir}/*").each do |path|
# skip the build dir
next if Pathname.new(File.expand_path(path)) == Pathname.new(build_dir)
if File.directory? path
add_child(path)
else
add_file(path)
end
end
end
# Finds the file at the given path in the directory if one exists,
# otherwise nil.
#
# @param String path the path of the file relative to the directory
# @param Boolean allow_multiple if enabled, can return multiple paths
# according to glob rules
#
# @return [ManifestFile] the manifest file at that path if one exists
def find_by_path path, allow_multiple = false
if path[0] == '/'
# refer absolute paths to the manifest
return @manifest.find_by_path(path[1..-1], allow_multiple)
end
components = path.to_s.split(File::SEPARATOR).reject{|x| x == ""}
case components.size
when 0
[self]
when 1
path = [@dir, components[0]].join(File::SEPARATOR)
if (File.directory?(path) rescue false)
c = @children.find{|d|
Pathname.new(d.dir).cleanpath == Pathname.new(path).cleanpath
}
unless c
c = add_child(path)
end
[c]
else
@files.find_all{|f| f.matches? components[0], allow_multiple}
end
else
if components[0] == ".."
@parent.find_by_path components[1..-1].join(File::SEPARATOR)
else
child = @children.find{|d|
Pathname.new(d.dir).basename.to_s == components[0]
}
if child
child.find_by_path(components[1..-1].join(File::SEPARATOR),
allow_multiple)
else
[]
end
end
end
end
# Adds a child directory
def add_child path
if File.directory? path
build_dir = (@build_dir + File.basename(path)).cleanpath
md = ManifestDir.new(path, self, build_dir, @manifest)
@children << md
md
end
end
# Adds a file on the filesystem to the manifest
#
# @param String path The path of the file
def add_file path
file = File.basename(path)
full_path = Pathname.new(@dir).join(file).to_s
if File.exists?(full_path) && !file.start_with?(".")
mf = ManifestFile.new(full_path, @build_dir, @manifest, self)
# we don't want two files with the same source
extant_file = @files.find{|f| f.source == mf.source}
if extant_file
@files.delete(extant_file)
end
@files << mf
mf
end
end
# Removes a file from the manifest
#
# @param ManifestFile mf The file to be deleted
def remove_file mf
@files.delete(mf)
end
def build
unless File.directory?(@build_dir.to_s)
FileUtils.mkdir(@build_dir.to_s)
end
if (@files + @children).map {|m| m.build}.any?
@build_dir
else
FileUtils.rmdir(@build_dir.to_s)
nil
end
end
def to_s
""
end
end
class ManifestFile
attr_accessor :source, :build_path
attr_reader :last_built, :directives, :parent, :manifest, :updated
def initialize source, build_path, manifest, parent = nil, options = {:devel => false}
@parent = parent
@source = Pathname.new(source).cleanpath.to_s
@last_built = Time.at(0)
@cfile = Compilers.cfile_for_file(@source)
@directives = find_directives
@build_path = build_path
@manifest = manifest
@devel = true if options[:devel]
end
def invalidate
@last_built = Time.at(0)
@last_md5 = nil
end
# Gets the list of manifest files that this one depends on
# according to its directive list and the dependencies config
# option.
#
# Throws a FileNotFoundError if a dependency doesn't exist in the
# tree.
def dependencies
SlinkyError.batch_errors do
(@directives[:slinky_require].to_a +
@manifest.config.dependencies["/" + relative_source_path.to_s].to_a).map{|rf|
required = parent.find_by_path(rf, true).flatten
if required.empty?
error = "Could not find file #{rf} required by /#{relative_source_path}"
SlinkyError.raise FileNotFoundError, error
end
required
}.flatten
end
end
# Predicate which determines whether the supplied name is the same
# as the file's name, taking into account compiled file
# extensions. For example, if mf refers to "/tmp/test/hello.sass",
# `mf.matches? "hello.sass"` and `mf.matches? "hello.css"` should
# both return true.
#
# @param String a filename
# @param Bool match_glob if true, matches according to glob rules
# @return Bool True if the filename matches, false otherwise
def matches? s, match_glob = false
name = Pathname.new(@source).basename.to_s
output = output_path.basename.to_s
# check for stars that are not escaped
a = [""]
last = ""
s.each_char {|c|
if c != "*" || last == "\\"
a[-1] << c
else
a << ""
end
last = c
}
if match_glob && a.size > 1
r2 = a.reduce{|a, x| /#{a}.*#{x}/}
name.match(r2) || output.match(r2)
else
name == s || output == s
end
end
# Predicate which determines whether the file matches (see
# `ManifestFile#matches?`) the full path relative to the manifest
# root.
def matches_path? s, match_glob = false
p = Pathname.new(s)
dir = Pathname.new("/" + relative_source_path.to_s).dirname
matches?(p.basename.to_s, match_glob) &&
dir == p.dirname
end
# Predicate which determines whether the file is the supplied path
# or lies on supplied tree
def in_tree? path
full_path = @manifest.dir + "/" + path
abs_path = Pathname.new(full_path).expand_path.to_s
dir = Pathname.new(@source).dirname.expand_path.to_s
dir.start_with?(abs_path) || abs_path == @source
end
# Returns the path to which this file should be output. This is
# equal to the source path unless the file needs to be compiled,
# in which case the extension returned is the output extension
#
# @return Pathname the output path
def output_path
if @cfile
ext = /\.[^.]*$/
Pathname.new(@source.gsub(ext, ".#{@cfile.output_ext}"))
else
Pathname.new(@source)
end
end
# returns the source path relative to the manifest directory
def relative_source_path
Pathname.new(@source).relative_path_from(Pathname.new(@manifest.dir))
end
# Returns the output path relative to the manifest directory
def relative_output_path
output_path.relative_path_from(Pathname.new(@manifest.dir))
end
# Looks through the file for directives
# @return {Symbol => [String]} the directives in the file
def find_directives
_, _, ext = @source.match(EXTENSION_REGEX).to_a
directives = {}
# check if this file might include directives
if @cfile || DIRECTIVE_FILES.include?(ext)
# make sure the file isn't too big to scan
stat = File::Stat.new(@source)
if stat.size < 1024*1024
File.open(@source) {|f|
matches = f.read.scan(BUILD_DIRECTIVES).to_a
matches.each{|slice|
key, value = slice.compact
directives[key.to_sym] ||= []
directives[key.to_sym] << value[1..-2] if value
}
} rescue nil
end
end
@directives = directives
end
# If there are any build directives for this file, the file is
# read and the directives are handled appropriately and a new file
# is written to a temp location.
#
# @return String the path of the de-directivefied file
def handle_directives path, to = nil
if path && @directives.size > 0
out = File.read(path)
out.gsub!(DEPENDS_DIRECTIVE, "")
out.gsub!(EXTERNAL_DEPENDS_DIRECTIVE, "")
out.gsub!(REQUIRE_DIRECTIVE, "")
out.gsub!(SCRIPTS_DIRECTIVE){ @manifest.scripts_string }
out.gsub!(STYLES_DIRECTIVE){ @manifest.styles_string }
out.gsub!(PRODUCT_DIRECTIVE){
@manifest.product_string($2[1..-2])
}
to = to || Tempfile.new("slinky").path + ".cache"
File.open(to, "w+"){|f|
f.write(out)
}
to
else
path
end
end
# Takes a path and compiles the file if necessary.
# @return Pathname the path of the compiled file, or the original
# path if compiling is not necessary
def compile path, to = nil
if @cfile
cfile = @cfile.clone
cfile.source = path
cfile.print_name = @source
cfile.output_path = to if to
cfile.file do |cpath, _, _, _|
path = cpath
end
end
path ? Pathname.new(path) : nil
end
# Gets the md5 hash of the source file
def md5
Digest::MD5.hexdigest(File.read(@source)) rescue nil
end
# The list of paths to files external to the manifest that this file
# depends on
def external_dependencies
(@directives[:slinky_depends_external] || []).map{|ed|
Dir.glob(File.join(@manifest.dir, ed))
}.flatten
end
def external_dependencies_updated?
return false if external_dependencies.empty?
external_dependencies.map{|x| File.mtime(x)}.max > (@updated || Time.at(0))
end
# Gets manifest file ready for serving or building by handling the
# directives and compiling the file if neccesary.
# @param String path to which the file should be compiled
#
# @return String the path of the processed file, ready for serving
def process to = nil, should_compile = true
return if @processing # prevent infinite recursion
start_time = Time.now
hash = md5
if hash != @last_md5
find_directives
end
SlinkyError.batch_errors do
depends = @directives[:slinky_depends].map{|f|
ps = if f.start_with?("/")
@manifest.find_by_pattern(f)
else
parent.find_by_path(f, true)
end
unless ps.size > 0
SlinkyError.raise DependencyError,
"File #{f} depended on by #{@source} not found"
end
ps
}.flatten.compact if @directives[:slinky_depends]
depends ||= []
@processing = true
# process each file on which we're dependent, watching out for
# infinite loops
depends.each{|f| f.process }
@processing = false
# get hash of source file
if @last_path && hash == @last_md5 &&
depends.all?{|f| f.updated < start_time} &&
!external_dependencies_updated?
@last_path
else
@last_md5 = hash
@updated = Time.now
# mangle file appropriately
f = should_compile ? (compile @source) : @source
@last_path = handle_directives(f, to)
end
end
end
# Path to which the file will be built
def build_to
Pathname.new(@build_path) + output_path.basename
end
# Builds the file by handling and compiling it and then copying it
# to the build path
def build
return nil unless should_build
if !File.exists? @build_path
FileUtils.mkdir_p(@build_path)
end
to = build_to
path = process to
if path != to
FileUtils.cp(path.to_s, to.to_s)
@last_built = Time.now
end
to
end
def should_build
@manifest.files_for_all_products.include?(self) || ![".js", ".css"].include?(output_path.extname)
end
def inspect
to_s
end
def to_s
""
end
end
end