Design, create and test your API in Symfony

API Blueprint

A powerful high-level API description language for web APIs.

API Blueprint

Blueprint - format

FORMAT: 1A
HOST: https://erp.local/api

# ERP API Documentation
REST API documentation for ERP application

## API Authorization
User authorization is based on JWT: [https://jwt.io/](https://jwt.io/)

Header example:
```http
Authorization: Bearer eyJhbGciOiJSUzI1NiIsInR5cCI6IkpXUyJ9
```

## General responses
Api response that can occur on each request

**401 Unauthorized**

User not authorized with correct token

Example response:
```http
{
  "error": {
    "code": 401,
    "message": "Unauthorized"
  }
}
```

API - data structures

# Data Structures

## EmployeeData
+ id: 1 (required, number) - Unique identifier
+ firstName: Jan (required, string)
+ lastName: Kowalski (required, string)
+ legalBasis: contract (required, string)
+ position: accountant (required, string)
+ status: hired (required, string)

## EmployeeList
+ page: 1 (number)
+ limit: 10 (number)
+ pages: 1 (number)
+ total: 2 (number)
+ _links
    + self
        + href: "/api/employee/?page=1&limit=10"
    + first
        + href: "/api/employee/?page=1&limit=10"
    + last
        + href: "/api/employee/?page=1&limit=10"
+ _embedded
    + items (array)
        + (EmployeeData)
        + (EmployeeData)

API - groups

# Group Employee

## List [/employee]

### Get employees [GET]
Get a list of employees.

+ Response 200 (application/json)

    + Attributes (EmployeeList)

### Create new employee [POST]
Create a new employee from request data

+ Request (application/json)

    + Body

            {
                "firstName": "John",
                "lastName": "Doe",
                "legalBasis": "contract",
                "position": "accountant",
                "status": "hired"
            }

+ Response 201

    + Attributes (EmployeeData)

+ Response 400 (application/json)

    + Body

            {
                "error": "Invalid title"
            }

API - schema

### Suggest Employee [GET]
Get employee suggestions based on text input

+ Response 200 (application/json)

    + Attributes (array[EmployeeSuggestData])

    + Schema

            {
                "$schema": "http://json-schema.org/draft-04/schema#",
                "type": "array",
                "items": {
                    "type": "object",
                    "properties": {
                        "first_name": {
                            "type": "string"
                        },
                        "last_name": {
                            "type": "string"
                        }
                    }
                }
            }

Aglio

An API Blueprint renderer that supports multiple themes and outputs static HTML that can be served by any web host.

Aglio

Aglio

aglio --theme-variables slate 
    -i app/Resources/blueprints/api.apib 
    -o web/apidoc.html
tools/apidoc.sh

Aglio

Dredd

Dredd is a language-agnostic command-line tool for validating API documentation written in API Blueprint format against its backend implementation.

Frontend testuje backend !!!

Dredd - config

dry-run: null
hookfiles:
  - tests/dredd/hooks.php
language: vendor/ddelnano/dredd-hooks-php/bin/dredd-hooks-php
sandbox: false
server: null
server-wait: 3
init: false
custom: {}
names: false
only: []
reporter: []
output: []
header: []
sorted: false
user: null
inline-errors: false
details: false
method: []
color: true
level: info
timestamp: false
silent: false
path: []
blueprint: app/Resources/blueprints/api.apib
endpoint: 'http://localhost/api'

Dredd - hooks

<?php
use Dredd\Hooks;

$loader = require __DIR__.'/../../app/autoload.php';
$kernel = new AppKernel('prod', false);
$kernel->loadClassCache();
$kernel->boot();
$container = $kernel->getContainer();

$user = $container->get('dvs.user_bundle.repository.doctrine.user_repository')->findByUsername('admin');
$tokenManager = $container->get('lexik_jwt_authentication.jwt_manager');
$token = $tokenManager->create($user);

Hooks::beforeEach(function(&$transaction) use ($token) {
    $transaction->request->
        headers->Authorization = 'Bearer '.$token;
});

Dredd - in action

dredd

tools/test_api.sh

SyliusResourceBundle

Rapid RESTful API Development

SyliusResourceBundle

SyliusResourceBundle

sylius_resource:
    resources:
        dvs.employee:
            classes:
                model: Dvs\Domain\Employee\Entity\Employee
<?php

declare (strict_types = 1);

namespace Dvs\Domain\Employee\Entity;

use Sylius\Component\Resource\Model\ResourceInterface;

class Employee implements ResourceInterface
{
    /**
     * @var int
     */
    private $id;

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

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

    /**
     * @var string
     */
    private $legalBasis;

    /**
     * @var string
     */
    private $position;

    /**
     * @var string
     */
    private $status;

    /**
     * @return int
     */
    public function getId()
    {
        return $this->id;
    }

    /**
     * @return string
     */
    public function getFirstName()
    {
        return $this->firstName;
    }

    /**
     * @param string $firstName
     */
    public function setFirstName(string $firstName)
    {
        $this->firstName = $firstName;
    }

    /**
     * @return string
     */
    public function getLastName()
    {
        return $this->lastName;
    }

    /**
     * @param string $lastName
     */
    public function setLastName(string $lastName)
    {
        $this->lastName = $lastName;
    }

    /**
     * @return string
     */
    public function getLegalBasis()
    {
        return $this->legalBasis;
    }

    /**
     * @param string $legalBasis
     */
    public function setLegalBasis(string $legalBasis)
    {
        $this->legalBasis = $legalBasis;
    }

    /**
     * @return string
     */
    public function getPosition()
    {
        return $this->position;
    }

    /**
     * @param string $position
     */
    public function setPosition(string $position)
    {
        $this->position = $position;
    }

    /**
     * @return string
     */
    public function getStatus()
    {
        return $this->status;
    }

    /**
     * @param string $status
     */
    public function setStatus(string $status)
    {
        $this->status = $status;
    }

    /**
     * @return array
     */
    public function getSuggest()
    {
        $name = sprintf('%s %s', $this->firstName, $this->lastName);
        $reversedName = sprintf('%s %s', $this->lastName, $this->firstName);
        $input = [$this->firstName, $this->lastName, $name, $reversedName];
        $payload = ['first_name' => $this->firstName, 'last_name' => $this->lastName];

        return [
            'input' => $input,
            'output' => $name,
            'payload' => $payload,
        ];
    }
}

SRB - routing

dvs_employee:
    resource: |
        alias: dvs.employee
        path: employee
        except: ['delete']
    type: sylius.resource_api
    defaults:
        _sylius:
            sortable: true
            filterable: true
app/console debug:router
...
dvs_employee_index            GET         ANY      ANY    /api/employee                           
dvs_employee_create           POST        ANY      ANY    /api/employee                           
dvs_employee_update           PUT|PATCH   ANY      ANY    /api/employee/{id}                      
dvs_employee_show             GET         ANY      ANY    /api/employee/{id}           

SRB - validation

<?php


class UpdateNotificationSettingsType extends AbstractType
{
    /**
     * @param FormBuilderInterface $builder
     * @param array                $options
     */
    public function buildForm(FormBuilderInterface $builder, array $options)
    {
        $builder
            ->add('daysBeforeExpiration', 'number', [
                'constraints' => [
                    new NotBlank(),
                    new Range([
                        'min' => 0,
                        'max' => 180,
                    ]),
                ],
            ])
            ->add('method', 'choice', [
                'constraints' => [
                    new NotBlank(),
                ],
                'choices' => NotificationType::getChoices(),
            ]);
    }

    /**
     * @return string
     */
    public function getName()
    {
        return 'updateNotificationSettingsType';
    }
}
<?xml version="1.0" encoding="UTF-8" ?>
<constraint-mapping xmlns="http://symfony.com/schema/dic/constraint-mapping"
                    xmlns:xsi="http://www.w3.org/2001/XMLSchema-instance"
                    xsi:schemaLocation="http://symfony.com/schema/dic/constraint-mapping http://symfony.com/schema/dic/constraint-mapping/constraint-mapping-1.0.xsd">

    <class name="Dvs\Domain\Employee\Entity\Employee">
        <property name="firstName">
            <constraint name="NotBlank"/>
        </property>
        <property name="lastName">
            <constraint name="NotBlank"/>
        </property>
        <property name="legalBasis">
            <constraint name="NotBlank"/>
            <constraint name="Choice">
                <option name="callback">
                    <value>Dvs\Domain\Employee\Type\LegalBasisType</value>
                    <value>getValues</value>
                </option>
            </constraint>
        </property>
        <property name="position">
            <constraint name="NotBlank"/>
            <constraint name="Choice">
                <option name="callback">
                    <value>Dvs\Domain\Employee\Type\PositionType</value>
                    <value>getValues</value>
                </option>
            </constraint>
        </property>
        <property name="status">
            <constraint name="NotBlank"/>
            <constraint name="Choice">
                <option name="callback">
                    <value>Dvs\Domain\Employee\Type\StatusType</value>
                    <value>getValues</value>
                </option>
            </constraint>
        </property>
    </class>
</constraint-mapping>

SRB - list

{
  "page": 1,
  "limit": 10,
  "pages": 6,
  "total": 52,
  "_links": {
    "self": {
      "href": "/api/employee?page=1&limit=10"
    },
    "first": {
      "href": "/api/employee?page=1&limit=10"
    },
    "last": {
      "href": "/api/employee?page=6&limit=10"
    },
    "next": {
      "href": "/api/employee?page=2&limit=10"
    }
  },
  "_embedded": {
    "items": [
      {
        "id": 3,
        "firstName": "Dariusz",
        "lastName": "Szymczak",
        "legalBasis": "contract",
        "position": "fitter",
        "status": "candidate"
      },
      {
        "id": 19,
        "firstName": "Anita",
        "lastName": "Król",
        "legalBasis": "order",
        "position": "accountant",
        "status": "hired"
      },
      {
        "id": 27,
        "firstName": "Arkadiusz",
        "lastName": "Tomaszewski",
        "legalBasis": "none",
        "position": "accountant",
        "status": "dismissed"
      },
      {
        "id": 32,
        "firstName": "Dominik",
        "lastName": "Grabowska",
        "legalBasis": "order",
        "position": "staff",
        "status": "hired"
      },
      {
        "id": 35,
        "firstName": "Dariusz",
        "lastName": "Dudek",
        "legalBasis": "contract",
        "position": "fitter",
        "status": "dismissed"
      },
      {
        "id": 36,
        "firstName": "Alicja",
        "lastName": "Brzezińska",
        "legalBasis": "contract",
        "position": "management-assistant",
        "status": "dismissed"
      },
      {
        "id": 37,
        "firstName": "Dawid",
        "lastName": "Krawczyk",
        "legalBasis": "order",
        "position": "foreman",
        "status": "hired"
      },
      {
        "id": 40,
        "firstName": "Apolonia",
        "lastName": "Zakrzewska",
        "legalBasis": "recruitment",
        "position": "foreman",
        "status": "dismissed"
      },
      {
        "id": 42,
        "firstName": "Anita",
        "lastName": "Borowski",
        "legalBasis": "contract",
        "position": "fitter",
        "status": "hired"
      },
      {
        "id": 50,
        "firstName": "Bianka",
        "lastName": "Michalak",
        "legalBasis": "recruitment",
        "position": "fitter",
        "status": "dismissed"
      }
    ]
  }
}

SRB - sort & filter

GET /api/employee?sorting[firstName]=asc
GET /api/employee?criteria[position]=fitter
&criteria[status]=dismissed

SRB - want more ?

http://docs.sylius.org/en/latest/
bundles/SyliusResourceBundle/

Thx !

Your are the best team I've worked with

Design, document, create and test your API in Symfony

By Arkadiusz Kondas

Design, document, create and test your API in Symfony

Symfony + SyliusResourceBundle + API Blueprint + Aglio + Dredd = Perfect API

  • 1,318