lifer 0.6.1 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (36) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +38 -0
  3. data/Gemfile.lock +1 -1
  4. data/lib/lifer/brain.rb +15 -9
  5. data/lib/lifer/builder/html/from_erb.rb +24 -5
  6. data/lib/lifer/builder/html/from_liquid/drops/collection_drop.rb +3 -3
  7. data/lib/lifer/builder/html/from_liquid/drops/collections_drop.rb +4 -3
  8. data/lib/lifer/builder/html/from_liquid/drops/entry_drop.rb +3 -3
  9. data/lib/lifer/builder/html/from_liquid/drops/frontmatter_drop.rb +2 -3
  10. data/lib/lifer/builder/html/from_liquid/drops/settings_drop.rb +2 -1
  11. data/lib/lifer/builder/html/from_liquid/drops/tag_drop.rb +42 -0
  12. data/lib/lifer/builder/html/from_liquid/drops/tags_drop.rb +43 -0
  13. data/lib/lifer/builder/html/from_liquid/drops.rb +2 -0
  14. data/lib/lifer/builder/html/from_liquid/filters.rb +3 -5
  15. data/lib/lifer/builder/html/from_liquid/layout_tag.rb +1 -2
  16. data/lib/lifer/builder/html/from_liquid.rb +4 -2
  17. data/lib/lifer/builder/html.rb +1 -1
  18. data/lib/lifer/builder/rss.rb +62 -8
  19. data/lib/lifer/builder.rb +2 -1
  20. data/lib/lifer/config.rb +1 -1
  21. data/lib/lifer/dev/response.rb +2 -3
  22. data/lib/lifer/dev/server.rb +2 -3
  23. data/lib/lifer/entry/html.rb +10 -13
  24. data/lib/lifer/entry/markdown.rb +22 -90
  25. data/lib/lifer/entry/txt.rb +11 -14
  26. data/lib/lifer/entry.rb +252 -129
  27. data/lib/lifer/selection.rb +4 -4
  28. data/lib/lifer/shared/finder_methods.rb +1 -2
  29. data/lib/lifer/tag.rb +53 -0
  30. data/lib/lifer/templates/config.yaml +7 -0
  31. data/lib/lifer/utilities.rb +7 -8
  32. data/lib/lifer/version.rb +1 -1
  33. data/lib/lifer.rb +15 -4
  34. data/lifer.gemspec +2 -2
  35. data/locales/en.yml +2 -3
  36. metadata +7 -4
checksums.yaml CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: 39e8d5c991a955d4c09740207671d49716b6ef42168b250fbe301c6a443785d5
4
- data.tar.gz: 73e4ea67bc8e9af6f1503782c53c334c5d59f31a2afd351312eb8365b9b42af7
3
+ metadata.gz: ea41a2cd1a058f5a103121b6b7641ce5857ec74300ecad4bb36ec33578170650
4
+ data.tar.gz: f752dc7498a0457945d895fc591eefc7a27a296e01a5b547264a13fd8113c3b5
5
5
  SHA512:
6
- metadata.gz: eacb4bd82e882758d88e9ffc0de30de8cb9213f35454856a2bed3f487bb8dc568afdb07ac22d6c269ac1b4c9418a18789dcd124193de009456243a2eeedc6415
7
- data.tar.gz: a8dd94d4c24932faae325461a4530c7f931d7ee7315353dc31ca42dac61c325ca91edd5f4f990e0710f4c18d4fbb9b020f51c1f13d8a02012d7c7f4918f5f88e
6
+ metadata.gz: 4665579521538df261c0839f6c1a7ee6b24362ca75c03cc6d8fbf42dfffa028ad0bc39c1fe01ef2407459cb1be6bdf9d08e60cd5abacd1e233b0819bbcab6186
7
+ data.tar.gz: e1bfbd9fa63bd2de10022f2b5810cfcf518acc5112f51bb843c19c936bc269436807a606c335ab90693bafac53569315b0bfcbe245b9c8570a56ff6ab60af6f0
data/CHANGELOG.md CHANGED
@@ -1,5 +1,43 @@
1
1
  ## Next
2
2
 
3
+ ## v0.8.0
4
+
5
+ ### Tags
6
+
7
+ Entries now support tag frontmatter. This introduces a new way of making
8
+ associations between entries. Tags can encoded in entries as YAML arrays or as a
9
+ string (with comma and/or space delimiters):
10
+
11
+ ---
12
+ title: My Entry
13
+ tags: beautifulTag, lovelyTag
14
+ ---
15
+
16
+ Blah blah blah.
17
+
18
+ Then, in your ERB or Liquid templates you can get entries via tag:
19
+
20
+ <% tags.beautifulTag.entries.each do |entry| %>
21
+ <li><%= entry.title %></li>
22
+ <% end %>
23
+
24
+ ### Frontmatter support across all entry types
25
+
26
+ Before this release, frontmatter was only supported by Markdown files. This
27
+ started to be annoying, because of features I wanted like tags. I figured it
28
+ couldn't hurt to just check any entry file for frontmatter, so that's how it
29
+ works now.
30
+
31
+ ## v0.7.0
32
+
33
+ This release adds Atom feed support to the RSS builder. In your configuration
34
+ file, you can configure feed formats to `rss` (the default) or `atom` now:
35
+
36
+ my_collection:
37
+ rss:
38
+ format: atom
39
+ url: haha.xml
40
+
3
41
  ## v0.6.1
4
42
 
5
43
  This release just fixes a mistake I made, where I built and pushed a tag from a
data/Gemfile.lock CHANGED
@@ -1,7 +1,7 @@
1
1
  PATH
2
2
  remote: .
3
3
  specs:
4
- lifer (0.6.1)
4
+ lifer (0.8.0)
5
5
  i18n (< 2)
6
6
  kramdown (~> 2.4)
7
7
  liquid (~> 5.6, < 6)
data/lib/lifer/brain.rb CHANGED
@@ -65,7 +65,7 @@ class Lifer::Brain
65
65
  # @return [Array<Lifer::Collection>] All the collections for the current Lifer
66
66
  # project.
67
67
  def collections
68
- @collections ||= generate_collections + generate_selections
68
+ @collections ||= (generate_collections + generate_selections).to_a
69
69
  end
70
70
 
71
71
  # Returns the Lifer project's configuration object.
@@ -125,8 +125,7 @@ class Lifer::Brain
125
125
  # value. If the in-scope collection does not have a configured setting, this
126
126
  # method will return fallback settings (unless `:strict` is `true`).
127
127
  #
128
- # Example usage:
129
- #
128
+ # @example Usage:
130
129
  # setting(:my, :great, :setting)
131
130
  #
132
131
  # @overload setting(path, ..., collection: nil, strict: false)
@@ -140,6 +139,19 @@ class Lifer::Brain
140
139
  config.setting *name, collection_name: collection&.name, strict: strict
141
140
  end
142
141
 
142
+ # Given the tag manifest, this returns an array of all tags for the current
143
+ # project. This method is preferrable for accessing and querying for tags.
144
+ #
145
+ # @return [Array<Lifer::Tag>]
146
+ def tags = tag_manifest.to_a
147
+
148
+ # The tag manifest tracks the unique tags added to the project as they're added.
149
+ # The writer method for this instance variable is used internally by Lifer when
150
+ # adding new tags.
151
+ #
152
+ # @return [Set<Lifer::Tag>]
153
+ def tag_manifest = (@tag_manifest ||= Set.new)
154
+
143
155
  private
144
156
 
145
157
  attr_reader :config_file_location
@@ -160,12 +172,6 @@ class Lifer::Brain
160
172
  path.start_with?("/") ? path : File.join(root, path)
161
173
  end
162
174
 
163
- # FIXME:
164
- # Do collections work with sub-subdirectories? For example, what if the
165
- # configured collection maps to a directory:
166
- #
167
- # subdirectory_one/sub_subdirectory_one
168
- #
169
175
  # @return [Set<Lifer::Collection>]
170
176
  def generate_collections
171
177
  config.collectionables
@@ -56,10 +56,16 @@ class Lifer::Builder::HTML
56
56
  end
57
57
 
58
58
  # @private
59
- # Each collection name is provided as a local variable. This allows you to
60
- # make ERB files that contain loops like:
59
+ # Each collection and tag name is provided as a local variable. This allows
60
+ # you to make ERB files that contain loops like:
61
61
  #
62
- # <% my_collection_name.entries.each do |entry| %>
62
+ # <% collections.my_collection_name.entries.each do |entry| %>
63
+ # <%= entry.title %>
64
+ # <% end %>
65
+ #
66
+ # or:
67
+ #
68
+ # <% tags.my_tag_name.entries.each do |entry| %>
63
69
  # <%= entry.title %>
64
70
  # <% end %>
65
71
  #
@@ -75,18 +81,31 @@ class Lifer::Builder::HTML
75
81
  end
76
82
  end
77
83
 
78
- collections = collection_context_class.new Lifer.collections.to_a
84
+ Lifer.tags.each do |tag|
85
+ binding.local_variable_set tag.name, tag
86
+
87
+ tag_context_class.define_method(tag.name) do
88
+ tag
89
+ end
90
+ end
91
+
92
+ collections = collection_context_class.new Lifer.collections
93
+ tags = tag_context_class.new Lifer.tags
79
94
 
80
95
  binding.local_variable_set :collections, collections
81
96
  binding.local_variable_set :settings, Lifer.settings
97
+ binding.local_variable_set :tags, tags
82
98
  binding.local_variable_set :content,
83
99
  ERB.new(entry.to_html).result(binding)
84
100
  }
85
101
  end
86
102
 
87
- # @private
88
103
  def collection_context_class
89
104
  @collection_context_class ||= Class.new(Array) do end
90
105
  end
106
+
107
+ def tag_context_class
108
+ @tag_context_class ||= Class.new(Array) do end
109
+ end
91
110
  end
92
111
  end
@@ -1,9 +1,9 @@
1
1
  module Lifer::Builder::HTML::FromLiquid::Drops
2
2
  # This drop allows users to access Lifer collection information from within
3
- # Liquid templates. Example:
3
+ # Liquid templates.
4
4
  #
5
+ # @example Usage
5
6
  # {{ collection.name }}
6
- #
7
7
  # {% for entries in collection.entries %}
8
8
  # {{ entry.title }}
9
9
  # {% endfor %}
@@ -28,7 +28,7 @@ module Lifer::Builder::HTML::FromLiquid::Drops
28
28
  # @return [Array<EntryDrop>]
29
29
  def entries
30
30
  @entries ||= lifer_collection.entries.map {
31
- EntryDrop.new _1, collection: self
31
+ EntryDrop.new _1, collection: self, tags: _1.tags
32
32
  }
33
33
  end
34
34
 
@@ -1,7 +1,8 @@
1
1
  module Lifer::Builder::HTML::FromLiquid::Drops
2
2
  # This drop allows users to iterate over their Lifer collections in Liquid
3
- # templates. Example:
3
+ # templates.
4
4
  #
5
+ # @example Usage
5
6
  # {% for collection in collections %}
6
7
  # {{ collection.name }}
7
8
  # {% endfor %}
@@ -17,7 +18,6 @@ module Lifer::Builder::HTML::FromLiquid::Drops
17
18
  #
18
19
  # @yield [CollectionDrop] All available collection drops.
19
20
  def each(&block)
20
-
21
21
  collections.each(&block)
22
22
  end
23
23
 
@@ -27,8 +27,9 @@ module Lifer::Builder::HTML::FromLiquid::Drops
27
27
  def to_a = @collections
28
28
 
29
29
  # Dynamically define Liquid accessors based on the Lifer project's
30
- # collection names. For example, to get the root collection's name:
30
+ # collection names.
31
31
  #
32
+ # @example Get the root collection's name.
32
33
  # {{ collections.root.name }}
33
34
  #
34
35
  # @param arg [String] The name of a collection.
@@ -2,17 +2,17 @@ module Lifer::Builder::HTML::FromLiquid::Drops
2
2
  # This drop represents a Lifer entry and allows users to access entry
3
3
  # metadata and content in Liquid templates.
4
4
  #
5
- # Example usage:
6
- #
5
+ # @example Usage
7
6
  # <h1>{{ entry.title }}</h1>
8
7
  # <small>Published on <datetime>{{ entry.date }}</datetime></small>
9
8
  #
10
9
  class EntryDrop < Liquid::Drop
11
10
  attr_accessor :lifer_entry, :collection
12
11
 
13
- def initialize(lifer_entry, collection:)
12
+ def initialize(lifer_entry, collection:, tags:)
14
13
  @lifer_entry = lifer_entry
15
14
  @collection = collection
15
+ @tags = tags
16
16
  end
17
17
 
18
18
  # The entry author (or authors).
@@ -2,8 +2,7 @@ module Lifer::Builder::HTML::FromLiquid::Drops
2
2
  # Markdown entries may contain YAML frontmatter. And if they do, we need a way
3
3
  # for the Liquid templates to access that data.
4
4
  #
5
- # Example usage:
6
- #
5
+ # @example Usage
7
6
  # {{ entry.frontmatter.any_available_frontmatter_key }}
8
7
  #
9
8
  class FrontmatterDrop < Liquid::Drop
@@ -18,8 +17,8 @@ module Lifer::Builder::HTML::FromLiquid::Drops
18
17
  def to_s = frontmatter.to_json
19
18
 
20
19
  # Dynamically define Liquid accessors based on the Lifer settings object.
21
- # For example, to get a collections URI strategy:
22
20
  #
21
+ # @example Get a collection's URI strategy.
23
22
  # {{ settings.my_collection.uri_strategy }}
24
23
  #
25
24
  # @param arg [String] The name of a collection.
@@ -2,6 +2,7 @@ module Lifer::Builder::HTML::FromLiquid::Drops
2
2
  # This drop allows users to access the current Lifer project settings from
3
3
  # Liquid templates. Example:
4
4
  #
5
+ # @example Usage
5
6
  # {{ settings.my_collection.uri_strategy }}
6
7
  #
7
8
  class SettingsDrop < Liquid::Drop
@@ -15,8 +16,8 @@ module Lifer::Builder::HTML::FromLiquid::Drops
15
16
  def to_s = settings.to_json
16
17
 
17
18
  # Dynamically define Liquid accessors based on the Lifer settings object.
18
- # For example, to get a collections URI strategy:
19
19
  #
20
+ # @example Get a collections URI strategy:
20
21
  # {{ settings.my_collection.uri_strategy }}
21
22
  #
22
23
  # @param arg [String] The name of a collection.
@@ -0,0 +1,42 @@
1
+ module Lifer::Builder::HTML::FromLiquid::Drops
2
+ # This drop allows users to access Lifer tag information from within
3
+ # Liquid templates.
4
+ #
5
+ # @example Usage
6
+ # {{ tag.name }}
7
+ # {% for entries in tag.entries %}
8
+ # {{ entry.title }}
9
+ # {% endfor %}
10
+ #
11
+ class TagDrop < Liquid::Drop
12
+ attr_accessor :lifer_tag
13
+
14
+ def initialize(lifer_tag) = (@lifer_tag = lifer_tag)
15
+
16
+ # The tag name.
17
+ #
18
+ # @return [Symbol]
19
+ def name = (@name ||= lifer_tag.name)
20
+
21
+ # Gets all entries in a tag and converts them to entry drops that can
22
+ # be accessed in Liquid templates. Example:
23
+ #
24
+ # {% for entry in tags..entries %}
25
+ # {{ entry.title }}
26
+ # {% endfor %}
27
+ #
28
+ # @return [Array<EntryDrop>]
29
+ def entries
30
+ @entries ||= lifer_tag.entries.map { |lifer_entry|
31
+ EntryDrop.new lifer_entry,
32
+ collection: CollectionDrop.new(lifer_entry.collection),
33
+ tags: lifer_entry.tags.map { TagDrop.new _1 }
34
+ }
35
+ end
36
+
37
+ # The tag's layout file path.
38
+ #
39
+ # @return [String] The path to the layout file.
40
+ def layout_file = (@lifer_tag.layout_file)
41
+ end
42
+ end
@@ -0,0 +1,43 @@
1
+ module Lifer::Builder::HTML::FromLiquid::Drops
2
+ # This drop allows users to iterate over their Lifer tags in Liquid
3
+ # templates.
4
+ #
5
+ # @example Usage
6
+ # {% for tag in tags %}
7
+ # {{ tag.name }}
8
+ # {% endfor %}
9
+ #
10
+ # {% for entry in tags.name-of-tag.entries %}
11
+ # {{ entry.title }}
12
+ # {% endfor %}
13
+ #
14
+ class TagsDrop < Liquid::Drop
15
+ attr_accessor :tags
16
+
17
+ def initialize
18
+ @tags = Lifer.tags.map { TagDrop.new _1 }
19
+ end
20
+
21
+ # Allow tags to be iterable in Liquid templates.
22
+ #
23
+ # @yield [CollectionDrop] All available collection drops.
24
+ def each(&block) = tags.each(&block)
25
+
26
+ # Allow tags to be rendered as an array in Liquid templates.
27
+ #
28
+ # @return [Array]
29
+ def to_a = @tags
30
+
31
+ # Dynamically define Liquid accessors based on the Lifer project's
32
+ # collection names.
33
+ #
34
+ # @example Get the "tagName" tag's entries.
35
+ # {{ tags.tagName.entries }}
36
+ #
37
+ # @param arg [String] The name of a collection.
38
+ # @return [CollectionDrop, NilClass]
39
+ def liquid_method_missing(arg)
40
+ tags.detect { arg.to_s == _1.name.to_s }
41
+ end
42
+ end
43
+ end
@@ -13,3 +13,5 @@ require_relative "drops/collections_drop"
13
13
  require_relative "drops/entry_drop"
14
14
  require_relative "drops/frontmatter_drop"
15
15
  require_relative "drops/settings_drop"
16
+ require_relative "drops/tag_drop"
17
+ require_relative "drops/tags_drop"
@@ -1,8 +1,7 @@
1
1
  # This module provides Liquid filters to be used within Liquid templates.
2
2
  # In many cases these utilities exist to be pseudo-compatible with Jekyll.
3
3
  #
4
- # For example, a filter (in a Liquid template):
5
- #
4
+ # @example A filter in a Liquid template.
6
5
  # {{ entry.date | date_to_xmlschema }}
7
6
  #
8
7
  module Lifer::Builder::HTML::FromLiquid::Filters
@@ -17,10 +16,9 @@ module Lifer::Builder::HTML::FromLiquid::Filters
17
16
 
18
17
  # Transforms a string to kabab-case.
19
18
  #
20
- # For example:
19
+ # @example Result
20
+ # handleize("hello_there") #=> "hello-there"
21
21
  #
22
- # Before: hello_there
23
- # After: hello-there
24
22
  # @param input [String] A string.
25
23
  # @return [String] The transformed string.
26
24
  def handleize(input) = Util.handleize(input)
@@ -11,8 +11,7 @@ class Lifer::Builder::HTML::FromLiquid
11
11
  # frontmatter when we didn't need to. Maybe in the long run this was a bad
12
12
  # call? I don't know.
13
13
  #
14
- # Example usage (from a Liquid template):
15
- #
14
+ # @example Usage from a Liquid template.
16
15
  # {% layout "path/to/my_liquid_layout_template" %}
17
16
  #
18
17
  # (The required `endlayout` tag will be appended to the end of the file
@@ -5,7 +5,6 @@ require_relative "from_liquid/filters"
5
5
  require_relative "from_liquid/layout_tag"
6
6
  require_relative "from_liquid/liquid_env"
7
7
 
8
-
9
8
  class Lifer::Builder::HTML
10
9
  # If the HTML builder is given a Liquid template, it uses this class to parse
11
10
  # the Liquid into HTML. Lifer project metadata is provided as context. For
@@ -64,13 +63,16 @@ class Lifer::Builder::HTML
64
63
 
65
64
  def context
66
65
  collections = Drops::CollectionsDrop.new
66
+ tags = Drops::TagsDrop.new
67
67
  collection = collections
68
68
  .to_a
69
69
  .detect { _1.name.to_sym == entry.collection.name }
70
+ entry_tags = tags.to_a.select { entry.tags.include? _1 }
70
71
 
71
72
  {
72
73
  "collections" => collections,
73
- "entry" => Drops::EntryDrop.new(entry, collection:),
74
+ "tags" => tags,
75
+ "entry" => Drops::EntryDrop.new(entry, collection:, tags: entry_tags),
74
76
  "parse_options" => parse_options,
75
77
  "render_options" => render_options,
76
78
  "settings" => Drops::SettingsDrop.new
@@ -74,7 +74,7 @@ class Lifer::Builder::HTML < Lifer::Builder
74
74
  # For the given collection, ensure all required directories and
75
75
  # subdirectories exist so the entry output can be safely written to.
76
76
  #
77
- # @param entry [Lifer::Collection] A collection.
77
+ # @param collection [Lifer::Collection] A collection.
78
78
  # @return [void]
79
79
  def generate_output_directories_for(collection)
80
80
  directories = collection.entries
@@ -1,6 +1,9 @@
1
1
  require "fileutils"
2
2
  require "rss"
3
3
 
4
+ # FIXME: Can this feed builder just take a type for Atom feeds instead of us
5
+ # creating a separate builder?
6
+
4
7
  # Builds a simple, UTF-8, RSS 2.0[1] feed using the Ruby standard library's RSS
5
8
  # features.
6
9
  #
@@ -26,6 +29,8 @@ require "rss"
26
29
  #
27
30
  # - `count:` - The limit of RSS feed items that should be included in the
28
31
  # output document. Leave unset or set to `0` to include all entries.
32
+ # - `format:` - The RSS format to build with. Either `rss` or `atom` are
33
+ # supported. `rss` is the default format.
29
34
  # - `managing_editor:` - the contents of the `<managingEditor>` node of the
30
35
  # RSS document. When unset, Lifer builds a valid `<managingEditor>` value
31
36
  # using the collection or root `author` value and a null email address to
@@ -47,10 +52,27 @@ class Lifer::Builder::RSS < Lifer::Builder
47
52
  #
48
53
  DEFAULT_MANAGING_EDITOR_EMAIL = "editor@null.invalid"
49
54
 
55
+ # All of the available formats that this builder can build. Where the keys
56
+ # are formats accepted as input in configuration files and the values are
57
+ # the formats that the RSS builder will output.
58
+ #
59
+ FORMATS = {atom: "atom", rss: "rss2.0"}
60
+
61
+ # The default format for this RSS feed builder. Where the key is the format
62
+ # accepted as input in configuration files and the value is the format that
63
+ # the RSS builder will output.
64
+
65
+ # The name of the format type, as needed by `RSS::Maker`, used by default by
66
+ # this feed builder.
67
+ #
68
+ DEFAULT_MAkER_FORMAT_NAME = FORMATS[:rss]
69
+
70
+
50
71
  self.name = :rss
51
72
  self.settings = [
52
73
  rss: [
53
74
  :count,
75
+ :format,
54
76
  :managing_editor,
55
77
  :url
56
78
  ]
@@ -69,6 +91,14 @@ class Lifer::Builder::RSS < Lifer::Builder
69
91
  end
70
92
  end
71
93
 
94
+ def feed_format(collection)
95
+ format = Lifer.setting(:rss, :format, collection:)&.to_sym
96
+
97
+ return FORMATS[format] if FORMATS.keys.include? format
98
+
99
+ DEFAULT_MAkER_FORMAT_NAME
100
+ end
101
+
72
102
  # Traverses and renders an RSS feed for feedable collection.
73
103
  #
74
104
  # @return [void]
@@ -150,8 +180,10 @@ class Lifer::Builder::RSS < Lifer::Builder
150
180
  end
151
181
  end
152
182
 
153
- # FIXME: Using the W3C feed validation checker[1], I found that RSS feed items
154
- # generated by Lifer are missing some recommended functionality. Reports
183
+ # @fixme Using the W3C feed validation checker[1], I found that RSS and Atom
184
+ # feed items generated by Lifer are missing some recommended functionality.
185
+ #
186
+ # RSS reports:
155
187
  #
156
188
  # > This feed is valid, but interoperability with the widest range of feed
157
189
  # > readers could be improved by implementing the following recommendations.
@@ -169,6 +201,14 @@ class Lifer::Builder::RSS < Lifer::Builder
169
201
  # related to our Markdown parser implementation than the RSS feed generation
170
202
  # itself.
171
203
  #
204
+ # Atom reports:
205
+ #
206
+ # > Two or more entries with the same value for <atom:updated>
207
+ # > (https://validator.w3.org/feed/docs/warning/DuplicateUpdated.html)
208
+ # >
209
+ # > Missing <atom:link> with rel="self".
210
+ # > (https://validator.w3.org/feed/docs/warning/MissingSelf.html)
211
+ #
172
212
  # [1]: https://validator.w3.org/feed/check.cgi
173
213
  #
174
214
  def rss_entry(rss_feed, lifer_entry)
@@ -185,7 +225,7 @@ class Lifer::Builder::RSS < Lifer::Builder
185
225
  end
186
226
  end
187
227
 
188
- # FIXME: Using the W3C feed validation checker[1], I found that RSS feeds
228
+ # @fixme Using the W3C feed validation checker[1], I found that RSS feeds
189
229
  # generated by Lifer are missing some recommended functionality. Reports:
190
230
  #
191
231
  # > Missing atom:link with rel="self"
@@ -196,20 +236,34 @@ class Lifer::Builder::RSS < Lifer::Builder
196
236
  def rss_feed_for(collection, &block)
197
237
  feed_object = nil
198
238
 
199
- ::RSS::Maker.make "rss2.0" do |feed|
239
+ ::RSS::Maker.make feed_format(collection) do |feed|
200
240
  feed.channel.description =
201
241
  Lifer.setting(:description, collection: collection) ||
202
242
  Lifer.setting(:site_title, collection: collection)
203
243
 
204
244
  feed.channel.language = Lifer.setting(:language, collection: collection)
205
245
 
206
- feed.channel.lastBuildDate = Time.now.to_s
207
-
208
- feed.channel.link = "%s/%s" % [
246
+ channel_link = "%s/%s" % [
209
247
  Lifer.setting(:global, :host),
210
- Lifer.setting(:rss, collection: collection)
248
+ Lifer.setting(:rss, :url, collection:)
211
249
  ]
212
250
 
251
+ # The W3C Atom validator claims that the <id> should be a "canonicalized"
252
+ # URL with a slash at the end.
253
+ #
254
+ channel_link = channel_link + "/" unless channel_link.end_with?("/")
255
+
256
+ feed.channel.lastBuildDate = Time.now.to_s
257
+ feed.channel.link = channel_link
258
+
259
+ # Additional channel fields for Atom format feeds.
260
+ #
261
+ if feed_format(collection) == "atom"
262
+ feed.channel.author = Lifer.setting(:author, collection: collection)
263
+ feed.channel.id = channel_link
264
+ feed.channel.updated = Time.now.to_s
265
+ end
266
+
213
267
  feed.channel.managingEditor = managing_editor(collection)
214
268
  feed.channel.title = Lifer.setting(:title, collection: collection)
215
269
  feed.channel.webMaster =
data/lib/lifer/builder.rb CHANGED
@@ -18,7 +18,8 @@ class Lifer::Builder
18
18
  # Every builder class must have execute method. This is the entrypoint for
19
19
  # instantiating *and* executing any builder.
20
20
  #
21
- # @param root [string] An absolute path to the Lifer project root directory.
21
+ # @param root [string] An absolute path to the Lifer project root
22
+ # directory.
22
23
  # @return [NotImplementedError] A builder subclass must implement this
23
24
  # method.
24
25
  def execute(root:) = (raise NotImplementedError)
data/lib/lifer/config.rb CHANGED
@@ -33,7 +33,7 @@ class Lifer::Config
33
33
  # Provides "implicit settings" that may not be set anywhere but really do
34
34
  # require a value.
35
35
  #
36
- # FIXME: I don't think this really belongs here. But in some cases we need
36
+ # @fixme I don't think this really belongs here. But in some cases we need
37
37
  # to provide the implicit setting key and a default value when calling the
38
38
  # `#setting` method. It would be nicer if the HTML builder handled this,
39
39
  # somehow.
@@ -33,9 +33,8 @@ module Lifer::Dev
33
33
  [File.read(path)]
34
34
  end
35
35
 
36
- # FIXME:
37
- # It would be very nice to not manually manage this list of content types.
38
- # Is there a nice, dependency-free way to do this?
36
+ # @fixme It would be very nice to not manually manage this list of
37
+ # content types. Is there a nice, dependency-free way to do this?
39
38
  #
40
39
  def content_type
41
40
  case File.extname(path)
@@ -71,8 +71,8 @@ module Lifer::Dev
71
71
  # @private
72
72
  # On reload, we rebuild the Lifer project.
73
73
  #
74
- # FIXME:
75
- # Partial rebuilds would be a nice enhancement for performance reasons.
74
+ # @fixme Partial rebuilds would be a nice enhancement for performance
75
+ # reasons.
76
76
  #
77
77
  def reload!
78
78
  if @changes
@@ -84,7 +84,6 @@ module Lifer::Dev
84
84
 
85
85
  # @private
86
86
  # @return [Lifer::Dev::Router] Our dev server router.
87
- #
88
87
  def router
89
88
  return @router if @router && !test_mode?
90
89