feat: first entities in database

This commit is contained in:
2026-04-23 12:21:13 +02:00
parent 28141deb8b
commit 13ac446793
10 changed files with 568 additions and 1 deletions

172
lib/bds/posts.ex Normal file
View File

@@ -0,0 +1,172 @@
defmodule BDS.Posts do
@moduledoc false
import Ecto.Query
alias BDS.Posts.Post
alias BDS.Repo
alias BDS.Slug
def create_post(attrs) do
now = System.system_time(:second)
project_id = attr(attrs, :project_id)
title = normalize_title(attr(attrs, :title))
base_slug = title |> default_slug_source() |> Slug.slugify()
%Post{}
|> Post.changeset(%{
id: Ecto.UUID.generate(),
project_id: project_id,
title: title,
slug: unique_slug(project_id, base_slug),
excerpt: attr(attrs, :excerpt),
content: attr(attrs, :content),
status: :draft,
author: attr(attrs, :author),
created_at: now,
updated_at: now,
published_at: nil,
file_path: "",
checksum: attr(attrs, :checksum),
tags: attr(attrs, :tags) || [],
categories: attr(attrs, :categories) || [],
template_slug: attr(attrs, :template_slug),
language: attr(attrs, :language),
do_not_translate: false,
published_title: nil,
published_content: nil,
published_tags: nil,
published_categories: nil,
published_excerpt: nil
})
|> Repo.insert()
end
def update_post(post_id, attrs) do
case Repo.get(Post, post_id) do
nil ->
{:error, :not_found}
post ->
with :ok <- validate_slug_change(post, attrs) do
now = System.system_time(:second)
updates =
attrs
|> normalize_updates(post)
|> Map.put(:updated_at, now)
|> maybe_reopen_published_post(post)
post
|> Post.changeset(updates)
|> Repo.update()
else
{:error, changeset} -> {:error, changeset}
end
end
end
defp normalize_updates(attrs, _post) do
%{}
|> maybe_put(:title, normalize_optional_title(attr(attrs, :title), attrs))
|> maybe_put(:slug, attr(attrs, :slug))
|> maybe_put(:excerpt, attr(attrs, :excerpt))
|> maybe_put(:content, attr(attrs, :content))
|> maybe_put(:status, attr(attrs, :status))
|> maybe_put(:author, attr(attrs, :author))
|> maybe_put(:published_at, attr(attrs, :published_at))
|> maybe_put(:file_path, attr(attrs, :file_path))
|> maybe_put(:checksum, attr(attrs, :checksum))
|> maybe_put(:tags, attr(attrs, :tags))
|> maybe_put(:categories, attr(attrs, :categories))
|> maybe_put(:template_slug, attr(attrs, :template_slug))
|> maybe_put(:language, attr(attrs, :language))
|> maybe_put(:do_not_translate, attr(attrs, :do_not_translate))
|> maybe_put(:published_title, attr(attrs, :published_title))
|> maybe_put(:published_content, attr(attrs, :published_content))
|> maybe_put(:published_tags, attr(attrs, :published_tags))
|> maybe_put(:published_categories, attr(attrs, :published_categories))
|> maybe_put(:published_excerpt, attr(attrs, :published_excerpt))
end
defp validate_slug_change(%Post{published_at: published_at} = post, attrs) when not is_nil(published_at) do
case attr(attrs, :slug) do
nil ->
:ok
slug when slug == post.slug ->
:ok
_slug ->
{:error,
post
|> Post.changeset(%{})
|> Ecto.Changeset.add_error(:slug, "cannot change slug after first publish")}
end
end
defp validate_slug_change(_post, _attrs), do: :ok
defp maybe_reopen_published_post(updates, %Post{status: :published} = post) do
if published_content_change?(updates, post) do
Map.put(updates, :status, :draft)
else
updates
end
end
defp maybe_reopen_published_post(updates, _post), do: updates
defp published_content_change?(updates, post) do
Enum.any?([:title, :excerpt, :content, :author, :language, :template_slug, :tags, :categories, :do_not_translate], fn field ->
case Map.fetch(updates, field) do
{:ok, value} -> value != Map.get(post, field)
:error -> false
end
end)
end
defp unique_slug(project_id, base_slug) do
normalized = if base_slug in [nil, ""], do: "untitled", else: base_slug
if slug_available?(project_id, normalized) do
normalized
else
find_unique_slug(project_id, normalized, 2)
end
end
defp find_unique_slug(project_id, base_slug, suffix) do
candidate = "#{base_slug}-#{suffix}"
if slug_available?(project_id, candidate) do
candidate
else
find_unique_slug(project_id, base_slug, suffix + 1)
end
end
defp slug_available?(project_id, slug) do
not Repo.exists?(from post in Post, where: post.project_id == ^project_id and post.slug == ^slug)
end
defp maybe_put(map, _key, nil), do: map
defp maybe_put(map, key, value), do: Map.put(map, key, value)
defp normalize_title(nil), do: ""
defp normalize_title(title), do: title
defp normalize_optional_title(_title, attrs) do
if has_attr?(attrs, :title), do: normalize_title(attr(attrs, :title)), else: nil
end
defp default_slug_source(""), do: "untitled"
defp default_slug_source(title), do: title
defp has_attr?(attrs, key) do
Map.has_key?(attrs, key) or Map.has_key?(attrs, Atom.to_string(key))
end
defp attr(attrs, key) do
Map.get(attrs, key) || Map.get(attrs, Atom.to_string(key))
end
end

74
lib/bds/posts/post.ex Normal file
View File

@@ -0,0 +1,74 @@
defmodule BDS.Posts.Post do
@moduledoc false
use Ecto.Schema
import Ecto.Changeset
alias BDS.Types.StringList
@primary_key {:id, :string, autogenerate: false}
@foreign_key_type :string
@statuses [:draft, :published, :archived]
schema "posts" do
belongs_to :project, BDS.Projects.Project, type: :string
field :title, :string
field :slug, :string
field :excerpt, :string
field :content, :string
field :status, Ecto.Enum, values: @statuses, default: :draft
field :author, :string
field :created_at, :integer
field :updated_at, :integer
field :published_at, :integer
field :file_path, :string, default: ""
field :checksum, :string
field :tags, StringList, default: []
field :categories, StringList, default: []
field :template_slug, :string
field :language, :string
field :do_not_translate, :boolean, default: false
field :published_title, :string
field :published_content, :string
field :published_tags, :string
field :published_categories, :string
field :published_excerpt, :string
end
def changeset(post, attrs) do
post
|> cast(
attrs,
[
:id,
:project_id,
:title,
:slug,
:excerpt,
:content,
:status,
:author,
:created_at,
:updated_at,
:published_at,
:file_path,
:checksum,
:tags,
:categories,
:template_slug,
:language,
:do_not_translate,
:published_title,
:published_content,
:published_tags,
:published_categories,
:published_excerpt
],
empty_values: [nil]
)
|> validate_required([:id, :project_id, :slug, :status, :created_at, :updated_at, :do_not_translate])
|> assoc_constraint(:project)
|> unique_constraint(:slug, name: :posts_project_slug_idx)
end
end

84
lib/bds/projects.ex Normal file
View File

@@ -0,0 +1,84 @@
defmodule BDS.Projects do
@moduledoc false
import Ecto.Query
alias Ecto.Multi
alias BDS.Projects.Project
alias BDS.Repo
alias BDS.Slug
def list_projects do
Repo.all(from project in Project, order_by: [asc: project.created_at])
end
def get_project!(id), do: Repo.get!(Project, id)
def create_project(attrs) do
now = System.system_time(:second)
name = attr(attrs, :name) || ""
slug = unique_slug(attr(attrs, :slug) || Slug.slugify(name))
%Project{}
|> Project.changeset(%{
id: Ecto.UUID.generate(),
name: name,
slug: slug,
description: attr(attrs, :description),
data_path: attr(attrs, :data_path),
created_at: now,
updated_at: now,
is_active: false
})
|> Repo.insert()
end
def set_active_project(project_id) do
case Repo.get(Project, project_id) do
nil ->
{:error, :not_found}
project ->
now = System.system_time(:second)
Multi.new()
|> Multi.update_all(:clear_previous, from(p in Project, where: p.is_active == true),
set: [is_active: false, updated_at: now]
)
|> Multi.update(:activate, Project.changeset(project, %{is_active: true, updated_at: now}))
|> Repo.transaction()
|> case do
{:ok, %{activate: active_project}} -> {:ok, active_project}
{:error, _step, reason, _changes} -> {:error, reason}
end
end
end
defp unique_slug(base_slug) do
normalized = if base_slug in [nil, ""], do: "project", else: base_slug
if slug_available?(normalized) do
normalized
else
find_unique_slug(normalized, 2)
end
end
defp find_unique_slug(base_slug, suffix) do
candidate = "#{base_slug}-#{suffix}"
if slug_available?(candidate) do
candidate
else
find_unique_slug(base_slug, suffix + 1)
end
end
defp slug_available?(slug) do
not Repo.exists?(from project in Project, where: project.slug == ^slug)
end
defp attr(attrs, key) do
Map.get(attrs, key) || Map.get(attrs, Atom.to_string(key))
end
end

View File

@@ -0,0 +1,30 @@
defmodule BDS.Projects.Project do
@moduledoc false
use Ecto.Schema
import Ecto.Changeset
@primary_key {:id, :string, autogenerate: false}
@foreign_key_type :string
schema "projects" do
field :name, :string
field :slug, :string
field :description, :string
field :data_path, :string
field :created_at, :integer
field :updated_at, :integer
field :is_active, :boolean, default: false
has_many :posts, BDS.Posts.Post, foreign_key: :project_id
end
def changeset(project, attrs) do
project
|> cast(attrs, [:id, :name, :slug, :description, :data_path, :created_at, :updated_at, :is_active],
empty_values: [nil]
)
|> validate_required([:id, :name, :slug, :created_at, :updated_at, :is_active])
|> unique_constraint(:slug)
end
end

23
lib/bds/slug.ex Normal file
View File

@@ -0,0 +1,23 @@
defmodule BDS.Slug do
@moduledoc false
@german_transliterations %{"ß" => "ss"}
def slugify(nil), do: ""
def slugify(value) when is_binary(value) do
value
|> replace_german_characters()
|> String.normalize(:nfd)
|> String.replace(~r/[^\p{ASCII}]/u, "")
|> String.downcase()
|> String.replace(~r/[^a-z0-9]+/u, "-")
|> String.replace(~r/^-+|-+$/u, "")
end
defp replace_german_characters(value) do
Enum.reduce(@german_transliterations, value, fn {source, target}, acc ->
String.replace(acc, source, target)
end)
end
end

View File

@@ -0,0 +1,45 @@
defmodule BDS.Types.StringList do
@moduledoc false
use Ecto.Type
def type, do: :string
def cast(value) when is_list(value) do
if Enum.all?(value, &is_binary/1) do
{:ok, value}
else
:error
end
end
def cast(nil), do: {:ok, []}
def cast(_value), do: :error
def load(nil), do: {:ok, []}
def load(value) when is_binary(value) do
case Jason.decode(value) do
{:ok, decoded} ->
if is_list(decoded) and Enum.all?(decoded, &is_binary/1) do
{:ok, decoded}
else
:error
end
_ -> :error
end
end
def dump(nil), do: {:ok, "[]"}
def dump(value) when is_list(value) do
if Enum.all?(value, &is_binary/1) do
{:ok, Jason.encode!(value)}
else
:error
end
end
def dump(_value), do: :error
end