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 CHANGED
@@ -1,7 +1,7 @@
1
1
  ---
2
2
  SHA256:
3
- metadata.gz: a2f61316d5e10ff4212470b5539741a4204f35976f97d193ef7aabfd944c62a5
4
- data.tar.gz: 03b9aafb098ab04f2aa1306fa34512d9627801d468910fb48cd5e10a736b7e47
3
+ metadata.gz: 1b6414082e8f6f1d9b18034f2f227890b85ac73b22ab50c25cb2e350812c1a90
4
+ data.tar.gz: 5891afc57d102fd67422bf581edc4de839b05efbe10d104529a817391c0e7cad
5
5
  SHA512:
6
- metadata.gz: 5c1f46e3c16622144cc513b3496d4d0bb9f14231d1b92960b63249b3962753355ed92d5a27ed12ad3bd930e0e56ef871b4551bb4d888e11384191f5747580b19
7
- data.tar.gz: bcc5771d87e311dfe52d1c2bc79d9386c8f3677b7f3eafb66100815a9d76b2c8e2b5ce60855424d276c1cc8d8fb5e1c42f2298c5d662edf1ee85d03f8ec1a9c8
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.7.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.1)
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 you to
60
- # make ERB files that contain loops like:
59
+ # Each collection and tag name is provided as a local variable. This allows
60
+ # you to make ERB files that contain loops like:
61
61
  #
62
- # <% my_collection_name.entries.each do |entry| %>
62
+ # <% collections.my_collection_name.entries.each do |entry| %>
63
+ # <%= entry.title %>
64
+ # <% end %>
65
+ #
66
+ # or:
67
+ #
68
+ # <% tags.my_tag_name.entries.each do |entry| %>
63
69
  # <%= entry.title %>
64
70
  # <% end %>
65
71
  #
@@ -75,18 +81,31 @@ class Lifer::Builder::HTML
75
81
  end
76
82
  end
77
83
 
78
- collections = collection_context_class.new Lifer.collections.to_a
84
+ Lifer.tags.each do |tag|
85
+ binding.local_variable_set tag.name, tag
86
+
87
+ tag_context_class.define_method(tag.name) do
88
+ tag
89
+ end
90
+ end
91
+
92
+ collections = collection_context_class.new Lifer.collections
93
+ tags = tag_context_class.new Lifer.tags
79
94
 
80
95
  binding.local_variable_set :collections, collections
81
96
  binding.local_variable_set :settings, Lifer.settings
97
+ binding.local_variable_set :tags, tags
82
98
  binding.local_variable_set :content,
83
99
  ERB.new(entry.to_html).result(binding)
84
100
  }
85
101
  end
86
102
 
87
- # @private
88
103
  def collection_context_class
89
104
  @collection_context_class ||= Class.new(Array) do end
90
105
  end
106
+
107
+ def tag_context_class
108
+ @tag_context_class ||= Class.new(Array) do end
109
+ end
91
110
  end
92
111
  end
@@ -28,7 +28,7 @@ module Lifer::Builder::HTML::FromLiquid::Drops
28
28
  # @return [Array<EntryDrop>]
29
29
  def entries
30
30
  @entries ||= lifer_collection.entries.map {
31
- EntryDrop.new _1, collection: self
31
+ EntryDrop.new _1, collection: self, tags: _1.tags
32
32
  }
33
33
  end
34
34
 
@@ -18,7 +18,6 @@ module Lifer::Builder::HTML::FromLiquid::Drops
18
18
  #
19
19
  # @yield [CollectionDrop] All available collection drops.
20
20
  def each(&block)
21
-
22
21
  collections.each(&block)
23
22
  end
24
23
 
@@ -4,14 +4,17 @@ module Lifer::Builder::HTML::FromLiquid::Drops
4
4
  #
5
5
  # @example Usage
6
6
  # <h1>{{ entry.title }}</h1>
7
- # <small>Published on <datetime>{{ entry.date }}</datetime></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
@@ -13,3 +13,5 @@ require_relative "drops/collections_drop"
13
13
  require_relative "drops/entry_drop"
14
14
  require_relative "drops/frontmatter_drop"
15
15
  require_relative "drops/settings_drop"
16
+ require_relative "drops/tag_drop"
17
+ require_relative "drops/tags_drop"
@@ -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.date | date_to_xmlschema }}
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
- "entry" => Drops::EntryDrop.new(entry, collection:),
74
+ "tags" => tags,
75
+ "entry" => Drops::EntryDrop.new(entry, collection:, tags: entry_tags),
74
76
  "parse_options" => parse_options,
75
77
  "render_options" => render_options,
76
78
  "settings" => Drops::SettingsDrop.new
@@ -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
- DEFAULT_MAkER_FORMAT_NAME = FORMATS[:rss]
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
- DEFAULT_MAkER_FORMAT_NAME
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
- rss_feed_item.updated = Time.now.to_s
224
- rss_feed_item.content_encoded = lifer_entry.to_html
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
 
@@ -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.date }.reverse
60
+ @entries_collection.sort_by { |entry| entry.published_at }.reverse
61
61
  when :oldest
62
- @entries_collection.sort_by { |entry| entry.date }
62
+ @entries_collection.sort_by { |entry| entry.published_at }
63
63
  end
64
64
  )
65
65
  end
@@ -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
@@ -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 frontmatter[:summary] if frontmatter[:summary]
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
- # @private
134
- def filename_date
135
- return unless file && File.basename(file).match?(FILENAME_DATE_FORMAT)
136
-
137
- 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
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?
@@ -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")
@@ -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 = full_text
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
- # 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
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
- if (new_entry = subclass_for(file)&.new(file:, collection:))
49
- Lifer.entry_manifest << new_entry
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
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
- Lifer.entry_manifest.select { |entry| entry.class == self }
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
- # Checks whether the given filename is supported entry type (using only its
65
- # file extension).
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
- # @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 }
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
- private
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
- def supported_file_extensions
78
- @supported_file_extensions ||= subclasses.flat_map(&:input_extensions)
159
+ setting
79
160
  end
80
161
 
81
- # @private
82
- # Retrieve the entry subclass based on the current filename.
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
- # @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
- }
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
- # @private
93
- def error!(file)
94
- raise StandardError, I18n.t("entry.not_found", file:)
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
- # 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
+ # 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
- def feedable?
110
- if (setting = self.class.include_in_feeds).nil?
111
- raise NotImplementedError,
112
- I18n.t("entry.feedable_error", entry_class: self.class)
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
- setting
116
- end
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
- # 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
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
- # 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)
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
- # 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: "/")
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
- def title
159
- raise NotImplementedError, I18n.t("shared.not_implemented_method")
160
- end
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
- def to_html
163
- raise NotImplementedError, I18n.t("shared.not_implemented_method")
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
@@ -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
- results = Parallel.map(collection) do |collection_item|
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
@@ -1,3 +1,3 @@
1
1
  module Lifer
2
- VERSION = "0.7.0"
2
+ VERSION = "0.9.0"
3
3
  end
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
- markdown:
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.7.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-03 00:00:00.000000000 Z
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/0.7.0/README.md
211
- source_code_uri: https://github.com/benjaminwil/lifer/tree/0.7.0
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: []