57
lib/graphql/resolvers/invitation.ex
Normal file
57
lib/graphql/resolvers/invitation.ex
Normal file
@@ -0,0 +1,57 @@
|
||||
defmodule Mobilizon.GraphQL.Resolvers.Invitation do
|
||||
@moduledoc """
|
||||
Handles the invitation-related GraphQL calls
|
||||
"""
|
||||
|
||||
alias Mobilizon.Actors
|
||||
alias Mobilizon.Actors.Actor
|
||||
alias Mobilizon.Invitations
|
||||
import Mobilizon.Web.Gettext
|
||||
|
||||
defp authorize_group_admin(%Actor{id: actor_id}, group_id) do
|
||||
if Actors.administrator?(actor_id, group_id) do
|
||||
:ok
|
||||
else
|
||||
{:error, dgettext("errors", "Profile is not administrator for the group")}
|
||||
end
|
||||
end
|
||||
|
||||
def create_invitation(_parent, %{group_id: group_id} = args, %{
|
||||
context: %{current_actor: %Actor{} = updater_actor}
|
||||
}) do
|
||||
with :ok <- authorize_group_admin(updater_actor, group_id),
|
||||
{:ok, invitation} <- Invitations.create_invitation(args) do
|
||||
{:ok, invitation}
|
||||
else
|
||||
{:error, _} ->
|
||||
{:error, dgettext("errors", "could not create invitation")}
|
||||
|
||||
error ->
|
||||
error
|
||||
end
|
||||
end
|
||||
|
||||
def list_invitations(_parent, %{group_id: group_id}, %{
|
||||
context: %{current_actor: %Actor{} = updater_actor}
|
||||
}) do
|
||||
with :ok <- authorize_group_admin(updater_actor, group_id) do
|
||||
{:ok, Invitations.list_invitations(group_id)}
|
||||
end
|
||||
end
|
||||
|
||||
def update_invitation(_parent, %{group_id: group_id, token: token} = args, %{
|
||||
context: %{current_actor: %Actor{} = updater_actor}
|
||||
}) do
|
||||
with :ok <- authorize_group_admin(updater_actor, group_id) do
|
||||
Invitations.update_invitation_by_token(group_id, token, args)
|
||||
end
|
||||
end
|
||||
|
||||
def delete_invitation(_parent, %{group_id: group_id, token: token}, %{
|
||||
context: %{current_actor: %Actor{} = updater_actor}
|
||||
}) do
|
||||
with :ok <- authorize_group_admin(updater_actor, group_id) do
|
||||
Invitations.delete_invitation_by_token(group_id, token)
|
||||
end
|
||||
end
|
||||
end
|
||||
@@ -56,6 +56,7 @@ defmodule Mobilizon.GraphQL.Schema do
|
||||
import_types(Schema.FollowedGroupActivityType)
|
||||
import_types(Schema.AuthApplicationType)
|
||||
import_types(Schema.ConversationType)
|
||||
import_types(Schema.InvitationType)
|
||||
|
||||
@desc "A struct containing the id of the deleted object"
|
||||
object :deleted_object do
|
||||
@@ -177,6 +178,7 @@ defmodule Mobilizon.GraphQL.Schema do
|
||||
import_fields(:post_queries)
|
||||
import_fields(:statistics_queries)
|
||||
import_fields(:auth_application_queries)
|
||||
import_fields(:invitation_queries)
|
||||
end
|
||||
|
||||
@desc """
|
||||
@@ -205,6 +207,7 @@ defmodule Mobilizon.GraphQL.Schema do
|
||||
import_fields(:push_mutations)
|
||||
import_fields(:activity_setting_mutations)
|
||||
import_fields(:auth_application_mutations)
|
||||
import_fields(:invitation_mutations)
|
||||
end
|
||||
|
||||
@desc """
|
||||
|
||||
50
lib/graphql/schema/invitation.ex
Normal file
50
lib/graphql/schema/invitation.ex
Normal file
@@ -0,0 +1,50 @@
|
||||
defmodule Mobilizon.GraphQL.Schema.InvitationType do
|
||||
@moduledoc """
|
||||
Schema representation for Invitation
|
||||
"""
|
||||
use Absinthe.Schema.Notation
|
||||
alias Mobilizon.GraphQL.Resolvers.Invitation
|
||||
|
||||
@desc "A local invitation to a Mobilizon group"
|
||||
object :invitation do
|
||||
meta(:authorize, :user)
|
||||
field(:token, :string, description: "The invitation token")
|
||||
field(:label, :string, description: "The invitation label")
|
||||
end
|
||||
|
||||
object :invitation_mutations do
|
||||
@desc "Create an invitation for a group"
|
||||
field :create_invitation, type: :invitation do
|
||||
arg(:group_id, non_null(:id), description: "ID of the group")
|
||||
arg(:label, :string, description: "Label")
|
||||
middleware(Rajska.QueryAuthorization, permit: :user, scope: false)
|
||||
resolve(&Invitation.create_invitation/3)
|
||||
end
|
||||
|
||||
@desc "Update an invitation for a group"
|
||||
field :update_invitation, type: :invitation do
|
||||
arg(:group_id, non_null(:id), description: "ID of the group")
|
||||
arg(:token, :string, description: "Token")
|
||||
arg(:label, :string, description: "Label")
|
||||
middleware(Rajska.QueryAuthorization, permit: :user, scope: false)
|
||||
resolve(&Invitation.update_invitation/3)
|
||||
end
|
||||
|
||||
@desc "Delete an invitation for a group"
|
||||
field :delete_invitation, type: :invitation do
|
||||
arg(:group_id, non_null(:id), description: "ID of the group")
|
||||
arg(:token, :string, description: "Token")
|
||||
middleware(Rajska.QueryAuthorization, permit: :user, scope: false)
|
||||
resolve(&Invitation.delete_invitation/3)
|
||||
end
|
||||
end
|
||||
|
||||
object :invitation_queries do
|
||||
@desc "List all invitations for a group"
|
||||
field :list_invitations, non_null(list_of(:invitation)) do
|
||||
arg(:group_id, non_null(:id), description: "ID of the group")
|
||||
middleware(Rajska.QueryAuthorization, permit: :user, scope: false)
|
||||
resolve(&Invitation.list_invitations/3)
|
||||
end
|
||||
end
|
||||
end
|
||||
29
lib/mobilizon/invitations/invitation.ex
Normal file
29
lib/mobilizon/invitations/invitation.ex
Normal file
@@ -0,0 +1,29 @@
|
||||
defmodule Mobilizon.Invitation do
|
||||
@moduledoc """
|
||||
Represents a local invitation to a group.
|
||||
"""
|
||||
|
||||
use Ecto.Schema
|
||||
import Ecto.Changeset
|
||||
alias Mobilizon.Actors.Actor
|
||||
|
||||
schema "invitations" do
|
||||
# We need read_after_writes because the uuid is generated by
|
||||
# the database gen_random_uuid() function
|
||||
field :token, :string, read_after_writes: true
|
||||
|
||||
# label can't be NULL in database
|
||||
# default: "" permit to set it to "" if label is nil
|
||||
field :label, :string, default: ""
|
||||
|
||||
belongs_to :group, Actor
|
||||
|
||||
timestamps()
|
||||
end
|
||||
|
||||
def changeset(invitation, attrs) do
|
||||
invitation
|
||||
|> cast(attrs, [:label, :group_id])
|
||||
|> validate_required([:group_id])
|
||||
end
|
||||
end
|
||||
42
lib/mobilizon/invitations/invitations.ex
Normal file
42
lib/mobilizon/invitations/invitations.ex
Normal file
@@ -0,0 +1,42 @@
|
||||
defmodule Mobilizon.Invitations do
|
||||
@moduledoc """
|
||||
The Invitations context.
|
||||
"""
|
||||
|
||||
alias Mobilizon.Invitation
|
||||
alias Mobilizon.Storage.Repo
|
||||
import Ecto.Query
|
||||
|
||||
def get_invitation(id), do: Repo.get(Invitation, id)
|
||||
|
||||
def create_invitation(attrs \\ %{}) do
|
||||
%Invitation{}
|
||||
|> Invitation.changeset(attrs)
|
||||
|> Repo.insert()
|
||||
end
|
||||
|
||||
def update_invitation_by_token(group_id, token, attrs) do
|
||||
case Repo.get_by(Mobilizon.Invitation, token: token, group_id: group_id) do
|
||||
nil ->
|
||||
{:error, "Invitation not found"}
|
||||
|
||||
%Mobilizon.Invitation{} = invitation ->
|
||||
invitation
|
||||
|> Invitation.changeset(attrs)
|
||||
|> Repo.update()
|
||||
end
|
||||
end
|
||||
|
||||
def delete_invitation_by_token(group_id, token) do
|
||||
case Repo.get_by(Mobilizon.Invitation, token: token, group_id: group_id) do
|
||||
nil -> {:error, "Invitation not found"}
|
||||
invitation -> Repo.delete(invitation)
|
||||
end
|
||||
end
|
||||
|
||||
def list_invitations(group_id) do
|
||||
Invitation
|
||||
|> where([i], i.group_id == ^group_id)
|
||||
|> Repo.all()
|
||||
end
|
||||
end
|
||||
Reference in New Issue
Block a user