Graphql-php

GraphQL

这是基于 JavaScript 参考实现GraphQL规范的PHP 实现

相关项目

要求

  • PHP版本> = 7.1
  • EXT-MBSTRING

安装

运行以下命令以通过Composer安装程序包:

composer require digiaonline/graphql

下面是一个简单的示例,演示如何从GraphQL模式文件构建可执行模式,该文件包含
星球大战主题模式的Schema定义语言(SDL)(Schema定义本身,见下文)。在
此示例中,我们使用该SDL构建可执行模式,并使用它来查询英雄的名称。该
查询的结果是一个关联数组,其结构类似于我们运行的查询。

use Digia\GraphQL\Language\FileSourceBuilder;
use function Digia\GraphQL\buildSchema;
use function Digia\GraphQL\graphql;

$sourceBuilder = new FileSourceBuilder(__DIR__ . '/star-wars.graphqls');

$schema = buildSchema($sourceBuilder->build(), [
    'Query' => [
        'hero' => function ($rootValue, $arguments) {
            return getHero($arguments['episode'] ?? null);
        },
    ],
]);

$result = graphql($schema, '
query HeroNameQuery {
  hero {
    name
  }
}');

\print_r($result);

上面的脚本产生以下输出:

Array
(
    [data] => Array
    (
        [hero] => Array
        (
            [name] => "R2-D2"
        )

    )

)

此示例中使用的GraphQL架构文件包含以下内容:

schema {
    query: Query
}

type Query {
    hero(episode: Episode): Character
    human(id: String!): Human
    droid(id: String!): Droid
}

interface Character {
    id: String!
    name: String
    friends: [Character]
    appearsIn: [Episode]
}

type Human implements Character {
    id: String!
    name: String
    friends: [Character]
    appearsIn: [Episode]
    homePlanet: String
}

type Droid implements Character {
    id: String!
    name: String
    friends: [Character]
    appearsIn: [Episode]
    primaryFunction: String
}

enum Episode { NEWHOPE, EMPIRE, JEDI }

创建架构

要对GraphQL API执行查询,首先需要定义API的结构。这是
通过创建模式来完成的。有两种方法可以执行此操作,您可以使用SDL执行此操作,也可以通过编程方式执行此操作。
但是,我们强烈建议您使用SDL,因为它更易于使用。要从
SDL 创建可执行Schema,您需要调用该buildSchema函数。

buildSchema函数有三个参数:

  • $source``Schema定义(SDL)作为Source实例
  • $resolverRegistry关联数组或ResolverRegistry包含所有解析器的实例
  • $options 用于构建Schema的选项,其中还包括自定义类型和指令

要创建Source实例,您可以使用提供的FileSourceBuilderMultiFileSourceBuilder类。

Resolver registry

Resolver registry本质上是一个简单的映射,其类型名称将作为其键,其对应的解析器
实例将作为其值。对于较小的项目,您可以使用关联数组和lambda函数(即:匿名函数)来定义
Resolver registry。但是,在较大的项目中,我们建议您实现自己的解析器。您可以
在“ 解析器”(#Resolvers)部分下阅读有关解析器的更多信息。

关联数组示例:

$schema = buildSchema($source, [
    'Query' => [
        'hero' => function ($rootValue, $arguments) {
            return getHero($arguments['episode'] ?? null);
        },
    ],
]);

解析器类示例:

$schema = buildSchema($source, [
    'Query' => [
        'hero' => new HeroResolver(),
    ],
]);

解析器中间件

如果您发现自己在多个解析器中编写相同的逻辑,则应考虑使用中间件。解析器
中间件允许您跨多个解析器有效地管理功能。

在中间件示例之前:

// use Digia\GraphQL\Schema\Resolver\ResolverRegistry; 
$resolves=new ResolverRegistry([
    'Query' => [
        'hero' => function ($rootValue, $arguments) {
            return getHero($arguments['episode'] ?? null);
        },
    ],
], [
    new BeforeMiddleware()
])
$schema = buildSchema($source,$resolves);
class BeforeMiddleware implements ResolverMiddlewareInterface
{
    public function resolve(callable $resolveCallback, $rootValue, array $arguments, $context, ResolveInfo $info) {
        $newRootValue = $this->doSomethingBefore();
        return $resolveCallback($newRootValue, $arguments, $context, $info);
    }
}

中间件示例后:

// use Digia\GraphQL\Schema\Resolver\ResolverRegistry; 
$resolves=new ResolverRegistry([
    'Query' => [
        'hero' => function ($rootValue, $arguments) {
            return getHero($arguments['episode'] ?? null);
        },
    ],
], [
    new AfterMiddleware()
])
$schema = buildSchema($source,$resolves);
class AfterMiddleware implements ResolverMiddlewareInterface
{
    public function resolve(callable $resolveCallback, $rootValue, array $arguments, $context, ResolveInfo $info) {
        $result = $resolveCallback($rootValue, $arguments, $context, $info);
        $this->doSomethingAfter();
        return $result;
    }
}

解析器中间件可用于许多事情; 例如日志记录,输入清理,性能
测量,授权和缓存。

如果您想了解有关Schema的更多信息,可以参考规范

执行

Queries

要对Schema执行查询,您需要调用该graphql函数并将其Schema和要执行的查询传递给它
。您还可以通过更改查询来运行Mutationssubscriptions

$query = '
query HeroNameQuery {
  hero {
    name
  }
}';

$result = graphql($schema, $query);

如果您想了解有关查询的更多信息,可以参考规范

Resolvers

Schema中的每个类型都有一个与之关联的解析器(Resolvers),允许解析实际值。但是,大多数
类型不需要自定义解析器,因为可以使用默认解析器解析它们。通常这些解析器
lambda函数(即:匿名函数),但您也可以通过扩展AbstractTypeResolverAbstractFieldResolver来定义自己的解析器。或者自行实现ResolverInterface

解析器函数接收四个参数:

  • $rootValue父对象,在某些情况下可为null
  • $arguments 在查询中提供给该字段的参数
  • $context 传递给每个可以保存重要上下文信息的解析器的值
  • $info 保存与当前查询相关的字段特定信息的值

Lambda函数示例:

function ($rootValue, array $arguments, $context, ResolveInfo $info): string {
    return [
        'type'       => 'Human',
        'id'         => '1000',
        'name'       => 'Luke Skywalker',
        'friends'    => ['1002', '1003', '2000', '2001'],
        'appearsIn'  => ['NEWHOPE', 'EMPIRE', 'JEDI'],
        'homePlanet' => 'Tatooine',
    ];
}

Type解析器示例:

class HumanResolver extends AbstractTypeResolver
{
    public function resolveName($rootValue, array $arguments, $context, ResolveInfo $info): string
    {
        return $rootValue['name'];
    }
}

Field解析器示例:

class NameResolver extends AbstractFieldResolver
{
    public function resolve($rootValue, array $arguments, $context, ResolveInfo $info): string
    {
       return $rootValue['name'];
    }
}

N + 1问题

解析器函数可以返回“值”,promisepromise数组。
下面的解析器函数说明了如何使用promise来解决N + 1问题,完整的例子可以在
这个测试用例中找到。

$movieType = newObjectType([
    'fields' => [
        'title'    => ['type' => stringType()],
        'director' => [
            'type'    => $directorType,
            'resolve' => function ($movie, $args) {
                DirectorBuffer::add($movie['directorId']);

                return new Promise(function (callable $resolve, callable $reject) use ($movie) {
                    DirectorBuffer::loadBuffered();
                    $resolve(DirectorBuffer::get($movie['directorId']));
                });
            }
        ]
    ]
]);

Variables

通过将查询传递给graphql函数,可以在执行查询时传入变量。

$query = '
query HeroNameQuery($id: ID!) {
  hero(id: $id) {
    name
  }
}';

$variables = ['id' => '1000'];

$result = graphql($schema, $query, null, null, $variables);

Context

如果您需要将一些重要的上下文信息传递给查询,可以使用$contextValues
参数graphql来执行此操作。此数据将作为$context参数传递给所有解析器。

$contextValues = [
    'currentlyLoggedInUser' => $currentlyLoggedInUser,
];

$result = graphql($schema, $query, null, $contextValues, $variables);

Scalars

模式中的叶节点称为标量(Scalars),每个标量(Scalars)可以解析出一些具体数据。
GraphQL中的内置或指定的标量如下:

  • Boolean
  • Float
  • Int
  • ID
  • String

自定义标量

除了指定的标量之外,您还可以定义自己的自定义标量,
并通过将它们传递给buildSchema函数作为$options的一部分参数来让您的Schema了解它们。

自定义日期标量类型示例:

$dateType = newScalarType([
    'name'         => 'Date',
    'serialize'    => function ($value) {
        if ($value instanceof DateTime) {
            return $value->format('Y-m-d');
        }
        return null;
    },
    'parseValue'   => function ($value) {
        if (\is_string($value)){
            return new DateTime($value);
        }
        return null;
    },
    'parseLiteral' => function ($node) {
        if ($node instanceof StringValueNode) {
            return new DateTime($node->getValue());
        }
        return null;
    },
]);

$schema = buildSchema($source, [
    'Query' => QueryResolver::class,
    [
        'types' => [$dateType],
    ],
]);

每个标量都必须被强制执行,由三个不同的函数完成。serialize函数将
PHP值转换为相应的输出值。parseValue函数将变量输入值转换为
相应的PHP值,parseLiteral函数将AST文字(抽象语法树)转换为相应的PHP值。

高级用法

如果您正在寻找本文档尚未涵盖的内容,最好的办法是查看本项目中的
测试。你会惊奇地发现那里有很多例子。

整合

Laravel

这是一个演示如何在Laravel项目中使用此库的示例。您需要一个application service来将此库公开给您的应用程序,一个服务提供者来注册该服务,一个控制器和一个
处理GraphQL POST请求的路由。

app/GraphQL/GraphQLService.php

class GraphQLService
{
    private $schema;

    public function __construct(Schema $schema)
    {
        $this->schema = $schema;
    }

    public function executeQuery(string $query, array $variables, ?string $operationName): array
    {
        return graphql($this->schema, $query, null, null, $variables, $operationName);
    }
}

app/GraphQL/GraphQLServiceProvider.php

class GraphQLServiceProvider
{
    public function register()
    {
        $this->app->singleton(GraphQLService::class, function () {
            $schemaDef = \file_get_contents(__DIR__ . '/schema.graphqls');

            $executableSchema = buildSchema($schemaDef, [
                'Query' => QueryResolver::class,
            ]);

            return new GraphQLService($executableSchema);
        });
    }
}

app/GraphQL/GraphQLController.php

class GraphQLController extends Controller
{
    private $graphqlService;

    public function __construct(GraphQLService $graphqlService)
    {
        $this->graphqlService = $graphqlService;
    }

    public function handle(Request $request): JsonResponse
    {
        $query         = $request->get('query');
        $variables     = $request->get('variables') ?? [];
        $operationName = $request->get('operationName');

        $result = $this->graphqlService->executeQuery($query, $variables, $operationName);

        return response()->json($result);
    }
}

routes/api.php

Route::post('/graphql', 'app\GraphQL\GraphQLController@handle');

文章来源

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,271评论 5 476
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,275评论 2 380
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,151评论 0 336
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,550评论 1 273
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,553评论 5 365
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,559评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,924评论 3 395
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,580评论 0 257
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,826评论 1 297
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,578评论 2 320
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,661评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,363评论 4 318
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,940评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,926评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,156评论 1 259
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 42,872评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,391评论 2 342

推荐阅读更多精彩内容