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
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
@@ -6,98 +6,29 @@ require_relative "../utilities"
|
|
6
6
|
|
7
7
|
# We should initialize each Markdown file in a Lifer project as a
|
8
8
|
# `Lifer::Entry::Markdown` object. This class contains convenience methods for
|
9
|
-
# parsing a Markdown file with frontmatter as a weblog post or article. Of
|
10
|
-
# all frontmatter key-values will be available for users to render as
|
11
|
-
# their template files.
|
9
|
+
# parsing a Markdown file with frontmatter as a weblog post or article. Of
|
10
|
+
# course, all frontmatter key-values will be available for users to render as
|
11
|
+
# they will in their template files.
|
12
12
|
#
|
13
|
-
#
|
13
|
+
# @fixme As we add other types of entries, especially ones that use frontmatter,
|
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
|
-
# FIXME:
|
90
|
-
# This would be easier to test and more appropriate as a module method
|
91
|
-
# takes text and options as arguments.
|
92
|
-
#
|
93
21
|
# If given a summary in the frontmatter of the entry, we can use this to
|
94
22
|
# provide a summary. Otherwise, we can truncate the first paragraph and use
|
95
23
|
# that as a summary, although that is a bit annoying. This is useful for
|
96
24
|
# indexes and feeds and so on.
|
97
25
|
#
|
26
|
+
# @fixme This would be easier to test and more appropriate as a module method
|
27
|
+
# takes text and options as arguments.
|
28
|
+
#
|
98
29
|
# @return [String] A summary of the entry.
|
99
30
|
def summary
|
100
|
-
return
|
31
|
+
return super if super
|
101
32
|
|
102
33
|
return if first_paragraph.nil?
|
103
34
|
return first_paragraph if first_paragraph.length <= TRUNCATION_THRESHOLD
|
@@ -119,9 +50,9 @@ class Lifer::Entry::Markdown < Lifer::Entry
|
|
119
50
|
|
120
51
|
# The HTML representation of the Markdown entry as parsed by Kramdown.
|
121
52
|
#
|
122
|
-
#
|
123
|
-
#
|
124
|
-
#
|
53
|
+
# @fixme Before converting a Kramdown document to Markdown, we should convert
|
54
|
+
# any relative URLs to absolute ones. This makes it more flexible to use
|
55
|
+
# HTML output where ever we want, especially in RSS feeds where feed
|
125
56
|
# readers may "wtf" a relative URL.
|
126
57
|
#
|
127
58
|
# @return [String] The HTML for the body of the entry.
|
@@ -131,11 +62,17 @@ class Lifer::Entry::Markdown < Lifer::Entry
|
|
131
62
|
|
132
63
|
private
|
133
64
|
|
134
|
-
#
|
135
|
-
|
136
|
-
|
137
|
-
|
138
|
-
|
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
|
139
76
|
end
|
140
77
|
|
141
78
|
# Using Kramdown we can detect the first paragraph of the entry.
|
@@ -150,11 +87,6 @@ class Lifer::Entry::Markdown < Lifer::Entry
|
|
150
87
|
)
|
151
88
|
end
|
152
89
|
|
153
|
-
# @private
|
154
|
-
def frontmatter?
|
155
|
-
full_text && full_text.match?(FRONTMATTER_REGEX)
|
156
|
-
end
|
157
|
-
|
158
90
|
# @private
|
159
91
|
def kramdown_paragraph_text(kramdown_element)
|
160
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")
|
@@ -34,8 +31,8 @@ class Lifer::Entry::TXT < Lifer::Entry
|
|
34
31
|
# While we don't actually output text to HTML, we need to implement this
|
35
32
|
# method so that the RSS feed builder can add text files as feed entries.
|
36
33
|
#
|
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,289 @@
|
|
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
|
47
85
|
|
48
|
-
|
49
|
-
|
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
|
96
|
+
|
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
|
-
|
60
|
-
|
61
|
-
|
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
|
62
129
|
end
|
63
130
|
|
64
|
-
#
|
65
|
-
#
|
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.
|
66
134
|
#
|
67
|
-
#
|
68
|
-
#
|
69
|
-
#
|
70
|
-
# @return [
|
71
|
-
def
|
72
|
-
|
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
|
73
141
|
end
|
74
142
|
|
75
|
-
|
143
|
+
# This method returns the full text of the entry, only removing the
|
144
|
+
# frontmatter. It should not parse anything other than frontmatter.
|
145
|
+
#
|
146
|
+
# @return [String] The body of the entry.
|
147
|
+
def body
|
148
|
+
return full_text.strip unless frontmatter?
|
76
149
|
|
77
|
-
|
78
|
-
@supported_file_extensions ||= subclasses.flat_map(&:input_extensions)
|
150
|
+
full_text.gsub(FRONTMATTER_REGEX, "").strip
|
79
151
|
end
|
80
152
|
|
81
|
-
#
|
82
|
-
#
|
153
|
+
# Since text files would only store dates as simple strings, it's nice to
|
154
|
+
# attempt to convert those into Ruby date or datetime objects.
|
83
155
|
#
|
84
|
-
# @
|
85
|
-
#
|
86
|
-
def
|
87
|
-
|
88
|
-
|
89
|
-
|
156
|
+
# @return [Time] A Ruby representation of the date and time provided by the
|
157
|
+
# entry frontmatter or filename.
|
158
|
+
def date
|
159
|
+
date_data = frontmatter[:date] || filename_date
|
160
|
+
|
161
|
+
case date_data
|
162
|
+
when Time then date_data
|
163
|
+
when String then DateTime.parse(date_data).to_time
|
164
|
+
else
|
165
|
+
Lifer::Message.log("entry.no_date_metadata", filename: file)
|
166
|
+
Lifer::Entry::DEFAULT_DATE
|
167
|
+
end
|
168
|
+
rescue ArgumentError => error
|
169
|
+
Lifer::Message.error("entry.date_error", filename: file, error:)
|
170
|
+
Lifer::Entry::DEFAULT_DATE
|
90
171
|
end
|
91
172
|
|
92
|
-
|
93
|
-
|
94
|
-
|
173
|
+
def feedable?
|
174
|
+
if (setting = self.class.include_in_feeds).nil?
|
175
|
+
raise NotImplementedError,
|
176
|
+
I18n.t("entry.feedable_error", entry_class: self.class)
|
177
|
+
end
|
178
|
+
|
179
|
+
setting
|
95
180
|
end
|
96
|
-
end
|
97
181
|
|
98
|
-
|
99
|
-
|
100
|
-
|
101
|
-
|
102
|
-
|
103
|
-
|
104
|
-
|
105
|
-
@file = Pathname file
|
106
|
-
@collection = collection
|
107
|
-
end
|
182
|
+
# Frontmatter is a widely supported YAML metadata block found at the top of
|
183
|
+
# text--often Markdown--files. We attempt to parse all entries for
|
184
|
+
# frontmatter.
|
185
|
+
#
|
186
|
+
# @return [Hash] A hash representation of the entry frontmatter.
|
187
|
+
def frontmatter
|
188
|
+
return {} unless frontmatter?
|
108
189
|
|
109
|
-
|
110
|
-
|
111
|
-
|
112
|
-
I18n.t("entry.feedable_error", entry_class: self.class)
|
190
|
+
Lifer::Utilities.symbolize_keys(
|
191
|
+
YAML.load(full_text[FRONTMATTER_REGEX, 1], permitted_classes: [Time])
|
192
|
+
)
|
113
193
|
end
|
114
194
|
|
115
|
-
|
116
|
-
|
195
|
+
# The full text of the entry.
|
196
|
+
#
|
197
|
+
# @return [String]
|
198
|
+
def full_text
|
199
|
+
@full_text ||= File.readlines(file).join if file
|
200
|
+
end
|
117
201
|
|
118
|
-
|
119
|
-
|
120
|
-
|
121
|
-
|
122
|
-
|
123
|
-
|
202
|
+
# Using the current Lifer configuration, we can calculate the expected
|
203
|
+
# permalink for the entry. For example:
|
204
|
+
#
|
205
|
+
# https://example.com/index.html
|
206
|
+
# https://example.com/blog/my-trip-to-toronto.html
|
207
|
+
#
|
208
|
+
# This would be useful for indexes and feeds and so on.
|
209
|
+
#
|
210
|
+
# @return [String] A permalink to the current entry.
|
211
|
+
def permalink(host: Lifer.setting(:global, :host))
|
212
|
+
cached_permalink_variable =
|
213
|
+
"@entry_permalink_" + Digest::SHA1.hexdigest(host)
|
124
214
|
|
125
|
-
|
126
|
-
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
|
131
|
-
|
132
|
-
|
133
|
-
|
134
|
-
def permalink(host: Lifer.setting(:global, :host))
|
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)
|
215
|
+
instance_variable_get(cached_permalink_variable) ||
|
216
|
+
instance_variable_set(
|
217
|
+
cached_permalink_variable,
|
218
|
+
File.join(
|
219
|
+
host,
|
220
|
+
Lifer::URIStrategy.find(collection.setting :uri_strategy)
|
221
|
+
.new(root: Lifer.root)
|
222
|
+
.output_file(self)
|
223
|
+
)
|
146
224
|
)
|
147
|
-
|
148
|
-
end
|
225
|
+
end
|
149
226
|
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
227
|
+
# The expected, absolute URI path to the entry. For example:
|
228
|
+
#
|
229
|
+
# /index.html
|
230
|
+
# /blog/my-trip-to-toronto.html
|
231
|
+
#
|
232
|
+
# @return [String] The absolute URI path to the entry.
|
233
|
+
def path = permalink(host: "/")
|
157
234
|
|
158
|
-
|
159
|
-
|
160
|
-
|
235
|
+
# If given a summary in the frontmatter of the entry, we can use this to
|
236
|
+
# provide a summary.
|
237
|
+
#
|
238
|
+
# Since subclasses may have more sophisticated access to the document, they
|
239
|
+
# may override this method with their own distinct implementations.
|
240
|
+
##
|
241
|
+
# @return [String] A summary of the entry.
|
242
|
+
def summary
|
243
|
+
return frontmatter[:summary] if frontmatter[:summary]
|
244
|
+
end
|
245
|
+
|
246
|
+
# Locates and returns all tags defined in the entry.
|
247
|
+
#
|
248
|
+
# @return [Array<Lifer::Tag>] The entry's tags.
|
249
|
+
def tags
|
250
|
+
@tags ||= candidate_tag_names
|
251
|
+
.map { Lifer::Tag.build_or_update(name: _1, entries: [self]) }
|
252
|
+
end
|
253
|
+
|
254
|
+
# Returns the title of the entry. Every entry subclass must implement this
|
255
|
+
# method so that builders have access to *some* kind of title for each entry.
|
256
|
+
#
|
257
|
+
# @return [String]
|
258
|
+
def title
|
259
|
+
raise NotImplementedError, I18n.t("shared.not_implemented_method")
|
260
|
+
end
|
261
|
+
|
262
|
+
def to_html
|
263
|
+
raise NotImplementedError, I18n.t("shared.not_implemented_method")
|
264
|
+
end
|
161
265
|
|
162
|
-
|
163
|
-
|
266
|
+
private
|
267
|
+
|
268
|
+
# It is conventional for users to use spaces or commas to delimit tags in
|
269
|
+
# other systems, so let's support that. But let's also support YAML-style
|
270
|
+
# arrays.
|
271
|
+
#
|
272
|
+
# @return [Array<String>] An array of candidate tag names.
|
273
|
+
def candidate_tag_names
|
274
|
+
case frontmatter[:tags]
|
275
|
+
when Array then frontmatter[:tags].map(&:to_s)
|
276
|
+
when String then frontmatter[:tags].split(TAG_DELIMITER_REGEX)
|
277
|
+
else []
|
278
|
+
end.uniq
|
279
|
+
end
|
280
|
+
|
281
|
+
def filename_date
|
282
|
+
return unless file && File.basename(file).match?(FILENAME_DATE_FORMAT)
|
283
|
+
|
284
|
+
File.basename(file).match(FILENAME_DATE_FORMAT)[1]
|
285
|
+
end
|
286
|
+
|
287
|
+
def frontmatter? = (full_text && full_text.match?(FRONTMATTER_REGEX))
|
164
288
|
end
|
165
289
|
end
|
166
|
-
|