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.
- checksums.yaml +4 -4
- data/CHANGELOG.md +38 -0
- data/Gemfile.lock +1 -1
- data/lib/lifer/brain.rb +15 -9
- data/lib/lifer/builder/html/from_erb.rb +24 -5
- data/lib/lifer/builder/html/from_liquid/drops/collection_drop.rb +3 -3
- data/lib/lifer/builder/html/from_liquid/drops/collections_drop.rb +4 -3
- data/lib/lifer/builder/html/from_liquid/drops/entry_drop.rb +3 -3
- data/lib/lifer/builder/html/from_liquid/drops/frontmatter_drop.rb +2 -3
- data/lib/lifer/builder/html/from_liquid/drops/settings_drop.rb +2 -1
- data/lib/lifer/builder/html/from_liquid/drops/tag_drop.rb +42 -0
- data/lib/lifer/builder/html/from_liquid/drops/tags_drop.rb +43 -0
- data/lib/lifer/builder/html/from_liquid/drops.rb +2 -0
- data/lib/lifer/builder/html/from_liquid/filters.rb +3 -5
- data/lib/lifer/builder/html/from_liquid/layout_tag.rb +1 -2
- data/lib/lifer/builder/html/from_liquid.rb +4 -2
- data/lib/lifer/builder/html.rb +1 -1
- data/lib/lifer/builder/rss.rb +62 -8
- data/lib/lifer/builder.rb +2 -1
- data/lib/lifer/config.rb +1 -1
- data/lib/lifer/dev/response.rb +2 -3
- data/lib/lifer/dev/server.rb +2 -3
- data/lib/lifer/entry/html.rb +10 -13
- data/lib/lifer/entry/markdown.rb +22 -90
- data/lib/lifer/entry/txt.rb +11 -14
- data/lib/lifer/entry.rb +252 -129
- data/lib/lifer/selection.rb +4 -4
- data/lib/lifer/shared/finder_methods.rb +1 -2
- data/lib/lifer/tag.rb +53 -0
- data/lib/lifer/templates/config.yaml +7 -0
- data/lib/lifer/utilities.rb +7 -8
- data/lib/lifer/version.rb +1 -1
- data/lib/lifer.rb +15 -4
- data/lifer.gemspec +2 -2
- data/locales/en.yml +2 -3
- metadata +7 -4
checksums.yaml
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
---
|
2
2
|
SHA256:
|
3
|
-
metadata.gz:
|
4
|
-
data.tar.gz:
|
3
|
+
metadata.gz: ea41a2cd1a058f5a103121b6b7641ce5857ec74300ecad4bb36ec33578170650
|
4
|
+
data.tar.gz: f752dc7498a0457945d895fc591eefc7a27a296e01a5b547264a13fd8113c3b5
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
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
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
|
-
#
|
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
|
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
|
-
|
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.
|
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.
|
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.
|
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
|
-
#
|
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
|
-
#
|
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
|
@@ -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
|
-
#
|
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
|
-
#
|
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
|
-
#
|
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
|
-
"
|
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
|
data/lib/lifer/builder/html.rb
CHANGED
@@ -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
|
77
|
+
# @param collection [Lifer::Collection] A collection.
|
78
78
|
# @return [void]
|
79
79
|
def generate_output_directories_for(collection)
|
80
80
|
directories = collection.entries
|
data/lib/lifer/builder/rss.rb
CHANGED
@@ -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
|
-
#
|
154
|
-
# generated by Lifer are missing some recommended functionality.
|
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
|
-
#
|
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
|
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
|
-
|
207
|
-
|
208
|
-
feed.channel.link = "%s/%s" % [
|
246
|
+
channel_link = "%s/%s" % [
|
209
247
|
Lifer.setting(:global, :host),
|
210
|
-
Lifer.setting(:rss,
|
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
|
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
|
-
#
|
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.
|
data/lib/lifer/dev/response.rb
CHANGED
@@ -33,9 +33,8 @@ module Lifer::Dev
|
|
33
33
|
[File.read(path)]
|
34
34
|
end
|
35
35
|
|
36
|
-
#
|
37
|
-
#
|
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)
|
data/lib/lifer/dev/server.rb
CHANGED
@@ -71,8 +71,8 @@ module Lifer::Dev
|
|
71
71
|
# @private
|
72
72
|
# On reload, we rebuild the Lifer project.
|
73
73
|
#
|
74
|
-
#
|
75
|
-
#
|
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
|
|