123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374 |
- import {
- ref,
- reactive,
- Ref,
- unref,
- watchEffect,
- watchSyncEffect,
- WatchStopHandle,
- set,
- isRef,
- } from "@nuxtjs/composition-api"
- import {
- createClient,
- TypedDocumentNode,
- OperationResult,
- dedupExchange,
- OperationContext,
- fetchExchange,
- makeOperation,
- GraphQLRequest,
- createRequest,
- subscriptionExchange,
- } from "@urql/core"
- import { authExchange } from "@urql/exchange-auth"
- import { offlineExchange } from "@urql/exchange-graphcache"
- import { makeDefaultStorage } from "@urql/exchange-graphcache/default-storage"
- import { devtoolsExchange } from "@urql/devtools"
- import { SubscriptionClient } from "subscriptions-transport-ws"
- import * as E from "fp-ts/Either"
- import * as TE from "fp-ts/TaskEither"
- import { pipe, constVoid } from "fp-ts/function"
- import { Source, subscribe, pipe as wonkaPipe, onEnd } from "wonka"
- import { keyDefs } from "./caching/keys"
- import { optimisticDefs } from "./caching/optimistic"
- import { updatesDef } from "./caching/updates"
- import { resolversDef } from "./caching/resolvers"
- import schema from "./backend-schema.json"
- import {
- authIdToken$,
- getAuthIDToken,
- probableUser$,
- waitProbableLoginToConfirm,
- } from "~/helpers/fb/auth"
- const BACKEND_GQL_URL =
- process.env.context === "production"
- ? "https://api.hoppscotch.io/graphql"
- : "https://api.hoppscotch.io/graphql"
- const storage = makeDefaultStorage({
- idbName: "hoppcache-v1",
- maxAge: 7,
- })
- const subscriptionClient = new SubscriptionClient(
- process.env.context === "production"
- ? "wss://api.hoppscotch.io/graphql"
- : "wss://api.hoppscotch.io/graphql",
- {
- reconnect: true,
- connectionParams: () => {
- return {
- authorization: `Bearer ${authIdToken$.value}`,
- }
- },
- }
- )
- authIdToken$.subscribe(() => {
- subscriptionClient.client?.close()
- })
- const createHoppClient = () =>
- createClient({
- url: BACKEND_GQL_URL,
- exchanges: [
- devtoolsExchange,
- dedupExchange,
- offlineExchange({
- schema: schema as any,
- keys: keyDefs,
- optimistic: optimisticDefs,
- updates: updatesDef,
- resolvers: resolversDef,
- storage,
- }),
- authExchange({
- addAuthToOperation({ authState, operation }) {
- if (!authState || !authState.authToken) {
- return operation
- }
- const fetchOptions =
- typeof operation.context.fetchOptions === "function"
- ? operation.context.fetchOptions()
- : operation.context.fetchOptions || {}
- return makeOperation(operation.kind, operation, {
- ...operation.context,
- fetchOptions: {
- ...fetchOptions,
- headers: {
- ...fetchOptions.headers,
- Authorization: `Bearer ${authState.authToken}`,
- },
- },
- })
- },
- willAuthError({ authState }) {
- return !authState || !authState.authToken
- },
- getAuth: async () => {
- if (!probableUser$.value) return { authToken: null }
- await waitProbableLoginToConfirm()
- return {
- authToken: getAuthIDToken(),
- }
- },
- }),
- fetchExchange,
- subscriptionExchange({
- forwardSubscription: (operation) =>
- // @ts-expect-error: An issue with the Urql typing
- subscriptionClient.request(operation),
- }),
- ],
- })
- export const client = ref(createHoppClient())
- authIdToken$.subscribe(() => {
- client.value = createHoppClient()
- })
- type MaybeRef<X> = X | Ref<X>
- type UseQueryOptions<T = any, V = object> = {
- query: TypedDocumentNode<T, V>
- variables?: MaybeRef<V>
- updateSubs?: MaybeRef<GraphQLRequest<any, object>[]>
- defer?: boolean
- pollDuration?: number | undefined
- }
- /**
- * 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
- }
- export const useGQLQuery = <DocType, DocVarType, DocErrorType extends string>(
- _args: UseQueryOptions<DocType, DocVarType>
- ) => {
- const stops: WatchStopHandle[] = []
- const args = reactive(_args)
- const loading: Ref<boolean> = ref(true)
- const isStale: Ref<boolean> = ref(true)
- const data: Ref<E.Either<GQLError<DocErrorType>, DocType>> = ref() as any
- if (!args.updateSubs) set(args, "updateSubs", [])
- const isPaused: Ref<boolean> = ref(args.defer ?? false)
- const pollDuration: Ref<number | null> = ref(args.pollDuration ?? null)
- const request: Ref<GraphQLRequest<DocType, DocVarType>> = ref(
- createRequest<DocType, DocVarType>(
- args.query,
- unref<DocVarType>(args.variables as any) as any
- )
- ) as any
- const source: Ref<Source<OperationResult> | undefined> = ref()
- // Toggles between true and false to cause the polling operation to tick
- const pollerTick: Ref<boolean> = ref(true)
- stops.push(
- watchEffect((onInvalidate) => {
- if (pollDuration.value !== null && !isPaused.value) {
- const handle = setInterval(() => {
- pollerTick.value = !pollerTick.value
- }, pollDuration.value)
- onInvalidate(() => {
- clearInterval(handle)
- })
- }
- })
- )
- stops.push(
- watchEffect(
- () => {
- const newRequest = createRequest<DocType, DocVarType>(
- args.query,
- unref<DocVarType>(args.variables as any) as any
- )
- if (request.value.key !== newRequest.key) {
- request.value = newRequest
- }
- },
- { flush: "pre" }
- )
- )
- stops.push(
- watchEffect(
- () => {
- // Just listen to the polling ticks
- // eslint-disable-next-line no-unused-expressions
- pollerTick.value
- source.value = !isPaused.value
- ? client.value.executeQuery<DocType, DocVarType>(request.value, {
- requestPolicy: "cache-and-network",
- })
- : undefined
- },
- { flush: "pre" }
- )
- )
- watchSyncEffect((onInvalidate) => {
- if (source.value) {
- loading.value = true
- isStale.value = false
- const invalidateStops = args.updateSubs!.map((sub) => {
- return wonkaPipe(
- client.value.executeSubscription(sub),
- onEnd(() => {
- if (source.value) execute()
- }),
- subscribe(() => {
- return execute()
- })
- ).unsubscribe
- })
- invalidateStops.push(
- wonkaPipe(
- source.value,
- onEnd(() => {
- loading.value = false
- isStale.value = false
- }),
- subscribe((res) => {
- if (res.operation.key === request.value.key) {
- data.value = pipe(
- // The target
- res.data as DocType | undefined,
- // Define what happens if data does not exist (it is an error)
- E.fromNullable(
- pipe(
- // Take the network error value
- res.error?.networkError,
- // If it null, set the left to the generic error name
- E.fromNullable(res.error?.message),
- E.match(
- // The left case (network error was null)
- (gqlErr) =>
- <GQLError<DocErrorType>>{
- type: "gql_error",
- error: parseGQLErrorString(
- gqlErr ?? ""
- ) as DocErrorType,
- },
- // The right case (it was a GraphQL Error)
- (networkErr) =>
- <GQLError<DocErrorType>>{
- type: "network_error",
- error: networkErr,
- }
- )
- )
- )
- )
- loading.value = false
- }
- })
- ).unsubscribe
- )
- onInvalidate(() => invalidateStops.forEach((unsub) => unsub()))
- }
- })
- const execute = (updatedVars?: DocVarType) => {
- if (updatedVars) {
- if (isRef(args.variables)) {
- args.variables.value = updatedVars
- } else {
- set(args, "variables", updatedVars)
- }
- }
- isPaused.value = false
- }
- const response = reactive({
- loading,
- data,
- isStale,
- execute,
- })
- return response
- }
- const parseGQLErrorString = (s: string) =>
- s.startsWith("[GraphQL] ") ? s.split("[GraphQL] ")[1] : s
- export const runMutation = <
- DocType,
- DocVariables extends object | undefined,
- DocErrors extends string
- >(
- mutation: TypedDocumentNode<DocType, DocVariables>,
- variables?: DocVariables,
- additionalConfig?: Partial<OperationContext>
- ): TE.TaskEither<GQLError<DocErrors>, DocType> =>
- pipe(
- TE.tryCatch(
- () =>
- client.value
- .mutation(mutation, variables, {
- requestPolicy: "cache-and-network",
- ...additionalConfig,
- })
- .toPromise(),
- () => constVoid() as never // The mutation function can never fail, so this will never be called ;)
- ),
- TE.chainEitherK((result) =>
- pipe(
- result.data,
- E.fromNullable(
- // Result is null
- pipe(
- result.error?.networkError,
- E.fromNullable(result.error?.message),
- E.match(
- // The left case (network error was null)
- (gqlErr) =>
- <GQLError<DocErrors>>{
- type: "gql_error",
- error: parseGQLErrorString(gqlErr ?? ""),
- },
- // The right case (it was a network error)
- (networkErr) =>
- <GQLError<DocErrors>>{
- type: "network_error",
- error: networkErr,
- }
- )
- )
- )
- )
- )
- )
|