jamesgolick-webby 0.9.5
Sign up to get free protection for your applications and to get access to all the features.
- data/History.txt +219 -0
- data/README.rdoc +92 -0
- data/Rakefile +62 -0
- data/bin/webby +41 -0
- data/bin/webby-gen +41 -0
- data/examples/blog/Sitefile +7 -0
- data/examples/blog/tasks/blog.rake +72 -0
- data/examples/blog/templates/atom_feed.erb +40 -0
- data/examples/blog/templates/blog/month.erb +22 -0
- data/examples/blog/templates/blog/post.erb +16 -0
- data/examples/blog/templates/blog/year.erb +22 -0
- data/examples/presentation/Sitefile +10 -0
- data/examples/presentation/content/css/uv/twilight.css +137 -0
- data/examples/presentation/content/presentation/_sample_code.txt +10 -0
- data/examples/presentation/content/presentation/index.txt +63 -0
- data/examples/presentation/content/presentation/s5/blank.gif +0 -0
- data/examples/presentation/content/presentation/s5/bodybg.gif +0 -0
- data/examples/presentation/content/presentation/s5/framing.css +23 -0
- data/examples/presentation/content/presentation/s5/iepngfix.htc +42 -0
- data/examples/presentation/content/presentation/s5/opera.css +7 -0
- data/examples/presentation/content/presentation/s5/outline.css +15 -0
- data/examples/presentation/content/presentation/s5/pretty.css +86 -0
- data/examples/presentation/content/presentation/s5/print.css +25 -0
- data/examples/presentation/content/presentation/s5/s5-core.css +9 -0
- data/examples/presentation/content/presentation/s5/slides.css +3 -0
- data/examples/presentation/content/presentation/s5/slides.js +553 -0
- data/examples/presentation/layouts/presentation.txt +43 -0
- data/examples/presentation/templates/_code_partial.erb +13 -0
- data/examples/presentation/templates/presentation.erb +40 -0
- data/examples/tumblog/Sitefile +9 -0
- data/examples/tumblog/content/css/tumblog.css +308 -0
- data/examples/tumblog/content/images/tumblog/permalink.gif +0 -0
- data/examples/tumblog/content/images/tumblog/rss.gif +0 -0
- data/examples/tumblog/content/tumblog/200806/the-noble-chicken/index.txt +12 -0
- data/examples/tumblog/content/tumblog/200807/historical-perspectives-on-the-classic-chicken-joke/index.txt +12 -0
- data/examples/tumblog/content/tumblog/200807/mad-city-chickens/index.txt +10 -0
- data/examples/tumblog/content/tumblog/200807/the-wisdom-of-the-dutch/index.txt +11 -0
- data/examples/tumblog/content/tumblog/200807/up-a-tree/index.txt +13 -0
- data/examples/tumblog/content/tumblog/index.txt +37 -0
- data/examples/tumblog/content/tumblog/rss.txt +37 -0
- data/examples/tumblog/layouts/tumblog/default.txt +44 -0
- data/examples/tumblog/layouts/tumblog/post.txt +15 -0
- data/examples/tumblog/lib/tumblog_helper.rb +32 -0
- data/examples/tumblog/tasks/tumblog.rake +30 -0
- data/examples/tumblog/templates/atom_feed.erb +40 -0
- data/examples/tumblog/templates/tumblog/conversation.erb +12 -0
- data/examples/tumblog/templates/tumblog/link.erb +10 -0
- data/examples/tumblog/templates/tumblog/photo.erb +13 -0
- data/examples/tumblog/templates/tumblog/post.erb +12 -0
- data/examples/tumblog/templates/tumblog/quote.erb +11 -0
- data/examples/webby/Sitefile +19 -0
- data/examples/webby/content/communicate/index.txt +28 -0
- data/examples/webby/content/css/background.gif +0 -0
- data/examples/webby/content/css/blueprint/print.css +76 -0
- data/examples/webby/content/css/blueprint/screen.css +696 -0
- data/examples/webby/content/css/coderay.css +96 -0
- data/examples/webby/content/css/site.css +196 -0
- data/examples/webby/content/css/uv/twilight.css +137 -0
- data/examples/webby/content/index.txt +37 -0
- data/examples/webby/content/learn/index.txt +28 -0
- data/examples/webby/content/reference/index.txt +204 -0
- data/examples/webby/content/release-notes/index.txt +21 -0
- data/examples/webby/content/release-notes/rel-0-9-0/index.txt +74 -0
- data/examples/webby/content/release-notes/rel-0-9-1/index.txt +93 -0
- data/examples/webby/content/release-notes/rel-0-9-2/index.txt +14 -0
- data/examples/webby/content/release-notes/rel-0-9-3/index.txt +47 -0
- data/examples/webby/content/robots.txt +6 -0
- data/examples/webby/content/script/jquery.corner.js +152 -0
- data/examples/webby/content/script/jquery.js +31 -0
- data/examples/webby/content/sitemap.txt +31 -0
- data/examples/webby/content/tips_and_tricks/index.txt +97 -0
- data/examples/webby/content/tutorial/index.txt +135 -0
- data/examples/webby/content/user-manual/index.txt +419 -0
- data/examples/webby/layouts/default.txt +49 -0
- data/examples/webby/templates/page.erb +10 -0
- data/examples/website/Sitefile +7 -0
- data/examples/website/content/css/blueprint/ie.css +26 -0
- data/examples/website/content/css/blueprint/plugins/buttons/icons/cross.png +0 -0
- data/examples/website/content/css/blueprint/plugins/buttons/icons/key.png +0 -0
- data/examples/website/content/css/blueprint/plugins/buttons/icons/tick.png +0 -0
- data/examples/website/content/css/blueprint/plugins/buttons/readme.txt +32 -0
- data/examples/website/content/css/blueprint/plugins/buttons/screen.css +97 -0
- data/examples/website/content/css/blueprint/plugins/fancy-type/readme.txt +14 -0
- data/examples/website/content/css/blueprint/plugins/fancy-type/screen.css +71 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/doc.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/email.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/external.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/feed.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/im.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/pdf.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/visited.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/icons/xls.png +0 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/readme.txt +18 -0
- data/examples/website/content/css/blueprint/plugins/link-icons/screen.css +40 -0
- data/examples/website/content/css/blueprint/plugins/rtl/readme.txt +10 -0
- data/examples/website/content/css/blueprint/plugins/rtl/screen.css +109 -0
- data/examples/website/content/css/blueprint/print.css +30 -0
- data/examples/website/content/css/blueprint/screen.css +251 -0
- data/examples/website/content/css/blueprint/src/forms.css +49 -0
- data/examples/website/content/css/blueprint/src/grid.css +212 -0
- data/examples/website/content/css/blueprint/src/grid.png +0 -0
- data/examples/website/content/css/blueprint/src/ie.css +59 -0
- data/examples/website/content/css/blueprint/src/print.css +85 -0
- data/examples/website/content/css/blueprint/src/reset.css +38 -0
- data/examples/website/content/css/blueprint/src/typography.css +105 -0
- data/examples/website/content/css/coderay.css +111 -0
- data/examples/website/content/css/site.css +67 -0
- data/examples/website/content/index.txt +19 -0
- data/examples/website/layouts/default.txt +61 -0
- data/examples/website/lib/breadcrumbs.rb +28 -0
- data/examples/website/templates/_partial.erb +10 -0
- data/examples/website/templates/page.erb +18 -0
- data/lib/webby.rb +233 -0
- data/lib/webby/apps.rb +12 -0
- data/lib/webby/apps/generator.rb +276 -0
- data/lib/webby/apps/main.rb +258 -0
- data/lib/webby/auto_builder.rb +157 -0
- data/lib/webby/builder.rb +172 -0
- data/lib/webby/core_ext/enumerable.rb +11 -0
- data/lib/webby/core_ext/hash.rb +28 -0
- data/lib/webby/core_ext/kernel.rb +26 -0
- data/lib/webby/core_ext/string.rb +163 -0
- data/lib/webby/core_ext/time.rb +9 -0
- data/lib/webby/filters.rb +85 -0
- data/lib/webby/filters/basepath.rb +97 -0
- data/lib/webby/filters/erb.rb +9 -0
- data/lib/webby/filters/haml.rb +18 -0
- data/lib/webby/filters/markdown.rb +16 -0
- data/lib/webby/filters/maruku.rb +16 -0
- data/lib/webby/filters/outline.rb +309 -0
- data/lib/webby/filters/sass.rb +17 -0
- data/lib/webby/filters/slides.rb +56 -0
- data/lib/webby/filters/textile.rb +16 -0
- data/lib/webby/filters/tidy.rb +76 -0
- data/lib/webby/filters/wiki_words.rb +14 -0
- data/lib/webby/helpers.rb +30 -0
- data/lib/webby/helpers/capture_helper.rb +141 -0
- data/lib/webby/helpers/coderay_helper.rb +69 -0
- data/lib/webby/helpers/graphviz_helper.rb +136 -0
- data/lib/webby/helpers/tag_helper.rb +65 -0
- data/lib/webby/helpers/tex_img_helper.rb +133 -0
- data/lib/webby/helpers/ultraviolet_helper.rb +63 -0
- data/lib/webby/helpers/url_helper.rb +241 -0
- data/lib/webby/journal.rb +126 -0
- data/lib/webby/link_validator.rb +160 -0
- data/lib/webby/renderer.rb +390 -0
- data/lib/webby/resources.rb +137 -0
- data/lib/webby/resources/db.rb +251 -0
- data/lib/webby/resources/layout.rb +54 -0
- data/lib/webby/resources/meta_file.rb +211 -0
- data/lib/webby/resources/page.rb +81 -0
- data/lib/webby/resources/partial.rb +85 -0
- data/lib/webby/resources/resource.rb +201 -0
- data/lib/webby/resources/static.rb +36 -0
- data/lib/webby/stelan/mktemp.rb +135 -0
- data/lib/webby/stelan/paginator.rb +165 -0
- data/lib/webby/tasks/build.rake +27 -0
- data/lib/webby/tasks/create.rake +25 -0
- data/lib/webby/tasks/deploy.rake +22 -0
- data/lib/webby/tasks/growl.rake +16 -0
- data/lib/webby/tasks/validate.rake +19 -0
- data/spec/core_ext/hash_spec.rb +47 -0
- data/spec/core_ext/string_spec.rb +110 -0
- data/spec/core_ext/time_spec.rb +19 -0
- data/spec/data/hooligans/bad_meta_data_1.txt +34 -0
- data/spec/data/hooligans/bad_meta_data_2.txt +34 -0
- data/spec/data/html/anchor.html +11 -0
- data/spec/data/html/external.html +10 -0
- data/spec/data/html/invalid-relative.html +10 -0
- data/spec/data/html/relative-anchor.html +10 -0
- data/spec/data/html/relative-invalid-anchor.html +10 -0
- data/spec/data/html/relative.html +10 -0
- data/spec/data/outline/basic.out +81 -0
- data/spec/data/outline/basic.txt +25 -0
- data/spec/data/outline/no_clobber.out +86 -0
- data/spec/data/outline/numbering.out +81 -0
- data/spec/data/outline/numbering_only.out +21 -0
- data/spec/data/outline/toc_range_1.out +66 -0
- data/spec/data/outline/toc_range_2.out +55 -0
- data/spec/data/outline/toc_style.out +81 -0
- data/spec/data/site/Sitefile +9 -0
- data/spec/data/site/content/_partial.txt +10 -0
- data/spec/data/site/content/css/coderay.css +111 -0
- data/spec/data/site/content/css/site.css +67 -0
- data/spec/data/site/content/css/tumblog.css +308 -0
- data/spec/data/site/content/images/tumblog/permalink.gif +0 -0
- data/spec/data/site/content/images/tumblog/rss.gif +0 -0
- data/spec/data/site/content/index.txt +19 -0
- data/spec/data/site/content/photos.txt +21 -0
- data/spec/data/site/content/tumblog/200806/the-noble-chicken/index.txt +12 -0
- data/spec/data/site/content/tumblog/200807/historical-perspectives-on-the-classic-chicken-joke/index.txt +12 -0
- data/spec/data/site/content/tumblog/200807/mad-city-chickens/index.txt +10 -0
- data/spec/data/site/content/tumblog/200807/the-wisdom-of-the-dutch/index.txt +11 -0
- data/spec/data/site/content/tumblog/200807/up-a-tree/index.txt +13 -0
- data/spec/data/site/content/tumblog/index.txt +37 -0
- data/spec/data/site/content/tumblog/rss.txt +37 -0
- data/spec/data/site/layouts/default.txt +58 -0
- data/spec/data/site/layouts/tumblog/default.txt +44 -0
- data/spec/data/site/layouts/tumblog/post.txt +15 -0
- data/spec/data/site/lib/breadcrumbs.rb +28 -0
- data/spec/data/site/lib/tumblog_helper.rb +32 -0
- data/spec/data/site/tasks/tumblog.rake +30 -0
- data/spec/data/site/templates/_partial.erb +10 -0
- data/spec/data/site/templates/atom_feed.erb +40 -0
- data/spec/data/site/templates/page.erb +18 -0
- data/spec/data/site/templates/presentation.erb +40 -0
- data/spec/data/site/templates/tumblog/conversation.erb +12 -0
- data/spec/data/site/templates/tumblog/link.erb +10 -0
- data/spec/data/site/templates/tumblog/photo.erb +13 -0
- data/spec/data/site/templates/tumblog/post.erb +12 -0
- data/spec/data/site/templates/tumblog/quote.erb +11 -0
- data/spec/spec.opts +1 -0
- data/spec/spec_helper.rb +58 -0
- data/spec/webby/apps/generator_spec.rb +117 -0
- data/spec/webby/apps/main_spec.rb +88 -0
- data/spec/webby/filters/basepath_spec.rb +167 -0
- data/spec/webby/filters/maruku_spec.rb +31 -0
- data/spec/webby/filters/outline_spec.rb +92 -0
- data/spec/webby/filters/textile_spec.rb +31 -0
- data/spec/webby/helpers/capture_helper_spec.rb +56 -0
- data/spec/webby/link_validator_spec.rb +154 -0
- data/spec/webby/renderer_spec.rb +139 -0
- data/spec/webby/resources/db_spec.rb +250 -0
- data/spec/webby/resources/layout_spec.rb +83 -0
- data/spec/webby/resources/meta_file_spec.rb +171 -0
- data/spec/webby/resources/page_spec.rb +111 -0
- data/spec/webby/resources/partial_spec.rb +58 -0
- data/spec/webby/resources/resource_spec.rb +219 -0
- data/spec/webby/resources/static_spec.rb +49 -0
- data/spec/webby/resources_spec.rb +69 -0
- data/tasks/mswin32.rake +38 -0
- data/tasks/website.rake +37 -0
- metadata +435 -0
@@ -0,0 +1,258 @@
|
|
1
|
+
require 'optparse'
|
2
|
+
require 'rake'
|
3
|
+
|
4
|
+
module Webby::Apps
|
5
|
+
|
6
|
+
class Main
|
7
|
+
|
8
|
+
# Create a new instance of Main, and run the +webby+ application given the
|
9
|
+
# command line _args_.
|
10
|
+
#
|
11
|
+
def self.run( args )
|
12
|
+
self.new.run args
|
13
|
+
end
|
14
|
+
|
15
|
+
attr_reader :cmd_line_options
|
16
|
+
|
17
|
+
# Create a new Main webby object for building websites.
|
18
|
+
#
|
19
|
+
def initialize
|
20
|
+
@stdout = $stdout
|
21
|
+
@cmd_line_options = {}
|
22
|
+
end
|
23
|
+
|
24
|
+
# Runs the main webby application. The command line arguments are passed
|
25
|
+
# in to this method as an array of strings. The command line arguments are
|
26
|
+
# parsed to figure out which rake task to invoke.
|
27
|
+
#
|
28
|
+
def run( args )
|
29
|
+
args = args.dup
|
30
|
+
|
31
|
+
parse args
|
32
|
+
init args
|
33
|
+
rake
|
34
|
+
end
|
35
|
+
|
36
|
+
# Parse the command line _args_ for options and commands to invoke.
|
37
|
+
#
|
38
|
+
def parse( args )
|
39
|
+
opts = OptionParser.new
|
40
|
+
opts.banner = 'Usage: webby [options] target [target args]'
|
41
|
+
|
42
|
+
opts.separator ''
|
43
|
+
|
44
|
+
desired_opts = %[--describe --prereqs --tasks --trace]
|
45
|
+
app.standard_rake_options.each do |options|
|
46
|
+
next unless desired_opts.include?(options.first)
|
47
|
+
opts.on(*options)
|
48
|
+
end
|
49
|
+
|
50
|
+
opts.separator ''
|
51
|
+
opts.separator 'autobuild options:'
|
52
|
+
|
53
|
+
opts.on('--web-server', 'Start a local web server') {
|
54
|
+
cmd_line_options[:use_web_server] = true
|
55
|
+
}
|
56
|
+
opts.on('--no-web-server', 'Do not start a local web server') {
|
57
|
+
cmd_line_options[:use_web_server] = false
|
58
|
+
}
|
59
|
+
|
60
|
+
opts.separator ''
|
61
|
+
opts.separator 'common options:'
|
62
|
+
|
63
|
+
opts.on_tail( '-h', '--help', 'show this message' ) do
|
64
|
+
@stdout.puts opts
|
65
|
+
exit
|
66
|
+
end
|
67
|
+
opts.on_tail( '--version', 'show version' ) do
|
68
|
+
@stdout.puts "Webby #{::Webby::VERSION}"
|
69
|
+
exit
|
70
|
+
end
|
71
|
+
|
72
|
+
opts.parse! args
|
73
|
+
|
74
|
+
ARGV.replace Array(args.shift)
|
75
|
+
args.delete_if do |arg|
|
76
|
+
if %r/^[A-Z_]+=/ =~ arg
|
77
|
+
ARGV << arg
|
78
|
+
next true
|
79
|
+
end
|
80
|
+
false
|
81
|
+
end
|
82
|
+
|
83
|
+
args
|
84
|
+
end
|
85
|
+
|
86
|
+
# Initialize the Rake application object and load the core rake tasks, the
|
87
|
+
# site specific rake tasks, and the site specific ruby code. Any extra
|
88
|
+
# command line arguments are converted into a page name and directory that
|
89
|
+
# might get created (depending upon the task invoked).
|
90
|
+
#
|
91
|
+
def init( args )
|
92
|
+
# Make sure we're in a folder with a Sitefile
|
93
|
+
options = app.standard_rake_options
|
94
|
+
[['--rakefile', 'Sitefile'],
|
95
|
+
['--no-search', nil],
|
96
|
+
['--silent', nil]].each {|opt, value| options.assoc(opt).last.call(value)}
|
97
|
+
|
98
|
+
unless app.have_rakefile
|
99
|
+
raise RuntimeError, "Sitefile not found"
|
100
|
+
end
|
101
|
+
|
102
|
+
import_default_tasks
|
103
|
+
import_website_tasks
|
104
|
+
require_lib_files
|
105
|
+
capture_command_line_args(args)
|
106
|
+
args
|
107
|
+
end
|
108
|
+
|
109
|
+
# Execute the rake command.
|
110
|
+
#
|
111
|
+
def rake
|
112
|
+
app.init 'webby'
|
113
|
+
app.load_rakefile
|
114
|
+
load_command_line_options
|
115
|
+
app.top_level
|
116
|
+
end
|
117
|
+
|
118
|
+
# Return the Rake application object.
|
119
|
+
#
|
120
|
+
def app
|
121
|
+
Rake.application
|
122
|
+
end
|
123
|
+
|
124
|
+
# Returns the options hash from the Rake application object.
|
125
|
+
#
|
126
|
+
def options
|
127
|
+
app.options
|
128
|
+
end
|
129
|
+
|
130
|
+
# Search for the "Sitefile" starting in the current directory and working
|
131
|
+
# upwards through the filesystem until the root of the filesystem is
|
132
|
+
# reached. If a "Sitefile" is not found, a RuntimeError is raised.
|
133
|
+
#
|
134
|
+
def find_sitefile
|
135
|
+
here = Dir.pwd
|
136
|
+
while ! app.have_rakefile
|
137
|
+
Dir.chdir("..")
|
138
|
+
if Dir.pwd == here || options.nosearch
|
139
|
+
fail "No Sitefile found"
|
140
|
+
end
|
141
|
+
here = Dir.pwd
|
142
|
+
end
|
143
|
+
end
|
144
|
+
|
145
|
+
def import_default_tasks
|
146
|
+
Dir.glob(::Webby.libpath(%w[webby tasks *.rake])).sort.each {|fn| import fn}
|
147
|
+
end
|
148
|
+
|
149
|
+
def import_website_tasks
|
150
|
+
Dir.glob(::File.join(%w[tasks *.rake])).sort.each {|fn| import fn}
|
151
|
+
end
|
152
|
+
|
153
|
+
def require_lib_files
|
154
|
+
Dir.glob(::File.join(%w[lib ** *.rb])).sort.each {|fn| require fn}
|
155
|
+
end
|
156
|
+
|
157
|
+
def capture_command_line_args(args)
|
158
|
+
args = OpenStruct.new(
|
159
|
+
:raw => args,
|
160
|
+
:rake => ARGV.dup
|
161
|
+
)
|
162
|
+
|
163
|
+
if args.raw.size > 1
|
164
|
+
::Webby.deprecated "multiple arguments used for page title",
|
165
|
+
"please quote the page title"
|
166
|
+
end
|
167
|
+
|
168
|
+
dashed = args.raw.join('-').downcase
|
169
|
+
spaced = args.raw.join(' ')
|
170
|
+
dir = ::File.dirname(dashed)
|
171
|
+
|
172
|
+
args.dir = ('.' == dir ? '' : dir)
|
173
|
+
args.slug = ::Webby::Resources.basename(dashed).to_url
|
174
|
+
args.title = ::Webby::Resources.basename(spaced).titlecase
|
175
|
+
|
176
|
+
# page should be dir/slug without leading /
|
177
|
+
args.page = ::File.join(args.dir, args.slug).gsub(/^\//, '')
|
178
|
+
|
179
|
+
ext = ::File.extname(dashed)
|
180
|
+
args.page << ext unless ext.empty?
|
181
|
+
|
182
|
+
::Webby.site.args = args
|
183
|
+
Object.const_set(:SITE, Webby.site)
|
184
|
+
args
|
185
|
+
end
|
186
|
+
|
187
|
+
# Load options from the command line into the ::Webby.site struct
|
188
|
+
#
|
189
|
+
def load_command_line_options
|
190
|
+
cmd_line_options.each do |key, value|
|
191
|
+
::Webby.site.__send__("#{key}=", value)
|
192
|
+
end
|
193
|
+
end
|
194
|
+
|
195
|
+
end # class Main
|
196
|
+
end # module Webby::Apps
|
197
|
+
|
198
|
+
# :stopdoc:
|
199
|
+
# Monkey patches so that rake displays the correct application name in the
|
200
|
+
# help messages.
|
201
|
+
#
|
202
|
+
class Rake::Application
|
203
|
+
def display_prerequisites
|
204
|
+
tasks.each do |t|
|
205
|
+
puts "#{name} #{t.name}"
|
206
|
+
t.prerequisites.each { |pre| puts " #{pre}" }
|
207
|
+
end
|
208
|
+
end
|
209
|
+
|
210
|
+
def display_tasks_and_comments
|
211
|
+
displayable_tasks = tasks.select { |t|
|
212
|
+
t.comment && t.name =~ options.show_task_pattern
|
213
|
+
}
|
214
|
+
if options.full_description
|
215
|
+
displayable_tasks.each do |t|
|
216
|
+
puts "#{name} #{t.name_with_args}"
|
217
|
+
t.full_comment.split("\n").each do |line|
|
218
|
+
puts " #{line}"
|
219
|
+
end
|
220
|
+
puts
|
221
|
+
end
|
222
|
+
else
|
223
|
+
width = displayable_tasks.collect { |t| t.name_with_args.length }.max || 10
|
224
|
+
max_column = truncate_output? ? terminal_width - name.size - width - 7 : nil
|
225
|
+
displayable_tasks.each do |t|
|
226
|
+
printf "#{name} %-#{width}s # %s\n",
|
227
|
+
t.name_with_args, max_column ? truncate(t.comment, max_column) : t.comment
|
228
|
+
end
|
229
|
+
end
|
230
|
+
end
|
231
|
+
|
232
|
+
# Provide standard execption handling for the given block.
|
233
|
+
def standard_exception_handling
|
234
|
+
begin
|
235
|
+
yield
|
236
|
+
rescue SystemExit => ex
|
237
|
+
# Exit silently with current status
|
238
|
+
exit(ex.status)
|
239
|
+
rescue SystemExit, OptionParser::InvalidOption => ex
|
240
|
+
# Exit silently
|
241
|
+
exit(1)
|
242
|
+
rescue Exception => ex
|
243
|
+
# Exit with error message
|
244
|
+
$stderr.puts "#{name} aborted!"
|
245
|
+
$stderr.puts ex.message
|
246
|
+
if options.trace
|
247
|
+
$stderr.puts ex.backtrace.join("\n")
|
248
|
+
else
|
249
|
+
$stderr.puts ex.backtrace.find {|str| str =~ /#{@rakefile}/ } || ""
|
250
|
+
$stderr.puts "(See full trace by running task with --trace)"
|
251
|
+
end
|
252
|
+
exit(1)
|
253
|
+
end
|
254
|
+
end
|
255
|
+
end # class Rake::Application
|
256
|
+
# :startdoc:
|
257
|
+
|
258
|
+
# EOF
|
@@ -0,0 +1,157 @@
|
|
1
|
+
require 'directory_watcher'
|
2
|
+
require 'launchy'
|
3
|
+
require 'webrick'
|
4
|
+
|
5
|
+
module Webby
|
6
|
+
|
7
|
+
# The AutoBuilder class is used to monitor the content and layouts folders
|
8
|
+
# and to compile the resource files only when they are modified. If a
|
9
|
+
# layout is modified, then all resources that depend upon the layout are
|
10
|
+
# compiled.
|
11
|
+
#
|
12
|
+
class AutoBuilder
|
13
|
+
|
14
|
+
# TODO: hit Ctrl-C once to rebuild everything, and hit it twice to stop the autobuild loop
|
15
|
+
|
16
|
+
# call-seq:
|
17
|
+
# AutoBuilder.run
|
18
|
+
#
|
19
|
+
# Creates a new AutoBuilder and sets it running. This method will only
|
20
|
+
# return when the user presses Ctrl-C.
|
21
|
+
#
|
22
|
+
def self.run
|
23
|
+
self.new.run
|
24
|
+
end
|
25
|
+
|
26
|
+
attr_reader :logger
|
27
|
+
|
28
|
+
# call-seq:
|
29
|
+
# AutoBuilder.new
|
30
|
+
#
|
31
|
+
# Create a new AutoBuilder class.
|
32
|
+
#
|
33
|
+
def initialize
|
34
|
+
@logger = Logging::Logger[self]
|
35
|
+
|
36
|
+
@builder = Builder.new
|
37
|
+
::Webby.load_files
|
38
|
+
|
39
|
+
@watcher = DirectoryWatcher.new '.', :interval => 2
|
40
|
+
@watcher.add_observer self
|
41
|
+
|
42
|
+
glob = []
|
43
|
+
glob << File.join(::Webby.site.layout_dir, '**', '*')
|
44
|
+
glob << File.join(::Webby.site.content_dir, '**', '*')
|
45
|
+
@watcher.glob = glob
|
46
|
+
|
47
|
+
@web_server = ::Webby.site.use_web_server ? WebServer.new : nil
|
48
|
+
end
|
49
|
+
|
50
|
+
# call-seq:
|
51
|
+
# update( *events )
|
52
|
+
#
|
53
|
+
# The update method is called by the DirectoryWatcher when files have been
|
54
|
+
# modified, added, or deleted. An array of events is passed to his method,
|
55
|
+
# and each event contains the event type and the path to the file.
|
56
|
+
#
|
57
|
+
def update( *events )
|
58
|
+
ary = events.find_all {|evt| evt.type != :removed}
|
59
|
+
return if ary.empty?
|
60
|
+
|
61
|
+
ary.each do |evt|
|
62
|
+
logger.debug "changed #{evt.path}"
|
63
|
+
next unless test ?f, evt.path
|
64
|
+
next if evt.path =~ ::Webby.exclude
|
65
|
+
Resources.new evt.path
|
66
|
+
end
|
67
|
+
|
68
|
+
logger.info 'running the build'
|
69
|
+
@builder.run :load_files => false, :verbose => false
|
70
|
+
rescue => err
|
71
|
+
logger.error err
|
72
|
+
end
|
73
|
+
|
74
|
+
# call-seq:
|
75
|
+
# run
|
76
|
+
#
|
77
|
+
# Starts the DirectoryWatcher running and waits till the user presses
|
78
|
+
# Ctrl-C to stop the watcher thread.
|
79
|
+
#
|
80
|
+
def run
|
81
|
+
logger.info 'starting autobuild (Ctrl-C to stop)'
|
82
|
+
|
83
|
+
Signal.trap('INT') {
|
84
|
+
@watcher.stop
|
85
|
+
@web_server.stop if @web_server
|
86
|
+
}
|
87
|
+
|
88
|
+
@watcher.start
|
89
|
+
if @web_server
|
90
|
+
@web_server.start
|
91
|
+
sleep 0.25
|
92
|
+
Launchy.open("http://localhost:#{::Webby.site.web_port}")
|
93
|
+
end
|
94
|
+
|
95
|
+
@watcher.join
|
96
|
+
@web_server.join if @web_server
|
97
|
+
end
|
98
|
+
|
99
|
+
# Wrapper class around the webrick web server.
|
100
|
+
#
|
101
|
+
class WebServer
|
102
|
+
|
103
|
+
# Create a new webrick server configured to serve pages from the output
|
104
|
+
# directory. Output will be directed to /dev/null.
|
105
|
+
#
|
106
|
+
def initialize
|
107
|
+
logger = WEBrick::Log.new(Kernel::DEV_NULL, WEBrick::Log::DEBUG)
|
108
|
+
access_log = [[ logger, WEBrick::AccessLog::COMBINED_LOG_FORMAT ]]
|
109
|
+
|
110
|
+
@thread = nil
|
111
|
+
@running = false
|
112
|
+
@server = WEBrick::HTTPServer.new(
|
113
|
+
:BindAddress => 'localhost',
|
114
|
+
:Port => ::Webby.site.web_port,
|
115
|
+
:DocumentRoot => ::Webby.site.output_dir,
|
116
|
+
:FancyIndexing => true,
|
117
|
+
:Logger => logger,
|
118
|
+
:AccessLog => access_log
|
119
|
+
)
|
120
|
+
end
|
121
|
+
|
122
|
+
# Returns +true+ if the server is running.
|
123
|
+
#
|
124
|
+
def running?
|
125
|
+
@running
|
126
|
+
end
|
127
|
+
|
128
|
+
# Start the webrick server running in a separate thread (so we don't
|
129
|
+
# block forever).
|
130
|
+
#
|
131
|
+
def start
|
132
|
+
return if running?
|
133
|
+
@running = true
|
134
|
+
@thread = Thread.new {@server.start}
|
135
|
+
end
|
136
|
+
|
137
|
+
# Stop the webrick server.
|
138
|
+
#
|
139
|
+
def stop
|
140
|
+
return if not running?
|
141
|
+
@running = false
|
142
|
+
@server.shutdown
|
143
|
+
end
|
144
|
+
|
145
|
+
# Join on the webserver thread.
|
146
|
+
#
|
147
|
+
def join
|
148
|
+
return if not running?
|
149
|
+
@thread.join
|
150
|
+
end
|
151
|
+
|
152
|
+
end # class WebServer
|
153
|
+
|
154
|
+
end # class AutoBuilder
|
155
|
+
end # module Webby
|
156
|
+
|
157
|
+
# EOF
|
@@ -0,0 +1,172 @@
|
|
1
|
+
require 'find'
|
2
|
+
require 'fileutils'
|
3
|
+
require 'erb'
|
4
|
+
|
5
|
+
module Webby
|
6
|
+
|
7
|
+
# The Builder class performs the work of scanning the content folder,
|
8
|
+
# creating Resource objects, and converting / copying the contents to the
|
9
|
+
# output folder as needed.
|
10
|
+
#
|
11
|
+
class Builder
|
12
|
+
|
13
|
+
class << self
|
14
|
+
|
15
|
+
# call-seq:
|
16
|
+
# Builder.run( :rebuild => false )
|
17
|
+
#
|
18
|
+
# Create a new instance of the Builder class and invoke the run method.
|
19
|
+
# If the <code>:rebuild</code> option is given as +true+, then all pages
|
20
|
+
# will be recreated / copied.
|
21
|
+
#
|
22
|
+
def run( opts = {} )
|
23
|
+
self.new.run opts
|
24
|
+
end
|
25
|
+
|
26
|
+
# call-seq:
|
27
|
+
# Builder.create( page, :from => template, :locals => {} )
|
28
|
+
#
|
29
|
+
# This mehod is used to create a new _page_ in the content folder based
|
30
|
+
# on the specified template. _page_ is the relative path to the new page
|
31
|
+
# from the <code>content/</code> folder. The _template_ is the name of
|
32
|
+
# the template to use from the <code>templates/</code> folder.
|
33
|
+
#
|
34
|
+
def create( page, opts = {} )
|
35
|
+
tmpl = opts[:from]
|
36
|
+
raise Error, "template not given" unless tmpl
|
37
|
+
|
38
|
+
name = ::Webby::Resources.basename(page)
|
39
|
+
ext = ::Webby::Resources.extname(page)
|
40
|
+
dir = ::File.dirname(page)
|
41
|
+
dir = '' if dir == '.'
|
42
|
+
|
43
|
+
if tmpl.pathmap('%n') =~ %r/^_/
|
44
|
+
page = ::File.join(::Webby.site.content_dir, dir, '_'+name)
|
45
|
+
page << '.' << (ext.empty? ? Webby.site.default_ext : ext)
|
46
|
+
elsif ::Webby.site.create_mode == 'directory' and name != 'index'
|
47
|
+
page = ::File.join(::Webby.site.content_dir, dir, name, 'index')
|
48
|
+
page << '.' << (ext.empty? ? Webby.site.default_ext : ext)
|
49
|
+
else
|
50
|
+
page = ::File.join(::Webby.site.content_dir, page)
|
51
|
+
page << '.' << Webby.site.default_ext if ext.empty?
|
52
|
+
end
|
53
|
+
raise Error, "#{page} already exists" if test ?e, page
|
54
|
+
|
55
|
+
Logging::Logger[self].info "creating #{page}"
|
56
|
+
FileUtils.mkdir_p ::File.dirname(page)
|
57
|
+
|
58
|
+
context = scope
|
59
|
+
opts[:locals].each do |k,v|
|
60
|
+
Thread.current[:value] = v
|
61
|
+
definition = "#{k} = Thread.current[:value]"
|
62
|
+
eval(definition, context)
|
63
|
+
end if opts.has_key?(:locals)
|
64
|
+
|
65
|
+
str = ERB.new(::File.read(tmpl), nil, '-').result(context)
|
66
|
+
::File.open(page, 'w') {|fd| fd.write str}
|
67
|
+
|
68
|
+
page
|
69
|
+
end
|
70
|
+
|
71
|
+
# call-seq:
|
72
|
+
# Builder.new_page_info => [page, title, directory]
|
73
|
+
#
|
74
|
+
def new_page_info
|
75
|
+
args = Webby.site.args
|
76
|
+
|
77
|
+
# TODO: maybe even get rid of this method altogether
|
78
|
+
raise "Usage: webby #{args.rake.first} 'path'" if args.raw.empty?
|
79
|
+
|
80
|
+
[args.page, args.title, args.dir]
|
81
|
+
end
|
82
|
+
|
83
|
+
private
|
84
|
+
|
85
|
+
# Returns the binding in the scope of the Builder class object.
|
86
|
+
#
|
87
|
+
def scope() binding end
|
88
|
+
|
89
|
+
end # class << self
|
90
|
+
|
91
|
+
# call-seq:
|
92
|
+
# Builder.new
|
93
|
+
#
|
94
|
+
# Creates a new Builder object for creating pages from the content and
|
95
|
+
# layout directories.
|
96
|
+
#
|
97
|
+
def initialize
|
98
|
+
@logger = Logging::Logger[self]
|
99
|
+
end
|
100
|
+
|
101
|
+
# call-seq:
|
102
|
+
# run( :rebuild => false, :load_files => true )
|
103
|
+
#
|
104
|
+
# Runs the Webby builder by loading in the layout files from the
|
105
|
+
# <code>layouts/</code> folder and the content from the
|
106
|
+
# <code>contents/</code> folder. Content is analyzed, and those that need
|
107
|
+
# to be copied or compiled (filtered using ERB, Texttile, Markdown, etc.)
|
108
|
+
# are handled. The results are placed in the <code>output/</code> folder.
|
109
|
+
#
|
110
|
+
# If the <code>:rebuild</code> flag is set to +true+, then all content is
|
111
|
+
# copied and/or compiled to the output folder.
|
112
|
+
#
|
113
|
+
# A content file can mark itself as dirty by setting the +dirty+ flag to
|
114
|
+
# +true+ in the meta-data of the file. This will cause the contenet to
|
115
|
+
# always be compiled when the builder is run. Conversely, setting the
|
116
|
+
# dirty flag to +false+ will cause the content to never be compiled or
|
117
|
+
# copied to the output folder.
|
118
|
+
#
|
119
|
+
# A content file needs to be built if the age of the file is less then the
|
120
|
+
# age of the output product -- i.e. the content file has been modified
|
121
|
+
# more recently than the output file.
|
122
|
+
#
|
123
|
+
def run( opts = {} )
|
124
|
+
opts[:load_files] = true unless opts.has_key?(:load_files)
|
125
|
+
verbose = opts.getopt(:verbose, true)
|
126
|
+
|
127
|
+
unless test(?d, output_dir)
|
128
|
+
journal.create output_dir
|
129
|
+
FileUtils.mkdir output_dir
|
130
|
+
end
|
131
|
+
|
132
|
+
::Webby.load_files if opts[:load_files]
|
133
|
+
|
134
|
+
Resources.pages.each do |page|
|
135
|
+
unless page.dirty? or opts[:rebuild]
|
136
|
+
journal.identical(page.destination) if verbose
|
137
|
+
next
|
138
|
+
end
|
139
|
+
|
140
|
+
# copy the resource to the output directory if it is static
|
141
|
+
if page.instance_of? Resources::Static
|
142
|
+
FileUtils.mkdir_p ::File.dirname(page.destination)
|
143
|
+
journal.create_or_update(page)
|
144
|
+
FileUtils.cp page.path, page.destination
|
145
|
+
FileUtils.chmod 0644, page.destination
|
146
|
+
|
147
|
+
# otherwise, layout the resource and write the results to
|
148
|
+
# the output directory
|
149
|
+
else Renderer.write(page) end
|
150
|
+
end
|
151
|
+
|
152
|
+
# touch the cairn so we know when the website was last generated
|
153
|
+
FileUtils.touch ::Webby.cairn
|
154
|
+
|
155
|
+
nil
|
156
|
+
end
|
157
|
+
|
158
|
+
def load_files
|
159
|
+
::Webby.deprecated "load_files", "it is being replaced by the Webby#load() method"
|
160
|
+
::Webby.load_files
|
161
|
+
end
|
162
|
+
|
163
|
+
%w(output_dir layout_dir content_dir).each do |key|
|
164
|
+
self.class_eval <<-CODE
|
165
|
+
def #{key}( ) ::Webby.site.#{key} end
|
166
|
+
CODE
|
167
|
+
end
|
168
|
+
|
169
|
+
end # class Builder
|
170
|
+
end # module Webby
|
171
|
+
|
172
|
+
# EOF
|