module Gumdrop WEB_PAGE_EXTS= %w(.html .htm) JETSAM_FILES= %w(**/.DS_Store .git* .git/**/* .svn/**/* **/.sass-cache/**/* Gumdrop) DEFAULT_CONFIG= { relative_paths: true, relative_paths_exts: WEB_PAGE_EXTS, default_layout: 'site', layout_exts: WEB_PAGE_EXTS, proxy_enabled: false, output_dir: "./output", source_dir: "./source", data_dir: './data', log: STDOUT, log_level: :info, ignore: JETSAM_FILES, blacklist: [], server_timeout: 5, server_port: 4567, env: :production, file_change_test: :checksum } class Site include Util::Configurable include Util::Eventable include Util::Loggable config_accessor :source_dir, :output_dir, :data_dir, :mode, :env, :ignore, :blacklist attr_reader :sitefile, :options, :root, :contents, :data, :layouts, :generators, :partials, :last_run # hmmm... attr_accessor :active_renderer, :active_builder # You shouldn't call this yourself! Access it via Gumdrop.site def initialize(sitefile, opts={}) Gumdrop.send :set_current_site, self @sitefile= sitefile.expand_path @options= Util::HashObject.from opts _options_updated! @root= File.dirname @sitefile @last_run= 0 @_preparations= [] @contents= ContentList.new @layouts= SpecialContentList.new ".layout" @partials= SpecialContentList.new @generators= [] @data= DataManager.new clear end def options=(opts={}) @options.merge!(opts) _options_updated! end def clear() @contents.clear() @layouts.clear() @partials.clear() @generators.clear() @data.clear() # @contents= ContentList.new # @layouts= SpecialContentList.new ".layout" # @partials= SpecialContentList.new # @generators= [] # @data= DataManager.new @is_scanned= false _reset_config! _load_sitefile self end def scan(force=false) if !@is_scanned or force clear if @is_scanned # ???? _content_scanner @is_scanned= true generate end self end def scan_only # For testing... if !@is_scanned or force clear if @is_scanned # ???? _content_scanner end self end def generate _execute_preparations _execute_generators self end def in_blacklist?(path) blacklist.any? do |pattern| path.path_match? pattern end end def ignore_path?(path) config.ignore.any? do |pattern| path.path_match? pattern end end def source_path @source_path ||= source_dir.expand_path(root) end def output_path @output_path ||= output_dir.expand_path(root) end def data_path @data_path ||= data_dir.expand_path(root) end def resolve(path=nil, opts={}) case when path.is_a?(Content) path when !path.nil? contents.first(path) || partials.first(path) when opts[:page] contents.first opts[:page] when opts[:partial] partials.first opts[:partial] when opts[:layout] layouts.first opts[:layout] when opts[:generator] generators.first opts[:generator] else nil end end def prepare(&block) @_preparations << block end # Events stop bubbling here. def parent nil end def config_did_change Gumdrop.init_logging end private def _reset_config! config.clear.merge! DEFAULT_CONFIG config.env= @options.env.to_sym if @options.env config.mode= @options.mode.nil? ? :unknown : @options.mode.to_sym end def _options_updated! config.env= @options.env.to_sym if @options.env config.mode= @options.mode.nil? ? :unknown : @options.mode.to_sym end def _load_sitefile clear_events load sitefile data.dir= data_path rescue Exception => ex msg= "There is an error in your Gumdrop file!" # $stderr.puts msg log.error msg log.error ex raise ex end def _content_scanner log.info "Gumdrop v#{ Gumdrop::VERSION } - #{ Time.new }" log.debug "(config)" log.debug config log.debug "(options)" log.debug @options log.debug "[Scanning: #{source_path}]" # Report ignore list ignore.each {|p| log.debug " ignoring: #{p}" } # Scan Filesystem event_block :scan do scanner= Util::Scanner.new(source_path, {}, &method(:_scanner_validator)) scanner.each do |path, rel| content= Content.new(path) layouts.add content and next if content.layout? partials.add content and next if content.partial? generators << Generator.new(content) and next if content.generator? contents.add content log.debug " including: #{ rel }" end contents.keys.size end @is_scanned= true end def _scanner_validator(source_path, full_path) return true if ignore_path? source_path # in_blacklist? source_path false end def _execute_preparations log.debug "[Executing Preparations]" dsl= Generator::DSL.new nil @_preparations.each do |block| if block.arity == 1 block.call dsl else dsl.instance_eval &block end end end def _execute_generators log.debug "[Executing Generators]" event_block :generate do generators.each do |generator| generator.execute() end end end end class << self # This will look for a nearby Gumdrop file and load the Site # the first time it is called. Each time after it will just # return the already created Site. def site(opts={}, force_new=false) opts= opts.to_symbolized_hash unless @current_site.nil? or force_new @current_site.options= opts unless opts.empty? @current_site else site_file= fetch_site_file unless site_file.nil? Site.new site_file, opts else nil end end end # Listen for life-cycle events in your Gumdrop file like this: # # Gumdrop.on :start do |event| # puts "Here we go!" # end # # Complete list of events fired by Gumdrop: # # :start # :before_scan # :scan # :after_scan # :before_generate # :generate # :after_generate # :before_generate_item # :generate_item # :after_generate_item # :before_build # :build # :after_build # :before_checksum # :checksum # :after_checksum # :before_render # :render # :after_render # :before_render_item # :render_item # :after_render_item # :before_write # :write # :after_write # :before_copy_file # :copy_file # :after_copy_file # :before_write_file # :write_file # :after_write_file # :end # # In the block parem `event` you will have access to :site, the # current executing site instance, and :payload, where applicable. # # :render_item is a special event because you can add/override the # compiled output by modifying `event.data.return_value` # def on(event_type, options={}, &block) site.on event_type, options, &block end # Yield a configuration object. You can update Gumdrop behavior # as well as add your own config information. def configure(&block) site.configure &block end def prepare(&block) site.prepare &block end # Short cut to the current Site config object. def config site.config end # The env Gumdrop is run in -- You can set this in the config # or, better, via command line: `gumdrop build -e test` def env site.env end # Mostly for internal use, but you can update/change it in the # configure block. def mode site.mode end # Returns true if this, or a parent, folder has a Gumdrop file. def in_site_folder?(filename="Gumdrop") !fetch_site_file(filename).nil? end # Specified paths will be added to the ignore list, preventing # matching files from appearing in the source tree def ignore(*paths) paths.each do |path| if path.is_a? Array config.ignore.concat path else config.ignore << path end end end # Specified paths will not be renderd to output (matching against # the source tree). def blacklist(*paths) paths.each do |path| if path.is_a? Array config.blacklist.concat path else config.blacklist << path end end end protected # Walks up the filesystem, starting from Dir.pwd, looking for # a Gumdrop file. Returns nil if not found. def fetch_site_file(filename="Gumdrop") here= Dir.pwd found= File.file? here / filename # TODO: Should be smarter -- This is a hack for Windows support "C:\" while !found and File.directory?(here) and File.dirname(here).length > 3 here= File.expand_path here /'..' found= File.file? here / filename end if found File.expand_path here / filename else nil end end def set_current_site(site) @current_site= site end end end