lifer 0.6.1 → 0.8.0

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (36) hide show
  1. checksums.yaml +4 -4
  2. data/CHANGELOG.md +38 -0
  3. data/Gemfile.lock +1 -1
  4. data/lib/lifer/brain.rb +15 -9
  5. data/lib/lifer/builder/html/from_erb.rb +24 -5
  6. data/lib/lifer/builder/html/from_liquid/drops/collection_drop.rb +3 -3
  7. data/lib/lifer/builder/html/from_liquid/drops/collections_drop.rb +4 -3
  8. data/lib/lifer/builder/html/from_liquid/drops/entry_drop.rb +3 -3
  9. data/lib/lifer/builder/html/from_liquid/drops/frontmatter_drop.rb +2 -3
  10. data/lib/lifer/builder/html/from_liquid/drops/settings_drop.rb +2 -1
  11. data/lib/lifer/builder/html/from_liquid/drops/tag_drop.rb +42 -0
  12. data/lib/lifer/builder/html/from_liquid/drops/tags_drop.rb +43 -0
  13. data/lib/lifer/builder/html/from_liquid/drops.rb +2 -0
  14. data/lib/lifer/builder/html/from_liquid/filters.rb +3 -5
  15. data/lib/lifer/builder/html/from_liquid/layout_tag.rb +1 -2
  16. data/lib/lifer/builder/html/from_liquid.rb +4 -2
  17. data/lib/lifer/builder/html.rb +1 -1
  18. data/lib/lifer/builder/rss.rb +62 -8
  19. data/lib/lifer/builder.rb +2 -1
  20. data/lib/lifer/config.rb +1 -1
  21. data/lib/lifer/dev/response.rb +2 -3
  22. data/lib/lifer/dev/server.rb +2 -3
  23. data/lib/lifer/entry/html.rb +10 -13
  24. data/lib/lifer/entry/markdown.rb +22 -90
  25. data/lib/lifer/entry/txt.rb +11 -14
  26. data/lib/lifer/entry.rb +252 -129
  27. data/lib/lifer/selection.rb +4 -4
  28. data/lib/lifer/shared/finder_methods.rb +1 -2
  29. data/lib/lifer/tag.rb +53 -0
  30. data/lib/lifer/templates/config.yaml +7 -0
  31. data/lib/lifer/utilities.rb +7 -8
  32. data/lib/lifer/version.rb +1 -1
  33. data/lib/lifer.rb +15 -4
  34. data/lifer.gemspec +2 -2
  35. data/locales/en.yml +2 -3
  36. metadata +7 -4
@@ -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
- # FIXME: This could probably get more sophisticated, but at the moment HTML
11
- # entries don't have any way to provide metadata about themselves. So let's
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
- # @return [Time] The publication date of the HTML entry.
15
- def date = Lifer::Entry::DEFAULT_DATE
16
-
17
- # Since HTML entries cannot provide metadata about themselves, we must extract
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 = full_text
35
+ def to_html = body
39
36
  end
@@ -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 course,
10
- # all frontmatter key-values will be available for users to render as they will in
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
- # FIXME: As we add other types of entries, especially ones that use frontmatter,
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 frontmatter[:summary] if frontmatter[:summary]
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
- # FIXME: Before converting a Kramdown document to Markdown, we chould
123
- # convert any relative URLs to absolute ones. This makes it more flexible to
124
- # use HTML output where ever we want, especially in RSS feeds where feed
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
- # @private
135
- def filename_date
136
- return unless file && File.basename(file).match?(FILENAME_DATE_FORMAT)
137
-
138
- File.basename(file).match(FILENAME_DATE_FORMAT)[1]
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?
@@ -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
- # FIXME: This could probably get more sophisticated, but at the moment HTML
11
- # entries don't have any way to provide metadata about themselves. So let's
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
- # @return [Time] The publication date of the HTML entry.
15
- def date = Lifer::Entry::DEFAULT_DATE
16
-
17
- # Since text entries cannot provide metadata about themselves, we must extract
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
- # FIXME: Maybe the `#to_html` methods should be renamed, then?
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 = full_text
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
- # An entry is a Lifer file that will be built into the output directory.
4
- # There are more than one subclass of entry: Markdown entries are the most
5
- # traditional, but HTML and text files are also very valid entries.
6
- #
7
- # This class provides a baseline of the functionality that all entry subclasses
8
- # should implement. It also provides the entry generator for *all* entry
9
- # subclasses.
10
- #
11
- # FIXME: Markdown entries are able to provide metadata via frontmatter, but
12
- # other entry types do not currently support frontmatter. Should they? Or is
13
- # there some nicer way to provide entry metadata for non-Markdown files in
14
- # 2024?
15
- #
16
- class Lifer::Entry
17
- class << self
18
- attr_accessor :include_in_feeds
19
- attr_accessor :input_extensions
20
- attr_accessor :output_extension
21
- end
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
- self.include_in_feeds = false
24
- self.input_extensions = []
25
- self.output_extension = nil
24
+ self.include_in_feeds = false
25
+ self.input_extensions = []
26
+ self.output_extension = nil
26
27
 
27
- require_relative "entry/html"
28
- require_relative "entry/markdown"
29
- require_relative "entry/txt"
28
+ require_relative "entry/html"
29
+ require_relative "entry/markdown"
30
+ require_relative "entry/txt"
30
31
 
31
- # We provide a default date for entries that have no date and entry types that
32
- # otherwise could not have a date due to no real way of getting that metadata.
33
- #
34
- DEFAULT_DATE = Time.new(1900, 01, 01, 0, 0, 0, "+00:00")
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
- attr_reader :file, :collection
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
- class << self
39
- # The entrypoint for generating entry objects. We should never end up with
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
- # @param file [String] The absolute filename of an entry file.
43
- # @param collection [Lifer::Collection] The collection for the entry.
44
- # @return [Lifer::Entry::HTML, Lifer::Entry::Markdown]
45
- def generate(file:, collection:)
46
- error!(file) unless File.exist?(file)
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
- if (new_entry = subclass_for(file)&.new(file:, collection:))
49
- Lifer.entry_manifest << new_entry
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
- # Whenever an entry is generated it should be added to the entry manifest.
55
- # This lets us get a list of all generated entries.
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
- # @return [Array<Lifer::Entry>] A list of all entries that currently exist.
58
- def manifest
59
- return Lifer.entry_manifest if self == Lifer::Entry
60
-
61
- Lifer.entry_manifest.select { |entry| entry.class == self }
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
- # Checks whether the given filename is supported entry type (using only its
65
- # file extension).
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
- # @param filename [String] The absolute filename to an entry.
68
- # @param file_extensions [Array<String>] An array of file extensions to
69
- # check against.
70
- # @return [Boolean]
71
- def supported?(filename, file_extensions= supported_file_extensions)
72
- file_extensions.any? { |ext| filename.end_with? ext }
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
- private
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
- def supported_file_extensions
78
- @supported_file_extensions ||= subclasses.flat_map(&:input_extensions)
150
+ full_text.gsub(FRONTMATTER_REGEX, "").strip
79
151
  end
80
152
 
81
- # @private
82
- # Retrieve the entry subclass based on the current filename.
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
- # @param filename [String] The current entry's filename.
85
- # @return [Class] The entry subclass for the current entry.
86
- def subclass_for(filename)
87
- Lifer::Entry.subclasses.detect { |klass|
88
- klass.input_extensions.any? { |ext| filename.end_with? ext }
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
- # @private
93
- def error!(file)
94
- raise StandardError, I18n.t("entry.not_found", file:)
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
- # When a new Markdown entry is initialized we expect the file to already
99
- # exist, and we expect to know which `Lifer::Collection` it belongs to.
100
- #
101
- # @param file [String] An absolute path to a file.
102
- # @param collection [Lifer::Collection] A collection.
103
- # @return [Lifer::Entry]
104
- def initialize(file:, collection:)
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
- def feedable?
110
- if (setting = self.class.include_in_feeds).nil?
111
- raise NotImplementedError,
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
- setting
116
- end
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
- # The full text of the entry.
119
- #
120
- # @return [String]
121
- def full_text
122
- @full_text ||= File.readlines(file).join if file
123
- end
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
- # Using the current Lifer configuration, we can calculate the expected
126
- # permalink for the entry. For example:
127
- #
128
- # https://example.com/index.html
129
- # https://example.com/blog/my-trip-to-toronto.html
130
- #
131
- # This would be useful for indexes and feeds and so on.
132
- #
133
- # @return [String] A permalink to the current entry.
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
- # The expected, absolute URI path to the entry. For example:
151
- #
152
- # /index.html
153
- # /blog/my-trip-to-toronto.html
154
- #
155
- # @return [String] The absolute URI path to the entry.
156
- def path = permalink(host: "/")
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
- def title
159
- raise NotImplementedError, I18n.t("shared.not_implemented_method")
160
- end
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
- def to_html
163
- raise NotImplementedError, I18n.t("shared.not_implemented_method")
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
-