123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104 |
- // Copyright (C) 2012-2025 Zammad Foundation, https://zammad-foundation.org/
- import { ApolloLink, createHttpLink, from } from '@apollo/client/core'
- import { BatchHttpLink } from '@apollo/client/link/batch-http'
- import { removeTypenameFromVariables } from '@apollo/client/link/remove-typename'
- import { getMainDefinition } from '@apollo/client/utilities'
- import ActionCableLink from 'graphql-ruby-client/subscriptions/ActionCableLink'
- import { consumer } from '#shared/server/action_cable/consumer.ts'
- import connectedStateLink from './link/connectedState.ts'
- import csrfLink from './link/csrf.ts'
- import debugLink from './link/debug.ts'
- import errorLink from './link/error.ts'
- import setAuthorizationLink from './link/setAuthorization.ts'
- import testFlagsLink from './link/testFlags.ts'
- import type { Operation } from '@apollo/client/core'
- import type { FragmentDefinitionNode, OperationDefinitionNode } from 'graphql'
- // Should subsequent HTTP calls be batched together?
- const enableBatchLink = true
- // Should queries and mutations be sent over ActionCable?
- const enableQueriesOverWebsocket = false
- const connectionSettings = {
- uri: '/graphql',
- credentials: 'same-origin', // Must have for CSRF validation via Rails.
- }
- const noBatchLink = createHttpLink(connectionSettings)
- const batchLink = new BatchHttpLink({
- ...connectionSettings,
- batchMax: 5,
- batchInterval: 20,
- })
- const operationIsLoginLogout = (
- definition: OperationDefinitionNode | FragmentDefinitionNode,
- ) => {
- return !!(
- definition.kind === 'OperationDefinition' &&
- definition.operation === 'mutation' &&
- definition.name?.value &&
- ['login', 'logout'].includes(definition.name?.value)
- )
- }
- // TODO: Maybe we can also add a generic solution with the query context to exclude operation for batching or
- // run operations over websocket.
- const operationIsFormUpdater = (
- definition: OperationDefinitionNode | FragmentDefinitionNode,
- ) => {
- return !!(
- definition.kind === 'OperationDefinition' &&
- definition.operation === 'query' &&
- definition.name?.value &&
- definition.name?.value === 'formUpdater'
- )
- }
- const requiresBatchLink = (op: Operation) => {
- if (!enableBatchLink) return false
- const definition = getMainDefinition(op.query)
- return (
- !operationIsLoginLogout(definition) && !operationIsFormUpdater(definition)
- )
- }
- const httpLink = ApolloLink.split(requiresBatchLink, batchLink, noBatchLink)
- const requiresHttpLink = (op: Operation) => {
- const definition = getMainDefinition(op.query)
- if (!enableQueriesOverWebsocket) {
- // Only subscriptions over websocket.
- return (
- !(
- definition.kind === 'OperationDefinition' &&
- definition.operation === 'subscription'
- ) && !operationIsFormUpdater(definition)
- )
- }
- // Everything over websocket except login/logout as that changes cookies.
- return operationIsLoginLogout(definition)
- }
- const actionCableLink = new ActionCableLink({ cable: consumer })
- const splitLink = ApolloLink.split(requiresHttpLink, httpLink, actionCableLink)
- const link = from([
- ...(VITE_TEST_MODE ? [testFlagsLink] : []),
- csrfLink,
- errorLink,
- setAuthorizationLink,
- debugLink,
- connectedStateLink,
- removeTypenameFromVariables(),
- splitLink,
- ])
- export default link
|