openstax_kitchen 1.0.0 → 4.0.0

Sign up to get free protection for your applications and to get access to all the features.
Files changed (182) hide show
  1. checksums.yaml +4 -4
  2. data/.devcontainer/devcontainer.json +37 -17
  3. data/.github/config.yml +14 -0
  4. data/.github/workflows/tests.yml +5 -15
  5. data/.gitignore +2 -2
  6. data/.inch.yml +6 -0
  7. data/.rubocop.yml +65 -0
  8. data/CHANGELOG.md +85 -2
  9. data/Gemfile +5 -3
  10. data/Gemfile.lock +66 -18
  11. data/README.md +69 -15
  12. data/Rakefile +5 -3
  13. data/bin/console +4 -3
  14. data/docker/Dockerfile +36 -0
  15. data/docker/Dockerfile.ci +10 -0
  16. data/docker/bash +5 -1
  17. data/docker/build +10 -0
  18. data/docker/ci +15 -0
  19. data/docker/run +9 -0
  20. data/docker/tag_and_push_latest +17 -0
  21. data/lefthook.yml +6 -0
  22. data/lib/kitchen/ancestor.rb +38 -1
  23. data/lib/kitchen/book_document.rb +20 -2
  24. data/lib/kitchen/book_element.rb +40 -5
  25. data/lib/kitchen/book_element_enumerator.rb +4 -0
  26. data/lib/kitchen/book_recipe.rb +15 -1
  27. data/lib/kitchen/chapter_element.rb +43 -6
  28. data/lib/kitchen/chapter_element_enumerator.rb +9 -1
  29. data/lib/kitchen/clipboard.rb +35 -4
  30. data/lib/kitchen/composite_chapter_element.rb +21 -6
  31. data/lib/kitchen/composite_page_element.rb +35 -7
  32. data/lib/kitchen/composite_page_element_enumerator.rb +9 -1
  33. data/lib/kitchen/config.rb +20 -6
  34. data/lib/kitchen/counter.rb +9 -2
  35. data/lib/kitchen/debug/print_recipe_error.rb +53 -35
  36. data/lib/kitchen/directions/.rubocop.yml +22 -0
  37. data/lib/kitchen/directions/bake_appendix.rb +4 -4
  38. data/lib/kitchen/directions/bake_chapter_glossary/main.rb +18 -0
  39. data/lib/kitchen/directions/bake_chapter_glossary/v1.rb +30 -0
  40. data/lib/kitchen/directions/bake_chapter_introductions.rb +7 -7
  41. data/lib/kitchen/directions/bake_chapter_key_concepts/main.rb +16 -0
  42. data/lib/kitchen/directions/bake_chapter_key_concepts/v1.rb +35 -0
  43. data/lib/kitchen/directions/bake_chapter_key_equations.rb +30 -20
  44. data/lib/kitchen/directions/bake_chapter_section_exercises/main.rb +11 -0
  45. data/lib/kitchen/directions/bake_chapter_section_exercises/v1.rb +28 -0
  46. data/lib/kitchen/directions/bake_chapter_summary.rb +45 -36
  47. data/lib/kitchen/directions/bake_chapter_title/main.rb +11 -0
  48. data/lib/kitchen/directions/bake_chapter_title/v1.rb +24 -0
  49. data/lib/kitchen/directions/bake_checkpoint.rb +44 -0
  50. data/lib/kitchen/directions/bake_composite_chapters.rb +14 -0
  51. data/lib/kitchen/directions/bake_composite_pages.rb +2 -2
  52. data/lib/kitchen/directions/bake_equations.rb +37 -0
  53. data/lib/kitchen/directions/bake_example.rb +39 -11
  54. data/lib/kitchen/directions/bake_figure.rb +8 -5
  55. data/lib/kitchen/directions/bake_first_elements.rb +16 -0
  56. data/lib/kitchen/directions/bake_footnotes/main.rb +2 -2
  57. data/lib/kitchen/directions/bake_footnotes/v1.rb +6 -5
  58. data/lib/kitchen/directions/bake_free_response/free_response.xhtml.erb +10 -0
  59. data/lib/kitchen/directions/bake_free_response/main.rb +11 -0
  60. data/lib/kitchen/directions/bake_free_response/v1.rb +29 -0
  61. data/lib/kitchen/directions/bake_further_research.rb +59 -0
  62. data/lib/kitchen/directions/bake_index/main.rb +2 -2
  63. data/lib/kitchen/directions/bake_index/v1.rb +46 -18
  64. data/lib/kitchen/directions/bake_link_placeholders.rb +24 -0
  65. data/lib/kitchen/directions/bake_math_in_paragraph.rb +5 -3
  66. data/lib/kitchen/directions/bake_non_introduction_pages.rb +26 -0
  67. data/lib/kitchen/directions/bake_notes/bake_autotitled_notes.rb +29 -0
  68. data/lib/kitchen/directions/bake_notes/bake_note_subtitle.rb +22 -0
  69. data/lib/kitchen/directions/bake_notes/bake_numbered_notes.rb +51 -0
  70. data/lib/kitchen/directions/bake_notes/bake_unclassified_notes.rb +30 -0
  71. data/lib/kitchen/directions/bake_numbered_exercise/main.rb +15 -0
  72. data/lib/kitchen/directions/bake_numbered_exercise/v1.rb +47 -0
  73. data/lib/kitchen/directions/bake_numbered_table/main.rb +4 -4
  74. data/lib/kitchen/directions/bake_numbered_table/v1.rb +37 -18
  75. data/lib/kitchen/directions/bake_page_abstracts.rb +30 -0
  76. data/lib/kitchen/directions/bake_preface/main.rb +11 -0
  77. data/lib/kitchen/directions/bake_preface/v1.rb +18 -0
  78. data/lib/kitchen/directions/bake_references/main.rb +16 -0
  79. data/lib/kitchen/directions/bake_references/v1.rb +48 -0
  80. data/lib/kitchen/directions/bake_stepwise.rb +8 -12
  81. data/lib/kitchen/directions/bake_suggested_reading.rb +31 -0
  82. data/lib/kitchen/directions/bake_theorem/main.rb +11 -0
  83. data/lib/kitchen/directions/bake_theorem/v1.rb +28 -0
  84. data/lib/kitchen/directions/bake_toc.rb +49 -22
  85. data/lib/kitchen/directions/bake_unit_title/main.rb +11 -0
  86. data/lib/kitchen/directions/bake_unit_title/v1.rb +23 -0
  87. data/lib/kitchen/directions/bake_unnumbered_tables.rb +7 -5
  88. data/lib/kitchen/directions/book_answer_key_container/eob_solutions_container.xhtml.erb +9 -0
  89. data/lib/kitchen/directions/book_answer_key_container/main.rb +11 -0
  90. data/lib/kitchen/directions/book_answer_key_container/v1.rb +13 -0
  91. data/lib/kitchen/directions/chapter_review_container/chapter_review.xhtml.erb +9 -0
  92. data/lib/kitchen/directions/chapter_review_container/main.rb +11 -0
  93. data/lib/kitchen/directions/chapter_review_container/v1.rb +13 -0
  94. data/lib/kitchen/directions/eoc_section_title_link_snippet.rb +20 -0
  95. data/lib/kitchen/directions/move_exercises_to_eoc/main.rb +27 -0
  96. data/lib/kitchen/directions/move_exercises_to_eoc/v1.rb +36 -0
  97. data/lib/kitchen/directions/move_exercises_to_eoc/v2.rb +49 -0
  98. data/lib/kitchen/directions/move_solutions_to_answer_key/main.rb +14 -0
  99. data/lib/kitchen/directions/move_solutions_to_answer_key/strategies/american_government.rb +19 -0
  100. data/lib/kitchen/directions/move_solutions_to_answer_key/strategies/calculus.rb +41 -0
  101. data/lib/kitchen/directions/move_solutions_to_answer_key/strategies/uphysics.rb +63 -0
  102. data/lib/kitchen/directions/move_solutions_to_answer_key/v1.rb +34 -0
  103. data/lib/kitchen/directions/move_title_text_into_span.rb +2 -2
  104. data/lib/kitchen/document.rb +83 -13
  105. data/lib/kitchen/element.rb +20 -3
  106. data/lib/kitchen/element_base.rb +373 -63
  107. data/lib/kitchen/element_enumerator.rb +8 -0
  108. data/lib/kitchen/element_enumerator_base.rb +297 -28
  109. data/lib/kitchen/element_enumerator_factory.rb +64 -53
  110. data/lib/kitchen/element_factory.rb +27 -12
  111. data/lib/kitchen/errors.rb +5 -0
  112. data/lib/kitchen/example_element.rb +21 -6
  113. data/lib/kitchen/example_element_enumerator.rb +9 -1
  114. data/lib/kitchen/exercise_element.rb +42 -0
  115. data/lib/kitchen/exercise_element_enumerator.rb +21 -0
  116. data/lib/kitchen/figure_element.rb +36 -5
  117. data/lib/kitchen/figure_element_enumerator.rb +9 -1
  118. data/lib/kitchen/metadata_element.rb +34 -0
  119. data/lib/kitchen/metadata_element_enumerator.rb +21 -0
  120. data/lib/kitchen/mixins/block_error_if.rb +24 -4
  121. data/lib/kitchen/note_element.rb +48 -20
  122. data/lib/kitchen/note_element_enumerator.rb +9 -1
  123. data/lib/kitchen/oven.rb +66 -15
  124. data/lib/kitchen/page_element.rb +84 -14
  125. data/lib/kitchen/page_element_enumerator.rb +9 -1
  126. data/lib/kitchen/pantry.rb +28 -1
  127. data/lib/kitchen/patches/nokogiri.rb +19 -2
  128. data/lib/kitchen/patches/renderable.rb +9 -3
  129. data/lib/kitchen/patches/string.rb +8 -0
  130. data/lib/kitchen/recipe.rb +69 -32
  131. data/lib/kitchen/reference_element.rb +27 -0
  132. data/lib/kitchen/references_element_enumerator.rb +20 -0
  133. data/lib/kitchen/search_history.rb +43 -4
  134. data/lib/kitchen/search_query.rb +106 -0
  135. data/lib/kitchen/selector.rb +24 -0
  136. data/lib/kitchen/selectors/base.rb +65 -0
  137. data/lib/kitchen/selectors/standard_1.rb +21 -0
  138. data/lib/kitchen/table_element.rb +55 -7
  139. data/lib/kitchen/table_element_enumerator.rb +9 -1
  140. data/lib/kitchen/templates/eob_section_title_template.xhtml.erb +10 -0
  141. data/lib/kitchen/templates/eoc_section_title_template.xhtml.erb +10 -0
  142. data/lib/kitchen/term_element.rb +15 -4
  143. data/lib/kitchen/term_element_enumerator.rb +9 -1
  144. data/lib/kitchen/transliterations.rb +7 -5
  145. data/lib/kitchen/type_casting_element_enumerator.rb +17 -1
  146. data/lib/kitchen/unit_element.rb +45 -0
  147. data/lib/kitchen/unit_element_enumerator.rb +20 -0
  148. data/lib/kitchen/utils.rb +10 -13
  149. data/lib/kitchen/version.rb +5 -1
  150. data/lib/locales/en.yml +18 -7
  151. data/lib/locales/pl.yml +24 -0
  152. data/lib/openstax_kitchen.rb +59 -0
  153. data/openstax_kitchen.gemspec +26 -20
  154. data/tutorials/00/solution1.rb +9 -0
  155. data/tutorials/00/solution2.rb +8 -0
  156. data/tutorials/01/solution1.rb +18 -0
  157. data/tutorials/01/solution2.rb +26 -0
  158. data/tutorials/02/solution1.rb +31 -0
  159. data/tutorials/03/{solution_1.rb → solution1.rb} +6 -4
  160. data/tutorials/03/solution2.rb +18 -0
  161. data/tutorials/04/{solution_1.rb → solution1.rb} +4 -2
  162. data/tutorials/04/{solution_2.rb → solution2.rb} +6 -4
  163. data/tutorials/05/solution1.rb +11 -0
  164. data/tutorials/check_it +16 -15
  165. data/tutorials/setup_my_recipes +7 -6
  166. metadata +148 -27
  167. data/Dockerfile +0 -19
  168. data/bin/normalize +0 -79
  169. data/books/chemistry2e/bake.rb +0 -133
  170. data/docker-compose.yml +0 -12
  171. data/docker/entrypoint +0 -9
  172. data/lib/kitchen.rb +0 -57
  173. data/lib/kitchen/directions/bake_chapter_glossary.rb +0 -34
  174. data/lib/kitchen/directions/bake_exercises.rb +0 -164
  175. data/lib/kitchen/directions/bake_notes.rb +0 -58
  176. data/tutorials/00/solution_1.rb +0 -7
  177. data/tutorials/00/solution_2.rb +0 -6
  178. data/tutorials/01/solution_1.rb +0 -16
  179. data/tutorials/01/solution_2.rb +0 -24
  180. data/tutorials/02/solution_1.rb +0 -29
  181. data/tutorials/03/solution_2.rb +0 -15
  182. data/tutorials/05/solution_1.rb +0 -9
@@ -1,9 +1,17 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Kitchen
4
+ # An enumerator for note elements
5
+ #
2
6
  class NoteElementEnumerator < ElementEnumeratorBase
3
7
 
8
+ # Returns a factory for this enumerator
9
+ #
10
+ # @return [ElementEnumeratorFactory]
11
+ #
4
12
  def self.factory
5
13
  ElementEnumeratorFactory.new(
6
- default_css_or_xpath: "div[data-type='note']", # TODO get from config?
14
+ default_css_or_xpath: Selector.named(:note),
7
15
  sub_element_class: NoteElement,
8
16
  enumerator_class: self
9
17
  )
data/lib/kitchen/oven.rb CHANGED
@@ -1,11 +1,18 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Kitchen
4
+ # A class for baking documents according to the instructions in recipes
5
+ #
2
6
  class Oven
3
7
 
4
- def self.bake(input_file:,
5
- config_file: nil,
6
- recipes:,
7
- output_file:)
8
-
8
+ # Bakes an input file using a recipe to produce an output file
9
+ #
10
+ # @param input_file [String] the path to the input file
11
+ # @param config_file [String] the path to the configuration file
12
+ # @param recipes [Array<Recipe>] an array of recipes with which to bake the document
13
+ # @param output_file [String] the path to the output file
14
+ #
15
+ def self.bake(input_file:, recipes:, output_file:, config_file: nil)
9
16
  profile = BakeProfile.new
10
17
  profile.started!
11
18
 
@@ -27,32 +34,76 @@ module Kitchen
27
34
  end
28
35
  profile.baked!
29
36
 
30
- File.open(output_file, "w") do |f|
31
- f.write doc.to_xhtml(indent:2)
37
+ File.open(output_file, 'w') do |f|
38
+ f.write doc.to_xhtml(indent: 2)
32
39
  end
33
40
  profile.written!
34
41
 
35
42
  profile
36
43
  end
37
44
 
45
+ # Stats on baking
46
+ #
38
47
  class BakeProfile
48
+ # Record that baking has started
39
49
  def started!; @started_at = Time.now; end
50
+
51
+ # Record that the input file has been opened
40
52
  def opened!; @opened_at = Time.now; end
53
+
54
+ # Record that the input file has been parsed
41
55
  def parsed!; @parsed_at = Time.now; end
56
+
57
+ # Record that the input file has been baked
42
58
  def baked!; @baked_at = Time.now; end
59
+
60
+ # Record that the output file has been written
43
61
  def written!; @written_at = Time.now; end
44
62
 
45
- def open_seconds; @opened_at - @started_at; end
46
- def parse_seconds; @parsed_at - @opened_at; end
47
- def bake_seconds; @baked_at - @parsed_at; end
48
- def write_seconds; @written_at - @baked_at; end
63
+ # Return the number of seconds it took to open the input file or nil if this
64
+ # info isn't available.
65
+ # @return [Float, nil]
66
+ def open_seconds
67
+ @opened_at - @started_at
68
+ rescue NoMethodError
69
+ nil
70
+ end
71
+
72
+ # Return the number of seconds it took to parse the input file after opening or
73
+ # nil if this info isn't available.
74
+ # @return [Float, nil]
75
+ def parse_seconds
76
+ @parsed_at - @opened_at
77
+ rescue NoMethodError
78
+ nil
79
+ end
80
+
81
+ # Return the number of seconds it took to bake the parsed file or nil if this
82
+ # info isn't available.
83
+ # @return [Float, nil]
84
+ def bake_seconds
85
+ @baked_at - @parsed_at
86
+ rescue NoMethodError
87
+ nil
88
+ end
89
+
90
+ # Return the number of seconds it took to write the baked file or nil if this
91
+ # info isn't available.
92
+ # @return [Float, nil]
93
+ def write_seconds
94
+ @written_at - @baked_at
95
+ rescue NoMethodError
96
+ nil
97
+ end
49
98
 
99
+ # Return the profile stats as a string
100
+ # @return [String]
50
101
  def to_s
51
102
  <<~STRING
52
- Open: #{open_seconds} s
53
- Parse: #{parse_seconds} s
54
- Bake: #{bake_seconds} s
55
- Write: #{write_seconds} s
103
+ Open: #{open_seconds || '??'} s
104
+ Parse: #{parse_seconds || '??'} s
105
+ Bake: #{bake_seconds || '??'} s
106
+ Write: #{write_seconds || '??'} s
56
107
  STRING
57
108
  end
58
109
  end
@@ -1,50 +1,120 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Kitchen
4
+ # An element for a page
5
+ #
2
6
  class PageElement < ElementBase
3
7
 
8
+ # Creates a new +PageElement+
9
+ #
10
+ # @param node [Nokogiri::XML::Node] the node this element wraps
11
+ # @param document [Document] this element's document
12
+ #
4
13
  def initialize(node:, document: nil)
5
14
  super(node: node,
6
15
  document: document,
7
- enumerator_class: PageElementEnumerator,
8
- short_type: :page)
16
+ enumerator_class: PageElementEnumerator)
17
+ end
18
+
19
+ # Returns the short type
20
+ # @return [Symbol]
21
+ #
22
+ def self.short_type
23
+ :page
9
24
  end
10
25
 
26
+ # Returns the title element. This method is aware that the title of the
27
+ # introduction page moves during the baking process.
28
+ #
29
+ # @raise [ElementNotFoundError] if no matching element is found
30
+ # @return [Element]
31
+ #
11
32
  def title
12
33
  # The selector for intro titles changes during the baking process
13
- first!(is_introduction? ?
14
- selectors.title_in_introduction_page :
15
- selectors.title_in_page)
34
+ first!(is_introduction? ? selectors.title_in_introduction_page : selectors.title_in_page)
35
+ end
36
+
37
+ # Returns the title's text regardless of whether the title has been baked
38
+ #
39
+ # @return [String]
40
+ #
41
+ def title_text
42
+ title.children.one? ? title.text : title.first('.os-text').text
43
+ end
44
+
45
+ # Returns an enumerator for titles.
46
+ #
47
+ # @return [ElementEnumerator]
48
+ #
49
+ def titles
50
+ search("div[data-type='document-title']")
16
51
  end
17
52
 
53
+ # Returns true if this page is an introduction
54
+ #
55
+ # @return [Boolean]
56
+ #
18
57
  def is_introduction?
19
- has_class?("introduction")
58
+ has_class?('introduction')
20
59
  end
21
60
 
61
+ # Returns true if this page is a preface
62
+ #
63
+ # @return [Boolean]
64
+ #
22
65
  def is_preface?
23
- has_class?("preface")
66
+ has_class?('preface')
24
67
  end
25
68
 
69
+ # Returns true if this page is an appendix
70
+ #
71
+ # @return [Boolean]
72
+ #
26
73
  def is_appendix?
27
- has_class?("appendix")
74
+ has_class?('appendix')
28
75
  end
29
76
 
77
+ # Returns the metadata element.
78
+ #
79
+ # @raise [ElementNotFoundError] if no matching element is found
80
+ # @return [Element]
81
+ #
30
82
  def metadata
31
83
  first!("div[data-type='metadata']")
32
84
  end
33
85
 
86
+ # Returns the summary element.
87
+ #
88
+ # @raise [ElementNotFoundError] if no matching element is found
89
+ # @return [Element]
90
+ #
34
91
  def summary
35
- first!("section.summary")
92
+ first!(selectors.page_summary)
36
93
  end
37
94
 
95
+ # Returns the exercises element.
96
+ #
97
+ # @raise [ElementNotFoundError] if no matching element is found
98
+ # @return [Element]
99
+ #
38
100
  def exercises
39
- first!("section.exercises")
101
+ first!('section.exercises')
40
102
  end
41
103
 
42
- def exercises_section
43
- search("")
104
+ # Returns the key concepts
105
+ #
106
+ # @return [Element]
107
+ #
108
+ def key_concepts
109
+ search('section.key-concepts')
44
110
  end
45
111
 
46
- def self.is_the_element_class_for?(node)
47
- node['data-type'] == "page"
112
+ # Returns the free response questions
113
+ #
114
+ # @return [Element]
115
+ #
116
+ def free_response
117
+ search('section.free-response')
48
118
  end
49
119
 
50
120
  end
@@ -1,9 +1,17 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Kitchen
4
+ # An enumerator for page elements
5
+ #
2
6
  class PageElementEnumerator < ElementEnumeratorBase
3
7
 
8
+ # Returns a factory for this enumerator
9
+ #
10
+ # @return [ElementEnumeratorFactory]
11
+ #
4
12
  def self.factory
5
13
  ElementEnumeratorFactory.new(
6
- default_css_or_xpath: "div[data-type='page']", # TODO get from config?
14
+ default_css_or_xpath: Selector.named(:page),
7
15
  sub_element_class: PageElement,
8
16
  enumerator_class: self
9
17
  )
@@ -1,3 +1,5 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Kitchen
2
4
  # A place to store labeled items during recipe work. Essentially, a slightly
3
5
  # improved hash.
@@ -5,22 +7,47 @@ module Kitchen
5
7
  class Pantry
6
8
  include Enumerable
7
9
 
10
+ # Adds an item to the pantry with the provided label
11
+ #
12
+ # @param item [Object] something to store
13
+ # @param label [String, Symbol] a label with which to retrieve this item later.
8
14
  def store(item, label:)
9
15
  @hash[label.to_sym] = item
10
16
  end
11
17
 
18
+ # Get an item from the pantry
19
+ #
20
+ # @param label [String, Symbol] the item's label
21
+ # @return [Object]
22
+ #
12
23
  def get(label)
13
24
  @hash[label.to_sym]
14
25
  end
15
26
 
27
+ # Get an item from the pantry, raising if not present
28
+ #
29
+ # @param label [String, Symbol] the item's label
30
+ # @raise [RecipeError] if there's no item for the label
31
+ # @return [Object]
32
+ #
16
33
  def get!(label)
17
34
  get(label) || raise(RecipeError, "There is no pantry item labeled '#{label}'")
18
35
  end
19
36
 
37
+ # Iterate over the pantry items
38
+ #
39
+ # @yield Gives each label and item pair to the block
40
+ # @yieldparam label [Symbol] the item's label
41
+ # @yieldparam item [Object] the item
42
+ #
20
43
  def each(&block)
21
- @hash.each{|k,v| block.call(k,v)}
44
+ @hash.each { |k, v| block.call(k, v) }
22
45
  end
23
46
 
47
+ # Returns the number of items in the pantry
48
+ #
49
+ # @return [Integer]
50
+ #
24
51
  def size
25
52
  @hash.keys.size
26
53
  end
@@ -1,16 +1,27 @@
1
+ # frozen_string_literal: true
2
+
1
3
  # Make debug output more useful (dumping entire document out is not useful)
2
4
  module Nokogiri
3
5
  module XML
6
+ # Monkey patches for Nokogiri::XML::Document
7
+ # @see https://www.rubydoc.info/github/sparklemotion/nokogiri/Nokogiri/XML/Document Nokogiri::XML::Document
4
8
  class Document
9
+ # Hides the guts of the document when printed out so you don't get 5MB dumped into your
10
+ # terminal
11
+ #
5
12
  def inspect
6
- "Nokogiri::XML::Document <hidden for brevity>"
13
+ 'Nokogiri::XML::Document <hidden for brevity>'
7
14
  end
8
15
 
16
+ # Alphabetizes all attributes within the document, useful for comparing one
17
+ # document to another (since attribute order isn't meaningful)
18
+ #
9
19
  def alphabetize_attributes!
10
20
  traverse do |child|
11
21
  next if child.text? || child.document?
22
+
12
23
  child_attributes = child.attributes
13
- child_attributes.each do |key, value|
24
+ child_attributes.each do |key, _value|
14
25
  child.remove_attribute(key)
15
26
  end
16
27
  sorted_keys = child_attributes.keys.sort
@@ -22,7 +33,13 @@ module Nokogiri
22
33
  end
23
34
  end
24
35
 
36
+ # Monkey patches for Nokogiri::XML::Node
37
+ # @see https://www.rubydoc.info/github/sparklemotion/nokogiri/Nokogiri/XML/Node Nokogiri::XML::Node
25
38
  class Node
39
+ # Calls to_s on the node
40
+ #
41
+ # @return [String]
42
+ #
26
43
  def inspect
27
44
  to_s
28
45
  end
@@ -1,6 +1,13 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Monkey patches for +Object+
4
+ #
1
5
  class Object
2
6
 
3
7
  # Adds a `render` method to a class for rendering an ERB template to a string.
8
+ #
9
+ # @param dir [String] a directory in which to find the template to be rendered,
10
+ # populated with a guess from the call stack if not provided.
4
11
  def self.renderable(dir: nil)
5
12
  dir ||= begin
6
13
  this_patch_file = __FILE__
@@ -8,11 +15,11 @@ class Object
8
15
  location.absolute_path == this_patch_file
9
16
  end
10
17
 
11
- location_that_called_renderable = caller_locations[(this_patch_file_caller_index || -1)+1]
18
+ location_that_called_renderable = caller_locations[(this_patch_file_caller_index || -1) + 1]
12
19
  File.dirname(location_that_called_renderable.path)
13
20
  end
14
21
 
15
- class_eval <<~METHOD
22
+ class_eval <<~METHOD, __FILE__, __LINE__ + 1
16
23
  def renderable_base_dir
17
24
  "#{dir}"
18
25
  end
@@ -25,7 +32,6 @@ class Object
25
32
  ERB.new(template).result(binding)
26
33
  end
27
34
  end
28
-
29
35
  end
30
36
 
31
37
  end
@@ -1,4 +1,12 @@
1
+ # frozen_string_literal: true
2
+
3
+ # Monkey patches for +String+
4
+ #
1
5
  class String
6
+ # Downcases the first letter of self, returning a new string
7
+ #
8
+ # @return [String]
9
+ #
2
10
  def uncapitalize
3
11
  sub(/^[A-Z]/, &:downcase)
4
12
  end
@@ -1,9 +1,28 @@
1
+ # frozen_string_literal: true
2
+
1
3
  module Kitchen
4
+ # An object that yields a +Document+ for modification (those modifications are
5
+ # the "recipe")
6
+ #
2
7
  class Recipe
3
8
 
9
+ # The document the recipe makes available for modification
10
+ # @return [Document]
4
11
  attr_reader :document
12
+
13
+ # The file location of the recipe
14
+ # @return [String]
5
15
  attr_reader :source_location
6
16
 
17
+ # An I18n backend specific to this recipe, may be nil
18
+ # @return [I18n::Backend::Simple, nil]
19
+ attr_reader :my_i18n_backend
20
+
21
+ # Sets the document so the recipe can yield it for modification
22
+ #
23
+ # @param document [Document] the document to modify
24
+ # @raise [StandardError] if not passed supported document type
25
+ #
7
26
  def document=(document)
8
27
  @document =
9
28
  case document
@@ -16,57 +35,75 @@ module Kitchen
16
35
 
17
36
  # Make a new Recipe
18
37
  #
38
+ # @param locales_dir [String, nil] the absolute path to a folder containing recipe-specific
39
+ # I18n translations. If not provided, Kitchen will look for a `locales` directory in the
40
+ # same directory as the recipe source. Recipe-specific translations override those in
41
+ # Kitchen. If no recipe-specific locales directory exists, Kitchen will just use its default
42
+ # translations.
43
+ # @yield A block for defining the steps of the recipe
19
44
  # @yieldparam doc [Document] an object representing an XML document
20
45
  #
21
- def initialize(&block)
46
+ def initialize(locales_dir: nil, &block)
47
+ raise(RecipeError, 'Recipes must be initialized with a block') unless block_given?
48
+
22
49
  @source_location = block.source_location[0]
23
50
  @block = block
24
- end
25
51
 
26
- def node!
27
- node || raise("The recipe's node has not been set")
52
+ load_my_i18n_backend(locales_dir)
28
53
  end
29
54
 
55
+ # Executes the block given to +Recipe.new+ on the document. Aka, does the baking.
56
+ #
30
57
  def bake
31
- begin
58
+ with_my_locales do
32
59
  @block.to_proc.call(document)
33
- rescue RecipeError => ee
34
- print_recipe_error_and_exit(ee)
35
- rescue ArgumentError => ee
36
- if if_any_stack_file_matches_source_location?(ee)
37
- print_recipe_error_and_exit(ee)
38
- else
39
- raise
40
- end
41
- rescue NoMethodError => ee
42
- if if_any_stack_file_matches_source_location?(ee)
43
- print_recipe_error_and_exit(ee)
44
- else
45
- raise
46
- end
47
- rescue NameError => ee
48
- if if_stack_starts_with_source_location?(ee)
49
- print_recipe_error_and_exit(ee)
50
- else
51
- raise
52
- end
53
- rescue ElementNotFoundError => ee
54
- print_recipe_error_and_exit(ee)
55
- rescue Nokogiri::CSS::SyntaxError => ee
56
- print_recipe_error_and_exit(ee)
57
60
  end
61
+ rescue RecipeError, ElementNotFoundError, Nokogiri::CSS::SyntaxError => e
62
+ print_recipe_error_and_exit(e)
63
+ rescue ArgumentError, NoMethodError => e
64
+ raise unless any_stack_file_matches_source_location?(e)
65
+
66
+ print_recipe_error_and_exit(e)
67
+ rescue NameError => e
68
+ raise unless stack_starts_with_source_location?(e)
69
+
70
+ print_recipe_error_and_exit(e)
58
71
  end
59
72
 
60
73
  protected
61
74
 
62
- def if_stack_starts_with_source_location?(error)
75
+ def stack_starts_with_source_location?(error)
63
76
  error.backtrace.first.start_with?(source_location)
64
77
  end
65
78
 
66
- def if_any_stack_file_matches_source_location?(error)
67
- error.backtrace.any? {|entry| entry.start_with?(@source_location)}
79
+ def any_stack_file_matches_source_location?(error)
80
+ error.backtrace.any? { |entry| entry.start_with?(@source_location) }
81
+ end
82
+
83
+ def load_my_i18n_backend(locales_dir)
84
+ locales_dir ||= begin
85
+ guessed_locales_dir = "#{File.dirname(@source_location)}/locales"
86
+ File.directory?(guessed_locales_dir) ? guessed_locales_dir : nil
87
+ end
88
+
89
+ return unless locales_dir
90
+
91
+ @my_i18n_backend = I18n::Backend::Simple.new
92
+ @my_i18n_backend.load_translations(Dir[File.expand_path("#{locales_dir}/*.yml")])
93
+ end
94
+
95
+ def with_my_locales
96
+ original_i18n_backend = I18n.backend
97
+ I18n.backend = I18n::Backend::Chain.new(my_i18n_backend, original_i18n_backend)
98
+ yield
99
+ ensure
100
+ I18n.backend = original_i18n_backend
68
101
  end
69
102
 
103
+ # Print the given recipe error and do a process exit
104
+ #
105
+ # @param error [RecipeError] the error
106
+ #
70
107
  def print_recipe_error_and_exit(error)
71
108
  Kitchen::Debug.print_recipe_error(error: error,
72
109
  source_location: source_location,