lifer 0.7.0 → 0.9.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 +43 -0
- data/Gemfile.lock +2 -2
- data/lib/lifer/brain.rb +14 -1
- data/lib/lifer/builder/html/from_erb.rb +24 -5
- data/lib/lifer/builder/html/from_liquid/drops/collection_drop.rb +1 -1
- data/lib/lifer/builder/html/from_liquid/drops/collections_drop.rb +0 -1
- data/lib/lifer/builder/html/from_liquid/drops/entry_drop.rb +15 -7
- 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 +1 -1
- data/lib/lifer/builder/html/from_liquid.rb +4 -2
- data/lib/lifer/builder/rss.rb +17 -5
- data/lib/lifer/collection.rb +2 -2
- data/lib/lifer/entry/html.rb +10 -13
- data/lib/lifer/entry/markdown.rb +12 -79
- data/lib/lifer/entry/txt.rb +10 -13
- data/lib/lifer/entry.rb +294 -129
- data/lib/lifer/tag.rb +53 -0
- data/lib/lifer/utilities.rb +4 -1
- data/lib/lifer/version.rb +1 -1
- data/lib/lifer.rb +22 -0
- 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: 1b6414082e8f6f1d9b18034f2f227890b85ac73b22ab50c25cb2e350812c1a90
|
4
|
+
data.tar.gz: 5891afc57d102fd67422bf581edc4de839b05efbe10d104529a817391c0e7cad
|
5
5
|
SHA512:
|
6
|
-
metadata.gz:
|
7
|
-
data.tar.gz:
|
6
|
+
metadata.gz: 7adb1041c558a2d60ef525bf1d12af4dd3d587a26f8ae0ef4e2f4c8fb9e04e14c648cf4bd72717aba143e747ccabe970ec1c859cd0c69b5259847887c0f99361
|
7
|
+
data.tar.gz: a9ec79fce82554f69ba1634ce414b7b63014c8545f8f8c5a6201c194d375d57d09a36128db4f6b242c279bfe0fe419f3b333034268415379c1b9138d372d352f
|
data/CHANGELOG.md
CHANGED
@@ -1,5 +1,48 @@
|
|
1
1
|
## Next
|
2
2
|
|
3
|
+
## v0.9.0
|
4
|
+
|
5
|
+
Atom feeds now support entries with both `#published_at` and `#updated_at`
|
6
|
+
timestamp. There is no standard equivalent way to provide this functonality for
|
7
|
+
RSS-format feeds, unfortunately. As part of this change, we removed all
|
8
|
+
`Entry#date` methods in favour of `Entry#published_at` onces. In Atom feeds, if
|
9
|
+
an article has no last updated date, the publication date is used instead.
|
10
|
+
|
11
|
+
Additionally, this release includes a new environment variable:
|
12
|
+
`LIFER_UNPARALLELIZED`. You can use this environment variable to run `lifer
|
13
|
+
build` or `lifer serve` without any parallelization turned on. This could be
|
14
|
+
useful for reproducing bugs and so on. Example usage:
|
15
|
+
|
16
|
+
LIFER_UNPARALLELIZED=1 lifer build
|
17
|
+
|
18
|
+
## v0.8.0
|
19
|
+
|
20
|
+
### Tags
|
21
|
+
|
22
|
+
Entries now support tag frontmatter. This introduces a new way of making
|
23
|
+
associations between entries. Tags can encoded in entries as YAML arrays or as a
|
24
|
+
string (with comma and/or space delimiters):
|
25
|
+
|
26
|
+
---
|
27
|
+
title: My Entry
|
28
|
+
tags: beautifulTag, lovelyTag
|
29
|
+
---
|
30
|
+
|
31
|
+
Blah blah blah.
|
32
|
+
|
33
|
+
Then, in your ERB or Liquid templates you can get entries via tag:
|
34
|
+
|
35
|
+
<% tags.beautifulTag.entries.each do |entry| %>
|
36
|
+
<li><%= entry.title %></li>
|
37
|
+
<% end %>
|
38
|
+
|
39
|
+
### Frontmatter support across all entry types
|
40
|
+
|
41
|
+
Before this release, frontmatter was only supported by Markdown files. This
|
42
|
+
started to be annoying, because of features I wanted like tags. I figured it
|
43
|
+
couldn't hurt to just check any entry file for frontmatter, so that's how it
|
44
|
+
works now.
|
45
|
+
|
3
46
|
## v0.7.0
|
4
47
|
|
5
48
|
This release adds Atom feed support to the RSS builder. In your configuration
|
data/Gemfile.lock
CHANGED
@@ -1,7 +1,7 @@
|
|
1
1
|
PATH
|
2
2
|
remote: .
|
3
3
|
specs:
|
4
|
-
lifer (0.
|
4
|
+
lifer (0.9.0)
|
5
5
|
i18n (< 2)
|
6
6
|
kramdown (~> 2.4)
|
7
7
|
liquid (~> 5.6, < 6)
|
@@ -44,7 +44,7 @@ GEM
|
|
44
44
|
kramdown (2.5.1)
|
45
45
|
rexml (>= 3.3.9)
|
46
46
|
language_server-protocol (3.17.0.3)
|
47
|
-
liquid (5.8.
|
47
|
+
liquid (5.8.2)
|
48
48
|
bigdecimal
|
49
49
|
strscan (>= 3.1.1)
|
50
50
|
listen (3.9.0)
|
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.
|
@@ -139,6 +139,19 @@ class Lifer::Brain
|
|
139
139
|
config.setting *name, collection_name: collection&.name, strict: strict
|
140
140
|
end
|
141
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
|
+
|
142
155
|
private
|
143
156
|
|
144
157
|
attr_reader :config_file_location
|
@@ -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
|
@@ -4,14 +4,17 @@ module Lifer::Builder::HTML::FromLiquid::Drops
|
|
4
4
|
#
|
5
5
|
# @example Usage
|
6
6
|
# <h1>{{ entry.title }}</h1>
|
7
|
-
# <small>
|
7
|
+
# <small>
|
8
|
+
# Published on <datetime>{{ entry.published_at }}</datetime>
|
9
|
+
# </small>
|
8
10
|
#
|
9
11
|
class EntryDrop < Liquid::Drop
|
10
12
|
attr_accessor :lifer_entry, :collection
|
11
13
|
|
12
|
-
def initialize(lifer_entry, collection:)
|
14
|
+
def initialize(lifer_entry, collection:, tags:)
|
13
15
|
@lifer_entry = lifer_entry
|
14
16
|
@collection = collection
|
17
|
+
@tags = tags
|
15
18
|
end
|
16
19
|
|
17
20
|
# The entry author (or authors).
|
@@ -29,11 +32,6 @@ module Lifer::Builder::HTML::FromLiquid::Drops
|
|
29
32
|
# @return [String]
|
30
33
|
def content = (@content ||= lifer_entry.to_html)
|
31
34
|
|
32
|
-
# The entry date (as a string).
|
33
|
-
#
|
34
|
-
# @return [String]
|
35
|
-
def date = (@date ||= lifer_entry.date)
|
36
|
-
|
37
35
|
# The entry frontmatter data.
|
38
36
|
#
|
39
37
|
# @return [FrontmatterDrop]
|
@@ -49,6 +47,11 @@ module Lifer::Builder::HTML::FromLiquid::Drops
|
|
49
47
|
# @return [String] The entry permalink.
|
50
48
|
def permalink = (@permalink ||= lifer_entry.permalink)
|
51
49
|
|
50
|
+
# The entry publication date (as a string).
|
51
|
+
#
|
52
|
+
# @return [String]
|
53
|
+
def published_at = (@published_at ||= lifer_entry.published_at)
|
54
|
+
|
52
55
|
# The summary of the entry.
|
53
56
|
#
|
54
57
|
# @return [String] The summary of the entry.
|
@@ -58,5 +61,10 @@ module Lifer::Builder::HTML::FromLiquid::Drops
|
|
58
61
|
#
|
59
62
|
# @return [String] The entry title.
|
60
63
|
def title = (@title ||= lifer_entry.title)
|
64
|
+
|
65
|
+
# The entry's last updated date (as a string).
|
66
|
+
#
|
67
|
+
# @return [String]
|
68
|
+
def updated_at = (@updated_at ||= lifer_entry.updated_at)
|
61
69
|
end
|
62
70
|
end
|
@@ -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
|
@@ -2,7 +2,7 @@
|
|
2
2
|
# In many cases these utilities exist to be pseudo-compatible with Jekyll.
|
3
3
|
#
|
4
4
|
# @example A filter in a Liquid template.
|
5
|
-
# {{ entry.
|
5
|
+
# {{ entry.published_at | date_to_xmlschema }}
|
6
6
|
#
|
7
7
|
module Lifer::Builder::HTML::FromLiquid::Filters
|
8
8
|
# @!visibility private
|
@@ -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/rss.rb
CHANGED
@@ -65,8 +65,7 @@ class Lifer::Builder::RSS < Lifer::Builder
|
|
65
65
|
# The name of the format type, as needed by `RSS::Maker`, used by default by
|
66
66
|
# this feed builder.
|
67
67
|
#
|
68
|
-
|
69
|
-
|
68
|
+
DEFAULT_MAKER_FORMAT_NAME = FORMATS[:rss]
|
70
69
|
|
71
70
|
self.name = :rss
|
72
71
|
self.settings = [
|
@@ -96,7 +95,7 @@ class Lifer::Builder::RSS < Lifer::Builder
|
|
96
95
|
|
97
96
|
return FORMATS[format] if FORMATS.keys.include? format
|
98
97
|
|
99
|
-
|
98
|
+
DEFAULT_MAKER_FORMAT_NAME
|
100
99
|
end
|
101
100
|
|
102
101
|
# Traverses and renders an RSS feed for feedable collection.
|
@@ -220,8 +219,21 @@ class Lifer::Builder::RSS < Lifer::Builder
|
|
220
219
|
rss_feed_item.link = lifer_entry.permalink
|
221
220
|
rss_feed_item.title = lifer_entry.title
|
222
221
|
rss_feed_item.summary = lifer_entry.summary
|
223
|
-
|
224
|
-
|
222
|
+
|
223
|
+
if feed_format(lifer_entry.collection) == "atom"
|
224
|
+
rss_feed_item.content.content = lifer_entry.to_html
|
225
|
+
rss_feed_item.published = lifer_entry.published_at
|
226
|
+
|
227
|
+
# Note: RSS does not provide a standard way to share last updated
|
228
|
+
# timestamps at all, while Atom does. This is the reason there is no
|
229
|
+
# equivalent call in the condition for RSS feeds.
|
230
|
+
#
|
231
|
+
rss_feed_item.updated =
|
232
|
+
lifer_entry.updated_at(fallback: lifer_entry.published_at)
|
233
|
+
else
|
234
|
+
rss_feed_item.content_encoded = lifer_entry.to_html
|
235
|
+
rss_feed_item.pubDate = lifer_entry.published_at.to_time.rfc2822
|
236
|
+
end
|
225
237
|
end
|
226
238
|
end
|
227
239
|
|
data/lib/lifer/collection.rb
CHANGED
@@ -57,9 +57,9 @@ class Lifer::Collection
|
|
57
57
|
cached_entries_variable,
|
58
58
|
case order
|
59
59
|
when :latest
|
60
|
-
@entries_collection.sort_by { |entry| entry.
|
60
|
+
@entries_collection.sort_by { |entry| entry.published_at }.reverse
|
61
61
|
when :oldest
|
62
|
-
@entries_collection.sort_by { |entry| entry.
|
62
|
+
@entries_collection.sort_by { |entry| entry.published_at }
|
63
63
|
end
|
64
64
|
)
|
65
65
|
end
|
data/lib/lifer/entry/html.rb
CHANGED
@@ -7,21 +7,18 @@ class Lifer::Entry::HTML < Lifer::Entry
|
|
7
7
|
self.input_extensions = ["html", "html.erb", "html.liquid"]
|
8
8
|
self.output_extension = :html
|
9
9
|
|
10
|
-
#
|
11
|
-
#
|
12
|
-
# just give them a default date to start.
|
10
|
+
# If there is no available metadata in the HTML file, we can extract a
|
11
|
+
# makeshift title from the permalink.
|
13
12
|
#
|
14
|
-
#
|
15
|
-
|
16
|
-
|
17
|
-
#
|
18
|
-
# a title from the permalink. Depending on the filename and URI strategy being
|
19
|
-
# used for the collection, it's possible that the extracted title would be
|
20
|
-
# "index", which is not very descriptive. If that's the case, we attempt to go
|
21
|
-
# up a directory to find a "non-index" title.
|
13
|
+
# Depending on the filename and URI strategy being used for the collection,
|
14
|
+
# it's possible that the extracted title would be "index", which is not very
|
15
|
+
# descriptive. If that's the case, we attempt to go up a directory to find a
|
16
|
+
# non-"index" title.
|
22
17
|
#
|
23
|
-
# @return [String] The extracted title of the entry.
|
18
|
+
# @return [String] The given or extracted title of the entry.
|
24
19
|
def title
|
20
|
+
return frontmatter[:title] if frontmatter[:title]
|
21
|
+
|
25
22
|
candidate = File.basename(permalink, ".html")
|
26
23
|
|
27
24
|
if candidate.include?("index") && !file.to_s.include?("index")
|
@@ -35,5 +32,5 @@ class Lifer::Entry::HTML < Lifer::Entry
|
|
35
32
|
# doesn't do much here.
|
36
33
|
#
|
37
34
|
# @return [String]
|
38
|
-
def to_html =
|
35
|
+
def to_html = body
|
39
36
|
end
|
data/lib/lifer/entry/markdown.rb
CHANGED
@@ -14,78 +14,10 @@ require_relative "../utilities"
|
|
14
14
|
# it may make sense to pull some of these methods into a separate module.
|
15
15
|
#
|
16
16
|
class Lifer::Entry::Markdown < Lifer::Entry
|
17
|
-
# If a filename contains a date, we should expect it to be in the following
|
18
|
-
# format.
|
19
|
-
#
|
20
|
-
FILENAME_DATE_FORMAT = /^(\d{4}-\d{1,2}-\d{1,2})-/
|
21
|
-
|
22
|
-
# We expect frontmatter to be provided in the following format.
|
23
|
-
#
|
24
|
-
FRONTMATTER_REGEX = /^---\n(.*)---\n/m
|
25
|
-
|
26
|
-
# We truncate anything that needs to be truncated (summaries, meta
|
27
|
-
# descriptions) at the following character count.
|
28
|
-
#
|
29
|
-
TRUNCATION_THRESHOLD = 120
|
30
|
-
|
31
17
|
self.include_in_feeds = true
|
32
18
|
self.input_extensions = ["md"]
|
33
19
|
self.output_extension = :html
|
34
20
|
|
35
|
-
# Given the entry's frontmatter, we should be able to get a list of authors.
|
36
|
-
# We always prefer authors (as opposed to a singular author) because it makes
|
37
|
-
# handling both cases easier in the long run.
|
38
|
-
#
|
39
|
-
# The return value here is likely an author's name. Whether that's a full
|
40
|
-
# name, a first name, or a handle is up to the end user.
|
41
|
-
#
|
42
|
-
# @return [Array<String>] An array of authors's names.
|
43
|
-
def authors
|
44
|
-
Array(frontmatter[:author] || frontmatter[:authors]).compact
|
45
|
-
end
|
46
|
-
|
47
|
-
# This method returns the full text of the entry, only removing the
|
48
|
-
# frontmatter. It should not parse anything other than frontmatter.
|
49
|
-
#
|
50
|
-
# @return [String] The body of the entry.
|
51
|
-
def body
|
52
|
-
return full_text.strip unless frontmatter?
|
53
|
-
|
54
|
-
full_text.gsub(FRONTMATTER_REGEX, "").strip
|
55
|
-
end
|
56
|
-
|
57
|
-
# Since Markdown files would only store dates as simple strings, it's nice to
|
58
|
-
# attempt to convert those into Ruby date or datetime objects.
|
59
|
-
#
|
60
|
-
# @return [Time] A Ruby representation of the date and time provided by the
|
61
|
-
# entry frontmatter or filename.
|
62
|
-
def date
|
63
|
-
date_data = frontmatter[:date] || filename_date
|
64
|
-
|
65
|
-
case date_data
|
66
|
-
when Time then date_data
|
67
|
-
when String then DateTime.parse(date_data).to_time
|
68
|
-
else
|
69
|
-
Lifer::Message.log("entry.markdown.no_date_metadata", filename: file)
|
70
|
-
Lifer::Entry::DEFAULT_DATE
|
71
|
-
end
|
72
|
-
rescue ArgumentError => error
|
73
|
-
Lifer::Message.error("entry.markdown.date_error", filename: file, error:)
|
74
|
-
Lifer::Entry::DEFAULT_DATE
|
75
|
-
end
|
76
|
-
|
77
|
-
# Frontmatter is a widely supported YAML metadata block found at the top of
|
78
|
-
# Markdown files. We should attempt to parse Markdown entries for it.
|
79
|
-
#
|
80
|
-
# @return [Hash] A hash representation of the entry frontmatter.
|
81
|
-
def frontmatter
|
82
|
-
return {} unless frontmatter?
|
83
|
-
|
84
|
-
Lifer::Utilities.symbolize_keys(
|
85
|
-
YAML.load(full_text[FRONTMATTER_REGEX, 1], permitted_classes: [Time])
|
86
|
-
)
|
87
|
-
end
|
88
|
-
|
89
21
|
# If given a summary in the frontmatter of the entry, we can use this to
|
90
22
|
# provide a summary. Otherwise, we can truncate the first paragraph and use
|
91
23
|
# that as a summary, although that is a bit annoying. This is useful for
|
@@ -96,7 +28,7 @@ class Lifer::Entry::Markdown < Lifer::Entry
|
|
96
28
|
#
|
97
29
|
# @return [String] A summary of the entry.
|
98
30
|
def summary
|
99
|
-
return
|
31
|
+
return super if super
|
100
32
|
|
101
33
|
return if first_paragraph.nil?
|
102
34
|
return first_paragraph if first_paragraph.length <= TRUNCATION_THRESHOLD
|
@@ -130,11 +62,17 @@ class Lifer::Entry::Markdown < Lifer::Entry
|
|
130
62
|
|
131
63
|
private
|
132
64
|
|
133
|
-
#
|
134
|
-
|
135
|
-
|
136
|
-
|
137
|
-
|
65
|
+
# It is conventional for users to use spaces or commas to delimit tags in
|
66
|
+
# other systems, so let's support that. But let's also support YAML-style
|
67
|
+
# arrays.
|
68
|
+
#
|
69
|
+
# @return [Array<String>] An array of candidate tag names.
|
70
|
+
def candidate_tag_names
|
71
|
+
case frontmatter[:tags]
|
72
|
+
when Array then frontmatter[:tags].map(&:to_s)
|
73
|
+
when String then frontmatter[:tags].split(TAG_DELIMITER_REGEX)
|
74
|
+
else []
|
75
|
+
end.uniq
|
138
76
|
end
|
139
77
|
|
140
78
|
# Using Kramdown we can detect the first paragraph of the entry.
|
@@ -149,11 +87,6 @@ class Lifer::Entry::Markdown < Lifer::Entry
|
|
149
87
|
)
|
150
88
|
end
|
151
89
|
|
152
|
-
# @private
|
153
|
-
def frontmatter?
|
154
|
-
full_text && full_text.match?(FRONTMATTER_REGEX)
|
155
|
-
end
|
156
|
-
|
157
90
|
# @private
|
158
91
|
def kramdown_paragraph_text(kramdown_element)
|
159
92
|
return if kramdown_element.nil?
|
data/lib/lifer/entry/txt.rb
CHANGED
@@ -7,21 +7,18 @@ class Lifer::Entry::TXT < Lifer::Entry
|
|
7
7
|
self.input_extensions = ["txt"]
|
8
8
|
self.output_extension = :txt
|
9
9
|
|
10
|
-
#
|
11
|
-
#
|
12
|
-
# just give them a default date to start.
|
10
|
+
# If there is no available metadata in the text file, we can extract a
|
11
|
+
# makeshift title from the permalink.
|
13
12
|
#
|
14
|
-
#
|
15
|
-
|
16
|
-
|
17
|
-
#
|
18
|
-
# a title from the permalink. Depending on the filename and URI strategy being
|
19
|
-
# used for the collection, it's possible that the extracted title would be
|
20
|
-
# "index", which is not very descriptive. If that's the case, we attempt to go
|
21
|
-
# up a directory to find a "non-index" title.
|
13
|
+
# Depending on the filename and URI strategy being used for the collection,
|
14
|
+
# it's possible that the extracted title would be "index", which is not very
|
15
|
+
# descriptive. If that's the case, we attempt to go up a directory to find a
|
16
|
+
# non-"index" title.
|
22
17
|
#
|
23
|
-
# @return [String] The extracted title of the entry.
|
18
|
+
# @return [String] The given or extracted title of the entry.
|
24
19
|
def title
|
20
|
+
return frontmatter[:title] if frontmatter[:title]
|
21
|
+
|
25
22
|
candidate = File.basename(permalink, ".txt")
|
26
23
|
|
27
24
|
if candidate.include?("index") && !file.to_s.include?("index")
|
@@ -37,5 +34,5 @@ class Lifer::Entry::TXT < Lifer::Entry
|
|
37
34
|
# @fixme Maybe the `#to_html` methods should be renamed, then?
|
38
35
|
#
|
39
36
|
# @return [String] The output HTML (not actually HTML).
|
40
|
-
def to_html =
|
37
|
+
def to_html = body
|
41
38
|
end
|
data/lib/lifer/entry.rb
CHANGED
@@ -1,166 +1,331 @@
|
|
1
1
|
require "digest/sha1"
|
2
2
|
|
3
|
-
|
4
|
-
#
|
5
|
-
#
|
6
|
-
#
|
7
|
-
#
|
8
|
-
#
|
9
|
-
# subclasses.
|
10
|
-
#
|
11
|
-
#
|
12
|
-
#
|
13
|
-
#
|
14
|
-
#
|
15
|
-
#
|
16
|
-
|
17
|
-
class
|
18
|
-
|
19
|
-
|
20
|
-
|
21
|
-
|
3
|
+
module Lifer
|
4
|
+
# An entry is a Lifer file that will be built into the output directory.
|
5
|
+
# There are more than one subclass of entry: Markdown entries are the most
|
6
|
+
# traditional, but HTML and text files are also very valid entries.
|
7
|
+
#
|
8
|
+
# This class provides a baseline of the functionality that all entry
|
9
|
+
# subclasses should implement. It also provides the entry generator for
|
10
|
+
# *all* entry subclasses.
|
11
|
+
#
|
12
|
+
# @fixme Markdown entries are able to provide metadata via frontmatter, but
|
13
|
+
# other entry types do not currently support frontmatter. Should they? Or is
|
14
|
+
# there some nicer way to provide entry metadata for non-Markdown files in
|
15
|
+
# 2024?
|
16
|
+
#
|
17
|
+
class Entry
|
18
|
+
class << self
|
19
|
+
attr_accessor :include_in_feeds
|
20
|
+
attr_accessor :input_extensions
|
21
|
+
attr_accessor :output_extension
|
22
|
+
end
|
22
23
|
|
23
|
-
|
24
|
-
|
25
|
-
|
24
|
+
self.include_in_feeds = false
|
25
|
+
self.input_extensions = []
|
26
|
+
self.output_extension = nil
|
26
27
|
|
27
|
-
|
28
|
-
|
29
|
-
|
28
|
+
require_relative "entry/html"
|
29
|
+
require_relative "entry/markdown"
|
30
|
+
require_relative "entry/txt"
|
30
31
|
|
31
|
-
|
32
|
-
|
33
|
-
|
34
|
-
|
32
|
+
# We provide a default date for entries that have no date and entry types that
|
33
|
+
# otherwise could not have a date due to no real way of getting that metadata.
|
34
|
+
#
|
35
|
+
DEFAULT_DATE = Time.new(1900, 01, 01, 0, 0, 0, "+00:00")
|
36
|
+
|
37
|
+
# If a filename contains a date, we should expect it to be in the following
|
38
|
+
# format.
|
39
|
+
#
|
40
|
+
FILENAME_DATE_FORMAT = /^(\d{4}-\d{1,2}-\d{1,2})-/
|
41
|
+
|
42
|
+
# We expect frontmatter to be provided in the following format.
|
43
|
+
#
|
44
|
+
FRONTMATTER_REGEX = /^---\n(.*)---\n/m
|
35
45
|
|
36
|
-
|
46
|
+
# If tags are represented in YAML frontmatter as a string, they're split on
|
47
|
+
# commas and/or spaces.
|
48
|
+
#
|
49
|
+
TAG_DELIMITER_REGEX = /[,\s]+/
|
37
50
|
|
38
|
-
|
39
|
-
#
|
40
|
-
# `Lifer::Entry` records: only subclasses.
|
51
|
+
# We truncate anything that needs to be truncated (summaries, meta
|
52
|
+
# descriptions) at the following character count.
|
41
53
|
#
|
42
|
-
|
43
|
-
|
44
|
-
|
45
|
-
|
46
|
-
|
54
|
+
TRUNCATION_THRESHOLD = 120
|
55
|
+
|
56
|
+
attr_reader :file, :collection
|
57
|
+
|
58
|
+
class << self
|
59
|
+
# The entrypoint for generating entry objects. We should never end up with
|
60
|
+
# `Lifer::Entry` records: only subclasses.
|
61
|
+
#
|
62
|
+
# @param file [String] The absolute filename of an entry file.
|
63
|
+
# @param collection [Lifer::Collection] The collection for the entry.
|
64
|
+
# @return [Lifer::Entry] An entry.
|
65
|
+
def generate(file:, collection:)
|
66
|
+
error!(file) unless File.exist?(file)
|
67
|
+
|
68
|
+
if (new_entry = subclass_for(file)&.new(file:, collection:))
|
69
|
+
Lifer.entry_manifest << new_entry
|
70
|
+
new_entry.tags
|
71
|
+
end
|
72
|
+
|
73
|
+
new_entry
|
74
|
+
end
|
75
|
+
|
76
|
+
# Whenever an entry is generated it should be added to the entry manifest.
|
77
|
+
# This lets us get a list of all generated entries.
|
78
|
+
#
|
79
|
+
# @return [Array<Lifer::Entry>] A list of all entries that currently exist.
|
80
|
+
def manifest
|
81
|
+
return Lifer.entry_manifest if self == Lifer::Entry
|
82
|
+
|
83
|
+
Lifer.entry_manifest.select { |entry| entry.class == self }
|
84
|
+
end
|
85
|
+
|
86
|
+
# Checks whether the given filename is supported entry type (using only its
|
87
|
+
# file extension).
|
88
|
+
#
|
89
|
+
# @param filename [String] The absolute filename to an entry.
|
90
|
+
# @param file_extensions [Array<String>] An array of file extensions to
|
91
|
+
# check against.
|
92
|
+
# @return [Boolean]
|
93
|
+
def supported?(filename, file_extensions= supported_file_extensions)
|
94
|
+
file_extensions.any? { |ext| filename.end_with? ext }
|
95
|
+
end
|
47
96
|
|
48
|
-
|
49
|
-
|
97
|
+
private
|
98
|
+
|
99
|
+
def supported_file_extensions
|
100
|
+
@supported_file_extensions ||= subclasses.flat_map(&:input_extensions)
|
101
|
+
end
|
102
|
+
|
103
|
+
# @private
|
104
|
+
# Retrieve the entry subclass based on the current filename.
|
105
|
+
#
|
106
|
+
# @param filename [String] The current entry's filename.
|
107
|
+
# @return [Class] The entry subclass for the current entry.
|
108
|
+
def subclass_for(filename)
|
109
|
+
Lifer::Entry.subclasses.detect { |klass|
|
110
|
+
klass.input_extensions.any? { |ext| filename.end_with? ext }
|
111
|
+
}
|
112
|
+
end
|
113
|
+
|
114
|
+
# @private
|
115
|
+
def error!(file)
|
116
|
+
raise StandardError, I18n.t("entry.not_found", file:)
|
50
117
|
end
|
51
|
-
new_entry
|
52
118
|
end
|
53
119
|
|
54
|
-
#
|
55
|
-
#
|
120
|
+
# When a new entry is initialized we expect the file to already exist, and
|
121
|
+
# we expect to know which `Lifer::Collection` it belongs to.
|
56
122
|
#
|
57
|
-
# @
|
58
|
-
|
59
|
-
|
123
|
+
# @param file [String] An absolute path to a file.
|
124
|
+
# @param collection [Lifer::Collection] A collection.
|
125
|
+
# @return [Lifer::Entry]
|
126
|
+
def initialize(file:, collection:)
|
127
|
+
@file = Pathname file
|
128
|
+
@collection = collection
|
129
|
+
end
|
60
130
|
|
61
|
-
|
131
|
+
# Given the entry's frontmatter, we should be able to get a list of authors.
|
132
|
+
# We always prefer authors (as opposed to a singular author) because it makes
|
133
|
+
# handling both cases easier in the long run.
|
134
|
+
#
|
135
|
+
# The return value here is likely an author's name. Whether that's a full
|
136
|
+
# name, a first name, or a handle is up to the end user.
|
137
|
+
#
|
138
|
+
# @return [Array<String>] An array of authors's names.
|
139
|
+
def authors
|
140
|
+
Array(frontmatter[:author] || frontmatter[:authors]).compact
|
62
141
|
end
|
63
142
|
|
64
|
-
#
|
65
|
-
#
|
143
|
+
# This method returns the full text of the entry, only removing the
|
144
|
+
# frontmatter. It should not parse anything other than frontmatter.
|
66
145
|
#
|
67
|
-
# @
|
68
|
-
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
file_extensions.any? { |ext| filename.end_with? ext }
|
146
|
+
# @return [String] The body of the entry.
|
147
|
+
def body
|
148
|
+
return full_text.strip unless frontmatter?
|
149
|
+
|
150
|
+
full_text.gsub(FRONTMATTER_REGEX, "").strip
|
73
151
|
end
|
74
152
|
|
75
|
-
|
153
|
+
def feedable?
|
154
|
+
if (setting = self.class.include_in_feeds).nil?
|
155
|
+
raise NotImplementedError,
|
156
|
+
I18n.t("entry.feedable_error", entry_class: self.class)
|
157
|
+
end
|
76
158
|
|
77
|
-
|
78
|
-
@supported_file_extensions ||= subclasses.flat_map(&:input_extensions)
|
159
|
+
setting
|
79
160
|
end
|
80
161
|
|
81
|
-
#
|
82
|
-
#
|
162
|
+
# Frontmatter is a widely supported YAML metadata block found at the top of
|
163
|
+
# text--often Markdown--files. We attempt to parse all entries for
|
164
|
+
# frontmatter.
|
83
165
|
#
|
84
|
-
# @
|
85
|
-
|
86
|
-
|
87
|
-
|
88
|
-
|
89
|
-
|
166
|
+
# @return [Hash] A hash representation of the entry frontmatter.
|
167
|
+
def frontmatter
|
168
|
+
return {} unless frontmatter?
|
169
|
+
|
170
|
+
Lifer::Utilities.symbolize_keys(
|
171
|
+
YAML.load(full_text[FRONTMATTER_REGEX, 1], permitted_classes: [Time])
|
172
|
+
)
|
90
173
|
end
|
91
174
|
|
92
|
-
#
|
93
|
-
|
94
|
-
|
175
|
+
# The full text of the entry.
|
176
|
+
#
|
177
|
+
# @return [String]
|
178
|
+
def full_text
|
179
|
+
@full_text ||= File.readlines(file).join if file
|
95
180
|
end
|
96
|
-
end
|
97
181
|
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
|
106
|
-
@
|
107
|
-
|
182
|
+
# Using the current Lifer configuration, we can calculate the expected
|
183
|
+
# permalink for the entry. For example:
|
184
|
+
#
|
185
|
+
# https://example.com/index.html
|
186
|
+
# https://example.com/blog/my-trip-to-toronto.html
|
187
|
+
#
|
188
|
+
# This would be useful for indexes and feeds and so on.
|
189
|
+
#
|
190
|
+
# @return [String] A permalink to the current entry.
|
191
|
+
def permalink(host: Lifer.setting(:global, :host))
|
192
|
+
cached_permalink_variable =
|
193
|
+
"@entry_permalink_" + Digest::SHA1.hexdigest(host)
|
108
194
|
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
|
195
|
+
instance_variable_get(cached_permalink_variable) ||
|
196
|
+
instance_variable_set(
|
197
|
+
cached_permalink_variable,
|
198
|
+
File.join(
|
199
|
+
host,
|
200
|
+
Lifer::URIStrategy.find(collection.setting :uri_strategy)
|
201
|
+
.new(root: Lifer.root)
|
202
|
+
.output_file(self)
|
203
|
+
)
|
204
|
+
)
|
113
205
|
end
|
114
206
|
|
115
|
-
|
116
|
-
|
207
|
+
# The expected, absolute URI path to the entry. For example:
|
208
|
+
#
|
209
|
+
# /index.html
|
210
|
+
# /blog/my-trip-to-toronto.html
|
211
|
+
#
|
212
|
+
# @return [String] The absolute URI path to the entry.
|
213
|
+
def path = permalink(host: "/")
|
117
214
|
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
215
|
+
# The entry's publication date. The published date can be inferred in a few
|
216
|
+
# ways. The priority is:
|
217
|
+
#
|
218
|
+
# 1. the frontmatter's `published_at` field
|
219
|
+
# 2. the frontmatter's `published` field
|
220
|
+
# 3. the frontamtter's `date` field
|
221
|
+
# 4. The date in the filename.
|
222
|
+
#
|
223
|
+
# Since text files would only store dates as simple strings, it's nice to
|
224
|
+
# attempt to convert those into Ruby date or datetime objects.
|
225
|
+
#
|
226
|
+
# @return [Time] A Ruby representation of the date and time provided by the
|
227
|
+
# entry frontmatter or filename.
|
228
|
+
def published_at
|
229
|
+
date_for frontmatter[:published_at],
|
230
|
+
frontmatter[:published],
|
231
|
+
frontmatter[:date],
|
232
|
+
filename_date,
|
233
|
+
missing_metadata_translation_key: "entry.no_published_at_metadata"
|
234
|
+
end
|
124
235
|
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
|
135
|
-
cached_permalink_variable =
|
136
|
-
"@entry_permalink_" + Digest::SHA1.hexdigest(host)
|
137
|
-
|
138
|
-
instance_variable_get(cached_permalink_variable) ||
|
139
|
-
instance_variable_set(
|
140
|
-
cached_permalink_variable,
|
141
|
-
File.join(
|
142
|
-
host,
|
143
|
-
Lifer::URIStrategy.find(collection.setting :uri_strategy)
|
144
|
-
.new(root: Lifer.root)
|
145
|
-
.output_file(self)
|
146
|
-
)
|
147
|
-
)
|
148
|
-
end
|
236
|
+
# If given a summary in the frontmatter of the entry, we can use this to
|
237
|
+
# provide a summary.
|
238
|
+
#
|
239
|
+
# Since subclasses may have more sophisticated access to the document, they
|
240
|
+
# may override this method with their own distinct implementations.
|
241
|
+
##
|
242
|
+
# @return [String] A summary of the entry.
|
243
|
+
def summary
|
244
|
+
return frontmatter[:summary] if frontmatter[:summary]
|
245
|
+
end
|
149
246
|
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
247
|
+
# Locates and returns all tags defined in the entry.
|
248
|
+
#
|
249
|
+
# @return [Array<Lifer::Tag>] The entry's tags.
|
250
|
+
def tags
|
251
|
+
@tags ||= candidate_tag_names
|
252
|
+
.map { Lifer::Tag.build_or_update(name: _1, entries: [self]) }
|
253
|
+
end
|
157
254
|
|
158
|
-
|
159
|
-
|
160
|
-
|
255
|
+
# Returns the title of the entry. Every entry subclass must implement this
|
256
|
+
# method so that builders have access to *some* kind of title for each entry.
|
257
|
+
#
|
258
|
+
# @return [String]
|
259
|
+
def title
|
260
|
+
raise NotImplementedError, I18n.t("shared.not_implemented_method")
|
261
|
+
end
|
262
|
+
|
263
|
+
def to_html
|
264
|
+
raise NotImplementedError, I18n.t("shared.not_implemented_method")
|
265
|
+
end
|
161
266
|
|
162
|
-
|
163
|
-
|
267
|
+
# The entry's last updated date. In the frontmatter, the last updated date
|
268
|
+
# can be specified using one of two fields. In priority order:
|
269
|
+
#
|
270
|
+
# 1. the `updated_at` field
|
271
|
+
# 2. the `updated` field
|
272
|
+
#
|
273
|
+
# The developer could set a fallback value as a fallback. For example, when
|
274
|
+
# building RSS feeds one might want the value of `#published_at` if there is
|
275
|
+
# no last updated date.
|
276
|
+
#
|
277
|
+
# @param fallback [Time, String, NilClass] Provide datetime data, a string
|
278
|
+
# that parses to a datetime object, or nil.
|
279
|
+
# @return [Time] A Ruby representation of the date and time provided by the
|
280
|
+
# entry frontmatter.
|
281
|
+
def updated_at(fallback: nil)
|
282
|
+
date_for frontmatter[:updated_at],
|
283
|
+
frontmatter[:updated],
|
284
|
+
default_date: fallback
|
285
|
+
end
|
286
|
+
|
287
|
+
private
|
288
|
+
|
289
|
+
# It is conventional for users to use spaces or commas to delimit tags in
|
290
|
+
# other systems, so let's support that. But let's also support YAML-style
|
291
|
+
# arrays.
|
292
|
+
#
|
293
|
+
# @return [Array<String>] An array of candidate tag names.
|
294
|
+
def candidate_tag_names
|
295
|
+
case frontmatter[:tags]
|
296
|
+
when Array then frontmatter[:tags].map(&:to_s)
|
297
|
+
when String then frontmatter[:tags].split(TAG_DELIMITER_REGEX)
|
298
|
+
else []
|
299
|
+
end.uniq
|
300
|
+
end
|
301
|
+
|
302
|
+
def date_for(
|
303
|
+
*candidate_date_fields,
|
304
|
+
default_date: Lifer::Entry::DEFAULT_DATE,
|
305
|
+
missing_metadata_translation_key: nil
|
306
|
+
)
|
307
|
+
date_data = candidate_date_fields.detect(&:itself)
|
308
|
+
|
309
|
+
case date_data
|
310
|
+
when Time then date_data
|
311
|
+
when String then DateTime.parse(date_data).to_time
|
312
|
+
else
|
313
|
+
if (translation_string = missing_metadata_translation_key)
|
314
|
+
Lifer::Message.log(translation_string, filename: file)
|
315
|
+
end
|
316
|
+
default_date
|
317
|
+
end
|
318
|
+
rescue ArgumentError => error
|
319
|
+
Lifer::Message.error("entry.date_error", filename: file, error:)
|
320
|
+
default_date
|
321
|
+
end
|
322
|
+
|
323
|
+
def filename_date
|
324
|
+
return unless file && File.basename(file).match?(FILENAME_DATE_FORMAT)
|
325
|
+
|
326
|
+
File.basename(file).match(FILENAME_DATE_FORMAT)[1]
|
327
|
+
end
|
328
|
+
|
329
|
+
def frontmatter? = (full_text && full_text.match?(FRONTMATTER_REGEX))
|
164
330
|
end
|
165
331
|
end
|
166
|
-
|
data/lib/lifer/tag.rb
ADDED
@@ -0,0 +1,53 @@
|
|
1
|
+
module Lifer
|
2
|
+
# A tag is a way to categorize entries. You've likely encountered tags in
|
3
|
+
# other software before. In Lifer, tags are sort of the inverse of
|
4
|
+
# collections. It's a nice way to associate entries across many collections.
|
5
|
+
#
|
6
|
+
# Because tags are used to link entries, we definitely do not want duplicate
|
7
|
+
# tags. So the only way to build or retrieve tags is via the
|
8
|
+
# `.build_or_update` class method, which helps us responsibly manage the
|
9
|
+
# global tag manifest.
|
10
|
+
#
|
11
|
+
class Tag
|
12
|
+
class << self
|
13
|
+
# Builds or updates a Lifer tag. On update, its list of entries gets
|
14
|
+
# freshened.
|
15
|
+
#
|
16
|
+
# @param name [String] The name of the tag.
|
17
|
+
# @param entries [Array<Lifer::Entry>] A list of entries that should be
|
18
|
+
# associated with the tag. This parameter is not a true writer, in that
|
19
|
+
# if the tag already exists, old entry associations won't be removed--
|
20
|
+
# only appended to.
|
21
|
+
# @return [Lifer:Tag] The new or updated tag.
|
22
|
+
def build_or_update(name:, entries: [])
|
23
|
+
update(name:, entries:) || build(name:, entries:)
|
24
|
+
end
|
25
|
+
|
26
|
+
private
|
27
|
+
|
28
|
+
def build(name:, entries:)
|
29
|
+
if (new_tag = new(name:, entries:))
|
30
|
+
Lifer.tag_manifest << new_tag
|
31
|
+
end
|
32
|
+
new_tag || false
|
33
|
+
end
|
34
|
+
|
35
|
+
def update(name:, entries:)
|
36
|
+
if (tag = Lifer.tags.detect { _1.name == name })
|
37
|
+
tag.instance_variable_set :@entries,
|
38
|
+
(tag.instance_variable_get(:@entries) | entries)
|
39
|
+
end
|
40
|
+
tag || false
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
attr_accessor :name
|
45
|
+
|
46
|
+
attr_reader :entries
|
47
|
+
|
48
|
+
def initialize(name:, entries:)
|
49
|
+
@name = name
|
50
|
+
@entries = entries
|
51
|
+
end
|
52
|
+
end
|
53
|
+
end
|
data/lib/lifer/utilities.rb
CHANGED
@@ -76,7 +76,10 @@ module Lifer::Utilities
|
|
76
76
|
# @raise [Exception] Any exception thrown by a child process.
|
77
77
|
# @return [Array] The mapped results of the operation.
|
78
78
|
def parallelized(collection, &block)
|
79
|
-
|
79
|
+
options = {}
|
80
|
+
options[:in_threads] = 0 if Lifer.parallelism_disabled?
|
81
|
+
|
82
|
+
results = Parallel.map(collection, **options) do |collection_item|
|
80
83
|
begin
|
81
84
|
yield collection_item
|
82
85
|
rescue => error
|
data/lib/lifer/version.rb
CHANGED
data/lib/lifer.rb
CHANGED
@@ -96,6 +96,16 @@ module Lifer
|
|
96
96
|
# project would be built to.
|
97
97
|
def output_directory = brain.output_directory
|
98
98
|
|
99
|
+
# Returns false if the Lifer project will be built with parallelism. This
|
100
|
+
# should return false almost always--unless you've explicitly set the
|
101
|
+
# `LIFER_UNPARALLELIZED` environment variable before running the program.
|
102
|
+
#
|
103
|
+
# This method is used internally by Lifer to determine whether features that
|
104
|
+
# would normally run in parallel should *not* run in parallel for some reason.
|
105
|
+
#
|
106
|
+
# @return [boolean] Returns whether parallelism is disabled.
|
107
|
+
def parallelism_disabled? = ENV["LIFER_UNPARALLELIZED"].is_a?(String)
|
108
|
+
|
99
109
|
# Register new settings so that they are "safe" and can be read from a Lifer
|
100
110
|
# configuration file. Unregistered settings are ignored.
|
101
111
|
#
|
@@ -141,6 +151,17 @@ module Lifer
|
|
141
151
|
#
|
142
152
|
# @return [Hash] The `Lifer::Config#settings`.
|
143
153
|
def settings = brain.config.settings
|
154
|
+
|
155
|
+
# All of the tags represented in Lifer entries for the current project.
|
156
|
+
#
|
157
|
+
# @return [Array<Lifer::Tag>] The complete list of tags.
|
158
|
+
def tags = brain.tags
|
159
|
+
|
160
|
+
# A set of all tags added to the project. Prefer using the `#tags` method
|
161
|
+
# for tag queries.
|
162
|
+
#
|
163
|
+
# @return [Set<Lifer::Tag>] The complete set of tags.
|
164
|
+
def tag_manifest = brain.tag_manifest
|
144
165
|
end
|
145
166
|
end
|
146
167
|
|
@@ -158,4 +179,5 @@ require_relative "lifer/builder"
|
|
158
179
|
require_relative "lifer/collection"
|
159
180
|
require_relative "lifer/entry"
|
160
181
|
require_relative "lifer/message"
|
182
|
+
require_relative "lifer/tag"
|
161
183
|
require_relative "lifer/uri_strategy"
|
data/lifer.gemspec
CHANGED
@@ -17,9 +17,9 @@ Gem::Specification.new do |spec|
|
|
17
17
|
spec.metadata["allowed_push_host"] = "https://rubygems.org"
|
18
18
|
|
19
19
|
spec.metadata["homepage_uri"] =
|
20
|
-
"%s/blob/%s/README.md" % [spec.homepage, Lifer::VERSION]
|
20
|
+
"%s/blob/%s/README.md" % [spec.homepage, "v#{Lifer::VERSION}"]
|
21
21
|
spec.metadata["source_code_uri"] =
|
22
|
-
"%s/tree/%s" % [spec.homepage, Lifer::VERSION]
|
22
|
+
"%s/tree/%s" % [spec.homepage, "v#{Lifer::VERSION}"]
|
23
23
|
spec.metadata["changelog_uri"] = "%s/blob/main/CHANGELOG.md" % spec.homepage
|
24
24
|
|
25
25
|
# Specify which files should be added to the gem when it is released. The
|
data/locales/en.yml
CHANGED
@@ -30,11 +30,10 @@ en:
|
|
30
30
|
content_type_not_implemented: no content type defined for files like %{path} yet
|
31
31
|
four_oh_four: 404 Not Found
|
32
32
|
entry:
|
33
|
+
date_error: "[%{filename}]: %{error}"
|
33
34
|
feedable_error: >
|
34
35
|
please set `%{entry_class}.include_in_feeds` to true or false
|
35
|
-
|
36
|
-
date_error: "[%{filename}]: %{error}"
|
37
|
-
no_date_metadata: "[%{filename}]: no date metadata"
|
36
|
+
no_published_at_metadata: "[%{filename}]: no `published_at` metadata"
|
38
37
|
not_found: >
|
39
38
|
file "%{file}" does not exist
|
40
39
|
config:
|
metadata
CHANGED
@@ -1,14 +1,14 @@
|
|
1
1
|
--- !ruby/object:Gem::Specification
|
2
2
|
name: lifer
|
3
3
|
version: !ruby/object:Gem::Version
|
4
|
-
version: 0.
|
4
|
+
version: 0.9.0
|
5
5
|
platform: ruby
|
6
6
|
authors:
|
7
7
|
- benjamin wil
|
8
8
|
autorequire:
|
9
9
|
bindir: bin
|
10
10
|
cert_chain: []
|
11
|
-
date: 2025-03-
|
11
|
+
date: 2025-03-30 00:00:00.000000000 Z
|
12
12
|
dependencies:
|
13
13
|
- !ruby/object:Gem::Dependency
|
14
14
|
name: i18n
|
@@ -167,6 +167,8 @@ files:
|
|
167
167
|
- lib/lifer/builder/html/from_liquid/drops/entry_drop.rb
|
168
168
|
- lib/lifer/builder/html/from_liquid/drops/frontmatter_drop.rb
|
169
169
|
- lib/lifer/builder/html/from_liquid/drops/settings_drop.rb
|
170
|
+
- lib/lifer/builder/html/from_liquid/drops/tag_drop.rb
|
171
|
+
- lib/lifer/builder/html/from_liquid/drops/tags_drop.rb
|
170
172
|
- lib/lifer/builder/html/from_liquid/filters.rb
|
171
173
|
- lib/lifer/builder/html/from_liquid/layout_tag.rb
|
172
174
|
- lib/lifer/builder/html/from_liquid/liquid_env.rb
|
@@ -188,6 +190,7 @@ files:
|
|
188
190
|
- lib/lifer/selection/included_in_feeds.rb
|
189
191
|
- lib/lifer/shared.rb
|
190
192
|
- lib/lifer/shared/finder_methods.rb
|
193
|
+
- lib/lifer/tag.rb
|
191
194
|
- lib/lifer/templates/cli.txt.erb
|
192
195
|
- lib/lifer/templates/config.yaml
|
193
196
|
- lib/lifer/templates/its-a-living.png
|
@@ -207,8 +210,8 @@ licenses:
|
|
207
210
|
- MIT
|
208
211
|
metadata:
|
209
212
|
allowed_push_host: https://rubygems.org
|
210
|
-
homepage_uri: https://github.com/benjaminwil/lifer/blob/
|
211
|
-
source_code_uri: https://github.com/benjaminwil/lifer/tree/
|
213
|
+
homepage_uri: https://github.com/benjaminwil/lifer/blob/v0.9.0/README.md
|
214
|
+
source_code_uri: https://github.com/benjaminwil/lifer/tree/v0.9.0
|
212
215
|
changelog_uri: https://github.com/benjaminwil/lifer/blob/main/CHANGELOG.md
|
213
216
|
post_install_message:
|
214
217
|
rdoc_options: []
|