chore: more god module refactor
Co-authored-by: Copilot <copilot@github.com>
This commit is contained in:
352
lib/bds/generation/data.ex
Normal file
352
lib/bds/generation/data.ex
Normal file
@@ -0,0 +1,352 @@
|
||||
defmodule BDS.Generation.Data do
|
||||
@moduledoc false
|
||||
|
||||
import BDS.Generation.Paths, only: [local_date_parts!: 1]
|
||||
import BDS.Generation.Progress, only: [report_snapshot_stage_progress: 4]
|
||||
|
||||
alias BDS.DocumentFields
|
||||
alias BDS.Frontmatter
|
||||
alias BDS.Posts.Post
|
||||
alias BDS.Posts.Translation
|
||||
alias BDS.Projects
|
||||
alias BDS.Repo
|
||||
|
||||
import Ecto.Query
|
||||
|
||||
@spec generation_data(map(), keyword()) :: map()
|
||||
def generation_data(plan, opts \\ []) do
|
||||
project = Projects.get_project!(plan.project_id)
|
||||
project_data_dir = Projects.project_data_dir(project)
|
||||
list_excluded_categories = excluded_list_categories(plan)
|
||||
on_snapshot_progress = Keyword.get(opts, :on_snapshot_progress)
|
||||
|
||||
published_candidates =
|
||||
Repo.all(
|
||||
from post in Post,
|
||||
where: post.project_id == ^plan.project_id and post.status == :published,
|
||||
order_by: [desc: post.created_at, desc: post.published_at, asc: post.slug]
|
||||
)
|
||||
|
||||
draft_candidates =
|
||||
Repo.all(
|
||||
from post in Post,
|
||||
where: post.project_id == ^plan.project_id and post.status == :draft,
|
||||
order_by: [desc: post.created_at, desc: post.published_at, asc: post.slug]
|
||||
)
|
||||
|
||||
post_snapshot_candidates = published_candidates ++ draft_candidates
|
||||
|
||||
snapshots_by_id =
|
||||
post_snapshot_candidates
|
||||
|> Enum.with_index(1)
|
||||
|> Enum.reduce(%{}, fn {post, index}, acc ->
|
||||
:ok = report_snapshot_stage_progress(on_snapshot_progress, :posts, index, length(post_snapshot_candidates))
|
||||
|
||||
case published_post_snapshot(project_data_dir, post) do
|
||||
nil -> acc
|
||||
snapshot -> Map.put(acc, post.id, snapshot)
|
||||
end
|
||||
end)
|
||||
|
||||
published_posts =
|
||||
published_candidates
|
||||
|> merge_generation_snapshots(snapshots_by_id)
|
||||
|> then(fn published ->
|
||||
draft_candidates
|
||||
|> merge_generation_snapshots(snapshots_by_id)
|
||||
|> Enum.reduce(Map.new(published, &{&1.id, &1}), fn post, acc -> Map.put(acc, post.id, post) end)
|
||||
|> Map.values()
|
||||
end)
|
||||
|> Enum.sort_by(&{-(&1.created_at || 0), -(&1.published_at || 0), to_string(&1.slug)})
|
||||
|
||||
published_list_posts =
|
||||
(published_candidates ++ draft_candidates)
|
||||
|> Enum.reject(fn post -> list_excluded_post?(post, list_excluded_categories) end)
|
||||
|> merge_generation_snapshots(snapshots_by_id)
|
||||
|> Enum.uniq_by(& &1.id)
|
||||
|> Enum.sort_by(&{-(&1.created_at || 0), -(&1.published_at || 0), to_string(&1.slug)})
|
||||
|
||||
{published_route_posts, translations_by_post} =
|
||||
build_generation_route_posts(
|
||||
plan.project_id,
|
||||
project_data_dir,
|
||||
published_posts,
|
||||
on_snapshot_progress
|
||||
)
|
||||
|
||||
%{
|
||||
project: project,
|
||||
project_data_dir: project_data_dir,
|
||||
published_posts: published_posts,
|
||||
published_list_posts: published_list_posts,
|
||||
published_route_posts: published_route_posts,
|
||||
translations_by_post: translations_by_post,
|
||||
post_index: build_generation_post_index(published_list_posts)
|
||||
}
|
||||
end
|
||||
|
||||
@spec flattened_generation_translations(map()) :: [Translation.t() | map()]
|
||||
def flattened_generation_translations(translations_by_post) do
|
||||
translations_by_post
|
||||
|> Map.values()
|
||||
|> List.flatten()
|
||||
end
|
||||
|
||||
@spec translation_lookup_map([Translation.t() | map()]) :: map()
|
||||
def translation_lookup_map(published_translations) do
|
||||
Map.new(published_translations, fn translation ->
|
||||
{{translation.translation_for, translation.language}, translation}
|
||||
end)
|
||||
end
|
||||
|
||||
@spec resolve_posts_for_language([map()], String.t() | nil, map(), String.t() | nil) :: [map()]
|
||||
def resolve_posts_for_language(posts, target_language, translations_by_post_language, main_language) do
|
||||
target = String.downcase(to_string(target_language || ""))
|
||||
main = String.downcase(to_string(main_language || ""))
|
||||
|
||||
Enum.map(posts, fn post ->
|
||||
post_language = String.downcase(to_string(Map.get(post, :language) || ""))
|
||||
effective_language = if post_language == "", do: main, else: post_language
|
||||
|
||||
cond do
|
||||
is_binary(Map.get(post, :translation_source_slug)) ->
|
||||
post
|
||||
|
||||
effective_language == target ->
|
||||
post
|
||||
|
||||
true ->
|
||||
case Map.get(translations_by_post_language, {post.id, target_language}) do
|
||||
nil -> post
|
||||
translation -> build_localized_subtree_variant(post, translation)
|
||||
end
|
||||
end
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_generation_post_index([map()]) :: map()
|
||||
def build_generation_post_index(posts) do
|
||||
Enum.reduce(posts, %{posts_by_category: %{}, posts_by_tag: %{}, posts_by_year: %{}, posts_by_year_month: %{}, posts_by_year_month_day: %{}}, fn post, acc ->
|
||||
{year, month_value, day_value} = local_date_parts!(post.created_at)
|
||||
month = String.pad_leading(Integer.to_string(month_value), 2, "0")
|
||||
day = String.pad_leading(Integer.to_string(day_value), 2, "0")
|
||||
year_month = "#{year}/#{month}"
|
||||
year_month_day = "#{year}/#{month}/#{day}"
|
||||
|
||||
acc
|
||||
|> append_generation_index(:posts_by_year, year, post)
|
||||
|> append_generation_index(:posts_by_year_month, year_month, post)
|
||||
|> append_generation_index(:posts_by_year_month_day, year_month_day, post)
|
||||
|> then(fn indexed ->
|
||||
indexed = Enum.reduce(post.categories || [], indexed, &append_generation_index(&2, :posts_by_category, &1, post))
|
||||
Enum.reduce(post.tags || [], indexed, &append_generation_index(&2, :posts_by_tag, &1, post))
|
||||
end)
|
||||
end)
|
||||
end
|
||||
|
||||
## --- internals -----------------------------------------------------------
|
||||
|
||||
defp merge_generation_snapshots(posts, snapshots_by_id) do
|
||||
posts
|
||||
|> Enum.map(&Map.get(snapshots_by_id, &1.id))
|
||||
|> Enum.reject(&is_nil/1)
|
||||
end
|
||||
|
||||
defp excluded_list_categories(plan) do
|
||||
plan
|
||||
|> resolved_category_settings()
|
||||
|> Enum.filter(fn {_category, settings} -> settings.render_in_lists == false end)
|
||||
|> Enum.map(&elem(&1, 0))
|
||||
|> MapSet.new()
|
||||
end
|
||||
|
||||
defp resolved_category_settings(plan) do
|
||||
defaults = %{
|
||||
"article" => %{render_in_lists: true, show_title: true},
|
||||
"picture" => %{render_in_lists: true, show_title: true},
|
||||
"aside" => %{render_in_lists: true, show_title: false},
|
||||
"page" => %{render_in_lists: false, show_title: true}
|
||||
}
|
||||
|
||||
Enum.reduce(Map.get(plan, :category_settings, %{}) || %{}, defaults, fn {category, settings}, acc ->
|
||||
Map.put(acc, category, %{
|
||||
render_in_lists: category_setting_flag(settings, :render_in_lists, "render_in_lists", true),
|
||||
show_title: category_setting_flag(settings, :show_title, "show_title", true)
|
||||
})
|
||||
end)
|
||||
end
|
||||
|
||||
defp category_setting_flag(settings, atom_key, string_key, default) do
|
||||
case Map.get(settings, atom_key, Map.get(settings, string_key, default)) do
|
||||
false -> false
|
||||
_other -> true
|
||||
end
|
||||
end
|
||||
|
||||
defp list_excluded_post?(post, excluded_categories) do
|
||||
Enum.any?(post.categories || [], &MapSet.member?(excluded_categories, &1))
|
||||
end
|
||||
|
||||
defp published_post_snapshot(project_data_dir, %Post{} = post) do
|
||||
cond do
|
||||
is_binary(post.file_path) and post.file_path != "" ->
|
||||
project_data_dir
|
||||
|> Path.join(post.file_path)
|
||||
|> read_post_snapshot(post)
|
||||
|
||||
post.status == :published ->
|
||||
post
|
||||
|
||||
true ->
|
||||
nil
|
||||
end
|
||||
end
|
||||
|
||||
defp read_post_snapshot(full_path, %Post{} = fallback_post) do
|
||||
case File.read(full_path) do
|
||||
{:ok, contents} ->
|
||||
{:ok, %{fields: fields}} = Frontmatter.parse_document(contents)
|
||||
|
||||
%Post{fallback_post |
|
||||
id: DocumentFields.get(fields, "id", fallback_post.id),
|
||||
title: DocumentFields.get(fields, "title", fallback_post.title) || "",
|
||||
slug: DocumentFields.fetch!(fields, "slug"),
|
||||
excerpt: Map.get(fields, "excerpt"),
|
||||
content: nil,
|
||||
status: :published,
|
||||
author: Map.get(fields, "author"),
|
||||
language: Map.get(fields, "language", fallback_post.language),
|
||||
do_not_translate: DocumentFields.get(fields, "doNotTranslate", fallback_post.do_not_translate || false),
|
||||
template_slug: DocumentFields.get(fields, "templateSlug", fallback_post.template_slug),
|
||||
created_at: DocumentFields.get(fields, "createdAt", fallback_post.created_at),
|
||||
updated_at: DocumentFields.get(fields, "updatedAt", fallback_post.updated_at),
|
||||
published_at: DocumentFields.get(fields, "publishedAt", fallback_post.published_at),
|
||||
file_path: fallback_post.file_path,
|
||||
tags: Map.get(fields, "tags", fallback_post.tags || []),
|
||||
categories: Map.get(fields, "categories", fallback_post.categories || [])
|
||||
}
|
||||
|
||||
{:error, _reason} ->
|
||||
if fallback_post.status == :published, do: fallback_post, else: nil
|
||||
end
|
||||
end
|
||||
|
||||
defp build_generation_route_posts(project_id, project_data_dir, published_posts, on_snapshot_progress) do
|
||||
source_post_ids = Enum.map(published_posts, & &1.id)
|
||||
|
||||
translation_candidates =
|
||||
Repo.all(
|
||||
from translation in Translation,
|
||||
where: translation.project_id == ^project_id and translation.translation_for in ^source_post_ids,
|
||||
where: translation.status in [:published, :draft],
|
||||
order_by: [asc: translation.translation_for, asc: translation.language]
|
||||
)
|
||||
|
||||
translations_by_post =
|
||||
translation_candidates
|
||||
|> Enum.with_index(1)
|
||||
|> Enum.reduce(%{}, fn {translation, index}, acc ->
|
||||
:ok = report_snapshot_stage_progress(on_snapshot_progress, :translations, index, length(translation_candidates))
|
||||
|
||||
case published_translation_snapshot(project_data_dir, translation) do
|
||||
nil -> acc
|
||||
snapshot -> Map.update(acc, translation.translation_for, [snapshot], &[snapshot | &1])
|
||||
end
|
||||
end)
|
||||
|> Map.new(fn {post_id, translations} -> {post_id, Enum.reverse(translations)} end)
|
||||
|
||||
route_posts =
|
||||
Enum.flat_map(published_posts, fn post ->
|
||||
variants =
|
||||
translations_by_post
|
||||
|> Map.get(post.id, [])
|
||||
|> Enum.map(&build_published_translation_variant(post, &1))
|
||||
|
||||
[post | variants]
|
||||
end)
|
||||
|
||||
{route_posts, translations_by_post}
|
||||
end
|
||||
|
||||
defp published_translation_snapshot(project_data_dir, %Translation{} = translation) do
|
||||
cond do
|
||||
is_binary(translation.file_path) and translation.file_path != "" ->
|
||||
project_data_dir
|
||||
|> Path.join(translation.file_path)
|
||||
|> read_translation_snapshot(translation)
|
||||
|
||||
translation.status == :published ->
|
||||
translation
|
||||
|
||||
true ->
|
||||
nil
|
||||
end
|
||||
end
|
||||
|
||||
defp read_translation_snapshot(full_path, %Translation{} = fallback_translation) do
|
||||
case File.read(full_path) do
|
||||
{:ok, contents} ->
|
||||
{:ok, %{fields: fields}} = Frontmatter.parse_document(contents)
|
||||
|
||||
%Translation{fallback_translation |
|
||||
id: DocumentFields.get(fields, "id", fallback_translation.id),
|
||||
translation_for: DocumentFields.fetch!(fields, "translationFor"),
|
||||
language: DocumentFields.fetch!(fields, "language"),
|
||||
title: DocumentFields.get(fields, "title", fallback_translation.title) || "",
|
||||
excerpt: Map.get(fields, "excerpt", fallback_translation.excerpt),
|
||||
content: nil,
|
||||
status: :published,
|
||||
created_at: DocumentFields.get(fields, "createdAt", fallback_translation.created_at),
|
||||
updated_at: DocumentFields.get(fields, "updatedAt", fallback_translation.updated_at),
|
||||
published_at: DocumentFields.get(fields, "publishedAt", fallback_translation.published_at),
|
||||
file_path: fallback_translation.file_path
|
||||
}
|
||||
|
||||
{:error, _reason} ->
|
||||
if fallback_translation.status == :published, do: fallback_translation, else: nil
|
||||
end
|
||||
end
|
||||
|
||||
defp build_published_translation_variant(post, translation) do
|
||||
%{
|
||||
id: translation.id,
|
||||
project_id: post.project_id,
|
||||
title: translation.title,
|
||||
slug: "#{post.slug}.#{translation.language}",
|
||||
excerpt: translation.excerpt,
|
||||
content: nil,
|
||||
status: :published,
|
||||
author: Map.get(post, :author),
|
||||
created_at: post.created_at,
|
||||
updated_at: translation.updated_at,
|
||||
published_at: translation.published_at || post.published_at,
|
||||
file_path: translation.file_path,
|
||||
tags: Map.get(post, :tags, []),
|
||||
categories: Map.get(post, :categories, []),
|
||||
template_slug: Map.get(post, :template_slug),
|
||||
language: translation.language,
|
||||
do_not_translate: Map.get(post, :do_not_translate, false),
|
||||
translation_source_slug: post.slug,
|
||||
translation_canonical_language: Map.get(post, :language),
|
||||
translation_file_path: translation.file_path
|
||||
}
|
||||
end
|
||||
|
||||
defp build_localized_subtree_variant(post, translation) do
|
||||
%{
|
||||
post
|
||||
| id: translation.id,
|
||||
title: translation.title,
|
||||
excerpt: translation.excerpt,
|
||||
content: translation.content,
|
||||
language: translation.language,
|
||||
updated_at: translation.updated_at,
|
||||
published_at: translation.published_at || post.published_at,
|
||||
file_path: translation.file_path
|
||||
}
|
||||
end
|
||||
|
||||
defp append_generation_index(index, field, key, post) do
|
||||
update_in(index[field], fn grouped -> Map.update(grouped, key, [post], &[post | &1]) end)
|
||||
end
|
||||
end
|
||||
490
lib/bds/generation/outputs.ex
Normal file
490
lib/bds/generation/outputs.ex
Normal file
@@ -0,0 +1,490 @@
|
||||
defmodule BDS.Generation.Outputs do
|
||||
@moduledoc false
|
||||
|
||||
import BDS.Generation.Paths
|
||||
import BDS.Generation.Renderers
|
||||
import BDS.Generation.Sitemap, only: [render_feed: 3, render_atom: 3, render_calendar: 1]
|
||||
|
||||
@spec additional_languages(map()) :: [String.t()]
|
||||
def additional_languages(plan) do
|
||||
Enum.reject(plan.blog_languages, &(&1 == plan.language))
|
||||
end
|
||||
|
||||
@spec route_post_output_path(map(), String.t() | nil) :: String.t()
|
||||
def route_post_output_path(post, nil), do: post_output_path(post)
|
||||
def route_post_output_path(post, ""), do: post_output_path(post)
|
||||
def route_post_output_path(post, route_language), do: post_output_path(post, route_language)
|
||||
|
||||
@spec suppress_subtree_translation_variants([map()], [String.t()]) :: [map()]
|
||||
def suppress_subtree_translation_variants(route_posts, additional_languages) do
|
||||
subtree_languages = MapSet.new(additional_languages)
|
||||
|
||||
Enum.reject(route_posts, fn post ->
|
||||
is_binary(Map.get(post, :translation_source_slug)) and
|
||||
MapSet.member?(subtree_languages, to_string(Map.get(post, :language)))
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_validation_route_paths(map(), [map()], [map()], map(), String.t() | nil) :: [String.t()]
|
||||
def build_validation_route_paths(plan, route_posts, published_list_posts, post_index, route_language) do
|
||||
[
|
||||
core_route_paths(plan, published_list_posts, route_language),
|
||||
page_route_paths(plan, route_posts, route_language),
|
||||
single_route_paths(plan, route_posts, route_language),
|
||||
category_route_paths(plan, post_index.posts_by_category, route_language),
|
||||
tag_route_paths(plan, post_index.posts_by_tag, route_language),
|
||||
date_route_paths(plan, post_index, route_language)
|
||||
]
|
||||
|> List.flatten()
|
||||
|> Enum.uniq()
|
||||
end
|
||||
|
||||
@spec core_route_paths(map(), [map()], String.t() | nil) :: [String.t()]
|
||||
def core_route_paths(plan, published_list_posts, route_language) do
|
||||
if :core in plan.sections do
|
||||
root_route_paths(route_language, length(published_list_posts), plan.max_posts_per_page)
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
@spec page_route_paths(map(), [map()], String.t() | nil) :: [String.t()]
|
||||
def page_route_paths(plan, route_posts, route_language) do
|
||||
if :core in plan.sections do
|
||||
route_posts
|
||||
|> Enum.filter(&("page" in (&1.categories || [])))
|
||||
|> Enum.map(&page_output_path(&1.slug, route_language))
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
@spec single_route_paths(map(), [map()], String.t() | nil) :: [String.t()]
|
||||
def single_route_paths(plan, route_posts, route_language) do
|
||||
if :single in plan.sections do
|
||||
Enum.map(route_posts, &route_post_output_path(&1, route_language))
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
@spec category_route_paths(map(), map(), String.t() | nil) :: [String.t()]
|
||||
def category_route_paths(plan, posts_by_category, route_language) do
|
||||
if :category in plan.sections do
|
||||
Enum.flat_map(posts_by_category, fn {category, posts} ->
|
||||
paginated_archive_paths(
|
||||
route_language,
|
||||
["category", archive_route_segment(category)],
|
||||
length(posts),
|
||||
plan.max_posts_per_page
|
||||
)
|
||||
end)
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
@spec tag_route_paths(map(), map(), String.t() | nil) :: [String.t()]
|
||||
def tag_route_paths(plan, posts_by_tag, route_language) do
|
||||
if :tag in plan.sections do
|
||||
Enum.flat_map(posts_by_tag, fn {tag, posts} ->
|
||||
paginated_archive_paths(
|
||||
route_language,
|
||||
["tag", archive_route_segment(tag)],
|
||||
length(posts),
|
||||
plan.max_posts_per_page
|
||||
)
|
||||
end)
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
@spec date_route_paths(map(), map(), String.t() | nil) :: [String.t()]
|
||||
def date_route_paths(plan, post_index, route_language) do
|
||||
if :date in plan.sections do
|
||||
year_paths =
|
||||
Enum.flat_map(post_index.posts_by_year, fn {year, posts} ->
|
||||
paginated_archive_paths(
|
||||
route_language,
|
||||
[Integer.to_string(year)],
|
||||
length(posts),
|
||||
plan.max_posts_per_page
|
||||
)
|
||||
end)
|
||||
|
||||
month_paths =
|
||||
Enum.flat_map(post_index.posts_by_year_month, fn {year_month, posts} ->
|
||||
[year, month] = String.split(year_month, "/", parts: 2)
|
||||
|
||||
paginated_archive_paths(
|
||||
route_language,
|
||||
[year, month],
|
||||
length(posts),
|
||||
plan.max_posts_per_page
|
||||
)
|
||||
end)
|
||||
|
||||
day_paths =
|
||||
Enum.flat_map(post_index.posts_by_year_month_day, fn {year_month_day, posts} ->
|
||||
[year, month, day] = String.split(year_month_day, "/", parts: 3)
|
||||
|
||||
paginated_archive_paths(
|
||||
route_language,
|
||||
[year, month, day],
|
||||
length(posts),
|
||||
plan.max_posts_per_page
|
||||
)
|
||||
end)
|
||||
|
||||
year_paths ++ month_paths ++ day_paths
|
||||
else
|
||||
[]
|
||||
end
|
||||
end
|
||||
|
||||
@spec build_archive_outputs(map(), map(), map()) :: [{String.t(), iodata()}]
|
||||
def build_archive_outputs(plan, post_index, localized_post_indexes) do
|
||||
category_outputs =
|
||||
if :category in plan.sections do
|
||||
build_category_outputs(plan, post_index.posts_by_category, [plan.language]) ++
|
||||
Enum.flat_map(additional_languages(plan), fn language ->
|
||||
build_category_outputs(
|
||||
plan,
|
||||
Map.get(localized_post_indexes, language, %{posts_by_category: %{}}).posts_by_category,
|
||||
[language]
|
||||
)
|
||||
end)
|
||||
else
|
||||
[]
|
||||
end
|
||||
|
||||
tag_outputs =
|
||||
if :tag in plan.sections do
|
||||
build_tag_outputs(plan, post_index.posts_by_tag, [plan.language]) ++
|
||||
Enum.flat_map(additional_languages(plan), fn language ->
|
||||
build_tag_outputs(
|
||||
plan,
|
||||
Map.get(localized_post_indexes, language, %{posts_by_tag: %{}}).posts_by_tag,
|
||||
[language]
|
||||
)
|
||||
end)
|
||||
else
|
||||
[]
|
||||
end
|
||||
|
||||
date_outputs =
|
||||
if :date in plan.sections do
|
||||
build_date_outputs(plan, post_index, [plan.language]) ++
|
||||
Enum.flat_map(additional_languages(plan), fn language ->
|
||||
build_date_outputs(
|
||||
plan,
|
||||
Map.get(
|
||||
localized_post_indexes,
|
||||
language,
|
||||
%{posts_by_year: %{}, posts_by_year_month: %{}, posts_by_year_month_day: %{}}
|
||||
),
|
||||
[language]
|
||||
)
|
||||
end)
|
||||
else
|
||||
[]
|
||||
end
|
||||
|
||||
category_outputs ++ tag_outputs ++ date_outputs
|
||||
end
|
||||
|
||||
@spec build_category_outputs(map(), map(), [String.t()]) :: [{String.t(), iodata()}]
|
||||
def build_category_outputs(plan, posts_by_category, languages) do
|
||||
Enum.flat_map(posts_by_category, fn {category, posts} ->
|
||||
paginated_posts = Enum.chunk_every(posts, max(plan.max_posts_per_page, 1))
|
||||
category_slug = archive_route_segment(category)
|
||||
|
||||
Enum.with_index(paginated_posts, 1)
|
||||
|> Enum.flat_map(fn {page_posts, page_number} ->
|
||||
Enum.map(languages, fn language ->
|
||||
pagination = %{
|
||||
current_page: page_number,
|
||||
total_pages: length(paginated_posts),
|
||||
total_items: length(posts),
|
||||
items_per_page: max(plan.max_posts_per_page, 1),
|
||||
has_prev_page: page_number > 1,
|
||||
prev_page_href:
|
||||
if(page_number > 1,
|
||||
do:
|
||||
archive_href(
|
||||
route_language(plan.language, language),
|
||||
["category", category_slug],
|
||||
page_number - 1
|
||||
),
|
||||
else: ""
|
||||
),
|
||||
has_next_page: page_number < length(paginated_posts),
|
||||
next_page_href:
|
||||
if(page_number < length(paginated_posts),
|
||||
do:
|
||||
archive_href(
|
||||
route_language(plan.language, language),
|
||||
["category", category_slug],
|
||||
page_number + 1
|
||||
),
|
||||
else: ""
|
||||
)
|
||||
}
|
||||
|
||||
{
|
||||
archive_path(
|
||||
route_language(plan.language, language),
|
||||
["category", category_slug],
|
||||
page_number
|
||||
),
|
||||
render_archive_page(plan, category, page_posts, language, "category", pagination)
|
||||
}
|
||||
end)
|
||||
end)
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_tag_outputs(map(), map(), [String.t()]) :: [{String.t(), iodata()}]
|
||||
def build_tag_outputs(plan, posts_by_tag, languages) do
|
||||
Enum.flat_map(posts_by_tag, fn {tag, posts} ->
|
||||
tag_slug = archive_route_segment(tag)
|
||||
|
||||
build_paginated_archive_outputs(plan, languages, ["tag", tag_slug], posts, fn page_posts, language, pagination ->
|
||||
render_archive_page(plan, tag, page_posts, language, "tag", pagination)
|
||||
end)
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_date_outputs(map(), map(), [String.t()]) :: [{String.t(), iodata()}]
|
||||
def build_date_outputs(plan, post_index, languages) do
|
||||
year_outputs =
|
||||
Enum.flat_map(post_index.posts_by_year, fn {year, posts} ->
|
||||
build_paginated_archive_outputs(plan, languages, [Integer.to_string(year)], posts, fn page_posts, language, pagination ->
|
||||
render_date_archive_page(
|
||||
plan,
|
||||
Integer.to_string(year),
|
||||
%{kind: "year", year: year},
|
||||
page_posts,
|
||||
language,
|
||||
pagination
|
||||
)
|
||||
end)
|
||||
end)
|
||||
|
||||
month_outputs =
|
||||
Enum.flat_map(post_index.posts_by_year_month, fn {year_month, posts} ->
|
||||
[year, month] = String.split(year_month, "/", parts: 2)
|
||||
|
||||
build_paginated_archive_outputs(plan, languages, [year, month], posts, fn page_posts, language, pagination ->
|
||||
render_date_archive_page(
|
||||
plan,
|
||||
"#{year}-#{month}",
|
||||
%{kind: "month", year: String.to_integer(year), month: String.to_integer(month)},
|
||||
page_posts,
|
||||
language,
|
||||
pagination
|
||||
)
|
||||
end)
|
||||
end)
|
||||
|
||||
day_outputs =
|
||||
Enum.flat_map(post_index.posts_by_year_month_day, fn {year_month_day, posts} ->
|
||||
[year, month, day] = String.split(year_month_day, "/", parts: 3)
|
||||
|
||||
build_paginated_archive_outputs(plan, languages, [year, month, day], posts, fn page_posts, language, pagination ->
|
||||
render_date_archive_page(
|
||||
plan,
|
||||
"#{year}-#{month}-#{day}",
|
||||
%{kind: "day", year: String.to_integer(year), month: String.to_integer(month), day: String.to_integer(day)},
|
||||
page_posts,
|
||||
language,
|
||||
pagination
|
||||
)
|
||||
end)
|
||||
end)
|
||||
|
||||
year_outputs ++ month_outputs ++ day_outputs
|
||||
end
|
||||
|
||||
@spec build_core_outputs(map(), [map()], map()) :: [{String.t(), iodata()}]
|
||||
def build_core_outputs(plan, published_posts, localized_posts_by_language) do
|
||||
language = plan.language
|
||||
additional_languages = Enum.reject(plan.blog_languages, &(&1 == language))
|
||||
main_posts = build_list_posts(plan.base_url, published_posts, nil)
|
||||
|
||||
build_root_outputs(plan, language, main_posts) ++
|
||||
[
|
||||
{"404.html", render_not_found_output(plan, language)},
|
||||
{"feed.xml", render_feed(plan, language, published_posts)},
|
||||
{"atom.xml", render_atom(plan, language, published_posts)},
|
||||
{"calendar.json", render_calendar(published_posts)}
|
||||
] ++
|
||||
Enum.flat_map(additional_languages, fn localized_language ->
|
||||
localized_prefix = route_language(plan.language, localized_language)
|
||||
localized_source_posts = Map.get(localized_posts_by_language, localized_language, [])
|
||||
localized_posts = build_list_posts(plan.base_url, localized_source_posts, localized_prefix)
|
||||
|
||||
build_root_outputs(plan, localized_language, localized_posts) ++
|
||||
[
|
||||
{Path.join(localized_language, "404.html"), render_not_found_output(plan, localized_language)},
|
||||
{Path.join(localized_language, "feed.xml"), render_feed(plan, localized_language, localized_source_posts)},
|
||||
{Path.join(localized_language, "atom.xml"), render_atom(plan, localized_language, localized_source_posts)}
|
||||
]
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_page_outputs(String.t(), String.t(), [map()], map(), map()) :: [{String.t(), iodata()}]
|
||||
def build_page_outputs(project_id, main_language, published_posts, translations_by_post_language, localized_posts_by_language) do
|
||||
page_outputs =
|
||||
published_posts
|
||||
|> Enum.filter(&("page" in (&1.categories || [])))
|
||||
|> Enum.map(fn post ->
|
||||
canonical_variant = Map.get(translations_by_post_language, {post.id, main_language}, post)
|
||||
body = load_body(project_id, canonical_variant.file_path, canonical_variant.content)
|
||||
|
||||
{page_output_path(post.slug, nil),
|
||||
render_post_output(
|
||||
project_id,
|
||||
post.template_slug,
|
||||
%{
|
||||
id: canonical_variant.id,
|
||||
title: canonical_variant.title,
|
||||
content: body,
|
||||
slug: post.slug,
|
||||
language: canonical_variant.language,
|
||||
excerpt: canonical_variant.excerpt
|
||||
},
|
||||
fn -> render_post_page(canonical_variant.title, body, post.slug, canonical_variant.language) end
|
||||
)}
|
||||
end)
|
||||
|
||||
translation_page_outputs =
|
||||
localized_posts_by_language
|
||||
|> Enum.flat_map(fn {language, posts} ->
|
||||
posts
|
||||
|> Enum.filter(&("page" in (&1.categories || [])))
|
||||
|> Enum.map(fn post ->
|
||||
body = load_body(project_id, post.file_path, post.content)
|
||||
|
||||
{page_output_path(post.slug, language),
|
||||
render_post_output(
|
||||
project_id,
|
||||
post.template_slug,
|
||||
%{
|
||||
id: post.id,
|
||||
title: post.title,
|
||||
content: body,
|
||||
slug: post.slug,
|
||||
language: Map.get(post, :language),
|
||||
excerpt: post.excerpt
|
||||
},
|
||||
fn -> render_post_page(post.title, body, post.slug, Map.get(post, :language)) end
|
||||
)}
|
||||
end)
|
||||
end)
|
||||
|
||||
page_outputs ++ translation_page_outputs
|
||||
end
|
||||
|
||||
@spec build_root_outputs(map(), String.t(), [map()]) :: [{String.t(), iodata()}]
|
||||
def build_root_outputs(plan, language, posts) do
|
||||
total_pages = page_count(length(posts), plan.max_posts_per_page)
|
||||
|
||||
posts
|
||||
|> paginate_posts(plan.max_posts_per_page)
|
||||
|> Enum.with_index(1)
|
||||
|> Enum.map(fn {page_posts, page_number} ->
|
||||
route_language = route_language(plan.language, language)
|
||||
|
||||
{root_output_path(route_language, page_number),
|
||||
render_list_output(
|
||||
plan,
|
||||
language,
|
||||
plan.project_name,
|
||||
page_posts,
|
||||
%{kind: "core"},
|
||||
pagination_for_page(page_number, total_pages, length(posts), plan.max_posts_per_page, route_language, []),
|
||||
fn -> render_home(plan, language) end
|
||||
)}
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_paginated_archive_outputs(map(), [String.t()], [String.t()], [map()], (... -> iodata())) :: [{String.t(), iodata()}]
|
||||
def build_paginated_archive_outputs(plan, languages, segments, posts, render_fun) do
|
||||
total_pages = page_count(length(posts), plan.max_posts_per_page)
|
||||
|
||||
posts
|
||||
|> paginate_posts(plan.max_posts_per_page)
|
||||
|> Enum.with_index(1)
|
||||
|> Enum.flat_map(fn {page_posts, page_number} ->
|
||||
Enum.map(languages, fn language ->
|
||||
route_language = route_language(plan.language, language)
|
||||
|
||||
{archive_path(route_language, segments, page_number),
|
||||
render_fun.(
|
||||
page_posts,
|
||||
language,
|
||||
pagination_for_page(page_number, total_pages, length(posts), plan.max_posts_per_page, route_language, segments)
|
||||
)}
|
||||
end)
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_single_outputs(String.t(), String.t(), [map()], map(), map()) :: [{String.t(), iodata()}]
|
||||
def build_single_outputs(
|
||||
project_id,
|
||||
main_language,
|
||||
published_posts,
|
||||
translations_by_post_language,
|
||||
localized_posts_by_language
|
||||
) do
|
||||
post_outputs =
|
||||
Enum.map(published_posts, fn post ->
|
||||
canonical_variant = Map.get(translations_by_post_language, {post.id, main_language}, post)
|
||||
body = load_body(project_id, canonical_variant.file_path, canonical_variant.content)
|
||||
|
||||
{post_output_path(post),
|
||||
render_post_output(
|
||||
project_id,
|
||||
post.template_slug,
|
||||
%{
|
||||
id: canonical_variant.id,
|
||||
title: canonical_variant.title,
|
||||
content: body,
|
||||
slug: post.slug,
|
||||
language: canonical_variant.language,
|
||||
excerpt: canonical_variant.excerpt
|
||||
},
|
||||
fn ->
|
||||
render_post_page(canonical_variant.title, body, post.slug, canonical_variant.language)
|
||||
end
|
||||
)}
|
||||
end)
|
||||
|
||||
translation_outputs =
|
||||
localized_posts_by_language
|
||||
|> Enum.flat_map(fn {language, posts} ->
|
||||
Enum.map(posts, fn post ->
|
||||
body = load_body(project_id, post.file_path, post.content)
|
||||
|
||||
{post_output_path(post, language),
|
||||
render_post_output(
|
||||
project_id,
|
||||
post.template_slug,
|
||||
%{
|
||||
id: post.id,
|
||||
title: post.title,
|
||||
content: body,
|
||||
slug: post.slug,
|
||||
language: Map.get(post, :language),
|
||||
excerpt: post.excerpt
|
||||
},
|
||||
fn -> render_post_page(post.title, body, post.slug, Map.get(post, :language)) end
|
||||
)}
|
||||
end)
|
||||
end)
|
||||
|
||||
post_outputs ++ translation_outputs
|
||||
end
|
||||
end
|
||||
445
lib/bds/generation/validation.ex
Normal file
445
lib/bds/generation/validation.ex
Normal file
@@ -0,0 +1,445 @@
|
||||
defmodule BDS.Generation.Validation do
|
||||
@moduledoc false
|
||||
|
||||
import BDS.Generation.Paths,
|
||||
only: [
|
||||
archive_route_segment: 1,
|
||||
local_date_parts!: 1,
|
||||
normalize_url_path: 1,
|
||||
relative_path_to_url_path: 1,
|
||||
url_path_to_relative_index_path: 1
|
||||
]
|
||||
import BDS.Generation.Progress, only: [report_validation_compare_progress: 3]
|
||||
import BDS.Generation.Sitemap, only: [extract_locs: 1, loc_to_project_path: 2]
|
||||
|
||||
alias BDS.Slug
|
||||
|
||||
@spec generated_file_updated_at_map([map()]) :: map()
|
||||
def generated_file_updated_at_map(generated_files) do
|
||||
Map.new(generated_files, &{&1.relative_path, &1.updated_at})
|
||||
end
|
||||
|
||||
@spec build_post_timestamp_checks(String.t(), [map()], map()) :: [map()]
|
||||
def build_post_timestamp_checks(project_data_dir, published_route_posts, generated_file_updated_at) do
|
||||
Enum.map(published_route_posts, fn post ->
|
||||
relative_path = BDS.Generation.Paths.post_output_path(post)
|
||||
|
||||
%{
|
||||
post_url_path: relative_path_to_url_path(relative_path),
|
||||
post_file_path:
|
||||
source_full_path(
|
||||
project_data_dir,
|
||||
Map.get(post, :translation_file_path) || Map.get(post, :file_path)
|
||||
),
|
||||
generated_updated_at_ms: Map.get(generated_file_updated_at, relative_path, 0)
|
||||
}
|
||||
end)
|
||||
end
|
||||
|
||||
@spec build_language_post_timestamp_checks(String.t(), String.t(), [map()], map()) :: [map()]
|
||||
def build_language_post_timestamp_checks(
|
||||
project_data_dir,
|
||||
language,
|
||||
published_posts,
|
||||
generated_file_updated_at
|
||||
) do
|
||||
Enum.map(published_posts, fn post ->
|
||||
relative_path = BDS.Generation.Paths.post_output_path(post, language)
|
||||
|
||||
%{
|
||||
post_url_path: relative_path_to_url_path(relative_path),
|
||||
post_file_path: source_full_path(project_data_dir, Map.get(post, :file_path)),
|
||||
generated_updated_at_ms: Map.get(generated_file_updated_at, relative_path, 0)
|
||||
}
|
||||
end)
|
||||
end
|
||||
|
||||
defp source_full_path(_project_data_dir, file_path) when file_path in [nil, ""], do: nil
|
||||
defp source_full_path(project_data_dir, file_path), do: Path.join(project_data_dir, file_path)
|
||||
|
||||
@spec compare_sitemap_to_html(map()) :: map()
|
||||
def compare_sitemap_to_html(params) do
|
||||
post_timestamp_checks = Map.get(params, :post_timestamp_checks, [])
|
||||
index_paths = Path.wildcard(Path.join(params.html_dir, "**/index.html"))
|
||||
total_compare_steps = max(length(index_paths) + length(post_timestamp_checks), 1)
|
||||
|
||||
expected_path_set =
|
||||
params.sitemap_xml
|
||||
|> extract_locs()
|
||||
|> Enum.map(&loc_to_project_path(&1, params.base_url))
|
||||
|> Enum.reduce(MapSet.new(), &MapSet.put(&2, normalize_url_path(&1)))
|
||||
|> then(fn expected_paths ->
|
||||
Enum.reduce(Map.get(params, :additional_expected_paths, []), expected_paths, fn path, acc ->
|
||||
MapSet.put(acc, normalize_url_path(path))
|
||||
end)
|
||||
end)
|
||||
|
||||
{existing_html_path_set, zero_byte_html_path_set} =
|
||||
collect_html_index_paths(index_paths, params.html_dir, params.on_progress, total_compare_steps)
|
||||
|
||||
missing_url_paths =
|
||||
expected_path_set
|
||||
|> MapSet.to_list()
|
||||
|> Enum.reject(&MapSet.member?(existing_html_path_set, &1))
|
||||
|> Enum.sort()
|
||||
|
||||
extra_url_paths =
|
||||
existing_html_path_set
|
||||
|> MapSet.to_list()
|
||||
|> Enum.reject(&MapSet.member?(expected_path_set, &1))
|
||||
|> Kernel.++(
|
||||
zero_byte_html_path_set
|
||||
|> MapSet.to_list()
|
||||
|> Enum.reject(&MapSet.member?(expected_path_set, &1))
|
||||
)
|
||||
|> Enum.uniq()
|
||||
|> Enum.sort()
|
||||
|
||||
updated_post_url_paths =
|
||||
post_timestamp_checks
|
||||
|> Enum.with_index(1)
|
||||
|> Enum.reduce(MapSet.new(), fn {check, index}, acc ->
|
||||
:ok =
|
||||
report_validation_compare_progress(
|
||||
params.on_progress,
|
||||
length(index_paths) + index,
|
||||
total_compare_steps
|
||||
)
|
||||
|
||||
normalized_url_path = normalize_url_path(check.post_url_path)
|
||||
|
||||
cond do
|
||||
not MapSet.member?(expected_path_set, normalized_url_path) ->
|
||||
acc
|
||||
|
||||
normalized_url_path in missing_url_paths ->
|
||||
acc
|
||||
|
||||
is_nil(check.post_file_path) or check.post_file_path == "" ->
|
||||
acc
|
||||
|
||||
true ->
|
||||
html_path = Path.join(params.html_dir, url_path_to_relative_index_path(normalized_url_path))
|
||||
|
||||
case {File.stat(html_path, time: :posix), File.stat(check.post_file_path, time: :posix)} do
|
||||
{{:ok, html_stat}, {:ok, post_stat}} ->
|
||||
effective_generated_at_ms = max(mtime_ms(html_stat), check.generated_updated_at_ms || 0)
|
||||
|
||||
if mtime_ms(post_stat) > effective_generated_at_ms do
|
||||
MapSet.put(acc, normalized_url_path)
|
||||
else
|
||||
acc
|
||||
end
|
||||
|
||||
_other ->
|
||||
acc
|
||||
end
|
||||
end
|
||||
end)
|
||||
|> MapSet.to_list()
|
||||
|> Enum.sort()
|
||||
|
||||
%{
|
||||
missing_url_paths: missing_url_paths,
|
||||
extra_url_paths: extra_url_paths,
|
||||
updated_post_url_paths: updated_post_url_paths,
|
||||
expected_url_count: MapSet.size(expected_path_set),
|
||||
existing_html_url_count: MapSet.size(existing_html_path_set)
|
||||
}
|
||||
end
|
||||
|
||||
defp collect_html_index_paths(index_paths, html_dir, on_progress, total_compare_steps) do
|
||||
index_paths
|
||||
|> Enum.with_index(1)
|
||||
|> Enum.reduce({MapSet.new(), MapSet.new()}, fn {path, index}, {existing, zero_byte} ->
|
||||
:ok = report_validation_compare_progress(on_progress, index, total_compare_steps)
|
||||
|
||||
relative_dir =
|
||||
path
|
||||
|> Path.relative_to(html_dir)
|
||||
|> Path.dirname()
|
||||
|
||||
url_path =
|
||||
case relative_dir do
|
||||
"." -> "/"
|
||||
value -> normalize_url_path("/" <> value)
|
||||
end
|
||||
|
||||
case File.stat(path) do
|
||||
{:ok, %{size: size}} when size > 0 -> {MapSet.put(existing, url_path), zero_byte}
|
||||
{:ok, _stat} -> {existing, MapSet.put(zero_byte, url_path)}
|
||||
{:error, _reason} -> {existing, MapSet.put(zero_byte, url_path)}
|
||||
end
|
||||
end)
|
||||
end
|
||||
|
||||
defp mtime_ms(%{mtime: mtime}) when is_integer(mtime), do: mtime * 1000
|
||||
|
||||
defp mtime_ms(%{mtime: mtime}) do
|
||||
mtime
|
||||
|> NaiveDateTime.from_erl!()
|
||||
|> DateTime.from_naive!("Etc/UTC")
|
||||
|> DateTime.to_unix(:millisecond)
|
||||
end
|
||||
|
||||
@spec report_paths(map()) :: [String.t()]
|
||||
def report_paths(report) do
|
||||
Map.get(report, :missing_url_paths, []) ++ Map.get(report, :updated_post_url_paths, [])
|
||||
end
|
||||
|
||||
@spec plan_validation_paths([String.t()], [String.t()]) :: map()
|
||||
def plan_validation_paths(paths, additional_languages) do
|
||||
{main_plan, language_plans} =
|
||||
Enum.reduce(paths, {empty_validation_path_plan(), %{}}, fn path, {plan, language_plans} ->
|
||||
normalized_path = normalize_url_path(path)
|
||||
{language, stripped_path} = extract_language_path(normalized_path, additional_languages)
|
||||
|
||||
if is_binary(language) do
|
||||
language_plan = Map.get(language_plans, language, empty_validation_path_plan())
|
||||
next_language_plan = classify_validation_path(stripped_path, language_plan)
|
||||
{plan, Map.put(language_plans, language, next_language_plan)}
|
||||
else
|
||||
{classify_validation_path(normalized_path, plan), language_plans}
|
||||
end
|
||||
end)
|
||||
|
||||
Map.put(main_plan, :language_plans, language_plans)
|
||||
end
|
||||
|
||||
@spec empty_validation_path_plan() :: map()
|
||||
def empty_validation_path_plan do
|
||||
%{
|
||||
request_root_routes: false,
|
||||
requires_fallback_section_render: false,
|
||||
requested_category_slugs: MapSet.new(),
|
||||
requested_tag_slugs: MapSet.new(),
|
||||
requested_years: MapSet.new(),
|
||||
requested_year_months: MapSet.new(),
|
||||
requested_post_routes: [],
|
||||
language_plans: %{}
|
||||
}
|
||||
end
|
||||
|
||||
defp classify_validation_path(path, plan) do
|
||||
case Regex.run(~r|^/category/([^/]+)(?:/page/\d+)?$|, path) do
|
||||
[_, slug] ->
|
||||
update_in(plan.requested_category_slugs, &MapSet.put(&1, slug))
|
||||
|
||||
nil ->
|
||||
case Regex.run(~r|^/tag/([^/]+)(?:/page/\d+)?$|, path) do
|
||||
[_, slug] ->
|
||||
update_in(plan.requested_tag_slugs, &MapSet.put(&1, slug))
|
||||
|
||||
nil ->
|
||||
case Regex.run(~r|^/(\d{4})/(\d{2})/(\d{2})/([^/]+)$|, path) do
|
||||
[_, year, month, day, slug] ->
|
||||
update_in(plan.requested_post_routes, &[ %{year: String.to_integer(year), month: String.to_integer(month), day: String.to_integer(day), slug: slug} | &1 ])
|
||||
|
||||
nil ->
|
||||
case Regex.run(~r|^/(\d{4})/(\d{2})(?:/page/\d+)?$|, path) do
|
||||
[_, year, month] ->
|
||||
update_in(plan.requested_year_months, &MapSet.put(&1, "#{year}/#{month}"))
|
||||
|
||||
nil ->
|
||||
case Regex.run(~r|^/(\d{4})(?:/page/\d+)?$|, path) do
|
||||
[_, year] ->
|
||||
update_in(plan.requested_years, &MapSet.put(&1, String.to_integer(year)))
|
||||
|
||||
nil ->
|
||||
if path == "/" or Regex.match?(~r|^/page/\d+$|, path) do
|
||||
%{plan | request_root_routes: true}
|
||||
else
|
||||
%{plan | requires_fallback_section_render: true}
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
|
||||
@spec build_targeted_validation_plan(map(), [map()]) :: map()
|
||||
def build_targeted_validation_plan(initial_plan, published_posts) do
|
||||
if initial_plan.requires_fallback_section_render do
|
||||
initial_plan
|
||||
else
|
||||
available_category_slugs =
|
||||
published_posts
|
||||
|> Enum.flat_map(&(&1.categories || []))
|
||||
|> Enum.map(&Slug.slugify/1)
|
||||
|> MapSet.new()
|
||||
|
||||
available_tag_slugs =
|
||||
published_posts
|
||||
|> Enum.flat_map(&(&1.tags || []))
|
||||
|> Enum.map(&Slug.slugify/1)
|
||||
|> MapSet.new()
|
||||
|
||||
targeted_post_routes =
|
||||
Enum.reduce(initial_plan.requested_post_routes, MapSet.new(), fn route, acc ->
|
||||
MapSet.put(acc, route_key(route.year, route.month, route.day, route.slug))
|
||||
end)
|
||||
|
||||
enriched =
|
||||
Enum.reduce(initial_plan.requested_post_routes, %{initial_plan | requested_post_routes: targeted_post_routes}, fn route, acc ->
|
||||
case Enum.find(published_posts, &post_matches_route?(&1, route)) do
|
||||
nil ->
|
||||
acc
|
||||
|> update_in([:requested_years], &MapSet.put(&1, route.year))
|
||||
|> update_in([:requested_year_months], &MapSet.put(&1, route_month_key(route.year, route.month)))
|
||||
|> Map.put(:request_root_routes, true)
|
||||
|
||||
post ->
|
||||
{year, month, _day} = local_date_parts!(post.created_at)
|
||||
|
||||
acc
|
||||
|> update_in([:requested_category_slugs], fn set ->
|
||||
Enum.reduce(post.categories || [], set, &MapSet.put(&2, archive_route_segment(&1)))
|
||||
end)
|
||||
|> update_in([:requested_tag_slugs], fn set ->
|
||||
Enum.reduce(post.tags || [], set, &MapSet.put(&2, archive_route_segment(&1)))
|
||||
end)
|
||||
|> update_in([:requested_years], &MapSet.put(&1, year))
|
||||
|> update_in([:requested_year_months], &MapSet.put(&1, route_month_key(year, month)))
|
||||
|> Map.put(:request_root_routes, true)
|
||||
end
|
||||
end)
|
||||
|
||||
language_plans =
|
||||
initial_plan.language_plans
|
||||
|> Enum.map(fn {language, language_plan} ->
|
||||
{language, build_targeted_validation_plan(language_plan, published_posts)}
|
||||
end)
|
||||
|> Map.new()
|
||||
|
||||
%{
|
||||
enriched
|
||||
| requested_category_slugs: MapSet.intersection(enriched.requested_category_slugs, available_category_slugs),
|
||||
requested_tag_slugs: MapSet.intersection(enriched.requested_tag_slugs, available_tag_slugs),
|
||||
language_plans: language_plans
|
||||
}
|
||||
end
|
||||
end
|
||||
|
||||
defp post_matches_route?(post, route) do
|
||||
{year, month, day} = local_date_parts!(post.created_at)
|
||||
|
||||
post.slug == route.slug and year == route.year and month == route.month and day == route.day
|
||||
end
|
||||
|
||||
defp route_key(year, month, day, slug) do
|
||||
"#{year}/#{String.pad_leading(Integer.to_string(month), 2, "0")}/#{String.pad_leading(Integer.to_string(day), 2, "0")}/#{slug}"
|
||||
end
|
||||
|
||||
defp route_month_key(year, month) do
|
||||
"#{year}/#{String.pad_leading(Integer.to_string(month), 2, "0")}"
|
||||
end
|
||||
|
||||
defp extract_language_path(path, additional_languages) do
|
||||
case Regex.run(~r|^/([a-z]{2,3})(/.*)?$|, path) do
|
||||
[_, language, suffix] ->
|
||||
if language in additional_languages do
|
||||
{language, normalize_url_path(suffix)}
|
||||
else
|
||||
{nil, path}
|
||||
end
|
||||
|
||||
[_, language] ->
|
||||
if language in additional_languages do
|
||||
{language, "/"}
|
||||
else
|
||||
{nil, path}
|
||||
end
|
||||
|
||||
_other -> {nil, path}
|
||||
end
|
||||
end
|
||||
|
||||
@spec targeted_output?(String.t(), map(), String.t() | nil, [String.t()]) :: boolean()
|
||||
def targeted_output?(relative_path, targeted_plan, main_language, additional_languages) do
|
||||
{language, stripped_path} = extract_relative_output_language(relative_path, additional_languages)
|
||||
|
||||
plan =
|
||||
case language do
|
||||
nil -> targeted_plan
|
||||
value -> Map.get(targeted_plan.language_plans, value, empty_validation_path_plan())
|
||||
end
|
||||
|
||||
targeted_output_for_plan?(stripped_path, plan, main_language == language or is_nil(language))
|
||||
end
|
||||
|
||||
defp extract_relative_output_language(relative_path, additional_languages) do
|
||||
segments = String.split(relative_path, "/", trim: true)
|
||||
|
||||
case segments do
|
||||
[language | rest] ->
|
||||
if language in additional_languages do
|
||||
{language, Path.join(rest)}
|
||||
else
|
||||
{nil, relative_path}
|
||||
end
|
||||
|
||||
_other ->
|
||||
{nil, relative_path}
|
||||
end
|
||||
end
|
||||
|
||||
defp targeted_output_for_plan?(_relative_path, %{requires_fallback_section_render: true}, _main?), do: true
|
||||
|
||||
defp targeted_output_for_plan?(relative_path, plan, _main?) do
|
||||
cond do
|
||||
relative_path in ["index.html", "404.html", "feed.xml", "atom.xml"] ->
|
||||
plan.request_root_routes
|
||||
|
||||
Regex.match?(~r|^category/([^/]+)(?:/page/\d+)?/index\.html$|, relative_path) ->
|
||||
[_, slug] = Regex.run(~r|^category/([^/]+)(?:/page/\d+)?/index\.html$|, relative_path)
|
||||
MapSet.member?(plan.requested_category_slugs, slug)
|
||||
|
||||
Regex.match?(~r|^tag/([^/]+)/index\.html$|, relative_path) ->
|
||||
[_, slug] = Regex.run(~r|^tag/([^/]+)/index\.html$|, relative_path)
|
||||
MapSet.member?(plan.requested_tag_slugs, slug)
|
||||
|
||||
Regex.match?(~r|^(\d{4})/(\d{2})/(\d{2})/([^/]+)/index\.html$|, relative_path) ->
|
||||
[_, year, month, day, slug] = Regex.run(~r|^(\d{4})/(\d{2})/(\d{2})/([^/]+)/index\.html$|, relative_path)
|
||||
MapSet.member?(plan.requested_post_routes, route_key(String.to_integer(year), String.to_integer(month), String.to_integer(day), slug))
|
||||
|
||||
Regex.match?(~r|^(\d{4})/(\d{2})/index\.html$|, relative_path) ->
|
||||
[_, year, month] = Regex.run(~r|^(\d{4})/(\d{2})/index\.html$|, relative_path)
|
||||
MapSet.member?(plan.requested_year_months, "#{year}/#{month}")
|
||||
|
||||
Regex.match?(~r|^(\d{4})/index\.html$|, relative_path) ->
|
||||
[_, year] = Regex.run(~r|^(\d{4})/index\.html$|, relative_path)
|
||||
MapSet.member?(plan.requested_years, String.to_integer(year))
|
||||
|
||||
true ->
|
||||
false
|
||||
end
|
||||
end
|
||||
|
||||
@spec route_html_path?(String.t()) :: boolean()
|
||||
def route_html_path?(relative_path), do: String.ends_with?(relative_path, "index.html")
|
||||
|
||||
@spec prune_empty_parent_dirs(String.t(), String.t()) :: {non_neg_integer(), String.t()}
|
||||
def prune_empty_parent_dirs(current_dir, html_root) do
|
||||
cond do
|
||||
Path.expand(current_dir) == Path.expand(html_root) ->
|
||||
{0, current_dir}
|
||||
|
||||
true ->
|
||||
case File.ls(current_dir) do
|
||||
{:ok, []} ->
|
||||
case File.rmdir(current_dir) do
|
||||
:ok ->
|
||||
{count, last_dir} = prune_empty_parent_dirs(Path.dirname(current_dir), html_root)
|
||||
{count + 1, last_dir}
|
||||
|
||||
{:error, _reason} ->
|
||||
{0, current_dir}
|
||||
end
|
||||
|
||||
_other ->
|
||||
{0, current_dir}
|
||||
end
|
||||
end
|
||||
end
|
||||
end
|
||||
Reference in New Issue
Block a user