# Federation

Federation is an optional module which can be included in your configuration to enroll with a federated schema.

# Dependencies

caliban-federation only depends on caliban-core and is very unobtrusive.

To use, add the following dependency to your build.sbt file:

"com.github.ghostdogpr" %% "caliban-federation" % "2.9.1"

# Federating

Federation allows graphs to become part of a larger graph without having to share models or create brittle schema stitching code at the gateway level.

You can read more about federation and why it may be useful here (opens new window).

Federation creates a wrapper over your existing schema so that it can add the necessary hooks to support interaction with the gateway.

If you already have a graph you can add federation simply by adding the federated annotation:

import caliban.federation.v1._

val schema: GraphQL[R] = graphQL(RootResolver(Queries(
  characters = List(Character("Amos"))
)))

val federatedSchema: GraphQL[R] = schema @@ federated

This will wrap the bare minimum schema additions around your API so that the gateway will recognize your schema. To actually enable entity resolution you will need to do a bit of leg work.

First, any types that will be "resolvable" need to be annotated with a @key directive. You can use a helper function found in the federation package to help with that.

@GQLKey("name")
case class Character(name: String)

The "name" field is a field selector minus the outer braces.

If you need to extend a type from another service, you will need to define a stub version of it in the current service and annotate it with the @extends annotation

@GQLKey("season episode") 
@GQLExtend
case class Episode(@GQLExternal season: Int, @GQLExternal episode: Int, cast: List[Character])

Note the additional annotations we needed in this case. Extend is needed to tell the gateway that this type is defined within another service, while the External flags these fields as being owned by the other service (there are several other annotations available that you are encouraged to read about).

Once you have annotated your types, you need to tell Federation how to resolve those types. Federation uses a slightly different mechanism in resolving types from a standard GraphQL query, so for each type that you wish to support, you will need to add an EntityResolver:

EntityResolver[CharacterService, CharacterArgs, Character](args => 
  ZQuery.fromZIO(characters.getCharacter(args.name))
)  

EntityResolvers like normal field resolvers also supports a "metadata" variant which can be used to inspect the requested fields and potentially optimize the resulting query. You can use the provided helper method if you need to access the metadata field:

EntityResolver.fromMetadata[CharacterArgs](field => args => {
  if (field.fields.forall(_.name == "name")) ZQuery.succeed(Character(args.name, Nil, None))
  else ZQuery.fromZIO(characters.getCharacter(args.name))
})

In the above we need to define an resolver which takes an R environment type, an A which has an implicit ArgBuilder and an Option[Out] where Out has an implicit Schema[R, Out] available. Creating the above we can now add these resolvers to our federated schema like so:

schema @@ federated(aResolver, additionalResolvers:_*)

You can now use the resulting GraphQL[R] to start querying. You can also see the full code example here (opens new window)

# Tracing

Federated tracing is slightly different from standard apollo-tracing thus it comes with its own wrapper defined in the caliban-federation module.

import caliban.federation.tracing.ApolloFederatedTracing


val api = schema @@ federated(resolver, additionalResolvers: _*) @@ ApolloFederatedTracing.wrapper()

In federated tracing the gateway communicates with the implementing service via a header apollo-federation-include-trace, for now the only value it can send is ftv1. Thus if you detect this header then you should enable tracing otherwise you can disable it.

If you are using one of the wrappers you are done, they will automatically detect when the gateway enables tracing on a request. However, if you are calling the interpreter.execute independently or you have some other custom set up you will need to add one more step to enable tracing.

If you wish to enable it manually (after detecting the header with your preferred framework) you can call: request.withFederatedTracing which will return a new GraphQLRequest that informs the wrapper that it should include tracing data as part of the response extensions.

# Federation V2

Caliban can support the v2 federation specification as well. If your gateway supports the Federation V2 specification (opens new window), you can specify the supported feature set by using caliban.federation.v2_x where x is the minor version of the specification you wish to use.

Directive Caliban Type Version Caliban package
@shareable @GQLShareable v2.0 caliban.federation.v2_0
@inaccessable @GQLInaccessible v2.0 caliban.federation.v2_0
@override @GQLOverride v2.0 caliban.federation.v2_0
@tag @GQLTag v2.0 caliban.federation.v2_0
@composeDirective ComposeDirective v2.1 caliban.federation.v2_1
@interfaceObject @GQLInterfaceObject v2.3 caliban.federation.v2_3

The GQLKey field now also supports the resolvable argument.

Using the new federated aspect from any v2_x package will automatically make your graph available as a v2 schema, even if you aren't using the new directives.

For more information see the Federation V2 specification (opens new window).

# Customizing Federation

Federation 2.1 introduced a new schema level directive called @composeDirective which allows you to specify custom directives that should be visible to clients of the gateway (by default all directives are hidden to clients of the gateway)

GraphQL federation is an evolving specification and not all routers support all features. Caliban provides support for v2.0, v2.1 and v2.3 of the specification. If you need to use an earlier version or you need to customize some aspect of the federation directives (for instance by providing your own @composeDirectives) you can do so by simply extending the FederationV2 class.

// With a package object but you can also create a normal object
package object myFederation extends FederationV2(
  Versions.v2_3 :: 
    Link("https://myspecs.dev/myDirective/v1.0", List(
      Import("@myDirective"),
      Import("@anotherDirective", as = Some("@hello"))
    )) :: 
      ComposeDirective("@myDirective") :: 
      ComposeDirective("@hello") :: Nil
    )
 with FederationDirectivesV2_3

// Then import your new federation object instead of `caliban.federation.v2_3`
import myFederation._