2018-12-23 13:04:54 -07:00
|
|
|
# Pleroma: A lightweight social networking server
|
2018-12-31 08:41:47 -07:00
|
|
|
# Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
|
2018-12-23 13:04:54 -07:00
|
|
|
# SPDX-License-Identifier: AGPL-3.0-only
|
|
|
|
|
2018-04-29 07:02:46 -06:00
|
|
|
defmodule Pleroma.List do
|
|
|
|
use Ecto.Schema
|
2019-02-09 08:16:26 -07:00
|
|
|
|
|
|
|
import Ecto.Query
|
|
|
|
import Ecto.Changeset
|
|
|
|
|
|
|
|
alias Pleroma.Activity
|
|
|
|
alias Pleroma.Repo
|
|
|
|
alias Pleroma.User
|
2018-04-29 07:02:46 -06:00
|
|
|
|
|
|
|
schema "lists" do
|
2019-01-09 08:08:24 -07:00
|
|
|
belongs_to(:user, User, type: Pleroma.FlakeId)
|
2018-04-29 07:02:46 -06:00
|
|
|
field(:title, :string)
|
|
|
|
field(:following, {:array, :string}, default: [])
|
2019-05-17 06:56:37 -06:00
|
|
|
field(:ap_id, :string)
|
2018-04-29 07:02:46 -06:00
|
|
|
|
|
|
|
timestamps()
|
|
|
|
end
|
|
|
|
|
|
|
|
def title_changeset(list, attrs \\ %{}) do
|
|
|
|
list
|
|
|
|
|> cast(attrs, [:title])
|
|
|
|
|> validate_required([:title])
|
|
|
|
end
|
|
|
|
|
|
|
|
def follow_changeset(list, attrs \\ %{}) do
|
|
|
|
list
|
|
|
|
|> cast(attrs, [:following])
|
|
|
|
|> validate_required([:following])
|
|
|
|
end
|
|
|
|
|
2018-12-09 02:12:48 -07:00
|
|
|
def for_user(user, _opts) do
|
2018-04-29 07:02:46 -06:00
|
|
|
query =
|
|
|
|
from(
|
|
|
|
l in Pleroma.List,
|
|
|
|
where: l.user_id == ^user.id,
|
|
|
|
order_by: [desc: l.id],
|
|
|
|
limit: 50
|
|
|
|
)
|
|
|
|
|
|
|
|
Repo.all(query)
|
|
|
|
end
|
|
|
|
|
2018-04-29 07:02:46 -06:00
|
|
|
def get(id, %{id: user_id} = _user) do
|
2018-04-29 07:02:46 -06:00
|
|
|
query =
|
|
|
|
from(
|
|
|
|
l in Pleroma.List,
|
|
|
|
where: l.id == ^id,
|
|
|
|
where: l.user_id == ^user_id
|
|
|
|
)
|
|
|
|
|
|
|
|
Repo.one(query)
|
|
|
|
end
|
|
|
|
|
2019-05-13 03:15:14 -06:00
|
|
|
def get_by_ap_id(ap_id) do
|
2019-05-17 06:56:37 -06:00
|
|
|
Repo.get_by(__MODULE__, ap_id: ap_id)
|
2019-05-13 03:15:14 -06:00
|
|
|
end
|
|
|
|
|
2018-12-09 02:12:48 -07:00
|
|
|
def get_following(%Pleroma.List{following: following} = _list) do
|
2018-04-29 07:02:46 -06:00
|
|
|
q =
|
|
|
|
from(
|
|
|
|
u in User,
|
|
|
|
where: u.follower_address in ^following
|
|
|
|
)
|
|
|
|
|
2018-04-29 07:02:46 -06:00
|
|
|
{:ok, Repo.all(q)}
|
|
|
|
end
|
|
|
|
|
2018-05-30 07:33:37 -06:00
|
|
|
# Get lists the activity should be streamed to.
|
|
|
|
def get_lists_from_activity(%Activity{actor: ap_id}) do
|
|
|
|
actor = User.get_cached_by_ap_id(ap_id)
|
|
|
|
|
|
|
|
query =
|
|
|
|
from(
|
|
|
|
l in Pleroma.List,
|
|
|
|
where: fragment("? && ?", l.following, ^[actor.follower_address])
|
|
|
|
)
|
|
|
|
|
|
|
|
Repo.all(query)
|
|
|
|
end
|
|
|
|
|
2018-10-18 10:46:26 -06:00
|
|
|
# Get lists to which the account belongs.
|
|
|
|
def get_lists_account_belongs(%User{} = owner, account_id) do
|
2019-04-22 01:20:43 -06:00
|
|
|
user = User.get_cached_by_id(account_id)
|
2018-10-19 06:24:15 -06:00
|
|
|
|
2018-10-18 10:46:26 -06:00
|
|
|
query =
|
|
|
|
from(
|
|
|
|
l in Pleroma.List,
|
2018-10-19 06:24:15 -06:00
|
|
|
where:
|
|
|
|
l.user_id == ^owner.id and
|
|
|
|
fragment(
|
|
|
|
"? = ANY(?)",
|
|
|
|
^user.follower_address,
|
|
|
|
l.following
|
|
|
|
)
|
2018-10-18 10:46:26 -06:00
|
|
|
)
|
2018-10-19 06:24:15 -06:00
|
|
|
|
2018-10-18 10:46:26 -06:00
|
|
|
Repo.all(query)
|
|
|
|
end
|
|
|
|
|
2018-04-29 07:02:46 -06:00
|
|
|
def rename(%Pleroma.List{} = list, title) do
|
|
|
|
list
|
|
|
|
|> title_changeset(%{title: title})
|
|
|
|
|> Repo.update()
|
|
|
|
end
|
|
|
|
|
|
|
|
def create(title, %User{} = creator) do
|
2019-08-26 05:59:57 -06:00
|
|
|
changeset = title_changeset(%Pleroma.List{user_id: creator.id}, %{title: title})
|
|
|
|
|
|
|
|
if changeset.valid? do
|
|
|
|
Repo.transaction(fn ->
|
|
|
|
list = Repo.insert!(changeset)
|
|
|
|
|
|
|
|
list
|
|
|
|
|> change(ap_id: "#{creator.ap_id}/lists/#{list.id}")
|
|
|
|
|> Repo.update!()
|
|
|
|
end)
|
|
|
|
else
|
|
|
|
{:error, changeset}
|
|
|
|
end
|
2018-04-29 07:02:46 -06:00
|
|
|
end
|
|
|
|
|
|
|
|
def follow(%Pleroma.List{following: following} = list, %User{} = followed) do
|
|
|
|
update_follows(list, %{following: Enum.uniq([followed.follower_address | following])})
|
|
|
|
end
|
|
|
|
|
|
|
|
def unfollow(%Pleroma.List{following: following} = list, %User{} = unfollowed) do
|
|
|
|
update_follows(list, %{following: List.delete(following, unfollowed.follower_address)})
|
|
|
|
end
|
|
|
|
|
|
|
|
def delete(%Pleroma.List{} = list) do
|
|
|
|
Repo.delete(list)
|
|
|
|
end
|
|
|
|
|
|
|
|
def update_follows(%Pleroma.List{} = list, attrs) do
|
|
|
|
list
|
|
|
|
|> follow_changeset(attrs)
|
|
|
|
|> Repo.update()
|
|
|
|
end
|
2019-05-13 03:15:14 -06:00
|
|
|
|
|
|
|
def memberships(%User{follower_address: follower_address}) do
|
|
|
|
Pleroma.List
|
|
|
|
|> where([l], ^follower_address in l.following)
|
2019-05-17 06:56:37 -06:00
|
|
|
|> select([l], l.ap_id)
|
2019-05-13 03:15:14 -06:00
|
|
|
|> Repo.all()
|
|
|
|
end
|
|
|
|
|
|
|
|
def memberships(_), do: []
|
2019-07-11 06:29:24 -06:00
|
|
|
|
|
|
|
def member?(%Pleroma.List{following: following}, %User{follower_address: follower_address}) do
|
|
|
|
Enum.member?(following, follower_address)
|
|
|
|
end
|
|
|
|
|
|
|
|
def member?(_, _), do: false
|
2018-04-29 07:02:46 -06:00
|
|
|
end
|