| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207 |
- import {
- computed,
- ref,
- onMounted,
- onBeforeUnmount,
- reactive,
- Ref,
- } from "@nuxtjs/composition-api"
- import { DocumentNode } from "graphql/language"
- import {
- createClient,
- TypedDocumentNode,
- OperationResult,
- defaultExchanges,
- } from "@urql/core"
- import { devtoolsExchange } from "@urql/devtools"
- import * as E from "fp-ts/Either"
- import * as TE from "fp-ts/TaskEither"
- import { pipe, constVoid } from "fp-ts/function"
- import { subscribe } from "wonka"
- import clone from "lodash/clone"
- import { getAuthIDToken } from "~/helpers/fb/auth"
- const BACKEND_GQL_URL =
- process.env.CONTEXT === "production"
- ? "https://api.hoppscotch.io/graphql"
- : "https://api.hoppscotch.io/graphql"
- const client = createClient({
- url: BACKEND_GQL_URL,
- fetchOptions: () => {
- const token = getAuthIDToken()
- return {
- headers: {
- authorization: token ? `Bearer ${token}` : "",
- },
- }
- },
- exchanges: [devtoolsExchange, ...defaultExchanges],
- })
- /**
- * A wrapper type for defining errors possible in a GQL operation
- */
- export type GQLError<T extends string> =
- | {
- type: "network_error"
- error: Error
- }
- | {
- type: "gql_error"
- error: T
- }
- const DEFAULT_QUERY_OPTIONS = {
- noPolling: false,
- pause: undefined as Ref<boolean> | undefined,
- }
- type GQL_QUERY_OPTIONS = typeof DEFAULT_QUERY_OPTIONS
- type UseQueryLoading = {
- loading: true
- }
- type UseQueryLoaded<
- QueryFailType extends string = "",
- QueryReturnType = any
- > = {
- loading: false
- data: E.Either<GQLError<QueryFailType>, QueryReturnType>
- }
- type UseQueryReturn<QueryFailType extends string = "", QueryReturnType = any> =
- | UseQueryLoading
- | UseQueryLoaded<QueryFailType, QueryReturnType>
- export function isLoadedGQLQuery<QueryFailType extends string, QueryReturnType>(
- x: UseQueryReturn<QueryFailType, QueryReturnType>
- ): x is {
- loading: false
- data: E.Either<GQLError<QueryFailType>, QueryReturnType>
- } {
- return !x.loading
- }
- export function useGQLQuery<
- QueryReturnType = any,
- QueryFailType extends string = "",
- QueryVariables extends object = {}
- >(
- query: string | DocumentNode | TypedDocumentNode<any, QueryVariables>,
- variables?: QueryVariables,
- options: Partial<GQL_QUERY_OPTIONS> = DEFAULT_QUERY_OPTIONS
- ):
- | { loading: false; data: E.Either<GQLError<QueryFailType>, QueryReturnType> }
- | { loading: true } {
- type DataType = E.Either<GQLError<QueryFailType>, QueryReturnType>
- const finalOptions = Object.assign(clone(DEFAULT_QUERY_OPTIONS), options)
- const data = ref<DataType>()
- let subscription: { unsubscribe(): void } | null = null
- onMounted(() => {
- const gqlQuery = client.query<any, QueryVariables>(query, variables)
- const processResult = (result: OperationResult<any, QueryVariables>) =>
- pipe(
- // The target
- result.data as QueryReturnType | undefined,
- // Define what happens if data does not exist (it is an error)
- E.fromNullable(
- pipe(
- // Take the network error value
- result.error?.networkError,
- // If it null, set the left to the generic error name
- E.fromNullable(result.error?.name),
- E.match(
- // The left case (network error was null)
- (gqlErr) =>
- <GQLError<QueryFailType>>{
- type: "gql_error",
- error: gqlErr as QueryFailType,
- },
- // The right case (it was a GraphQL Error)
- (networkErr) =>
- <GQLError<QueryFailType>>{
- type: "network_error",
- error: networkErr,
- }
- )
- )
- )
- )
- if (finalOptions.noPolling) {
- gqlQuery.toPromise().then((result) => {
- data.value = processResult(result)
- })
- } else {
- subscription = pipe(
- gqlQuery,
- subscribe((result) => {
- data.value = processResult(result)
- })
- )
- }
- })
- onBeforeUnmount(() => {
- subscription?.unsubscribe()
- })
- return reactive({
- loading: computed(() => !data.value),
- data: data!,
- }) as
- | {
- loading: false
- data: DataType
- }
- | { loading: true }
- }
- export const runMutation = <
- MutationReturnType = any,
- MutationFailType extends string = "",
- MutationVariables extends {} = {}
- >(
- mutation: string | DocumentNode | TypedDocumentNode<any, MutationVariables>,
- variables?: MutationVariables
- ): TE.TaskEither<GQLError<MutationFailType>, NonNullable<MutationReturnType>> =>
- pipe(
- TE.tryCatch(
- () => client.mutation<MutationReturnType>(mutation, variables).toPromise(),
- () => constVoid() as never // The mutation function can never fail, so this will never be called ;)
- ),
- TE.chainEitherK((result) =>
- pipe(
- result.data as MutationReturnType, // If we have the result, then okay
- E.fromNullable(
- // Result is null
- pipe(
- result.error?.networkError, // Check for network error
- E.fromNullable(result.error?.name), // If it is null, then it is a GQL error
- E.match(
- // The left case (network error was null)
- (gqlErr) =>
- <GQLError<MutationFailType>>{
- type: "gql_error",
- error: gqlErr as MutationFailType,
- },
- // The right case (it was a GraphQL Error)
- (networkErr) =>
- <GQLError<MutationFailType>>{
- type: "network_error",
- error: networkErr,
- }
- )
- )
- )
- )
- )
- )
|