rails 3.2.22.2 → 4.2.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (216) hide show
  1. checksums.yaml +7 -0
  2. data/README.md +85 -0
  3. data/guides/CHANGELOG.md +31 -0
  4. data/guides/Rakefile +79 -0
  5. data/guides/assets/images/akshaysurve.jpg +0 -0
  6. data/guides/assets/images/belongs_to.png +0 -0
  7. data/guides/assets/images/book_icon.gif +0 -0
  8. data/guides/assets/images/bullet.gif +0 -0
  9. data/guides/assets/images/chapters_icon.gif +0 -0
  10. data/guides/assets/images/check_bullet.gif +0 -0
  11. data/guides/assets/images/credits_pic_blank.gif +0 -0
  12. data/guides/assets/images/csrf.png +0 -0
  13. data/guides/assets/images/edge_badge.png +0 -0
  14. data/guides/assets/images/favicon.ico +0 -0
  15. data/guides/assets/images/feature_tile.gif +0 -0
  16. data/guides/assets/images/footer_tile.gif +0 -0
  17. data/guides/assets/images/fxn.png +0 -0
  18. data/guides/assets/images/getting_started/article_with_comments.png +0 -0
  19. data/guides/assets/images/getting_started/challenge.png +0 -0
  20. data/guides/assets/images/getting_started/confirm_dialog.png +0 -0
  21. data/guides/assets/images/getting_started/forbidden_attributes_for_new_article.png +0 -0
  22. data/guides/assets/images/getting_started/form_with_errors.png +0 -0
  23. data/guides/assets/images/getting_started/index_action_with_edit_link.png +0 -0
  24. data/guides/assets/images/getting_started/new_article.png +0 -0
  25. data/guides/assets/images/getting_started/rails_welcome.png +0 -0
  26. data/guides/assets/images/getting_started/routing_error_no_controller.png +0 -0
  27. data/guides/assets/images/getting_started/routing_error_no_route_matches.png +0 -0
  28. data/guides/assets/images/getting_started/show_action_for_articles.png +0 -0
  29. data/guides/assets/images/getting_started/template_is_missing_articles_new.png +0 -0
  30. data/guides/assets/images/getting_started/unknown_action_create_for_articles.png +0 -0
  31. data/guides/assets/images/getting_started/unknown_action_new_for_articles.png +0 -0
  32. data/guides/assets/images/grey_bullet.gif +0 -0
  33. data/guides/assets/images/habtm.png +0 -0
  34. data/guides/assets/images/has_many.png +0 -0
  35. data/guides/assets/images/has_many_through.png +0 -0
  36. data/guides/assets/images/has_one.png +0 -0
  37. data/guides/assets/images/has_one_through.png +0 -0
  38. data/guides/assets/images/header_backdrop.png +0 -0
  39. data/guides/assets/images/header_tile.gif +0 -0
  40. data/guides/assets/images/i18n/demo_html_safe.png +0 -0
  41. data/guides/assets/images/i18n/demo_localized_pirate.png +0 -0
  42. data/guides/assets/images/i18n/demo_translated_en.png +0 -0
  43. data/guides/assets/images/i18n/demo_translated_pirate.png +0 -0
  44. data/guides/assets/images/i18n/demo_translation_missing.png +0 -0
  45. data/guides/assets/images/i18n/demo_untranslated.png +0 -0
  46. data/guides/assets/images/icons/README +5 -0
  47. data/guides/assets/images/icons/callouts/1.png +0 -0
  48. data/guides/assets/images/icons/callouts/10.png +0 -0
  49. data/guides/assets/images/icons/callouts/11.png +0 -0
  50. data/guides/assets/images/icons/callouts/12.png +0 -0
  51. data/guides/assets/images/icons/callouts/13.png +0 -0
  52. data/guides/assets/images/icons/callouts/14.png +0 -0
  53. data/guides/assets/images/icons/callouts/15.png +0 -0
  54. data/guides/assets/images/icons/callouts/2.png +0 -0
  55. data/guides/assets/images/icons/callouts/3.png +0 -0
  56. data/guides/assets/images/icons/callouts/4.png +0 -0
  57. data/guides/assets/images/icons/callouts/5.png +0 -0
  58. data/guides/assets/images/icons/callouts/6.png +0 -0
  59. data/guides/assets/images/icons/callouts/7.png +0 -0
  60. data/guides/assets/images/icons/callouts/8.png +0 -0
  61. data/guides/assets/images/icons/callouts/9.png +0 -0
  62. data/guides/assets/images/icons/caution.png +0 -0
  63. data/guides/assets/images/icons/example.png +0 -0
  64. data/guides/assets/images/icons/home.png +0 -0
  65. data/guides/assets/images/icons/important.png +0 -0
  66. data/guides/assets/images/icons/next.png +0 -0
  67. data/guides/assets/images/icons/note.png +0 -0
  68. data/guides/assets/images/icons/prev.png +0 -0
  69. data/guides/assets/images/icons/tip.png +0 -0
  70. data/guides/assets/images/icons/up.png +0 -0
  71. data/guides/assets/images/icons/warning.png +0 -0
  72. data/guides/assets/images/nav_arrow.gif +0 -0
  73. data/guides/assets/images/oscardelben.jpg +0 -0
  74. data/guides/assets/images/polymorphic.png +0 -0
  75. data/guides/assets/images/radar.png +0 -0
  76. data/guides/assets/images/rails4_features.png +0 -0
  77. data/guides/assets/images/rails_guides_kindle_cover.jpg +0 -0
  78. data/guides/assets/images/rails_guides_logo.gif +0 -0
  79. data/guides/assets/images/rails_logo_remix.gif +0 -0
  80. data/guides/assets/images/session_fixation.png +0 -0
  81. data/guides/assets/images/tab_grey.gif +0 -0
  82. data/guides/assets/images/tab_info.gif +0 -0
  83. data/guides/assets/images/tab_note.gif +0 -0
  84. data/guides/assets/images/tab_red.gif +0 -0
  85. data/guides/assets/images/tab_yellow.gif +0 -0
  86. data/guides/assets/images/tab_yellow.png +0 -0
  87. data/guides/assets/images/vijaydev.jpg +0 -0
  88. data/guides/assets/javascripts/guides.js +59 -0
  89. data/guides/assets/javascripts/jquery.min.js +4 -0
  90. data/guides/assets/javascripts/responsive-tables.js +43 -0
  91. data/guides/assets/javascripts/syntaxhighlighter/shBrushAS3.js +59 -0
  92. data/guides/assets/javascripts/syntaxhighlighter/shBrushAppleScript.js +75 -0
  93. data/guides/assets/javascripts/syntaxhighlighter/shBrushBash.js +59 -0
  94. data/guides/assets/javascripts/syntaxhighlighter/shBrushCSharp.js +65 -0
  95. data/guides/assets/javascripts/syntaxhighlighter/shBrushColdFusion.js +100 -0
  96. data/guides/assets/javascripts/syntaxhighlighter/shBrushCpp.js +97 -0
  97. data/guides/assets/javascripts/syntaxhighlighter/shBrushCss.js +91 -0
  98. data/guides/assets/javascripts/syntaxhighlighter/shBrushDelphi.js +55 -0
  99. data/guides/assets/javascripts/syntaxhighlighter/shBrushDiff.js +41 -0
  100. data/guides/assets/javascripts/syntaxhighlighter/shBrushErlang.js +52 -0
  101. data/guides/assets/javascripts/syntaxhighlighter/shBrushGroovy.js +67 -0
  102. data/guides/assets/javascripts/syntaxhighlighter/shBrushJScript.js +52 -0
  103. data/guides/assets/javascripts/syntaxhighlighter/shBrushJava.js +57 -0
  104. data/guides/assets/javascripts/syntaxhighlighter/shBrushJavaFX.js +58 -0
  105. data/guides/assets/javascripts/syntaxhighlighter/shBrushPerl.js +72 -0
  106. data/guides/assets/javascripts/syntaxhighlighter/shBrushPhp.js +88 -0
  107. data/guides/assets/javascripts/syntaxhighlighter/shBrushPlain.js +33 -0
  108. data/guides/assets/javascripts/syntaxhighlighter/shBrushPowerShell.js +74 -0
  109. data/guides/assets/javascripts/syntaxhighlighter/shBrushPython.js +64 -0
  110. data/guides/assets/javascripts/syntaxhighlighter/shBrushRuby.js +55 -0
  111. data/guides/assets/javascripts/syntaxhighlighter/shBrushSass.js +94 -0
  112. data/guides/assets/javascripts/syntaxhighlighter/shBrushScala.js +51 -0
  113. data/guides/assets/javascripts/syntaxhighlighter/shBrushSql.js +66 -0
  114. data/guides/assets/javascripts/syntaxhighlighter/shBrushVb.js +56 -0
  115. data/guides/assets/javascripts/syntaxhighlighter/shBrushXml.js +69 -0
  116. data/guides/assets/javascripts/syntaxhighlighter/shCore.js +17 -0
  117. data/guides/assets/stylesheets/fixes.css +16 -0
  118. data/guides/assets/stylesheets/kindle.css +11 -0
  119. data/guides/assets/stylesheets/main.css +713 -0
  120. data/guides/assets/stylesheets/print.css +52 -0
  121. data/guides/assets/stylesheets/reset.css +43 -0
  122. data/guides/assets/stylesheets/responsive-tables.css +50 -0
  123. data/guides/assets/stylesheets/style.css +13 -0
  124. data/guides/assets/stylesheets/syntaxhighlighter/shCore.css +226 -0
  125. data/guides/assets/stylesheets/syntaxhighlighter/shCoreDefault.css +328 -0
  126. data/guides/assets/stylesheets/syntaxhighlighter/shCoreDjango.css +331 -0
  127. data/guides/assets/stylesheets/syntaxhighlighter/shCoreEclipse.css +339 -0
  128. data/guides/assets/stylesheets/syntaxhighlighter/shCoreEmacs.css +324 -0
  129. data/guides/assets/stylesheets/syntaxhighlighter/shCoreFadeToGrey.css +328 -0
  130. data/guides/assets/stylesheets/syntaxhighlighter/shCoreMDUltra.css +324 -0
  131. data/guides/assets/stylesheets/syntaxhighlighter/shCoreMidnight.css +324 -0
  132. data/guides/assets/stylesheets/syntaxhighlighter/shCoreRDark.css +324 -0
  133. data/guides/assets/stylesheets/syntaxhighlighter/shThemeDefault.css +117 -0
  134. data/guides/assets/stylesheets/syntaxhighlighter/shThemeDjango.css +120 -0
  135. data/guides/assets/stylesheets/syntaxhighlighter/shThemeEclipse.css +128 -0
  136. data/guides/assets/stylesheets/syntaxhighlighter/shThemeEmacs.css +113 -0
  137. data/guides/assets/stylesheets/syntaxhighlighter/shThemeFadeToGrey.css +117 -0
  138. data/guides/assets/stylesheets/syntaxhighlighter/shThemeMDUltra.css +113 -0
  139. data/guides/assets/stylesheets/syntaxhighlighter/shThemeMidnight.css +113 -0
  140. data/guides/assets/stylesheets/syntaxhighlighter/shThemeRDark.css +113 -0
  141. data/guides/assets/stylesheets/syntaxhighlighter/shThemeRailsGuides.css +116 -0
  142. data/guides/bug_report_templates/action_controller_gem.rb +47 -0
  143. data/guides/bug_report_templates/action_controller_master.rb +54 -0
  144. data/guides/bug_report_templates/active_record_gem.rb +40 -0
  145. data/guides/bug_report_templates/active_record_master.rb +49 -0
  146. data/guides/rails_guides/generator.rb +248 -0
  147. data/guides/rails_guides/helpers.rb +53 -0
  148. data/guides/rails_guides/indexer.rb +68 -0
  149. data/guides/rails_guides/kindle.rb +119 -0
  150. data/guides/rails_guides/levenshtein.rb +39 -0
  151. data/guides/rails_guides/markdown/renderer.rb +82 -0
  152. data/guides/rails_guides/markdown.rb +167 -0
  153. data/guides/rails_guides.rb +63 -0
  154. data/guides/source/2_2_release_notes.md +435 -0
  155. data/guides/source/2_3_release_notes.md +621 -0
  156. data/guides/source/3_0_release_notes.md +611 -0
  157. data/guides/source/3_1_release_notes.md +559 -0
  158. data/guides/source/3_2_release_notes.md +568 -0
  159. data/guides/source/4_0_release_notes.md +279 -0
  160. data/guides/source/4_1_release_notes.md +730 -0
  161. data/guides/source/4_2_release_notes.md +850 -0
  162. data/guides/source/_license.html.erb +2 -0
  163. data/guides/source/_welcome.html.erb +19 -0
  164. data/guides/source/action_controller_overview.md +1249 -0
  165. data/guides/source/action_mailer_basics.md +752 -0
  166. data/guides/source/action_view_overview.md +1620 -0
  167. data/guides/source/active_job_basics.md +318 -0
  168. data/guides/source/active_model_basics.md +554 -0
  169. data/guides/source/active_record_basics.md +374 -0
  170. data/guides/source/active_record_callbacks.md +413 -0
  171. data/guides/source/active_record_migrations.md +1018 -0
  172. data/guides/source/active_record_postgresql.md +433 -0
  173. data/guides/source/active_record_querying.md +1783 -0
  174. data/guides/source/active_record_validations.md +1178 -0
  175. data/guides/source/active_support_core_extensions.md +3904 -0
  176. data/guides/source/active_support_instrumentation.md +499 -0
  177. data/guides/source/api_documentation_guidelines.md +361 -0
  178. data/guides/source/asset_pipeline.md +1360 -0
  179. data/guides/source/association_basics.md +2236 -0
  180. data/guides/source/caching_with_rails.md +379 -0
  181. data/guides/source/command_line.md +625 -0
  182. data/guides/source/configuring.md +1045 -0
  183. data/guides/source/constant_autoloading_and_reloading.md +1297 -0
  184. data/guides/source/contributing_to_ruby_on_rails.md +624 -0
  185. data/guides/source/credits.html.erb +80 -0
  186. data/guides/source/debugging_rails_applications.md +861 -0
  187. data/guides/source/development_dependencies_install.md +289 -0
  188. data/guides/source/documents.yaml +205 -0
  189. data/guides/source/engines.md +1412 -0
  190. data/guides/source/form_helpers.md +1024 -0
  191. data/guides/source/generators.md +676 -0
  192. data/guides/source/getting_started.md +2085 -0
  193. data/guides/source/i18n.md +1086 -0
  194. data/guides/source/index.html.erb +28 -0
  195. data/guides/source/initialization.md +704 -0
  196. data/guides/source/kindle/copyright.html.erb +1 -0
  197. data/guides/source/kindle/layout.html.erb +27 -0
  198. data/guides/source/kindle/rails_guides.opf.erb +52 -0
  199. data/guides/source/kindle/toc.html.erb +24 -0
  200. data/guides/source/kindle/toc.ncx.erb +64 -0
  201. data/guides/source/kindle/welcome.html.erb +5 -0
  202. data/guides/source/layout.html.erb +143 -0
  203. data/guides/source/layouts_and_rendering.md +1227 -0
  204. data/guides/source/maintenance_policy.md +78 -0
  205. data/guides/source/nested_model_forms.md +228 -0
  206. data/guides/source/plugins.md +444 -0
  207. data/guides/source/rails_application_templates.md +266 -0
  208. data/guides/source/rails_on_rack.md +336 -0
  209. data/guides/source/routing.md +1141 -0
  210. data/guides/source/ruby_on_rails_guides_guidelines.md +127 -0
  211. data/guides/source/security.md +1024 -0
  212. data/guides/source/testing.md +1123 -0
  213. data/guides/source/upgrading_ruby_on_rails.md +1154 -0
  214. data/guides/source/working_with_javascript_in_rails.md +407 -0
  215. data/guides/w3c_validator.rb +97 -0
  216. metadata +290 -44
@@ -0,0 +1,47 @@
1
+ # Activate the gem you are reporting the issue against.
2
+ gem 'rails', '4.0.0'
3
+
4
+ require 'rails'
5
+ require 'action_controller/railtie'
6
+
7
+ class TestApp < Rails::Application
8
+ config.root = File.dirname(__FILE__)
9
+ config.session_store :cookie_store, key: 'cookie_store_key'
10
+ secrets.secret_token = 'secret_token'
11
+ secrets.secret_key_base = 'secret_key_base'
12
+
13
+ config.logger = Logger.new($stdout)
14
+ Rails.logger = config.logger
15
+
16
+ routes.draw do
17
+ get '/' => 'test#index'
18
+ end
19
+ end
20
+
21
+ class TestController < ActionController::Base
22
+ include Rails.application.routes.url_helpers
23
+
24
+ def index
25
+ render text: 'Home'
26
+ end
27
+ end
28
+
29
+ require 'minitest/autorun'
30
+ require 'rack/test'
31
+
32
+ # Ensure backward compatibility with Minitest 4
33
+ Minitest::Test = MiniTest::Unit::TestCase unless defined?(Minitest::Test)
34
+
35
+ class BugTest < Minitest::Test
36
+ include Rack::Test::Methods
37
+
38
+ def test_returns_success
39
+ get '/'
40
+ assert last_response.ok?
41
+ end
42
+
43
+ private
44
+ def app
45
+ Rails.application
46
+ end
47
+ end
@@ -0,0 +1,54 @@
1
+ unless File.exist?('Gemfile')
2
+ File.write('Gemfile', <<-GEMFILE)
3
+ source 'https://rubygems.org'
4
+ gem 'rails', github: 'rails/rails'
5
+ gem 'arel', github: 'rails/arel'
6
+ GEMFILE
7
+
8
+ system 'bundle'
9
+ end
10
+
11
+ require 'bundler'
12
+ Bundler.setup(:default)
13
+
14
+ require 'rails'
15
+ require 'action_controller/railtie'
16
+
17
+ class TestApp < Rails::Application
18
+ config.root = File.dirname(__FILE__)
19
+ config.session_store :cookie_store, key: 'cookie_store_key'
20
+ secrets.secret_token = 'secret_token'
21
+ secrets.secret_key_base = 'secret_key_base'
22
+
23
+ config.logger = Logger.new($stdout)
24
+ Rails.logger = config.logger
25
+
26
+ routes.draw do
27
+ get '/' => 'test#index'
28
+ end
29
+ end
30
+
31
+ class TestController < ActionController::Base
32
+ include Rails.application.routes.url_helpers
33
+
34
+ def index
35
+ render text: 'Home'
36
+ end
37
+ end
38
+
39
+ require 'minitest/autorun'
40
+ require 'rack/test'
41
+
42
+ class BugTest < Minitest::Test
43
+ include Rack::Test::Methods
44
+
45
+ def test_returns_success
46
+ get '/'
47
+ assert last_response.ok?
48
+ end
49
+
50
+ private
51
+ def app
52
+ Rails.application
53
+ end
54
+ end
@@ -0,0 +1,40 @@
1
+ # Activate the gem you are reporting the issue against.
2
+ gem 'activerecord', '4.0.0'
3
+ require 'active_record'
4
+ require 'minitest/autorun'
5
+ require 'logger'
6
+
7
+ # Ensure backward compatibility with Minitest 4
8
+ Minitest::Test = MiniTest::Unit::TestCase unless defined?(Minitest::Test)
9
+
10
+ # This connection will do for database-independent bug reports.
11
+ ActiveRecord::Base.establish_connection(adapter: 'sqlite3', database: ':memory:')
12
+ ActiveRecord::Base.logger = Logger.new(STDOUT)
13
+
14
+ ActiveRecord::Schema.define do
15
+ create_table :posts do |t|
16
+ end
17
+
18
+ create_table :comments do |t|
19
+ t.integer :post_id
20
+ end
21
+ end
22
+
23
+ class Post < ActiveRecord::Base
24
+ has_many :comments
25
+ end
26
+
27
+ class Comment < ActiveRecord::Base
28
+ belongs_to :post
29
+ end
30
+
31
+ class BugTest < Minitest::Test
32
+ def test_association_stuff
33
+ post = Post.create!
34
+ post.comments << Comment.create!
35
+
36
+ assert_equal 1, post.comments.count
37
+ assert_equal 1, Comment.count
38
+ assert_equal post.id, Comment.first.post.id
39
+ end
40
+ end
@@ -0,0 +1,49 @@
1
+ unless File.exist?('Gemfile')
2
+ File.write('Gemfile', <<-GEMFILE)
3
+ source 'https://rubygems.org'
4
+ gem 'rails', github: 'rails/rails'
5
+ gem 'arel', github: 'rails/arel'
6
+ gem 'sqlite3'
7
+ GEMFILE
8
+
9
+ system 'bundle'
10
+ end
11
+
12
+ require 'bundler'
13
+ Bundler.setup(:default)
14
+
15
+ require 'active_record'
16
+ require 'minitest/autorun'
17
+ require 'logger'
18
+
19
+ # This connection will do for database-independent bug reports.
20
+ ActiveRecord::Base.establish_connection(adapter: 'sqlite3', database: ':memory:')
21
+ ActiveRecord::Base.logger = Logger.new(STDOUT)
22
+
23
+ ActiveRecord::Schema.define do
24
+ create_table :posts do |t|
25
+ end
26
+
27
+ create_table :comments do |t|
28
+ t.integer :post_id
29
+ end
30
+ end
31
+
32
+ class Post < ActiveRecord::Base
33
+ has_many :comments
34
+ end
35
+
36
+ class Comment < ActiveRecord::Base
37
+ belongs_to :post
38
+ end
39
+
40
+ class BugTest < Minitest::Test
41
+ def test_association_stuff
42
+ post = Post.create!
43
+ post.comments << Comment.create!
44
+
45
+ assert_equal 1, post.comments.count
46
+ assert_equal 1, Comment.count
47
+ assert_equal post.id, Comment.first.post.id
48
+ end
49
+ end
@@ -0,0 +1,248 @@
1
+ # ---------------------------------------------------------------------------
2
+ #
3
+ # This script generates the guides. It can be invoked via the
4
+ # guides:generate rake task within the guides directory.
5
+ #
6
+ # Guides are taken from the source directory, and the resulting HTML goes into the
7
+ # output directory. Assets are stored under files, and copied to output/files as
8
+ # part of the generation process.
9
+ #
10
+ # Some arguments may be passed via environment variables:
11
+ #
12
+ # WARNINGS
13
+ # If you are writing a guide, please work always with WARNINGS=1. Users can
14
+ # generate the guides, and thus this flag is off by default.
15
+ #
16
+ # Internal links (anchors) are checked. If a reference is broken levenshtein
17
+ # distance is used to suggest an existing one. This is useful since IDs are
18
+ # generated by Markdown from headers and thus edits alter them.
19
+ #
20
+ # Also detects duplicated IDs. They happen if there are headers with the same
21
+ # text. Please do resolve them, if any, so guides are valid XHTML.
22
+ #
23
+ # ALL
24
+ # Set to "1" to force the generation of all guides.
25
+ #
26
+ # ONLY
27
+ # Use ONLY if you want to generate only one or a set of guides. Prefixes are
28
+ # enough:
29
+ #
30
+ # # generates only association_basics.html
31
+ # ONLY=assoc ruby rails_guides.rb
32
+ #
33
+ # Separate many using commas:
34
+ #
35
+ # # generates only association_basics.html and migrations.html
36
+ # ONLY=assoc,migrations ruby rails_guides.rb
37
+ #
38
+ # Note that if you are working on a guide generation will by default process
39
+ # only that one, so ONLY is rarely used nowadays.
40
+ #
41
+ # GUIDES_LANGUAGE
42
+ # Use GUIDES_LANGUAGE when you want to generate translated guides in
43
+ # <tt>source/<GUIDES_LANGUAGE></tt> folder (such as <tt>source/es</tt>).
44
+ # Ignore it when generating English guides.
45
+ #
46
+ # EDGE
47
+ # Set to "1" to indicate generated guides should be marked as edge. This
48
+ # inserts a badge and changes the preamble of the home page.
49
+ #
50
+ # ---------------------------------------------------------------------------
51
+
52
+ require 'set'
53
+ require 'fileutils'
54
+
55
+ require 'active_support/core_ext/string/output_safety'
56
+ require 'active_support/core_ext/object/blank'
57
+ require 'action_controller'
58
+ require 'action_view'
59
+
60
+ require 'rails_guides/indexer'
61
+ require 'rails_guides/helpers'
62
+ require 'rails_guides/levenshtein'
63
+
64
+ module RailsGuides
65
+ class Generator
66
+ attr_reader :guides_dir, :source_dir, :output_dir, :edge, :warnings, :all
67
+
68
+ GUIDES_RE = /\.(?:erb|md)\z/
69
+
70
+ def initialize(output=nil)
71
+ set_flags_from_environment
72
+
73
+ if kindle?
74
+ check_for_kindlegen
75
+ register_kindle_mime_types
76
+ end
77
+
78
+ initialize_dirs(output)
79
+ create_output_dir_if_needed
80
+ end
81
+
82
+ def set_flags_from_environment
83
+ @edge = ENV['EDGE'] == '1'
84
+ @warnings = ENV['WARNINGS'] == '1'
85
+ @all = ENV['ALL'] == '1'
86
+ @kindle = ENV['KINDLE'] == '1'
87
+ @version = ENV['RAILS_VERSION'] || 'local'
88
+ @lang = ENV['GUIDES_LANGUAGE']
89
+ end
90
+
91
+ def register_kindle_mime_types
92
+ Mime::Type.register_alias("application/xml", :opf, %w(opf))
93
+ Mime::Type.register_alias("application/xml", :ncx, %w(ncx))
94
+ end
95
+
96
+ def generate
97
+ generate_guides
98
+ copy_assets
99
+ generate_mobi if kindle?
100
+ end
101
+
102
+ private
103
+
104
+ def kindle?
105
+ @kindle
106
+ end
107
+
108
+ def check_for_kindlegen
109
+ if `which kindlegen`.blank?
110
+ raise "Can't create a kindle version without `kindlegen`."
111
+ end
112
+ end
113
+
114
+ def generate_mobi
115
+ require 'rails_guides/kindle'
116
+ out = "#{output_dir}/kindlegen.out"
117
+ Kindle.generate(output_dir, mobi, out)
118
+ puts "(kindlegen log at #{out})."
119
+ end
120
+
121
+ def mobi
122
+ "ruby_on_rails_guides_#@version%s.mobi" % (@lang.present? ? ".#@lang" : '')
123
+ end
124
+
125
+ def initialize_dirs(output)
126
+ @guides_dir = File.join(File.dirname(__FILE__), '..')
127
+ @source_dir = "#@guides_dir/source/#@lang"
128
+ @output_dir = if output
129
+ output
130
+ elsif kindle?
131
+ "#@guides_dir/output/kindle/#@lang"
132
+ else
133
+ "#@guides_dir/output/#@lang"
134
+ end.sub(%r</$>, '')
135
+ end
136
+
137
+ def create_output_dir_if_needed
138
+ FileUtils.mkdir_p(output_dir)
139
+ end
140
+
141
+ def generate_guides
142
+ guides_to_generate.each do |guide|
143
+ output_file = output_file_for(guide)
144
+ generate_guide(guide, output_file) if generate?(guide, output_file)
145
+ end
146
+ end
147
+
148
+ def guides_to_generate
149
+ guides = Dir.entries(source_dir).grep(GUIDES_RE)
150
+
151
+ if kindle?
152
+ Dir.entries("#{source_dir}/kindle").grep(GUIDES_RE).map do |entry|
153
+ next if entry == 'KINDLE.md'
154
+ guides << "kindle/#{entry}"
155
+ end
156
+ end
157
+
158
+ ENV.key?('ONLY') ? select_only(guides) : guides
159
+ end
160
+
161
+ def select_only(guides)
162
+ prefixes = ENV['ONLY'].split(",").map(&:strip)
163
+ guides.select do |guide|
164
+ prefixes.any? { |p| guide.start_with?(p) || guide.start_with?("kindle") }
165
+ end
166
+ end
167
+
168
+ def copy_assets
169
+ FileUtils.cp_r(Dir.glob("#{guides_dir}/assets/*"), output_dir)
170
+ end
171
+
172
+ def output_file_for(guide)
173
+ if guide.end_with?('.md')
174
+ guide.sub(/md\z/, 'html')
175
+ else
176
+ guide.sub(/\.erb\z/, '')
177
+ end
178
+ end
179
+
180
+ def output_path_for(output_file)
181
+ File.join(output_dir, File.basename(output_file))
182
+ end
183
+
184
+ def generate?(source_file, output_file)
185
+ fin = File.join(source_dir, source_file)
186
+ fout = output_path_for(output_file)
187
+ all || !File.exist?(fout) || File.mtime(fout) < File.mtime(fin)
188
+ end
189
+
190
+ def generate_guide(guide, output_file)
191
+ output_path = output_path_for(output_file)
192
+ puts "Generating #{guide} as #{output_file}"
193
+ layout = kindle? ? 'kindle/layout' : 'layout'
194
+
195
+ File.open(output_path, 'w') do |f|
196
+ view = ActionView::Base.new(source_dir, :edge => @edge, :version => @version, :mobi => "kindle/#{mobi}")
197
+ view.extend(Helpers)
198
+
199
+ if guide =~ /\.(\w+)\.erb$/
200
+ # Generate the special pages like the home.
201
+ # Passing a template handler in the template name is deprecated. So pass the file name without the extension.
202
+ result = view.render(:layout => layout, :formats => [$1], :file => $`)
203
+ else
204
+ body = File.read(File.join(source_dir, guide))
205
+ result = RailsGuides::Markdown.new(view, layout).render(body)
206
+
207
+ warn_about_broken_links(result) if @warnings
208
+ end
209
+
210
+ f.write(result)
211
+ end
212
+ end
213
+
214
+ def warn_about_broken_links(html)
215
+ anchors = extract_anchors(html)
216
+ check_fragment_identifiers(html, anchors)
217
+ end
218
+
219
+ def extract_anchors(html)
220
+ # Markdown generates headers with IDs computed from titles.
221
+ anchors = Set.new
222
+ html.scan(/<h\d\s+id="([^"]+)/).flatten.each do |anchor|
223
+ if anchors.member?(anchor)
224
+ puts "*** DUPLICATE ID: #{anchor}, please make sure that there're no headings with the same name at the same level."
225
+ else
226
+ anchors << anchor
227
+ end
228
+ end
229
+
230
+ # Footnotes.
231
+ anchors += Set.new(html.scan(/<p\s+class="footnote"\s+id="([^"]+)/).flatten)
232
+ anchors += Set.new(html.scan(/<sup\s+class="footnote"\s+id="([^"]+)/).flatten)
233
+ return anchors
234
+ end
235
+
236
+ def check_fragment_identifiers(html, anchors)
237
+ html.scan(/<a\s+href="#([^"]+)/).flatten.each do |fragment_identifier|
238
+ next if fragment_identifier == 'mainCol' # in layout, jumps to some DIV
239
+ unless anchors.member?(fragment_identifier)
240
+ guess = anchors.min { |a, b|
241
+ Levenshtein.distance(fragment_identifier, a) <=> Levenshtein.distance(fragment_identifier, b)
242
+ }
243
+ puts "*** BROKEN LINK: ##{fragment_identifier}, perhaps you meant ##{guess}."
244
+ end
245
+ end
246
+ end
247
+ end
248
+ end
@@ -0,0 +1,53 @@
1
+ require 'yaml'
2
+
3
+ module RailsGuides
4
+ module Helpers
5
+ def guide(name, url, options = {}, &block)
6
+ link = content_tag(:a, :href => url) { name }
7
+ result = content_tag(:dt, link)
8
+
9
+ if options[:work_in_progress]
10
+ result << content_tag(:dd, 'Work in progress', :class => 'work-in-progress')
11
+ end
12
+
13
+ result << content_tag(:dd, capture(&block))
14
+ result
15
+ end
16
+
17
+ def documents_by_section
18
+ @documents_by_section ||= YAML.load_file(File.expand_path('../../source/documents.yaml', __FILE__))
19
+ end
20
+
21
+ def documents_flat
22
+ documents_by_section.flat_map {|section| section['documents']}
23
+ end
24
+
25
+ def finished_documents(documents)
26
+ documents.reject { |document| document['work_in_progress'] }
27
+ end
28
+
29
+ def docs_for_menu(position=nil)
30
+ if position.nil?
31
+ documents_by_section
32
+ elsif position == 'L'
33
+ documents_by_section.to(3)
34
+ else
35
+ documents_by_section.from(4)
36
+ end
37
+ end
38
+
39
+ def author(name, nick, image = 'credits_pic_blank.gif', &block)
40
+ image = "images/#{image}"
41
+
42
+ result = tag(:img, :src => image, :class => 'left pic', :alt => name, :width => 91, :height => 91)
43
+ result << content_tag(:h3, name)
44
+ result << content_tag(:p, capture(&block))
45
+ content_tag(:div, result, :class => 'clearfix', :id => nick)
46
+ end
47
+
48
+ def code(&block)
49
+ c = capture(&block)
50
+ content_tag(:code, c)
51
+ end
52
+ end
53
+ end
@@ -0,0 +1,68 @@
1
+ require 'active_support/core_ext/object/blank'
2
+ require 'active_support/core_ext/string/inflections'
3
+
4
+ module RailsGuides
5
+ class Indexer
6
+ attr_reader :body, :result, :warnings, :level_hash
7
+
8
+ def initialize(body, warnings)
9
+ @body = body
10
+ @result = @body.dup
11
+ @warnings = warnings
12
+ end
13
+
14
+ def index
15
+ @level_hash = process(body)
16
+ end
17
+
18
+ private
19
+
20
+ def process(string, current_level=3, counters=[1])
21
+ s = StringScanner.new(string)
22
+
23
+ level_hash = {}
24
+
25
+ while !s.eos?
26
+ re = %r{^h(\d)(?:\((#.*?)\))?\s*\.\s*(.*)$}
27
+ s.match?(re)
28
+ if matched = s.matched
29
+ matched =~ re
30
+ level, idx, title = $1.to_i, $2, $3.strip
31
+
32
+ if level < current_level
33
+ # This is needed. Go figure.
34
+ return level_hash
35
+ elsif level == current_level
36
+ index = counters.join(".")
37
+ idx ||= '#' + title_to_idx(title)
38
+
39
+ raise "Parsing Fail" unless @result.sub!(matched, "h#{level}(#{idx}). #{index} #{title}")
40
+
41
+ key = {
42
+ :title => title,
43
+ :id => idx
44
+ }
45
+ # Recurse
46
+ counters << 1
47
+ level_hash[key] = process(s.post_match, current_level + 1, counters)
48
+ counters.pop
49
+
50
+ # Increment the current level
51
+ last = counters.pop
52
+ counters << last + 1
53
+ end
54
+ end
55
+ s.getch
56
+ end
57
+ level_hash
58
+ end
59
+
60
+ def title_to_idx(title)
61
+ idx = title.strip.parameterize.sub(/^\d+/, '')
62
+ if warnings && idx.blank?
63
+ puts "BLANK ID: please put an explicit ID for section #{title}, as in h5(#my-id)"
64
+ end
65
+ idx
66
+ end
67
+ end
68
+ end
@@ -0,0 +1,119 @@
1
+ #!/usr/bin/env ruby
2
+
3
+ unless `which kindlerb`
4
+ abort "Please gem install kindlerb"
5
+ end
6
+
7
+ require 'nokogiri'
8
+ require 'fileutils'
9
+ require 'yaml'
10
+ require 'date'
11
+
12
+ module Kindle
13
+ extend self
14
+
15
+ def generate(output_dir, mobi_outfile, logfile)
16
+ output_dir = File.absolute_path(output_dir)
17
+ Dir.chdir output_dir do
18
+ puts "=> Using output dir: #{output_dir}"
19
+ puts "=> Arranging html pages in document order"
20
+ toc = File.read("toc.ncx")
21
+ doc = Nokogiri::XML(toc).xpath("//ncx:content", 'ncx' => "http://www.daisy.org/z3986/2005/ncx/")
22
+ html_pages = doc.select {|c| c[:src]}.map {|c| c[:src]}.uniq
23
+
24
+ generate_front_matter(html_pages)
25
+
26
+ generate_sections(html_pages)
27
+
28
+ generate_document_metadata(mobi_outfile)
29
+
30
+ puts "Creating MOBI document with kindlegen. This make take a while."
31
+ cmd = "kindlerb . > #{File.absolute_path logfile} 2>&1"
32
+ puts cmd
33
+ system(cmd)
34
+ puts "MOBI document generated at #{File.expand_path(mobi_outfile, output_dir)}"
35
+ end
36
+ end
37
+
38
+ def generate_front_matter(html_pages)
39
+ frontmatter = []
40
+ html_pages.delete_if {|x|
41
+ if x =~ /(toc|welcome|credits|copyright).html/
42
+ frontmatter << x unless x =~ /toc/
43
+ true
44
+ end
45
+ }
46
+ html = frontmatter.map {|x|
47
+ Nokogiri::HTML(File.open(x)).at("body").inner_html
48
+ }.join("\n")
49
+
50
+ fdoc = Nokogiri::HTML(html)
51
+ fdoc.search("h3").each do |h3|
52
+ h3.name = 'h4'
53
+ end
54
+ fdoc.search("h2").each do |h2|
55
+ h2.name = 'h3'
56
+ h2['id'] = h2.inner_text.gsub(/\s/, '-')
57
+ end
58
+ add_head_section fdoc, "Front Matter"
59
+ File.open("frontmatter.html",'w') {|f| f.puts fdoc.to_html}
60
+ html_pages.unshift "frontmatter.html"
61
+ end
62
+
63
+ def generate_sections(html_pages)
64
+ FileUtils::rm_rf("sections/")
65
+ html_pages.each_with_index do |page, section_idx|
66
+ FileUtils::mkdir_p("sections/%03d" % section_idx)
67
+ doc = Nokogiri::HTML(File.open(page))
68
+ title = doc.at("title").inner_text.gsub("Ruby on Rails Guides: ", '')
69
+ title = page.capitalize.gsub('.html', '') if title.strip == ''
70
+ File.open("sections/%03d/_section.txt" % section_idx, 'w') {|f| f.puts title}
71
+ doc.xpath("//h3[@id]").each_with_index do |h3,item_idx|
72
+ subsection = h3.inner_text
73
+ content = h3.xpath("./following-sibling::*").take_while {|x| x.name != "h3"}.map {|x| x.to_html}
74
+ item = Nokogiri::HTML(h3.to_html + content.join("\n"))
75
+ item_path = "sections/%03d/%03d.html" % [section_idx, item_idx]
76
+ add_head_section(item, subsection)
77
+ item.search("img").each do |img|
78
+ img['src'] = "#{Dir.pwd}/#{img['src']}"
79
+ end
80
+ item.xpath("//li/p").each {|p| p.swap(p.children); p.remove}
81
+ File.open(item_path, 'w') {|f| f.puts item.to_html}
82
+ end
83
+ end
84
+ end
85
+
86
+ def generate_document_metadata(mobi_outfile)
87
+ puts "=> Generating _document.yml"
88
+ x = Nokogiri::XML(File.open("rails_guides.opf")).remove_namespaces!
89
+ cover_jpg = "#{Dir.pwd}/images/rails_guides_kindle_cover.jpg"
90
+ cover_gif = cover_jpg.sub(/jpg$/, 'gif')
91
+ puts `convert #{cover_jpg} #{cover_gif}`
92
+ document = {
93
+ 'doc_uuid' => x.at("package")['unique-identifier'],
94
+ 'title' => x.at("title").inner_text.gsub(/\(.*$/, " v2"),
95
+ 'publisher' => x.at("publisher").inner_text,
96
+ 'author' => x.at("creator").inner_text,
97
+ 'subject' => x.at("subject").inner_text,
98
+ 'date' => x.at("date").inner_text,
99
+ 'cover' => cover_gif,
100
+ 'masthead' => nil,
101
+ 'mobi_outfile' => mobi_outfile
102
+ }
103
+ puts document.to_yaml
104
+ File.open("_document.yml", 'w'){|f| f.puts document.to_yaml}
105
+ end
106
+
107
+ def add_head_section(doc, title)
108
+ head = Nokogiri::XML::Node.new "head", doc
109
+ title_node = Nokogiri::XML::Node.new "title", doc
110
+ title_node.content = title
111
+ title_node.parent = head
112
+ css = Nokogiri::XML::Node.new "link", doc
113
+ css['rel'] = 'stylesheet'
114
+ css['type'] = 'text/css'
115
+ css['href'] = "#{Dir.pwd}/stylesheets/kindle.css"
116
+ css.parent = head
117
+ doc.at("body").before head
118
+ end
119
+ end