Learn WunderGraph
TypeScript Operations

In the previous section, we've used GraphQL to join data across two APIs. Let's see how we can use TypeScript Operations to achieve the same result and go beyond what's possible with pure GraphQL.

Let's start by defining two GraphQL Operations first, so we can fetch a city and weather data.

# wundergraph/operations/internal/Country.graphql
query ($code: String!) {
  countries: countries_countries(filter: { code: { eq: $code } }) {
    code
    name
    capital
  }
}

And the second one for fetching the weather data:

# wundergraph/operations/internal/Weather.graphql
query ($city: String!) {
  getCityByName: weather_getCityByName(name: $city) {
    name
    id
    weather {
      summary {
        title
        description
      }
      temperature {
        actual
        feelsLike
        min
        max
      }
    }
  }
}

Note that both Operations are placed in the internal directory. That's because we don't want to expose them to the public API, but only use them internally.

Last step, we need to define a TypeScript Operation to glue it all together:

// wundergraph/operations/weather/CountryWeather.ts
import { createOperation, z } from '../../generated/wundergraph.factory';
 
export default createOperation.query({
  input: z.object({
    countryCode: z.string(),
  }),
  handler: async ({ operations, input }) => {
    const country = await operations.query({
      operationName: 'internal/Country',
      input: {
        code: input.countryCode,
      },
    });
    if (!country.data?.countries[0].capital) {
      throw new Error('No capital found');
    }
    const weather = await operations.query({
      operationName: 'internal/Weather',
      input: {
        city: country.data?.countries[0].capital,
      },
    });
    const out: {
      country: string;
      capital: string;
      weather: {
        title: string;
        description: string;
      };
    } = {
      country: country.data?.countries[0].name || '',
      capital: country.data?.countries[0].capital || '',
      weather: {
        title: weather.data?.getCityByName?.weather?.summary?.title || '',
        description: weather.data?.getCityByName?.weather?.summary?.description || '',
      },
    };
    return out;
  },
});

TypeScript Operations follow more or less the same structure as GraphQL Operations, the flow is just a bit different. We don't derive a JSON Schema from the GraphQL Variables, but instead use the zod library to define the input schema.

We're then free to put whatever code we want into the handler function. In this case, we're using the operations object passed to the handler to execute the two GraphQL Operations we've defined before.

Let's call our operation:

curl -X GET http://localhost:9991/operations/weather/CountryWeather?countryCode=DE

And we get the following response, it's freezingly cold in Berlin right now:

{
  "data": {
    "country": "Germany",
    "capital": "Berlin",
    "weather": {
      "title": "Snow",
      "description": "light snow"
    }
  }
}