155 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			Elixir
		
	
	
	
	
	
			
		
		
	
	
			155 lines
		
	
	
	
		
			3.5 KiB
		
	
	
	
		
			Elixir
		
	
	
	
	
	
| # Pleroma: A lightweight social networking server
 | |
| # Copyright © 2017-2019 Pleroma Authors <https://pleroma.social/>
 | |
| # SPDX-License-Identifier: AGPL-3.0-only
 | |
| 
 | |
| defmodule Pleroma.List do
 | |
|   use Ecto.Schema
 | |
| 
 | |
|   import Ecto.Query
 | |
|   import Ecto.Changeset
 | |
| 
 | |
|   alias Pleroma.Activity
 | |
|   alias Pleroma.Repo
 | |
|   alias Pleroma.User
 | |
| 
 | |
|   schema "lists" do
 | |
|     belongs_to(:user, User, type: Pleroma.FlakeId)
 | |
|     field(:title, :string)
 | |
|     field(:following, {:array, :string}, default: [])
 | |
|     field(:ap_id, :string)
 | |
| 
 | |
|     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
 | |
| 
 | |
|   def for_user(user, _opts) do
 | |
|     query =
 | |
|       from(
 | |
|         l in Pleroma.List,
 | |
|         where: l.user_id == ^user.id,
 | |
|         order_by: [desc: l.id],
 | |
|         limit: 50
 | |
|       )
 | |
| 
 | |
|     Repo.all(query)
 | |
|   end
 | |
| 
 | |
|   def get(id, %{id: user_id} = _user) do
 | |
|     query =
 | |
|       from(
 | |
|         l in Pleroma.List,
 | |
|         where: l.id == ^id,
 | |
|         where: l.user_id == ^user_id
 | |
|       )
 | |
| 
 | |
|     Repo.one(query)
 | |
|   end
 | |
| 
 | |
|   def get_by_ap_id(ap_id) do
 | |
|     Repo.get_by(__MODULE__, ap_id: ap_id)
 | |
|   end
 | |
| 
 | |
|   def get_following(%Pleroma.List{following: following} = _list) do
 | |
|     q =
 | |
|       from(
 | |
|         u in User,
 | |
|         where: u.follower_address in ^following
 | |
|       )
 | |
| 
 | |
|     {:ok, Repo.all(q)}
 | |
|   end
 | |
| 
 | |
|   # 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
 | |
| 
 | |
|   # Get lists to which the account belongs.
 | |
|   def get_lists_account_belongs(%User{} = owner, account_id) do
 | |
|     user = User.get_cached_by_id(account_id)
 | |
| 
 | |
|     query =
 | |
|       from(
 | |
|         l in Pleroma.List,
 | |
|         where:
 | |
|           l.user_id == ^owner.id and
 | |
|             fragment(
 | |
|               "? = ANY(?)",
 | |
|               ^user.follower_address,
 | |
|               l.following
 | |
|             )
 | |
|       )
 | |
| 
 | |
|     Repo.all(query)
 | |
|   end
 | |
| 
 | |
|   def rename(%Pleroma.List{} = list, title) do
 | |
|     list
 | |
|     |> title_changeset(%{title: title})
 | |
|     |> Repo.update()
 | |
|   end
 | |
| 
 | |
|   def create(title, %User{} = creator) do
 | |
|     list = %Pleroma.List{user_id: creator.id, title: title}
 | |
| 
 | |
|     Repo.transaction(fn ->
 | |
|       list = Repo.insert!(list)
 | |
| 
 | |
|       list
 | |
|       |> change(ap_id: "#{creator.ap_id}/lists/#{list.id}")
 | |
|       |> Repo.update!()
 | |
|     end)
 | |
|   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
 | |
| 
 | |
|   def memberships(%User{follower_address: follower_address}) do
 | |
|     Pleroma.List
 | |
|     |> where([l], ^follower_address in l.following)
 | |
|     |> select([l], l.ap_id)
 | |
|     |> Repo.all()
 | |
|   end
 | |
| 
 | |
|   def memberships(_), do: []
 | |
| 
 | |
|   def member?(%Pleroma.List{following: following}, %User{follower_address: follower_address}) do
 | |
|     Enum.member?(following, follower_address)
 | |
|   end
 | |
| 
 | |
|   def member?(_, _), do: false
 | |
| end
 | 
