Chris Colborne

Build a GraphQL server with Laravel

July 26, 2018

One of the cool new things coming out from Facebook Open Source is GraphQL.

What is GraphQL?

GraphQL is a specification for communicating with APIs. You can think of GraphQL as an equivalent to REST, where GraphQL and REST are both just specifications for how to communicate with an API.

Why is GraphQL awesome?

GraphQL is awesome because it

  • has a single endpoint
  • allows for relational queries
  • eliminates over-requesting and under-requesting
  • can return all the data in one request
  • can mitigate the need for versioning as the client determines what it wants, not the server
  • can combine disparate data sources into a single API

How do we do it?

To use GraphQL, we need two parts. First we need a GraphQL server, and then we need a GraphQL client. In this post we’ll build out a GraphQL server in Laravel, and then in a following post, we’ll build a client app which uses our new server to display some data.

GraphQL plus Laravel

Let’s go!

We’ll be building upon Steve Luscher’s awesome zero-to-graphql talk and repository. In the talk he spun up a GraphQL endpoint in Django, Rails and Node inside 30 minutes.

It exposes a person model as follows:

class Person
{
    /** @var string */
    public $id;

    /** @var string */
    public $firstName;

    /** @var string */
    public $lastName;

    /** @var string */
    public $username;

    /** @var string */
    public $email;

    /** @var Person[] */
    public $friends;
}

It also provides two queries, allPeople to list all the people, and person to retrieve a particular Person by ID.

We’ll use Laravel and Laravel GraphQL to create our very own GraphQL server with Laravel that meets this schema.

Set up a new Laravel app

First up we’ll use the Laravel CLI to create a new application.

$ laravel new zero-graphql

Add a database

I’m using Laradock to serve and provide a MySQL database for our users, but you could just as easily use Homestead or a locally installed Mysql database with Valet.

Configuration of this step is beyond the scope of this post, but see the repo and Laradock docs for a configured Laradock setup, or the docs for Homestead or Valet

Add Laravel GraphQL

Next up we’ll add the Laravel GraphQL library to provide some GraphQL primitives, as well as the GraphiQL web based API client for testing the endpoint.

# Require the library
$ composer require folklore/graphql
# publish the graphql config and graphiql route
$ artisan vendor:publish --provider="Folklore\GraphQL\ServiceProvider"

Add Person Type

As GraphQL is a strongly typed definition, we need to define the types we’ll use, in our case the Person type. Thankfully the Laravel GraphQL library provides an artisan command to scaffold out our Person type.

$ ./artisan make:graphql:type PersonType

And then add our fields to the newly generated type file.

// app/GraphQL/Type/PersonType.php
...
return [
    'id' => [
        'type' => Type::nonNull(Type::int()),
        'description' => 'The id of the user'
    ],
    'first_name' => [
        'type' => Type::nonNull(Type::string()),
        'description' => 'The firstName of the user'
    ],
    'last_name' => [
        'type' => Type::nonNull(Type::string()),
        'description' => 'The lastName of the user'
    ],
    'username' => [
        'type' => Type::nonNull(Type::string()),
        'description' => 'The username of the user'
    ],
    'email' => [
        'type' => Type::nonNull(Type::string()),
        'description' => 'The email of user'
    ],
    'friends' => [
        'type' => Type::listOf(GraphQL::type('Person')),
        'description' => 'Person\'s friends'
    ]
];

Add People Data

So we have something for our GraphQL API to return, let’s add some people to the database. We’ll create our Person model and migrations and seeders for each of the People and Friendships tables. Again we can use artisan to scaffold out the files.

$ ./artisan make:model Person -m
$ ./artisan make:seeder PeopleTableSeeder
$ ./artisan make:seeder FriendshipsTableSeeder

We’ll add our columns to our people and our friendships table and some seed data for each.

// database/migrations/2018_04_16_045416_create_people_table.php
...
    public function up()
    {
        Schema::create('people', function (Blueprint $table) {
            $table->increments('id');
            $table->string('first_name');
            $table->string('last_name');
            $table->string('username');
            $table->string('email');
            $table->timestamps();
        });
    }
...
// database/seeds/PeopleTableSeeder.php

use Illuminate\Database\Seeder;

class PeopleTableSeeder extends Seeder
{
    /**
     * Run the database seeds.
     *
     * @return void
     */
    public function run()
    {
        DB::table('people')->insert([
            'id' => 1,
            'first_name' => 'Kyle',
            'last_name' => 'Reyes',
            'username' => 'kreyes',
            'email' => '[email protected]',
        ]);
        DB::table('people')->insert([
            'id' => 2,
            'first_name' => 'Emma',
            'last_name' => 'Thomsen',
            'username' => 'ethomsen',
            'email' => '[email protected]',
        ]);
        DB::table('people')->insert([
            'id' => 3,
            'first_name' => 'Crispim',
            'last_name' => 'Ramos',
            'username' => 'cramos',
            'email' => '[email protected]',
        ]);
    }
}
// database/migrations/2018_04_16_051549_create_friendships_table.php
...
    public function up()
    {
        Schema::create('friendships', function (Blueprint $table) {
            $table->integer('person_id');
            $table->integer('friend_id');
            $table->timestamps();
        });
    }
...
// database/seeds/FriendshipsTableSeeder.php

use Illuminate\Database\Seeder;

class FriendshipsTableSeeder extends Seeder
{
    /**
     * Run the database seeds.
     *
     * @return void
     */
    public function run()
    {
        DB::table('friendships')->insert([
            'person_id' => 1,
            'friend_id' => 2,
        ]);
        DB::table('friendships')->insert([
            'person_id' => 1,
            'friend_id' => 3,
        ]);
        DB::table('friendships')->insert([
            'person_id' => 2,
            'friend_id' => 1,
        ]);
        DB::table('friendships')->insert([
            'person_id' => 2,
            'friend_id' => 3,
        ]);
        DB::table('friendships')->insert([
            'person_id' => 3,
            'friend_id' => 1,
        ]);
        DB::table('friendships')->insert([
            'person_id' => 3,
            'friend_id' => 2,
        ]);
    }
}

And finally our Person model

<?php
// app/Person.php

namespace App;

use Illuminate\Database\Eloquent\Model;

class Person extends Model
{
    public function friends()
    {
        return $this->belongsToMany('App\Person', 'friendships', 'person_id', 'friend_id');
    }
}

GraphQL Queries

Now we’re ready to add our GraphQL queries.

All People Query

The all people query allows us to list all people in the database.

# Query to return everyone
$ ./artisan make:graphql:query AllPeopleQuery

Update the name to allPeople

protected $attributes = [
    'name' => 'allPeople'
];

Update the type to a list of People

public function type()
{
    return Type::listOf(GraphQL::type('Person'));
}

And add our resolver

public function resolve($root, $args)
{
    return Person::all();
}

Person Query

The person query allows us to retrieve a single person from the database by id.

# Query to return a person by ID
$ ./artisan make:graphql:query PersonQuery

Update the name to person

protected $attributes = [
    'name' => 'person'
];

Update the type to a Person

public function type()
{
    return GraphQL::type('Person');
}

And add our resolver

public function args()
{
    // this allows us to query a particular id
    // it is passed to resolve() below
    return [
        'id' => ['name' => 'id', 'type' => Type::nonNull(Type::int())],
    ];
}

public function resolve($root, $args)
{
    if (isset($args['id'])) {
        return Person::where('id', $args['id'])->get()->first();
    }
}

GraphiQL

GraphiQL is a handy web based GraphQL client, baked in when we installed Laravel GraphQL. It’s similar to something like Postman which you might have used for testing REST APIs.

You can access it by going to http://localhost/graphiql.

Once you’re there, you can test out the queries we defined earlier.

allPeople Query

Get the username, first name and last name of all users, as well as their friends’ usernames.

Notice we can nest inside the query to return the username of each person’s friends as well. This is another of GraphQL’s benefits.

Request
{
  allPeople {
    username
    first_name
    last_name
    friends {
      username
    }
  }
}
Response
{
  "data": {
    "allPeople": [
      {
        "username": "kreyes",
        "first_name": "Kyle",
        "last_name": "Reyes",
        "friends": [
          {
            "username": "ethomsen"
          },
          {
            "username": "cramos"
          }
        ]
      },
      {
        "username": "ethomsen",
        "first_name": "Emma",
        "last_name": "Thomsen",
        "friends": [
          {
            "username": "kreyes"
          },
          {
            "username": "cramos"
          }
        ]
      },
      {
        "username": "cramos",
        "first_name": "Crispim",
        "last_name": "Ramos",
        "friends": [
          {
            "username": "kreyes"
          },
          {
            "username": "ethomsen"
          }
        ]
      }
    ]
  }
}

Person Query

Get the username of user id 1, along with the username, first name, last name and email of all his friends.

Request
{
  person(id: 1) {
    username
    friends {
      username
      first_name
      last_name
      email
    }
  }
}
Response
{
  "data": {
    "person": {
      "username": "kreyes",
      "friends": [
        {
          "username": "ethomsen",
          "first_name": "Emma",
          "last_name": "Thomsen",
          "email": "[email protected]"
        },
        {
          "username": "cramos",
          "first_name": "Crispim",
          "last_name": "Ramos",
          "email": "[email protected]"
        }
      ]
    }
  }
}

Other cool GraphQL endpoints

Here is a list of cool GraphQL endpoints to play around with and use for inspiration.

https://github.com/APIs-guru/graphql-apis

Next Steps

Check out the Github repo to clone the finished code.

And stay tuned for an upcoming post where we’ll build a client app which uses our new server to display the data!


Like what you've read?

Why not subscribe to my mailing list to get my latest articles by email.

I respect the privacy of your email address. You can unsubscribe at any time.

Written by Chris Colborne, an Aussie software engineer from Brisbane, Australia. Follow me on Twitter

Chris Colborne © 2023