import {
  ClientError,
  InferClientInput,
  InferClientOutput,
} from "@mercurjs/client"
import {
  QueryKey,
  UseMutationOptions,
  UseQueryOptions,
  useMutation,
  useQuery,
} from "@tanstack/react-query"
import { sdk, fetchQuery } from "../../lib/client"
import { queryClient } from "../../lib/query-client"
import { queryKeysFactory } from "../../lib/query-key-factory"

const INVITES_QUERY_KEY = "invites" as const
const invitesQueryKeys = queryKeysFactory(INVITES_QUERY_KEY)

export const useInvite = (
  id: string,
  options?: Omit<
    UseQueryOptions<
      InferClientOutput<typeof sdk.admin.invites.$id.query>,
      ClientError,
      InferClientOutput<typeof sdk.admin.invites.$id.query>,
      QueryKey
    >,
    "queryFn" | "queryKey"
  >
) => {
  const { data, ...rest } = useQuery({
    queryKey: invitesQueryKeys.detail(id),
    queryFn: () => sdk.admin.invites.$id.query({ $id: id }),
    ...options,
  })

  return { ...data, ...rest }
}

export const useInvites = (
  query?: InferClientInput<typeof sdk.admin.invites.query>,
  options?: Omit<
    UseQueryOptions<
      InferClientOutput<typeof sdk.admin.invites.query>,
      ClientError,
      InferClientOutput<typeof sdk.admin.invites.query>,
      QueryKey
    >,
    "queryFn" | "queryKey"
  >
) => {
  const { data, ...rest } = useQuery({
    queryFn: () => sdk.admin.invites.query({ ...query }),
    queryKey: invitesQueryKeys.list(query),
    ...options,
  })

  return { ...data, ...rest }
}

export const useCreateInvite = (
  options?: UseMutationOptions<
    InferClientOutput<typeof sdk.admin.invites.mutate>,
    ClientError,
    InferClientInput<typeof sdk.admin.invites.mutate>
  >
) => {
  return useMutation({
    mutationFn: (payload) => sdk.admin.invites.mutate(payload),
    onSuccess: (data, variables, context) => {
      queryClient.invalidateQueries({ queryKey: invitesQueryKeys.lists() })
      options?.onSuccess?.(data, variables, context)
    },
    ...options,
  })
}

export const useResendInvite = (
  id: string,
  options?: UseMutationOptions<
    InferClientOutput<typeof sdk.admin.invites.$id.resend.mutate>,
    ClientError,
    void
  >
) => {
  return useMutation({
    mutationFn: () => sdk.admin.invites.$id.resend.mutate({ $id: id }),
    onSuccess: (data, variables, context) => {
      queryClient.invalidateQueries({ queryKey: invitesQueryKeys.lists() })
      queryClient.invalidateQueries({ queryKey: invitesQueryKeys.detail(id) })
      options?.onSuccess?.(data, variables, context)
    },
    ...options,
  })
}

export const useDeleteInvite = (
  id: string,
  options?: UseMutationOptions<
    InferClientOutput<typeof sdk.admin.invites.$id.delete>,
    ClientError,
    void
  >
) => {
  return useMutation({
    mutationFn: () => sdk.admin.invites.$id.delete({ $id: id }),
    onSuccess: (data, variables, context) => {
      queryClient.invalidateQueries({ queryKey: invitesQueryKeys.lists() })
      queryClient.invalidateQueries({ queryKey: invitesQueryKeys.detail(id) })
      options?.onSuccess?.(data, variables, context)
    },
    ...options,
  })
}

type AcceptInvitePayload = {
  email?: string | null
  first_name?: string | null
  last_name?: string | null
  auth_token: string
}

export const useAcceptInvite = (
  inviteToken: string,
  options?: UseMutationOptions<{ user: unknown }, ClientError, AcceptInvitePayload>
) => {
  return useMutation({
    mutationFn: ({ auth_token, ...body }) =>
      fetchQuery('/admin/invites/accept', {
        method: 'POST',
        query: { token: inviteToken },
        body,
        headers: { Authorization: `Bearer ${auth_token}` },
      }),
    onSuccess: (data, variables, context) => {
      options?.onSuccess?.(data, variables, context)
    },
    ...options,
  })
}
