NuxtJS Web 开发实用指南(八)

原文:zh.annas-archive.org/md5/95454EEF6B1A13DFE0FAD028BE716A19

译者:飞龙

协议:CC BY-NC-SA 4.0

为 Nuxt 创建一个与框架无关的 PHP API

在之前的章节中,比如第八章《添加服务器端框架》和第九章《添加服务器端数据库》,你学习了如何使用 Nuxt 的默认服务器与 Node.js JavaScript 框架(如 Koa 和 Express)创建 API。在第十二章《创建用户登录和 API 身份验证》中,你学习了如何使用相同的 Node.js JavaScript 框架 Koa 在外部服务器上创建 API。

在这一章中,我们将指导你如何使用 PHP(超文本预处理器)在外部服务器上创建 API。在第九章《添加服务器端数据库》中,你还学习了如何使用 MongoDB 来管理数据库。然而,在这一章中,我们将使用 MySQL,而你在第十二章《创建用户登录和 API 身份验证》中使用了 Koa。

在这一章中,最重要的是,你将学习关于 PHP 标准和 PHP 标准建议(PSRs)的所有知识。特别是,你将学习关于 PSR-4 用于自动加载,PSR-7 用于 HTTP 消息,以及 PSR-15 用于组合中间件组件和处理 HTTP 服务器请求。我们将整合来自不同供应商(如 Zend Framework 和 The PHP League)基于这些 PSR 标准的包,为我们的 Nuxt 应用创建一个与框架无关的 PHP RESTful API。

在这一章中,我们将涵盖以下主题:

  • 介绍 PHP

  • 理解 HTTP 消息和 PHP 标准

  • 使用 PHP 数据库框架编写 CRUD 操作

  • 与 Nuxt 集成

让我们开始吧!

第十六章:介绍 PHP

PHP 已经走过了很长的路。它早在 Node.js 之前就存在了,由 Rasmus Lerdorf 于 1994 年创建。最初它代表的是“个人主页”。PHP 的参考实现现在由 PHP 组织(https://www.php.net/)生产。PHP 最初是作为一个模板语言开发的,允许我们将 HTML 与 PHP 代码本身混合在一起,就像 Twig(https://twig.symfony.com/)和 Pug(https://pugjs.org/)现在所做的那样。

现在,PHP 不仅仅是一个模板语言。多年来,它已经发展成为一种通用脚本语言和面向对象语言,特别适用于服务器端 Web 开发。您仍然可以用它来制作模板,但在现代 PHP 开发中,我们应该充分利用它的全部功能。如果您想了解 PHP 还能做什么,请访问www.php.net/manual/en/intro-whatcando.php

在撰写本书时,PHP 的当前稳定版本是 7.4.x。如果您刚开始使用 PHP,请从 PHP 7.4 开始。如果您正在使用 PHP 7.2 或 7.3,您应该考虑将其升级到 PHP 7.4,因为它包含了几个错误修复。有关此版本更改的更多信息,请访问www.php.net/ChangeLog-7.php

在本书中,我们将指导您如何在支持 Apache2 的 Ubuntu 上安装或升级到 PHP 7.4。让我们开始吧!

安装或升级 PHP

如果您使用的是 macOS,请使用此指南:phptherightway.com/mac_setup。如果您使用的是 Windows,请使用此指南:phptherightway.com/windows_setup

我们正在使用 Apache2 HTTP 服务器,但如果您的计算机上已安装了 Nginx HTTP 服务器,也可以使用它。现在,按照以下简单步骤安装 PHP:

  1. 运行以下命令更新 Ubuntu 服务器上的本地软件包并安装 Apache2:
$ sudo apt update
$ sudo apt install apache2
  1. 安装 Apache2 后,使用-v选项进行验证:
$ apache2 -v
Server version: Apache/2.4.41 (Ubuntu)
Server built: 2019-08-14T14:36:32

您可以使用以下命令停止、启动和启用 Apache2 服务,以便在服务器启动时始终启动:

$ sudo systemctl stop apache2
$ sudo systemctl start apache2
$ sudo systemctl enable apache2

您可以使用以下命令检查 Apache2 的状态:

$ sudo systemctl status apache2

您应该始终在终端中获得active (running)的输出:

apache2.service - The Apache HTTP Server
 Loaded: loaded (/lib/systemd/system/apache2.service; enabled; vendor preset: enabled)
 Active: active (running) since Thu 2020-08-06 13:17:25 CEST; 52min ago
 //...
  1. 运行以下命令安装 PHP 7.4:
$ sudo apt update
$ sudo apt install php
  1. 您还应该安装与 PHP 7.4 相关的可能在开发 PHP 应用程序时需要的模块和扩展:
$ sudo apt install -y php7.4-{bcmath,bz2,curl,gd,intl,json,mbstring,xml,zip,mysql}
  1. 禁用 PHP 7.3(如果您使用的是 PHP 7.3),然后启用 PHP 7.4:
$ sudo a2dismod php7.3
$ sudo a2enmod php7.4

如果您是第一次安装 PHP,则无需禁用旧版本。如果您想卸载 PHP 及其所有相关模块,可以使用以下命令:

$ sudo apt-get purge 'php*'
  1. 重新启动 Apache2 和 PHP 服务:
$ sudo service apache2 restart
  1. 现在,您可以使用以下命令验证刚刚安装的 PHP:
$ php -v

您应该获得以下版本信息:

PHP 7.4.8 (cli) (built: Jul 13 2020 16:46:22) ( NTS )
Copyright (c) The PHP Group
Zend Engine v3.4.0, Copyright (c) Zend Technologies
 with Zend OPcache v7.4.8, Copyright (c), by Zend Technologies

现在您已经安装了 Apache2 和 PHP 7.4,接下来应该做的是配置 PHP。我们将在下一节中进行。

配置 PHP 7.4

现在 Apache2 和 PHP 已安装,您可能希望配置 PHP,以便根据您的 PHP 应用程序的需要使用它。默认的 PHP 配置文件位于/etc/php/7.4/apache2/php.ini,因此请按照以下步骤配置您的 PHP 7.4 版本:

  1. 运行以下命令以编辑或配置 PHP 7.4:
$ sudo nano /etc/php/7.4/apache2/php.ini

您可能需要更改已上传文件的upload_max_filesize的默认允许量:

upload_max_filesize = 2M

您可以在php.net/upload-max-filesize找到有关此配置的更多信息。

对于 PHP 应用程序来说,上传文件的最大限制为 2 MB 可能被认为是很小的。因此,请根据您的需求进行更改,如下所示:

upload_max_filesize = 32M

以下是一些其他重要的行/PHP 指令需要考虑:

post_max_size = 48M
memory_limit = 256M
max_execution_time = 600

您可以在www.php.net/manual/en/ini.core.php找到有关上述 PHP 指令和其他配置 PHP 的指令的更多信息。

  1. 重启 Apache 以使上述修改的 PHP 设置生效:
$ sudo service apache2 restart

PHP 7.4 非常强大。如果您不想在本地开发机器上安装 Apache,您可以只安装它并用于开发站点。在下一节中,您将学习如何在没有 Apache 服务器的情况下使用 PHP 7.4。

使用内置的 PHP Web 服务器运行 PHP 应用程序

自 PHP 5.4 以来,您可以使用内置的 PHP Web 服务器运行 PHP 脚本和应用程序,而无需像 Apache 或 Nginx 这样的常见 Web 服务器。只要安装了 PHP 7.4,您就可以跳过上述 Apache 安装。要启动 PHP 服务器,只需从项目的根目录打开终端并运行以下命令:

$ php -S 0.0.0.0:8181

如果您想从特定的文档根目录开始应用程序,例如从名为public的项目目录中的www目录开始,请执行以下操作:

$ cd ~/www
$ php -S localhost:8181 -t public

让我们创建一个经典的“Hello World”示例,这个内置的 PHP Web 服务器将提供,以查看是否一切设置正确:

  1. 创建一个简单的 PHP 文件中的“Hello World”消息页面,如下所示:
// public/index.php
<?php
echo 'Hello world!';
  1. 转到您的项目目录,并使用上述命令启动内置的 PHP Web 服务器。终端应显示以下信息:
[Sun Mar 22 09:12:37 2020] PHP 7.4.4 Development Server (http://localhost:8181) started
  1. 现在,在浏览器上加载localhost:8181。您应该在屏幕上看到 Hello world!,没有任何错误。

如果您想了解这个内置的 Web 服务器,请访问www.php.net/features.commandline.webserver

接下来,您将学习如何使用一些 PHP 标准。您还将了解 HTTP 消息是什么,以及为什么我们需要为现代 PHP 应用程序使用 PSR。

理解 HTTP 消息和 PSR

超文本传输协议HTTP)是客户端计算机和 Web 服务器之间的通信协议。诸如 Chrome、Safari 或 Firefox 之类的网络浏览器可以是 Web 客户端或用户代理,而计算机上监听某个端口的 Web 应用程序可以是 Web 服务器。Web 客户端不仅仅是浏览器,还包括任何可以与 Web 服务器通信的应用程序,比如 cURL 或 Telnet。

客户端通过互联网打开连接,向服务器发出请求,并等待直到收到服务器的响应。请求包含请求信息,而响应包含状态信息和请求的内容。这两种交换的数据称为 HTTP 消息。它们只是用 ASCII 编码的文本体,并且跨越多行,具有以下结构:

Start-line
HTTP Headers

Body

这看起来非常简单和直接,不是吗?尽管可能是这样,让我们详细说明一下这个结构:

  • Start-line描述了实现的请求方法(例如GETPUTPOST)、请求目标(通常是 URI)和响应的 HTTP 版本或状态(例如 200、404 或 500)以及 HTTP 版本。Start-line始终是单行。

  • HTTP Headers行描述了请求或响应的特定细节(元信息),例如HostUser-AgentServerContent-type等。

  • 空白行表示请求的所有元信息已经发送。

  • Body(或消息体)包含请求的交换数据(例如 HTML 表单的内容)或响应的内容(例如 HTML 文档的内容)。消息体是可选的(有时,在请求中不需要它来请求服务器的数据)。

现在,让我们使用 cURL 来看看 HTTP 请求和响应的数据是如何交换的:

  1. 使用内置的 PHP Web 服务器在localhost:8181上提供您在上一节中学到的 PHP“Hello World”应用程序:
$ php -S localhost:8181 -t public
  1. 在您的终端上打开一个新标签,并运行以下 cURL 脚本:
$ curl http://0.0.0.0:8181 \
 --trace-ascii \
 /dev/stdout

您应该看到请求消息显示在第一部分中,如下所示:

== Info: Trying 0.0.0.0:8181...
== Info: TCP_NODELAY set
== Info: Connected to 0.0.0.0 (127.0.0.1) port 8181 (0)
=> Send header, 76 bytes (0x4c)
0000: GET / HTTP/1.1
0010: Host: 0.0.0.0:8181
0024: User-Agent: curl/7.65.3
003d: Accept: /
004a:

在这里,您可以看到空行表示为004a:,请求中根本没有消息正文。响应消息显示在第二部分中,如下所示:

== Info: Mark bundle as not supporting multiuse
<= Recv header, 17 bytes (0x11)
0000: HTTP/1.1 200 OK
<= Recv header, 20 bytes (0x14)
0000: Host: 0.0.0.0:8181
<= Recv header, 37 bytes (0x25)
0000: Date: Sat, 21 Mar 2020 20:33:09 GMT
<= Recv header, 19 bytes (0x13)
0000: Connection: close
<= Recv header, 25 bytes (0x19)
0000: X-Powered-By: PHP/7.4.4
<= Recv header, 40 bytes (0x28)
0000: Content-type: text/html; charset=UTF-8
<= Recv header, 2 bytes (0x2)
0000:
<= Recv data, 12 bytes (0xc)
0000: Hello world!
== Info: Closing connection 0

在响应的起始行中,您可以看到状态是200 OK。但在前面的示例中,我们没有发送任何数据,因此请求消息中没有消息正文。让我们创建另一个非常基本的 PHP 脚本,如下所示:

  1. 创建一个带有 PHP print函数的 PHP 页面,以便显示POST数据,如下所示:
// public/index.php
<?php
print_r($_POST);
  1. 使用内置的 PHP Web 服务器在localhost:8181上提供页面:
$ php -S localhost:8181 -t public
  1. 在终端上使用 cURL 发送一些数据:
$ curl http://0.0.0.0:8181 \
 -d "param1=value1&param2=value2" \
 --trace-ascii \
 /dev/stdout

这次,请求消息将显示在第一部分中,以及消息正文:

== Info: Trying 0.0.0.0:8181...
== Info: TCP_NODELAY set
== Info: Connected to 0.0.0.0 (127.0.0.1) port 8181 (0)
=> Send header, 146 bytes (0x92)
0000: POST / HTTP/1.1
0011: Host: 0.0.0.0:8181
0025: User-Agent: curl/7.65.3
003e: Accept: /
004b: Content-Length: 27
005f: Content-Type: application/x-www-form-urlencoded
0090:
=> Send data, 27 bytes (0x1b)
0000: param1=value1&param2=value2
== Info: upload completely sent off: 27 out of 27 bytes

响应消息显示在第二部分中,如下所示:

== Info: Mark bundle as not supporting multiuse
<= Recv header, 17 bytes (0x11)
0000: HTTP/1.1 200 OK
<= Recv header, 20 bytes (0x14)
0000: Host: 0.0.0.0:8181
<= Recv header, 37 bytes (0x25)
0000: Date: Sat, 21 Mar 2020 20:43:06 GMT
<= Recv header, 19 bytes (0x13)
0000: Connection: close
<= Recv header, 25 bytes (0x19)
0000: X-Powered-By: PHP/7.4.4
<= Recv header, 40 bytes (0x28)
0000: Content-type: text/html; charset=UTF-8
<= Recv header, 2 bytes (0x2)
0000:
<= Recv data, 56 bytes (0x38)
0000: Array.(. [param1] => value1\. [param2] => value2.).
Array
(
    [param1] => value1
    [param2] => value2
)
== Info: Closing connection 0
  1. 在这里,您还可以在终端上使用 cURL 查看PUT方法的请求消息和请求消息:
$ curl -X PUT http://0.0.0.0:8181 \
 -d "param1=value1&param2=value2" \
 --trace-ascii \
 /dev/stdout
  1. 对于 cURL 上的DELETE方法也是如此,如下所示:
$ curl -X DELETE http://0.0.0.0:8181 \
 -d "param1=value1&param2=value2" \
 --trace-ascii \
 /dev/stdout
  1. 最后但同样重要的是,我们还可以使用 Google Chrome 中的开发者工具来帮助我们检查交换的数据。让我们创建另一个简单的 PHP 脚本,它将从 URI 接收数据:
// public/index.php
<?php
print_r($_GET);
  1. 通过使用0.0.0.0:8181/?param1=value1&param2=value2在浏览器上发送一些数据。通过这样做,数据将作为param1=value1&param2=value2发送,如下截图所示:

外链图片转存失败,源站可能有防盗链机制,建议将图片保存下来直接上传

如果您想了解更多关于 HTTP 和 HTTP 消息的信息,请访问developer.mozilla.org/en-US/docs/Web/HTTP了解 HTTP 的一般信息,以及developer.mozilla.org/en-US/docs/Web/HTTP/Messages了解特定的 HTTP 消息。

在涉及服务器端开发时,HTTP 消息最好封装在对象中,以便更容易处理。例如,Node.js 具有内置的 HTTP 模块(nodejs.dev/the-nodejs-http-module)用于 HTTP 通信,在其中,您可以从http.createServer()方法的回调中获取 HTTP 消息对象,用于创建 HTTP 服务器:

const http = require('http')

http.createServer((request, response) => {
  response.writeHead(200, {'Content-Type': 'text/plain'})
  response.end('Hello World')
}).listen(8080)

如果您正在使用 Node.js 框架,例如 Koa,您可以在ctx中找到 HTTP 消息对象,如下所示:

const Koa = require('koa')
const app = new Koa()

app.use(async ctx => {
  ctx
  ctx.request
  ctx.response
})

在前面的代码中,ctx是 Koa 上下文,而ctx.request是 HTTP 请求消息,ctx.response是 HTTP 响应消息。在 Express 中也是一样的;您可以按如下方式找到 HTTP 消息:

const express = require('express')
const app = express()

app.get('/', (req, res) => res.send('Hello World!'))

与 Node.js 不同,PHP 从来没有内置的 HTTP 消息对象。有很多方法可以手动和直接获取和设置 Web 数据,就像我们在之前的 PHP 示例中看到的那样,通过使用超全局变量($_GET$_POST)和内置函数(echoprint_r)。如果您想捕获传入的请求,您可以根据情况使用$_GET$_POST$_FILE$_COOKIE$_SESSION或其他超全局变量(www.php.net/manual/en/language.variables.superglobals.php)。

返回响应也是一样的:您可以使用全局函数,如echoprintheader手动设置响应头。过去,PHP 开发人员和框架有他们自己实现 HTTP 消息的方式。这导致了不同框架有不同的抽象来表示HTTP 消息的时代,任何基于特定实现的 HTTP 消息的应用程序几乎无法在项目中与其他框架一起使用。这种行业标准的缺乏使得框架的组件紧密耦合。如果您没有从框架开始,最终您会自己构建一个框架。

但是今天,PHP 社区已经学习并执行了 PHP 标准和建议。您不必完全遵守这些标准和建议;如果您有哲学原因要求您这样做,您可以忽略它们。但它们是一个良好意图的衡量标准,至少在商业和合作方面结束 PHP 战争。最重要的是,PHP 开发人员可以专注于 PHP 标准,而不是以框架无关的方式。当我们谈论 PHP 标准时,我们倾向于指的是 PSR,这是由 PHP 框架互操作性组(PHP-FIG)定义和发布的 PHP 规范。PSR-7:HTTP 消息接口是 PHP-FIG 成员提出的规范之一,并根据他们同意的既定协议进行了投票。

PSR-7 于 2015 年 5 月正式被接受。它基本上用于标准化 HTTP 消息接口。在深入研究 PSR-7 之前,我们还应该了解一些其他 PSR 编号,特别是 PSR-12(取代 PSR-2),PSR-4 和 PSR-15。我们将在本书中为您介绍它们,以便您可以编写可重用的,与框架无关的应用程序和组件,这些应用程序和组件可以独立使用,也可以与其他框架进行互操作,无论它们是全栈还是微框架。让我们开始吧!

为什么要使用 PSR?

在内部,PHP 从不告诉开发人员他们应该如何编写他们的 PHP 代码。例如,Python 使用缩进来指示一块代码,而对于其他编程语言,如 PHP 和 JavaScript,代码中的缩进是为了可读性。以下是 Python 将接受的示例:

age = 20
if age == 20:
  print("age is 20")

如果没有缩进,Python 将返回错误:

if age == 20:
print("age is 20")

空格的数量取决于编码者的偏好,但您必须至少使用一个空格,并且在同一块中的其他行中使用相同数量的空格;否则,Python 将返回错误:

if age == 20:
 print("age is 20")
  print("age is 20")

另一方面,在 PHP 中,您可以编写以下内容:

if (age == 20) {
print("age is 20");
}

PHP 中也可以使用以下内容:

if (age == 20) {
 print("age is 20");
  print("age is 20");
}

Python 在内部强制执行代码的可读性和整洁性。PHP 没有。您可以想象,如果没有一些基本的强制措施,并且根据编码者的经验,PHP 代码可能会变得非常混乱,丑陋和难以阅读。也许 PHP Web 开发的低门槛在其中起了作用。因此,您的 PHP 代码必须遵循通用的代码风格,以便于协作和维护。

有一些特定框架的 PHP 编码标准,但它们基本上是基于(或类似于)PSR 标准的:

从实用的角度来看,您的代码应该遵循您所依赖的框架,以及特定的框架。但是,如果您只是从框架中使用一些组件或库,那么您可以遵守任何组合的 PSR,或者由 PEAR 制定的编码标准。PEAR 编码标准可以在pear.php.net/manual/en/standards.php找到。

本书侧重于各种 PSR,因为本章旨在创建与框架无关的 PHP 应用程序。您不必同意 PSR,但如果您正在寻找一个标准来开始项目,并且在您的组织内没有自己的标准,那么这可能是一个很好的开始。您可以在www.php-fig.org/psr/找到更多关于 PSR 的信息。

除了我们在这里提到的内容之外,您还应该查看phptherightway.com/PHP: The Right Way。它概述了现代 PHP 编码人员可以用作参考的事项,从设置 PHP,使用Composer进行依赖管理(我们将在本章后面介绍),编码风格指南(其中推荐使用 PSR),依赖注入,数据库,模板化,测试框架等等。对于想要避免过去错误并在网络上找到权威 PHP 教程链接的新 PHP 编码人员来说,这是一个很好的开始。对于需要快速参考和来自 PHP 社区的更新的经验丰富的 PHP 编码人员来说,这也是一个很好的地方,或者是他们在过去几年中可能错过的任何内容。

现在,让我们开始研究 PSR,从PSR-12开始。

PSR-12 - 扩展编码风格指南

PSR-12 是 PSR-2 的修订编码风格指南,考虑了 PHP 7。PSR-12 规范于 2019 年 8 月 9 日获得批准。自 2012 年接受 PSR-2 以来,PHP 已经进行了许多更改,对编码风格指南产生了一些影响,其中最显着的是返回类型声明,这是在 PHP 7 中引入的,而在 PSR-2 中没有描述。因此,应该定义一个标准来使用它们,以便它们可以被更广泛的 PHP 社区采用,然后再由个别的 PHP 编码人员实施他们的标准,这可能最终会发生冲突。

例如,PHP 7 中添加的返回类型声明简单地指定了函数应该返回的值的类型。让我们看一下以下采用返回类型声明的函数:

declare(strict_types = 1);

function returnInt(int $value): int
{
    return $value;
}

print(returnInt(2));

您将得到2作为整数的正确结果。但是,让我们看看如果您改变returnInt函数内的代码会发生什么,如下所示:

function returnInt(int $value): int
{
    return $value + 1.0;
}

PHP 将放弃以下错误:

PHP Fatal error: Uncaught TypeError: Return value of returnInt() must be of the type int, float returned in ...

因此,为了满足 PHP 7 的这一新功能的需求,PSR-12 要求您在冒号后使用一个空格,后面是带有返回类型声明的方法的类型声明。此外,冒号和声明必须与参数列表的右括号在同一行,两个字符之间没有空格。让我们看一个简单的例子,其中有一个return类型声明:

class Fruit
{
    public function setName(int $arg1, $arg2): string
    {
        return 'kiwi';
    }
}

在 PSR-2 和 PSR-12 中有一些规则保持不变。例如,在这两个 PSR 中,您不能使用制表符进行缩进,而是使用四个单个空格。但是在 PSR-2 中关于块列表的规则已经修订。现在,在 PSR-12 中,使用语句导入类、函数和常量的块必须用单个空行分隔,即使只有一个导入它们的地方。让我们快速看一下符合此规则的一些代码:

<?php

/**
 * The block of comments...
 */

declare(strict_types=1);

namespace VendorName\PackageName;

use VendorName\PackageName\{ClassX as X, ClassY, ClassZ as Z};
use VendorName\PackageName\SomeNamespace\ClassW as W;

use function VendorName\PackageName\{functionX, functionY, functionZ};

use const VendorName\PackageName\{ConstantX, ConstantY, ConstantZ};

/**
 * The block of comments...
 */
class Fruit
{
    //...
}

现在,您应该注意,在 PSR-12 中,您必须在开头的<?php标记后使用一个空行。但是,在 PSR-2 中,这是不必要的。例如,您可以编写以下内容:

<?php
namespace VendorName\PackageName;

use FruitClass;
use VegetableClass as Veg;

值得知道,PSR-2 是从 PSR-1 扩展而来的,它是一个基本的编码标准,但自从 PSR-12 被接受以来,PSR-2 现在已正式弃用。

要为您的代码实施这些 PSR,请访问以下网站:

如果您想了解 PHP 7 的新功能,例如标量类型声明和返回类型声明,请访问www.php.net/manual/en/migration70.new-features.php

PSR-12 帮助 PHP 编码人员编写更易读和结构化的代码,因此在使用 PHP 编写代码时值得采用它。现在,让我们继续讨论PSR-4,它允许我们在 PHP 中使用自动加载。

PSR-4 – 自动加载器

在 PHP 的旧日子里,如果您想将第三方库引入您的 PHP 项目,或者从单独的 PHP 文件中引入您的函数和类,您将使用includerequire语句。随着 PHP 自动加载的到来,您将使用__autoload魔术方法(自 PHP 7.2 起已弃用)或spl_autoload来自动调用您的代码。然后在 PHP 5.3 中出现了真正的命名空间支持,开发人员和框架可以设计他们的方法来防止命名冲突。但仍然远非理想,因为不同方法之间存在冲突。您可以想象一种情况,您有两个框架 - 框架 A 和框架 B - 以及个别开发人员彼此不同意并实施自己的方法来实现相同的结果。这是疯狂的。

今天,我们遵守 PSR-4(它是 PSR-0 的后继者)来标准化自动加载的方法,并将开发人员和框架绑定在一起。它指定了从文件路径自动加载类的标准。它还描述了文件的位置。因此,一个完全限定的类名应该遵循以下形式:

\<NamespaceName>(\<SubNamespaceNames>)\<ClassName>

在这个规则中,我们有以下内容:

  • 完全限定类的命名空间必须具有顶级供应商命名空间,这是上述代码中的<NamespaceName>部分。

  • 在上面的代码中,您可以使用一个或多个子命名空间,如<SubNamespaceNames>部分所示。

  • 然后,您必须使用您的类名结束命名空间,如上述代码中的<ClassName>部分所示。

因此,如果您正在编写自动加载程序,建议使用此标准。但是,您不必(也可能不应该)费力地编写自己的自动加载程序来符合 PSR-4。这是因为您可以使用Composer来帮助您做到这一点。Composer 是 PHP 的包管理器。它类似于 Node.js 中的 npm。它最初是在 2012 年发布的。从那时起,它已被所有现代 PHP 框架和 PHP 编码人员使用。这意味着您可以更多地专注于您的代码开发,而不必过多担心要引入项目环境中的不同包和库的互操作性。

在开始之前,请确保您的系统上已安装 Composer。根据您的系统,您可以按照以下指南安装 Composer:

当前版本为 1.10.9。按照以下步骤安装 Composer 并利用其提供的自动加载程序:

  1. 通过在终端中运行以下脚本在当前目录中安装 Composer:
$ php -r "copy('https://getcomposer.org/installer', 'composer-setup.php');"
$ php -r "if (hash_file('sha384', 'composer-setup.php') === 'e5325b19b381bfd88ce90a5ddb7823406b2a38cff6bb704b0acc289a09c8128d4a8ce2bbafcd1fcbdc38666422fe2806') { echo 'Installer verified'; } else { echo 'Installer corrupt'; unlink('composer-setup.php'); } echo PHP_EOL;"
  1. 按照以下步骤运行 Composer 设置文件:
$ sudo php composer-setup.php

您应该在终端中获得以下输出:

All settings correct for using Composer
Downloading...

Composer (version 1.10.9) successfully installed to: /home/lau/composer.phar
Use it: php composer.phar
  1. 按照以下步骤删除 Composer 设置文件:
$ php -r "unlink('composer-setup.php');"
  1. 通过在终端上运行php composer.phar来验证安装。如果您想全局使用 Composer,则将 Composer 移动到/usr/local/bin(如果您使用 Linux/Unix):
$ sudo mv composer.phar /usr/local/bin/composer
  1. 现在,您可以全局运行 Composer。要验证它,只需运行以下命令:
$ composer

您应该看到 Composer 的标志,以及其可用的命令和选项:

   ______
  / ____/___ ____ ___ ____ ____ ________ _____
 / / / __ \/ __ __ \/ __ \/ __ \/ ___/ _ \/ ___/
/ /___/ /_/ / / / / / / /_/ / /_/ (__ ) __/ /
\____/\____/_/ /_/ /_/ .___/\____/____/\___/_/
                    /_/
Composer version 1.10.9 2020-07-16 12:57:00
...
...

或者,您可以使用-V选项直接检查您安装的版本:

$ composer -V
Composer version 1.10.9 2020-07-16 12:57:00
  1. 现在,您已经在系统上安装了 Composer,只需通过终端导航到项目的根目录,并使用composer require,然后是<package-name>,来安装您项目中需要的任何第三方软件包(也称为依赖项),如下所示:
$ composer require monolog/monolog
  1. 安装所需的软件包后,您可以转到项目根目录。您应该看到已创建一个包含项目依赖项的composer.json文件,其中包含require键:
{
    "require": {
        "monolog/monolog": "².0"
    }
}
  1. 如果您想下次再次安装所有依赖项,只需运行install命令,如下所示:
$ composer install
  1. 当您安装了项目的依赖项,无论是使用require还是install命令,Composer 都会生成一个包含所有依赖项的/vendor/文件夹。autoload.php文件将始终生成并位于/vendor/文件夹内。然后,您可以包含此文件并立即开始使用这些软件包提供的类,如下所示:
require __DIR__ . '/vendor/autoload.php';

$log = new Monolog\Logger('name');
$log->pushHandler(new Monolog\Handler\StreamHandler('path/to/your.log', Monolog\Logger::WARNING));
$log->addWarning('Foo');
$log->error('Bar');
  1. 最重要的是,您甚至可以通过向composer.json文件添加autoload键以及自定义命名空间来将您的类添加到自动加载程序。例如,您可以将类存储在项目根目录中的/src/文件夹中,与/vendor/目录位于同一级别:
{
    "autoload": {
        "psr-4": {"Spectre\\": "src/"}
    }
}

如果您的源文件位于多个位置,您可以使用数组[]将其与您的自定义命名空间关联起来,如下所示:

{
    "autoload": {
        "psr-4": {
            "Spectre\\": [
                "module1/",
                "module2/"
            ]
        }
    }
}

Composer 将为Spectre命名空间注册一个 PSR-4 自动加载程序。之后,您可以开始编写您的类。例如,您可以创建一个包含Spectre\Foo类的/src/Foo.php文件。之后,只需在终端上运行dump-autoload来重新生成/vendor/目录中的autoload.php文件。您还可以添加多个自定义命名空间到autoload字段,如下所示:

{
    "autoload": {
        "psr-4": {
            "Spectre\\": [
                //...
            ],
            "AnotherNamespace\\": [
                //...
            ]
        }
    }
}

除了 PSR-4,Composer 还支持 PSR-0。您可以在composer.json文件中添加 PSR-0 键。

有关如何在 Composer 中使用 PSR-0 的更多信息和示例,请访问getcomposer.org/doc/04-schema.mdautoload。但是,请注意,PSR-0 现在已经不推荐使用。如果您想阅读有关这两个 PSR 的更多信息,请访问www.php-fig.org/psr/psr-0/了解 PSR 0(不推荐使用),以及www.php-fig.org/psr/psr-4/了解 PSR-4。如果您想了解我们在 PHP 中用于记录的 Monolog,请访问github.com/Seldaek/monolog。如果您想了解 PHP 中自动加载类的更多信息,请访问www.php.net/manual/en/language.oop5.autoload.php

一旦您掌握了关于 PSR-12 和 PSR-4 的知识,您将更容易构建符合其他 PSR 的 PHP 应用程序。本书关注的另外两个 PSR 是 PSR-7 和 PSR-15。让我们先来看一下PSR-7

PSR-7 - HTTP 消息接口

早些时候,我们提到 PHP 没有 HTTP 请求和响应消息对象,这就是为什么 PHP 框架和编码人员在过去提出了不同的抽象来表示(或“模拟”)HTTP 消息。幸运的是,在 2015 年,PSR-7 出现了,结束了这些“分歧”和差异。

PSR-7 是一组通用接口(抽象),用于在 HTTP 通信中指定 HTTP 消息和 URI 的公共方法。在面向对象编程(OOP)中,接口实际上是对象(类)必须实现的操作(公共方法)的抽象,而不定义这些操作的复杂性和细节。例如,以下表格显示了当组合 HTTP 消息类时,必须实现的方法,以便符合 PSR-7 规范。

用于访问和修改请求和响应对象的指定方法如下:

访问修改
getProtocolVersion()withProtocolVersion($version)
getHeaders()withHeader($name, $value)
hasHeader($name)withAddedHeader($name, $value)
getHeader($name)``getHeaderLine($name)withoutHeader($name)
getBody()withBody(StreamInterface $body)

用于访问和修改请求对象的指定方法如下:

访问修改

|

  • getRequestTarget()

  • getMethod()

  • getUri()

  • getServerParams()

  • getCookieParams()

  • getQueryParams()

  • getUploadedFiles()

  • 获取解析后的主体getParsedBody()

  • 获取属性getAttributes()

  • getAttribute($name, $default = null)

|

  • 使用withMethod($method)方法

  • withRequestTarget($requestTarget)

  • 使用withUri(UriInterface $uri, $preserveHost = false)方法

  • withCookieParams(array $cookies)

  • withQueryParams(array $query)

  • withUploadedFiles(array $uploadedFiles)

  • withParsedBody($data)

  • withAttribute($name, $value)

  • withoutAttribute($name)

|

用于访问和修改响应对象的指定方法如下:

访问修改

|

  • 获取状态码getStatusCode()

  • getReasonPhrase()

|

  • withStatus($code, $reasonPhrase = '')

|

自从 2015 年 5 月 18 日接受了 PSR-7 以来,许多基于它制作的软件包已经问世。只要实现了 PSR-7 中指定的接口和方法,就可以开发自己的版本。然而,除非你有充分的理由这样做,否则你可能会“重复造轮子”,因为已经有了 PSR-7 HTTP 消息软件包。因此,为了快速开始,让我们使用 Zend Framework 中的zend-diactoros。我们将“重用”你在前几节中学到的 PSR 知识(PSR-12 和 PSR-4)来创建一个简单的基于 HTTP 消息的“Hello World”服务器端应用程序。让我们开始吧:

  1. 在应用程序根目录中创建一个/public/目录,并在其中添加一个index.php文件。将以下行添加到其中以引导应用程序环境:
// public/index.php
chdir(dirname(__DIR__));
require_once 'vendor/autoload.php';

在这两行代码中,我们已将当前目录从/path/to/public更改为/path/to,以便我们可以通过写vendor/autoload.php而不是../vendor/autoload.php来导入autoload.php文件。

__DIR__(魔术)常量用于获取当前文件的目录路径,即index.php,位于/path/to/public/目录中。然后使用dirname函数获取父目录的路径,即/path/to。然后使用chdir函数来改变当前目录。

请注意,在接下来关于 PSR 的章节中,我们将使用这种模式来引导应用程序环境并导入自动加载文件。请访问以下链接以了解更多关于之前提到的常量和函数:

还要注意,您必须通过使用内置的 PHP Web 服务器在终端上运行所有传入的 PHP 应用程序,如下所示:

**$ php -S localhost:8181 -t public** 
  1. 通过 Composer 将zend-diactoros安装到应用程序的根目录:
$ composer require zendframework/zend-diactoros
  1. 要整理传入的请求,您应该在/public/目录中的index.php文件中创建一个请求对象,如下所示:
$request = Zend\Diactoros\ServerRequestFactory::fromGlobals(
    $_SERVER,
    $_GET,
    $_POST,
    $_COOKIE,
    $_FILES
);
  1. 现在,我们可以创建一个响应对象并对响应进行操作,如下所示:
$response = new Zend\Diactoros\Response();
$response->getBody()->write("Hello ");
  1. 请注意write方法在流接口(StreamInterface)中指定,我们也可以通过多次调用此方法来追加更多数据:
$response->getBody()->write("World!");
  1. 如果需要,我们可以操作标头。
$response = $response
    ->withHeader('Content-Type', 'text/plain');
  1. 请注意,头部应在数据写入主体后添加。然后,您已经成功将您在本章开头学到的简单 PHP“Hello World”应用程序转换为具有 PSR-7 的现代 PHP 应用程序!但是,如果您在终端中使用php -S localhost:8181 -t public运行此 PSR-7“Hello World”应用程序在浏览器上,您将看不到任何内容。这是因为我们没有使用PSR-15 HTTP 服务器请求处理程序PSR-7 HTTP 响应发射器将响应发送到浏览器,我们将在下一节中介绍。如果您现在想看到输出,可以使用getBody方法访问数据,然后使用echo
echo $response->getBody();
  1. 如果您通过 Chrome 的开发者工具检查页面的Content-type,您将得到text/html而不是我们用withHeader方法修改的text/plain。我们将在下一章中使用发射器获得正确的内容类型。

有关zend-diactoros及其高级用法的更多信息,请访问docs.zendframework.com/zend-diactoros/。除了 Zend Framework 的zend-diactoros之外,您还可以使用其他框架和库的 HTTP 消息包:

您应该查看www.php-fig.org/psr/psr-7/上的 PSR-7 文档,以获取有关此 PSR 的更多信息。如果您是 PHP 接口的新手,请访问www.php.net/manual/en/language.oop5.interfaces.php进行进一步阅读。

从 PSR-7 文档中,您可以找到本书中未提及的其他公共方法。它们应该在任何 PSR-7 HTTP 消息包中都可以找到,比如zend-diactoros。了解这些方法很有用,这样您就知道可以用它们做什么。您还可以在运行时使用内置的 PHP get_class_methods 方法列出您可以在请求和响应对象中使用的所有方法。例如,对于request对象,您可以执行以下操作:

$request = Zend\Diactoros\ServerRequestFactory::fromGlobals(
    //...
);
print_r(get_class_methods($request));

您将获得一个可以调用的请求方法列表的数组。对于response对象也是一样;通过这样做,您将获得一个响应方法列表的数组:

$response = new Zend\Diactoros\Response();
print_r(get_class_methods($response));

现在,让我们继续并查看PSR-15,在那里我们将了解如何向客户端(浏览器)发出响应。

PSR-15 - HTTP 服务器请求处理程序(请求处理程序)

PSR-7 是 PHP 社区的一个重要进步,但它只是实现目标的一半,可能使 PHP 编码人员摆脱庞大的 MVC 框架,并允许他们从一系列可重用的中间件中组合出不可知的 PHP 应用。它只定义了 HTTP 消息(请求和响应);它从未定义如何处理它们。因此,我们需要一个请求处理程序来处理请求以产生响应。

与 PSR-7 一样,PSR-15 是一组常见接口,但它们更进一步,并指定了请求处理程序(HTTP 服务器请求处理程序)和中间件(HTTP 服务器请求中间件)的标准。它于 2018 年 1 月 22 日被接受。我们将在下一节中介绍 HTTP 服务器请求中间件。现在,让我们了解 PSR-15 接口中的 HTTP 服务器请求处理程序RequestHandlerInterface

// Psr\Http\Server\RequestHandlerInterface

namespace Psr\Http\Server;

use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;

interface RequestHandlerInterface
{
    public function handle(ServerRequestInterface $request) : 
     ResponseInterface;
}

正如您所看到的,这是一个非常简单的接口。它只有一个指定的公共方法handle,它只接受一个 PSR-7 HTTP 请求消息,并且必须返回一个 PSR-7 HTTP 响应消息。我们将使用 Zend Framework 的zend-httphandlerrunner组件来实现这个接口,以提供我们可以用来发出 PSR-7 响应的实用工具。让我们将其连接到应用程序:

  1. 通过 Composer 安装zend-httphandlerrunner
$ composer require zendframework/zend-httphandlerrunner
  1. 一旦我们在项目环境中安装好了,我们可以将之前创建的响应发送到浏览器,如下所示:
//...
$response = $response
    ->withHeader('Content-Type', 'text/plain');

(new Zend\HttpHandlerRunner\Emitter\SapiEmitter)->emit($response);

如果您通过 Chrome 的开发者工具再次检查页面的Content-Type,您将获得正确的内容类型,即text/plain

有关zend-httphandlerrunner的更多信息,请访问docs.zendframework.com/zend-httphandlerrunner/。有关 PSR-15 的更多信息,请访问www.php-fig.org/psr/psr-15/

除了zend-httphandlerrunner,您还可以使用 Narrowspark 的 Http Response Emitter github.com/narrowspark/http-emitter 来处理请求并发出响应。现在,让我们继续看一下 PSR-15 的第二个接口MiddlewareInterface

PSR-15 - HTTP 服务器请求处理程序(中间件)

PSR-15 中的中间件接口具有以下抽象:

// Psr\Http\Server\MiddlewareInterface

namespace Psr\Http\Server;

use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\RequestHandlerInterface;

interface MiddlewareInterface
{
    public function process(
        ServerRequestInterface $request,
        RequestHandlerInterface $handler
    ) : ResponseInterface;
}

再次,您可以看到这是一个非常简单的接口。它只有一个指定的公共方法process用于中间件实现。实现这个接口的组件(中间件)将只接受一个 PSR-7 HTTP 请求消息和一个 PSR-15 HTTP 服务器请求处理程序,然后必须返回一个 PSR-7 HTTP 响应消息。

我们将使用 Zend Framework 的zend-stratigility组件来实现这个接口,以便我们可以在我们的应用程序中创建 PSR-15 中间件。让我们学习如何将其连接到应用程序:

  1. 通过 Composer 安装zend-stratigility
$ composer require zendframework/zend-stratigility
  1. 一旦我们在项目环境中安装了它,我们将导入middleware函数和MiddlewarePipe类,如下所示:
use function Zend\Stratigility\middleware;

$app = new Zend\Stratigility\MiddlewarePipe();

// Create a request
$request = Zend\Diactoros\ServerRequestFactory::fromGlobals(
    //...
);
  1. 然后,我们可以使用这个middleware函数创建三个中间件并将它们附加到管道中,如下所示:
$app->pipe(middleware(function ($request, $handler) {
    $response = $handler->handle($request);
    return $response
        ->withHeader('Content-Type', 'text/plain');
}));

$app->pipe(middleware(function ($request, $handler) {
    $response = $handler->handle($request);
    $response->getBody()->write("User Agent: " . 
     $request->getHeader('user-agent')[0]);
    return $response;
}));

$app->pipe(middleware(function ($request, $handler) {
    $response = new Zend\Diactoros\Response();
    $response->getBody()->write("Hello world!\n");
    $response->getBody()->write("Request method: " .
     $request->getMethod() . "\n");
    return $response;
}));
  1. 正如您所看到的,“Hello World”代码块我们之前创建的现在是一个与其他中间件堆叠在一起的中间件。最后,我们可以从这些中间件生成一个最终响应并将其发出到浏览器,如下所示:
$response = $app->handle($request);
(new Zend\HttpHandlerRunner\Emitter\SapiEmitter)->
  emit($response);

您应该在0.0.0.0:8181的浏览器上获得类似以下的结果:

Hello world!
Request method: GET
User Agent: Mozilla/5.0 (X11; Linux x86_64) AppleWebKit/537.36 
 (KHTML, like Gecko) Chrome/77.0.3865.90 Safari/537.36

有关zend-stratigility的更多信息,请访问docs.zendframework.com/zend-stratigility/

除了zend-stratigility,您还可以使用以下软件包来创建您的中间件:

所以,就是这样。借助几个可互操作的组件,我们已经启动了一个符合 PSR-12、PSR-7 和 PSR-15 的现代 PHP 应用程序,这意味着您可以自由地从广泛的供应商实现中(不可知论)选择用于 HTTP 消息、请求处理程序和中间件的标准。但我们还没有完成。您可能已经注意到,我们创建的应用程序只是一个在0.0.0.0:8181上运行的单页面应用程序。它没有其他路由,如/about/contact等。因此,我们需要一个实现 PSR-15 的路由器。我们将在下一节中介绍这个。

PSR-7/PSR-15 路由器

我们将使用来自 The League of Extraordinary Packages(一个 PHP 开发者组)的 Route,以便我们拥有一个 PSR-7 路由系统,并在其上调度我们的 PSR-15 中间件。简而言之,Route 是一个快速的 PSR-7 路由/调度程序包。

它是一个 PSR-15 服务器请求处理程序,可以处理一系列中间件的调用。它是建立在 Nikita Popov 的 FastRoute (github.com/nikic/FastRoute)之上。

让我们学习如何将其连接到应用程序:

  1. 通过 Composer 安装league/route
$ composer require league/route
  1. 安装后,我们可以按如下方式重构我们的“Hello World”组件以使用路由:
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;

$request = Zend\Diactoros\ServerRequestFactory::fromGlobals(
    //...
);

$router = new League\Route\Router;

$router->map('GET', '/', function (ServerRequestInterface $request) : ResponseInterface {
    $response = new Zend\Diactoros\Response;
    $response->getBody()->write('<h1>Hello, World!</h1>');
    return $response;
});
  1. 然后,我们只需使用 Route 的dispatch方法创建一个 PSR-7 HTTP 响应,并将其发送到浏览器:
$response = $router->dispatch($request);
(new Zend\HttpHandlerRunner\Emitter\SapiEmitter)->emit($response);

查看您可以在route.thephpleague.com/4.x/route使用的 HTTP 请求方法列表(getpostputdelete等)。此外,我们可以将中间件附加到我们的应用程序。

  1. 如果您想锁定整个应用程序,可以将中间件添加到路由器,如下所示:
use function Zend\Stratigility\middleware;

$router = new League\Route\Router;
$router->middleware(<middleware>);
  1. 如果您想锁定一组路由,可以将中间件添加到该组,如下所示:
$router
    ->group('/private', function ($router) {
        // ... add routes
    })
    ->middleware(<middleware>)
;
  1. 如果您想锁定特定路由,可以将中间件添加到该路由,如下所示:
$router
    ->map('GET', '/secret', <SomeController>)
    ->middleware(<middleware>)
;
  1. 例如,您可以使用zend-stratigility与 Route:
use function Zend\Stratigility\middleware;

$router = new League\Route\Router;
$router->middleware(middleware(function ($request, $handler) {
    //...
}));
  1. 如果您不想使用middleware函数,或者根本不想使用zend-stratigility,您可以创建匿名中间件,如下所示:
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;
use Psr\Http\Server\MiddlewareInterface;
use Psr\Http\Server\RequestHandlerInterface;

$router = new League\Route\Router;

$router->middleware(new class implements MiddlewareInterface {
    public function process(ServerRequestInterface $request, 
    RequestHandlerInterface $handler) : ResponseInterface
    {
        $response = $handler->handle($request);
        return $response->withHeader('X-Clacks-Overhead', 
        'GNU Terry Pratchett');
    }
});

只要您遵守 PSR7 和 PSR-15,通过在中间件中实现process方法,就无需zend-stratigility。如果您想在单独的 PHP 文件中创建基于类的中间件,请查看提供的示例route.thephpleague.com/4.x/middleware/

有关 The League of Extraordinary Packages 的 Route 的更多信息,请访问route.thephpleague.com/。您还可以查看由这个开发人员组创建的其他软件包,网址为thephpleague.com/。除了 The League of Extraordinary 的 Route 之外,您还可以使用以下基于 PSR-7 和 PSR-15 的 HTTP 路由器软件包:

您可能需要一个分发器来与其中一些软件包一起使用。使用 The League of Extraordinary Packages 的 Route 的优势在于它提供了一个路由器和一个分发器在一个软件包中。

有了这个,我们通过使用 PSR-12、PSR-4、PSR-7 和 PSR-15,编写了一个不可知的 PHP 应用程序。但是我们的 PHP API 还没有完成。还有一项任务要做——我们需要为 CRUD 操作添加一个数据库框架。我们将在下一节中指导您完成这项任务。

使用 PHP 数据库框架编写 CRUD 操作

正如您可能还记得的那样第九章,添加服务器端数据库CRUD代表create,read,update 和delete。在那一章中,我们使用 MongoDB 来创建 CRUD 操作。在本节中,我们将使用 MySQL 来创建后端身份验证。我们将在刚刚使用 PSRs 创建的 PHP 应用程序中使用 MySQL 和 PHP。因此,让我们首先创建我们在 MySQL 数据库中需要的表。

创建 MySQL 表

确保您已在本地计算机上安装了 MySQL 服务器并创建了一个名为nuxt-php的数据库。完成这些操作后,请按照以下步骤完成我们 API 的第一部分:

  1. 在数据库中插入以下 SQL 查询以创建表:
CREATE TABLE user (
  uuid varchar(255) NOT NULL,
  name varchar(255) NOT NULL,
  slug varchar(255) NOT NULL,
  created_on int(10) unsigned NOT NULL,
  updated_on int(10) unsigned NOT NULL,
  UNIQUE KEY slug (slug)
) ENGINE=MyISAM DEFAULT CHARSET=utf8;

您可能已经注意到的第一件事是,我们使用的是uuid而不是像我们在第十二章中所做的id。UUID 代表通用唯一标识符。可能有一些原因和好处会让您选择 UUID 而不是自动递增键来索引数据库表中的记录。例如,您可以在不连接到数据库的情况下创建 UUID。它在应用程序中几乎是唯一的,因此您可以轻松地从不同的数据库中组合数据而永远不会发生冲突。为了在 PHP 应用程序中生成 UUID,我们可以使用 Ben Ramsey 的ramsey/uuid来帮助我们生成 RFC 4122(tools.ietf.org/html/rfc4122)版本 1、3、4 和 5 的 UUID。

  1. 所以,让我们通过 Composer 安装ramsey/uuid
$ composer require ramsey/uuid
  1. 现在,您可以使用这个包来生成 UUID 的第一个版本,如下所示:
use Ramsey\Uuid\Uuid;

$uuid1 = Uuid::uuid1();
echo $uuid1->toString();

如果您想了解更多关于这个包的信息,请访问github.com/ramsey/uuid

现在,让我们学习如何使用 PHP 来处理 MySQL 数据库,并了解为什么我们需要一个数据库框架来加快 PHP 开发速度。

使用 Medoo 作为数据库框架

在 PHP 的旧时代,开发人员使用 MySQL 函数(www.php.net/manual/en/ref.mysql.php)来管理 MySQL 数据库。然后,MySQLi 扩展(www.php.net/manual/en/book.mysqli.php)取代了现在已经弃用的 MySQL 函数。然而,现在,开发人员被鼓励使用PHP 数据对象PDO)(www.php.net/manual/en/book.pdo.php)。PDO 是一个内置的 PHP 接口抽象,就像 PSR-7 和 PSR-15 一样。它是一个数据访问抽象层,为访问和管理数据库(例如 MySQL 和 PostgreSQL)提供了一个一致的接口(统一的 API),这意味着无论你使用哪种数据库,你都可以使用相同的函数来查询和获取数据。它支持以下数据库:

|

  • CUBRID

  • MS SQL Server

  • Firebird

  • IBM

|

  • Informix

  • MySQL

  • Oracle

  • ODBC 和 DB2

|

  • PostgreSQL

  • SQLite

  • 4D

|

请注意,PDO 是一个数据访问抽象层,而不是数据库抽象层。因此,取决于您使用的数据库,必须安装该数据库的 PDO 驱动程序才能使用 PDO。我们正在使用 MySQL 数据库,因此必须确保安装了PDO_MYSQL驱动程序。在 Ubuntu 中,您可以使用以下命令来检查您是否已启用 PDO 扩展,并且PDO_MYSQL驱动程序已安装在您的环境中:

$ php -m

您应该会得到一系列 PHP 模块。查找PDOpdo_mysql

[PHP Modules]
...
PDO
pdo_mysql
...

另一个更具体的选项是检查 PDO 及其驱动程序,如下所示:

$ php -m|grep -i pdo
PDO
pdo_mysql

如果您只想搜索 PDO 驱动程序,请执行以下操作:

$ php -m|grep -i pdo_
pdo_mysql

您还可以创建一个带有phpinfo()的 PHP 页面来查找它们。或者,您可以使用getAvailableDrivers方法,如下所示:

print_r(PDO::getAvailableDrivers());

您应该会得到一系列 PDO 驱动程序,如下所示:

Array
(
    [0] => mysql
)

或者,还有一些内置的 PHP 函数可以帮助您:

extension_loaded ('PDO'); // returns boolean
extension_loaded('pdo_mysql'); // returns boolean
get_loaded_extensions(); // returns array

如果您没有看到任何 PDO 驱动程序,则必须安装 MySQL 支持的驱动程序。请按照以下步骤执行:

  1. 搜索软件包名称(Ubuntu):
$ apt-cache search php7.4|grep mysql
php7.4-mysql - MySQL module for PHP
  1. 安装php7.4-mysql并重新启动 Apache 服务器:
$ sudo apt-get install php7.4-mysql
$ sudo service apache2 restart

一旦您安装了PDO_MYSQL驱动程序,就可以立即开始编写 CRUD 操作。例如,让我们编写一个insert操作,如下所示:

  1. 创建 MySQL 数据库连接:
$servername = "localhost";
$username = "<username>";
$password = "<password>";
$dbname = "<dbname>";
$connection = new PDO(
    "mysql:host=$servername;dbname=$dbname",
    $username,
    $password
)

请注意,<username><password><dbname>是实际连接详细信息的占位符。您必须根据自己的数据库设置进行更改。

  1. 准备 SQL 查询并“绑定”参数:
$stmt = $connection->prepare("
    INSERT INTO user (
        uuid,
        name,
        slug,
        created_on,
        updated_on
    ) VALUES (
        :uuid,
        :name,
        :slug,
        :created_on,
        :updated_on
    )
");
$stmt->bindParam(':uuid', $uuid);
$stmt->bindParam(':name', $name);
$stmt->bindParam(':slug', $slug);
$stmt->bindParam(':created_on', $createdOn);
$stmt->bindParam(':updated_on', $updatedOn);
  1. 插入一行新数据:
$uuid = "25769c6c-d34d-4bfe-ba98-e0ee856f3e7a";
$name = "John Doe";
$slug = "john-doe";
$createdOn = (new DateTime())->getTimestamp();
$updatedOn = $createdOn;
$stmt->execute();

这并不理想,因为您必须每次“准备”语句并绑定参数,这需要相当多的行来操作。因此,我们应该选择一个 PHP 数据库框架来加速开发。Medoo (medoo.in/)是其中的一个选择。它非常轻量级,非常容易集成和使用。

让我们安装并连接到我们的应用程序:

  1. 通过 Composer 安装 Medoo:
$ composer require catfan/medoo
  1. 如果一切都设置好了,您可以导入 Medoo 并传递一个配置数组来启动数据库连接,就像我们之前在原始方法中所做的那样:
use Medoo\Medoo;

$database = new Medoo([
  'database_type' => 'mysql',
  'database_name' => '<dbname>',
  'server' => 'localhost',
  'username' => '<username>',
  'password' => '<password>'
]);

通过这个数据库框架建立与 MySQL 数据库的连接就到此为止。您可以在本书的 GitHub 存储库中的/chapter-16/nuxt-php/proxy/backend/core/mysql.php中找到此片段的实际用法。我们将在接下来的部分中向您展示如何实现它,但现在让我们探索如何使用 Medoo 编写一些基本的 CRUD 操作。

插入记录

当您想要向表中插入新记录时,可以使用insert方法,如下所示:

$database->insert('user', [
    'uuid' => '41263659-3c1f-305a-bfac-6a7c9eab0507',
    'name' => 'Jane',
    'slug' => 'jane',
    'created_on' => '1568072289'
]);

如果您想了解有关此方法的更多细节,请访问medoo.in/api/insert

查询记录

当您想要列出表中的记录时,可以使用select方法,如下所示:

$database->select('user', [
    'uuid',
    'name',
    'slug',
    'created_on',
    'updated_on',
]);

select方法会给您一个记录列表。如果您只想选择特定行,可以使用get方法,如下所示:

$database->get('user', [
    'uuid',
    'name',
    'slug',
    'created_on',
    'updated_on',
    ], [
    'slug' => 'jane'
]);

如果您想了解更多细节,请访问medoo.in/api/select查看select方法和medoo.in/api/get查看get方法。

更新记录

当您想要修改表中记录的数据时,可以使用update方法,如下所示:

$database->update('user', [
    'name' => 'Janey',
    'slug' => 'jane',
    'updated_on' => '1568091701'
], [
    'uuid' => '41263659-3c1f-305a-bfac-6a7c9eab0507'
]);

如果您想了解有关此方法的更多细节,请访问medoo.in/api/update

删除记录

当您想要从表中删除记录时,可以使用delete方法,如下所示:

$database->delete('user', [
    'uuid' => '41263659-3c1f-305a-bfac-6a7c9eab0507'
]);

如果您想了解有关此方法的更多细节,请访问medoo.in/api/delete

这就是如何使用 Medoo 和 PDO 编写基本 CRUD 操作的全部内容。

请查看 Medoo 的文档medoo.in/doc以了解您可以使用的其他方法。还有其他替代方案,如github.com/doctrine/dbal上的 Doctrine DBAL 和github.com/illuminate/database上的 Eloquent。

在本节中,您学习了一些 PSR 和 CRUD 操作。接下来,我们将介绍如何将它们全部整合到 Nuxt 中。由于 PHP 和 JavaScript 是两种不同的语言,它们之间唯一的交流方式是通过 API 中的 JSON。

但在编写启用该功能的脚本之前,我们应该研究这两个程序的跨域应用程序结构。自从第十二章以来,我们一直在为我们的 Nuxt 应用程序使用跨域应用程序结构,创建用户登录和 API 身份验证,所以这对您来说应该很熟悉。让我们开始吧!

构建跨域应用程序目录

再次,就像构建跨域应用程序目录时一样,以下是我们对 Nuxt 和我们的 PHP API 的整体视图:

// Nuxt app
front-end
├── package.json
├── nuxt.config.js
└── pages
    ├── index.vue
    └── ...

// PHP API
backend
├── composer.json
├── vendor
│ └── ...
├── ...
└── ...

就 Nuxt 的目录结构而言,它保持不变。我们只需对 API 目录的结构进行轻微更改,如下所示:

// PHP API
backend
├── composer.json
├── middlewares.php
├── routes.php
├── vendor
│ └── ...
├── public
│ └── index.php
├── static
│ └── ...
├── config
│ └── ...
├── core
│ └── ...
├── middleware
│ └── ...
└── module
    └── ...

PHP API 的目录结构是一个建议。您可以始终设计自己喜欢并最适合您的结构。因此,一目了然,我们有以下内容:

  • /vendor/目录是存放所有第三方包或依赖项的地方。

  • /public/目录只包含一个index.php文件,该文件启动我们的 API。

  • /static/目录用于静态文件,例如网站图标。

  • /config/目录存储配置文件,例如 MySQL 文件。

  • /core/目录存储我们可以在整个应用程序中使用的常见对象和函数。

  • /middleware/目录存储我们的 PSR-15 中间件。

  • /module/目录存储我们稍后将创建的自定义模块,就像我们在第十二章中所做的那样,创建用户登录和 API 身份验证,使用 Koa。

  • composer.json文件始终位于根级别。

  • middlewares.php文件是从/middleware/目录导入中间件的核心位置。

  • routes.php文件是从/module/目录导入路由的核心位置。

一旦您准备好结构,就可以开始编写顶级代码,将来自不同位置和目录的其他代码粘合到/public/目录中的index.php文件中,从而形成一个单一的应用程序。所以,让我们开始吧:

  1. foreach循环放在routes.php文件中,以迭代稍后将创建的每个模块:
// backend/routes.php
$modules = require './config/routes.php';

foreach ($modules as $module) {
    require './module/' . $module . 'index.php';
}
  1. /config/目录中创建一个routes.php文件,该文件将列出您的模块的文件名,如下所示:
// backend/config/routes.php
return [
    'Home/',
    'User/'.
    //...
];
  1. 在这个 PHP API 中,middlewares.php文件将导入一个用于装饰 CRUD 操作输出的中间件片段:
// backend/middlewares.php
require './middleware/outputDecorator.php';

此装饰器将以以下格式以 JSON 格式打印 CRUD 操作的输出:

{"status":<status code>,"data":<data>}
  1. /middleware/目录中创建一个名为outputDecorator.php的文件,其中包含以下代码。这将以前述格式包装操作的输出:
// backend/middleware/outputDecorator.php
use function Zend\Stratigility\middleware;

$router->middleware(middleware(function ($request, $handler) {
    $response = $handler->handle($request);
    $existingContent = (string) $response->getBody();
    $contentDecoded = json_decode($existingContent, true);
    $status = $response->getStatusCode();
    $data = [
        "status" => $status,
        "data" => $contentDecoded
    ];
    $payload = json_encode($data);

    $response->getBody()->rewind();
    $response->getBody()->write($payload);

    return $response
        ->withHeader('Content-Type', 'application/json')
        ->withStatus($status);
}));

在这里,我们使用zend-stratigility组件的middleware方法来创建装饰器中间件。然后,我们使用 The League of Extraordinary 的league/route路由器来使用此中间件锁定整个应用程序。

  1. /core/目录中创建一个名为mysql.php的文件,该文件返回 MySQL 连接的 Medoo 实例:
// backend/core/mysql.php
$dbconfig = require './config/mysql.php';
$mysql = new Medoo\Medoo([
    'database_type' => $dbconfig['type'],
    'database_name' => $dbconfig['name'],
    'server' => $dbconfig['host'],
    'username' => $dbconfig['username'],
    'password' => $dbconfig['password']
]);
return $mysql;
  1. 正如我们之前提到的,/public/目录只包含一个index.php文件。这用于启动我们的程序,因此它包含了您之前学习的关于 PSRs 的脚本:
// backend/public/index.php
chdir(dirname(__DIR__));
require_once 'vendor/autoload.php';

$request = Zend\Diactoros\ServerRequestFactory::fromGlobals(
    //...
);

$router = new League\Route\Router;
try {
    require 'middlewares.php';
    require 'routes.php';
    $response = $router->dispatch($request);
} catch(Exception $exception) {
    // handle errors
}

(new Zend\HttpHandlerRunner\Emitter\SapiEmitter)->emit($response);

在这里,您可以看到middlewares.phproutes.php文件被导入到这个文件中以生成一个 PSR-7 响应。它们被包裹在trycatch块中,以捕获任何 HTTP 错误,比如 404 和 506 错误。因此,模块的任何输出和错误都将通过最后一行传递到浏览器。希望这给您提供了一个对这个简单 API 的整体了解。现在,让我们继续深入学习/module/目录,以更详细地了解如何创建模块和路由。

创建 API 的公共路由及其模块

创建 API 的公共路由及其模块与您在本书前几章中学习构建的 API 非常相似;主要区别在于语言。以前我们使用 JavaScript 和 Node.js 框架 Koa,而在本章中的 API 中,我们使用 PHP 和 PSRs 来创建一个与框架无关的 API。所以,让我们开始吧:

  1. /module/目录中创建两个目录:一个名为Home,另一个名为User。这两个子目录是 API 中的模块。在每个模块中,创建一个/_routes/目录和一个index.php文件,该文件将从/_routes/目录导入路由,如下所示:

└── module
    ├── Home
    │ ├── index.php
    │ └── _routes
    │ └── hello_world.php
    └── User
        ├── index.php
        └── _routes
           └── ...
  1. Home模块中,输出一个“Hello world!”消息,并将其映射到/路由,如下所示:
// module/Home/_routes/hello_world.php
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;

$router->get('/', function (ServerRequestInterface $request) : 
  ResponseInterface {
    return new Zend\Diactoros\Response\JsonResponse(
     'Hello world!');
});
  1. User模块中,编写 CRUD 操作,以便我们可以创建、读取、更新和删除用户。因此,在/_routes/目录中,创建五个文件,分别为fetch_user.phpfetch_users.phpinsert_user.phpupdate_user.phpdelete_user.php。在这些文件中,我们将在/Controller/目录中为每个 CRUD 操作映射路由:
└── User
    ├── index.php
    ├── _routes
    │ ├── delete_user.php
    │ ├── fetch_user.php
    │ └── ...
    └── Controller
        └── ...
  1. 例如,在fetch_users.php文件中,我们将定义一个/users路由来列出所有用户,如下所示:
// module/User/_routes/fetch_users.php
use Psr\Http\Message\ResponseInterface;
use Psr\Http\Message\ServerRequestInterface;

$router->get('/users', function (ServerRequestInterface $request) : ResponseInterface {
    $database = require './core/mysql.php';
    $users = (new Spectre\User\Controller\
     Fetch\Users($database))->fetch();
    return new Zend\Diactoros\Response\JsonResponse($users);
});

在这里,您可以看到我们将 Medoo 实例作为$database导入,并将其传递给执行Read操作的控制器,然后调用fetch方法以获取所有可用用户。

  1. 接下来,我们要做的是创建一些 CRUD 目录:InsertFetchUpdateDelete。在每个 CRUD 目录中,我们将把 PSR-4 类存储在/Controller/目录中,如下所示:
└── Controller
    ├── Controller.php
    ├── Insert
    │ └── User.php
    ├── Fetch
    │ ├── User.php
    │ └── Users.php
    ├── Update
    │ └── User.php
    └── Delete
        └── User.php
  1. 首先,创建一个abstract类,可以被 CRUD 目录中的类扩展。这个类将只在其构造函数中接受Medoo\Medoo数据库,如下所示:
// module/User/Controller/Controller.php
namespace Spectre\User\Controller;

use Medoo\Medoo;

abstract class Controller
{
    protected $database;

    public function __construct(Medoo $database)
    {
        $this->database = $database;
    }
}
  1. 导入上述abstract类并将其扩展到需要连接到 MySQL 数据库的任何其他类中,如下所示:
// module/User/Controller/Fetch/Users.php
namespace Spectre\User\Controller\Fetch;

use Spectre\User\Controller\Controller;

class Users extends Controller
{
    public function fetch()
    {
        $columns = [
            'uuid',
            'name',
            'slug',
            'created_on',
            'updated_on',
        ];
        return $this->database->select('user', $columns);
    }
}

在这个类中,我们使用select方法从 MySQL 数据库的user表中获取所有用户。Medoo 将返回一个包含用户列表的Array,如果没有用户,则返回一个空的Array。然后,使用zend-diactoros中的JsonResponse方法将结果转换为 JSON,在fetch_users.php文件中。

最后,它将被装饰在/middleware/目录中的中间件。这将产生以下输出:

{"status":200,"data":[{"uuid":"...","name":"Jane","slug":"jane",...},{...},{...}]}

关于 PHP API 就是这样了。很简单,不是吗?在这个练习中,我们将跳过在 API 端处理 CORS 的任务,因为我们将使用 Nuxt Axios 和 Proxy 模块在我们即将创建的 Nuxt 应用程序中无缝轻松地处理 CORS。所以,让我们开始吧!

您可以在本书的 GitHub 存储库中的/chapter-16/nuxt-php/proxy/backend/中找到这个 PHP API,以及这个 API 的其余 CRUD 类在/chapter-16/nuxt-php/proxy/backend/module/User/Controller/中。

与 Nuxt 集成

@nuxtjs/axios模块与@nuxtjs/proxy模块很好地集成在一起,在许多情况下非常有用。防止 CORS 问题是使用这两个模块的好处之一。您在第六章中学习了如何安装和使用它们,编写插件和模块。让我们回顾一下:

  1. 通过 npm 安装@nuxtjs/axios@nuxtjs/proxy模块:
$ npm install @nuxtjs/axios
$ npm install @nuxtjs/proxy
  1. 在 Nuxt 配置文件的modules选项中注册@nuxtjs/axios,如下所示:
// nuxt.config.js
module.exports = {
  modules: [
    '@nuxtjs/axios'
  ],

  axios: {
    proxy: true
  },

  proxy: {
    '/api/': { target: 'http://0.0.0.0:8181', 
     pathRewrite: {'^/api/': ''} }
  }
}

请注意,当您与@nuxtjs/axios一起使用@nuxtjs/proxy时,不需要注册@nuxtjs/proxy模块,只要它已安装并在package.jsondependencies字段中。

在上述配置中,我们使用/api/作为http://0.0.0.0:8181的代理,这是我们的 PHP API 运行的地方。因此,每当我们在任何 API 端点请求中使用/api/时,它都会调用0.0.0.0:8181。例如,假设您正在进行 API 调用,如下所示:

$axios.get('/api/users')

@nuxtjs/axios@nuxtjs/proxy模块将把/api/users端点转换为以下内容:

http://0.0.0.0:8181/api/users

但由于我们在 PHP API 的路由中不使用/api/,所以我们在配置中使用pathRewrite在调用过程中将其移除。然后,由@nuxtjs/axios@nuxtjs/proxy模块发送到 API 的实际 URL 如下:

http://0.0.0.0:8181/users

再次访问以下链接,了解有关这两个模块的更多信息:

安装和配置完成后,我们可以开始创建用于与 PHP API 通信的前端 UI。我们将在下一节中讨论这个问题。

创建 CRUD 页面

再次强调,这对您来说并不是完全新的任务,因为这几乎与您在第九章中学习创建的 CRUD 页面相同,即添加服务器端数据库。让我们回顾一下:

  1. /pages/users/目录中创建以下页面以发送和获取数据:
users
├── index.vue
├── _slug.vue
├── add
│ └── index.vue
├── update
│ └── _slug.vue
└── delete
    └── _slug.vue
  1. 例如,使用以下脚本来获取所有可用用户:
// pages/users/index.vue
export default {
  async asyncData ({ error, $axios }) {
    try {
      let { data } = await $axios.get('/api/users')
      return {
        users: data.data
      }
    } catch (err) {
      // handle errors.
    }
  }
}

这个 Nuxt 应用程序中的脚本、模板和目录结构与您在第九章中学习创建的应用程序相同,即添加服务器端数据库。不同之处在于在那一章中使用了_id,但在这一章中,我们使用_slug。到目前为止,您应该能够独立完成其余的 CRUD 页面。但是,您可以随时回顾第九章中的以下部分,添加服务器端数据库,以获取更多信息:

  • 创建添加页面

  • 创建更新页面

  • 创建删除页面

创建了这些页面后,可以使用npm run dev运行 Nuxt 应用程序。您应该在浏览器上看到应用程序在localhost:3000上运行。

你可以在本书的 GitHub 存储库中的/chapter-16/nuxt-php/proxy/frontend/nuxt-universal/中找到此应用程序的完整源代码。

如果你不想在这个 Nuxt 应用中使用@nuxtjs/axios@nuxtjs/proxy模块,你可以在本书的 GitHub 存储库中的/chapter-16/nuxt-php/cors/中找到有关如何在 Nuxt 应用中为 PHP API 启用 CORS 的完整源代码。

你还可以在本书的 GitHub 存储库中的/chapter-16/nuxt-php/中找到名为user.sql的数据库副本。

现在,让我们总结一下你在这一长章节中学到的东西。我们希望你喜欢这一章,并且觉得它很有启发性。

总结

在本章中,你不仅成功地将 Nuxt 应用程序与 API 解耦,类似于你在第十二章中所做的,创建用户登录和 API 身份验证,而且还成功地用不同的语言 PHP 编写了一个 API,PHP 是 Web 开发中最流行的服务器端脚本语言之一。你学会了如何安装 PHP 和 Apache 以运行 PHP 应用程序,或者使用内置的 PHP Web 服务器进行开发,同时遵守 PSR-12、PSR4、PSR7 和 PSR-15,以构建一个现代的框架无关的应用程序。你还学会了使用 PHP 数据库框架 Medoo 来编写 CRUD 操作,重用了来自第九章的 Nuxt 应用程序,添加服务器端数据库,但进行了一些修改,并完美地将前端 UI 和后端 API 粘合在一起。现在,你还更详细地了解了 HTTP 消息,并知道如何使用 PDO 进行现代 PHP 数据库管理。干得好。

在下一章中,你将了解 Nuxt 在实时应用方面还能做些什么。在那里,你将学习Socket.ioRethinkDB。我们将带你了解这两种技术的安装过程。然后,你将学习如何在 RethinkDB 数据库中执行实时 CRUD 操作,在 JavaScript 中使用 Socket.io 编写实时代码,并将它们与 Nuxt 应用集成。这将是另一个有趣和激动人心的章节,我们将会引导你完成。所以,敬请关注!

使用 Nuxt 创建实时应用程序

在本章中,我们将进一步探讨 Nuxt,看看如何使用它与其他框架一起制作实时应用程序。我们将继续使用 Koa 作为后端 API,但是通过 RethinkDB 和 Socket.IO 来“增强”它。换句话说,我们将使用这两个令人敬畏的框架和工具将我们的后端 API 转换为实时 API。同时,我们还将借助它们将我们的前端 Nuxt 应用程序转换为实时 Nuxt 应用程序。如果您愿意,您可以在单域方法上开发这两个实时应用程序。但是,本书更倾向于跨域方法,以便我们不会混淆前端和后端的依赖关系,并随着时间的推移而感到困惑。因此,这将是另一个您可以从中学习的有趣而令人兴奋的章节!

在本章中,我们将涵盖以下主题:

  • 介绍 RethinkDB

  • 将 RethinkDB 与 Koa 集成

  • 介绍 Socket.IO

  • 将 Socket.IO 与 Nuxt 集成

让我们开始吧!

第十七章:介绍 RethinkDB

RethinkDB 是用于实时应用程序的开源 JSON 数据库。每当数据库表中发生更改时,它会从数据库实时推送 JSON 数据到您的应用程序,您可以订阅这些实时订阅 - changefeeds。尽管 changefeeds 是 RethinkDB 实时功能的核心,但如果您愿意,您可以跳过此功能。您可以像使用 MongoDB 一样使用 RethinkDB 来存储和查询您的 NoSQL 数据库。

尽管您可以使用 MongoDB 中的更改流来访问实时数据更改,但这需要一些配置才能启动,而实时订阅在 RethinkDB 中默认情况下已准备就绪,您可以立即开始使用,无需任何配置。让我们首先在您的系统中安装RethinkDB 服务器,然后看看您如何在下一节中使用它。

安装 RethinkDB 服务器

在撰写本书时,RethinkDB 的当前稳定版本是2.4.0活死人之夜),于 2019 年 12 月 19 日发布。根据平台(Ubuntu 或 OS),有几种安装 RethinkDB 服务器的方法。您可以在rethinkdb.com/docs/install/上查看您平台的指南。请注意,Windows 在 2.4.0 中尚不受支持。有关 Windows 的更多信息,请访问rethinkdb.com/docs/install/windows

在本书中,我们将在Ubuntu 20.04 LTS(Focal Fossa)上安装 RethinkDB 2.4.0。如果您使用的是 Ubuntu 19.10(Eoan Ermine)、Ubuntu 19.04(Disco Dingo)或较旧版本的 Ubuntu,如 18.04 LTS(Bionic Beaver),操作方式是相同的。让我们开始吧:

  1. 将 RethinkDB 存储库添加到 Ubuntu 存储库列表中,如下所示:
$ source /etc/lsb-release && echo "deb https://download.rethinkdb.com/apt $DISTRIB_CODENAME main" | sudo tee /etc/apt/sources.list.d/rethinkdb.list
  1. 使用wget获取 RethinkDB 的公钥:
$ wget -qO- https://download.rethinkdb.com/apt/pubkey.gpg | sudo apt-key add -

对于上述命令行,您应该在终端上收到一个 OK 消息。

  1. 更新您的 Ubuntu 版本并安装 RethinkDB:
$ sudo apt update
$ sudo apt install rethinkdb
  1. 验证 RethinkDB:
$ rethinkdb -v

您应该在终端上获得以下输出:

rethinkdb 2.4.0~0eoan (CLANG 9.0.0 (tags/RELEASE_900/final))

RethinkDB 附带了一个管理 UI,供您在浏览器上管理数据库,地址为localhost:8080。这在项目开发过程中非常方便和有用。如果您想要卸载 RethinkDB 并删除所有数据库,可以使用以下命令进行操作:

$ sudo apt purge rethinkdb.
$ sudo rm -r /var/lib/rethinkdb

安装时附带的管理 UI 类似于您在上一章中用于管理 PHP API 的 MySQL 数据库的 PHP Adminer。您可以使用 RethinkDB 管理 UI 通过 UI 上的图形按钮或使用 JavaScript 中的 RethinkDB 查询语言(ReQL)添加数据库和表。我们将在下一节中探索管理 UI 和 ReQL。

介绍 ReQL

ReQL 是 RethinkDB 的查询语言,用于操作 RethinDB 数据库中的 JSON 文档。查询是通过在服务器端调用 RethinkDB 的内置可链接函数自动构建的。这些函数嵌入在各种编程语言的驱动程序中,包括 JavaScript、Python、Ruby 和 Java。您可以在以下链接中查看 ReQL 命令/函数:

本书将使用 JavaScript。让我们在管理 UI 上使用数据资源管理器,通过使用相应的 ReQL 命令执行一些 CRUD 操作。您可以导航到数据资源管理器所在的页面,或者将浏览器指向localhost:8080/#dataexplorer并开始使用查询,如下所示。数据资源管理器上的默认顶级命名空间是r,因此 ReQL 命令必须链接到此命名空间。

但是,在我们的应用程序中使用驱动程序时,我们可以更改这个r命名空间,并在下一节中使用任何我们喜欢的名称。现在,让我们在这个练习中坚持使用默认命名空间r

  1. 创建数据库:
r.dbCreate('nuxtdb')

点击运行按钮。您应该在屏幕上看到类似以下的结果,显示已创建一个数据库,数据库名称由您选择,并且 RethinkDB 生成了一个 ID:

{
  "config_changes": [
    {
      "new_val": {
      "id": "353d11a4-adc8-4958-a4ae-a82c996dcb9f" ,
      "name": "nuxtdb"
    } ,
      "old_val": null
    }
  ] ,
  "dbs_created": 1
}

如果您想了解有关dbCreate ReQL 命令的更多信息,请访问rethinkdb.com/api/javascript/db_create/

  1. 在现有数据库中创建表;例如,在nuxtdb数据库中创建一个user表:
r.db('nuxtdb').tableCreate('user')

点击运行按钮。您应该在屏幕上看到类似以下的结果,显示 RethinkDB 为您生成的一个带有 ID 的表已被创建,并显示您创建的表的其他信息:

{
  "config_changes": [{
    "new_val": {
      "db": "nuxtdb",
      "durability": "hard",
      "id": "259e0066-1ffe-4064-8b24-d1c82e515a4a",
      "indexes": [],
      "name": "user",
      "primary_key": "id",
      "shards": [{
        "nonvoting_replicas": [],
        "primary_replica": "lau_desktop_opw",
        "replicas": ["lau_desktop_opw"]
      }],
      "write_acks": "majority",
      "write_hook": null
    },
    "old_val": null
  }],
  "tables_created": 1
}

如果您想了解有关tableCreate ReQL 命令的更多信息,请访问rethinkdb.com/api/javascript/table_create/

  1. 将新文档插入user表中:
r.db('nuxtdb').table('user').insert([
 { name: "Jane Doe", slug: "jane" },
 { name: "John Doe", slug: "john" }
])

点击运行按钮。您应该在屏幕上看到类似以下的结果,显示 RethinkDB 为您生成的两个带有键的文档已被插入:

{
  "deleted": 0,
  "errors": 0,
  "generated_keys": [
    "7f7d768d-0efd-447d-8605-2d460a381944",
    "a144001c-d47e-4e20-a570-a29968980d0f"
  ],
  "inserted": 2,
  "replaced": 0,
  "skipped": 0,
  "unchanged": 0
}

如果您想了解有关tableinsert ReQL 命令的更多信息,请分别访问rethinkdb.com/api/javascript/table/rethinkdb.com/api/javascript/insert/

  1. user表中检索文档:
r.db('nuxtdb').table('user')

点击运行按钮。您应该在屏幕上看到类似以下的结果,显示user表中的两个文档:

[{
  "id": "7f7d768d-0efd-447d-8605-2d460a381944",
  "name": "Jane Doe",
  "slug": "jane"
}, {
  "id": "a144001c-d47e-4e20-a570-a29968980d0f",
  "name": "John Doe",
  "slug": "john"
}]

如果要计算表中的总文档数,可以将count方法链接到查询中,如下所示:

r.db('nuxtdb').table('user').count()

在注入新文档后,user表中应该有2个文档。

如果您想了解有关count ReQL 命令的更多信息,请访问rethinkdb.com/api/javascript/count/

  1. 更新user表中的文档,通过使用slug键过滤表:
r.db('nuxtdb').table('user')
.filter(
  r.row("slug").eq("john")
)
.update({
  name: "John Wick"
})

点击运行按钮。您应该在屏幕上看到以下结果,显示已替换一个文档:

{
  "deleted": 0,
  "errors": 0,
  "inserted": 0,
  "replaced": 1,
  "skipped": 0,
  "unchanged": 0
}

如果您想了解有关filterupdate ReQL 命令的更多信息,请分别访问rethinkdb.com/api/javascript/filter/rethinkdb.com/api/javascript/update/

另外,如果您想了解有关roweq ReQL 命令的更多信息,请分别访问rethinkdb.com/api/javascript/row/rethinkdb.com/api/javascript/eq/

  1. 通过使用slug键过滤表格来从user表中删除文档:
r.db('nuxtdb').table('user')
.filter(
  r.row("slug").eq("john")
)
.delete()

点击运行按钮。您应该在屏幕上看到以下结果,显示已删除一个文档:

{
  "deleted": 1,
  "errors": 0,
  "inserted": 0,
  "replaced": 0,
  "skipped": 0,
  "unchanged": 0
}

如果您想删除表中的所有文档,那么只需将delete方法链接到表而不进行过滤,如下所示:

r.db('nuxtdb').table('user').delete()

如果您想了解有关delete ReQL 命令的更多信息,请访问rethinkdb.com/api/javascript/delete/

在使用 ReQL 命令时,这很有趣也很容易,不是吗?您不必阅读所有 ReQL 命令并详细研究每个命令以提高生产力。您只需要知道您想要做什么,并根据您已经了解的编程语言从 ReQL 命令参考/API 页面中找到您需要的命令。接下来,您将找出如何将RethinkDB 客户端或驱动程序添加到您的应用程序中。让我们开始吧!

将 RethinkDB 与 Koa 集成

在本节中,我们将构建一个简单的 API,按照我们在上一章中创建的 PHP API 的方式列出、添加、更新和删除用户。在之前的 API 中,我们使用了 PHP 和 MySQL,而在本章中,我们将使用 JavaScript 和 RethinkDB。我们仍将使用 Koa 作为 API 的框架。但这一次,我们将重新构建 API 目录,使其结构与您已经熟悉的 Nuxt 应用程序和 PHP API 的目录结构保持一致(尽可能)。所以,让我们开始吧!

重构 API 目录

请记住,当使用 Vue CLI 时,您在项目中获得的默认目录结构,您在第十一章中了解过,编写路由中间件和服务器中间件?使用 Vue CLI 安装项目后,如果您查看项目目录,您将看到一个基本的项目结构,其中包含/src/目录,用于开发组件、页面和路由,如下所示:

├── package.json
├── babel.config.js
├── README.md
├── public
│ ├── index.html
│ └── favicon.ico
└── src
    ├── App.vue
    ├── main.js
    ├── router.js
    ├── components
    │ └── HelloWorld.vue
    └── assets
        └── logo.png

自第十二章以来,我们一直在为跨域应用程序使用这种标准结构,创建用户登录和 API 身份验证。例如,以下是您之前创建的 Koa API 的目录结构:

backend
├── package.json
├── backpack.config.js
├── static
│ └── ...
└── src
    ├── index.vue
    ├── ...
    ├── modules
    │ └── ...
    └── core
        └── ...

但是这一次,我们将在本章中要创建的 API 中消除/src/目录。因此,让我们将/src/目录中的所有内容移动到顶层,并重新配置应用程序的引导方式,如下所示:

  1. 在项目的根目录中创建以下文件和文件夹:
backend
├── package.json
├── backpack.config.js
├── middlewares.js
├── routes.js
├── configs
│ ├── index.js
│ └── rethinkdb.js
├── core
│ └── ...
├── middlewares
│ └── ...
├── modules
│ └── ...
└── public
    └── index.js

再次强调,这里的目录结构仅仅是一个建议;您可以根据自己的需求设计自己的目录结构,使其最适合您。但让我们来看看这个建议的目录,并研究这些文件和文件夹的用途:

  • /configs/目录用于存储应用程序的基本信息和 RethinkDB 数据库连接的详细信息。

  • /public/目录用于存储启动应用程序的文件。

  • /modules/目录用于存储应用程序的模块,例如我们将在接下来的章节中创建的'user'模块。

  • /core/目录用于存储可以在整个应用程序中使用的常用函数或类。

  • middlewares.js文件是从/middlewares//node_modules/目录导入中间件的核心位置。

  • routes.js文件是从/modules目录导入路由的核心位置。

  • backpack.config.js文件用于自定义我们应用程序的 webpack 配置。

  • package.json文件包含我们应用程序的脚本和依赖项,并始终位于根目录。

  1. 将入口文件指向/public/目录中的index.js文件:
// backpack.config.js
module.exports = {
  webpack: (config, options, webpack) => {
    config.entry.main = './public/index.js'
    return config
  }
}

请记住,Backpack 中的默认入口文件是/src/目录中的index.js文件。由于我们已将此索引文件移动到/public/目录,因此必须通过 Backpack 配置文件配置此入口点。

如果您想了解 webpack 中的入口点,请访问webpack.js.org/concepts/entry-points/

  1. 在返回 Backpack 配置文件中的config对象之前,在 webpack 配置中的resolve选项中为/configs/core/modules/middlewares路径添加别名:
// backpack.config.js
const path = require('path')

config.resolve = {
  alias: {
    Configs: path.resolve(__dirname, 'configs/'),
    Core: path.resolve(__dirname, 'core/'),
    Modules: path.resolve(__dirname, 'modules/'),
    Middlewares: path.resolve(__dirname, 'middlewares/')
  }
}

在我们的应用程序中使用别名来解析文件路径非常有用和方便。通常,我们使用相对路径导入文件,就像这样:

import notFound from '../../Middlewares/notFound'

现在,我们可以使用别名从任何地方导入文件,这样可以隐藏相对路径,从而使我们的代码更整洁:

import notFound from 'Middlewares/notFound'

如果您想了解 webpack 中的别名和解析选项,请访问webpack.js.org/configuration/resolve/resolvealias

一旦您准备好了上述结构并且入口文件已经排序好,您就可以开始将 CRUD 操作应用到此 API 中。但首先,您需要将RethinkDB JavaScript 客户端安装到您的项目中。所以,让我们开始吧!

添加并使用 RethinkDB JavaScript 客户端

根据您拥有的编程知识,您可以选择几个官方客户端驱动程序,包括 JavaScript、Ruby、Python 和 Java。还有许多社区支持的驱动程序,如 PHP、Perl 和 R。您可以在rethinkdb.com/docs/install-drivers/上查看它们。

在本书中,我们将使用 RethinkDB JavaScript 客户端驱动程序。我们将指导您通过以下步骤安装并使用此驱动程序进行 CRUD 操作:

  1. 通过 npm 安装 RethinkDB JavaScript 客户端驱动程序:
$ npm i rethinkdb
  1. 创建一个rethinkdb.js文件,其中包含 RethinkDB 服务器连接详细信息,放在/configs/目录中,如下所示:
// configs/rethinkdb.js
export default {
  host: 'localhost',
  port: 28015,
  dbname: 'nuxtdb'
}
  1. 创建一个名为connection.js的文件,其中包含在/core/目录中的上述连接详细信息,用于打开 RethinkDB 服务器连接,如下所示:
// core/database/rethinkdb/connection.js
import config from 'Configs/rethinkdb'
import rethink from'rethinkdb'

const c = async() => {
  const connection = await rethink.connect({
    host: config.host,
    port: config.port,
    db: config.dbname
  })
  return connection
}
export default c
  1. 此外,还可以创建一个名为open.js的开放连接中间件文件,并将其绑定到 Koa 上下文中作为连接到 RethinkDB 的另一种选择,放在/middlewares/目录中。
// middlewares/database/rdb/connection/open.js
import config from 'Configs/rethinkdb'
import rdb from'rethinkdb'

export default async (ctx, next) => {
  ctx._rdbConn = await rdb.connect({
    host: config.host,
    port: config.port,
    db: config.dbname
  })
  await next()
}

使用目录路径来描述你的中间件(或 CRUD 操作)是一个很好的做法,我们从 PHP 的 PSR-4 中学到了这一点,这样你就不必使用一个很长的名称来描述你的文件。例如,如果你没有使用描述性的目录路径,你可能想将这个中间件命名为rdb-connection-open.js,以尽可能清楚地描述它。但如果你使用目录路径来描述中间件,那么你可以简单地将文件命名为open.js

  1. /middlewares/目录中创建一个close.js文件,创建一个关闭连接中间件,并将其绑定到 Koa 上下文作为最后一个中间件,如下所示:
// middlewares/database/rdb/connection/close.js
import config from 'Configs/rethinkdb'
import rdb from'rethinkdb'

export default async (ctx, next) => {
  ctx._rdbConn.close()
  await next()
}
  1. 在根目录的middlewares.js文件中导入openclose连接中间件,并将它们注册到应用程序中,如下所示:
// middlewares.js
import routes from './routes'
import rdbOpenConnection from 'Middlewares/database/rdb/connection/open'
import rdbCloseConnection from 'Middlewares/database/rdb/connection/close'

export default (app) => {
  //...
  app.use(rdbOpenConnection)
  app.use(routes.routes(), routes.allowedMethods())
  app.use(rdbCloseConnection)
}

在这里,你可以看到open连接中间件在所有模块路由之前注册,而close连接中间件则在最后注册,这样它们分别被首先和最后调用。

  1. 在接下来的步骤中,我们将使用以下模板代码与 Koa 路由器和 RethinkDB 客户端驱动进行 CRUD 操作。例如,以下代码显示了我们如何将模板代码应用于从user模块中的user表中获取所有用户的操作:
// modules/user/_routes/index.js
import Router from 'koa-router'
import rdb from 'rethinkdb'

const router = new Router()
router.get('/', async (ctx, next) => {
  try {
    // perform verification on the incoming parameters...
    // perform a CRUD operation:
    let result = await rdb.table('user')
      .run(ctx._rdbConn)

    ctx.type = 'json'
    ctx.body = result
    await next()

  } catch (err) {
    ctx.throw(500, err)
  }
})
export default router

让我们来看看这段代码,了解它的作用。在这里,你可以看到我们在应用中使用了自定义的顶级命名空间rdb,用于 RethinkDB 客户端驱动,而不是你在localhost:8080上练习过的r命名空间。此外,在我们的应用中使用 RethinkDB 客户端驱动时,我们必须始终在 ReQL 命令的末尾调用run方法,以及 RethinkDB 服务器连接,以构造查询并将其传递到服务器执行。

此外,我们必须在代码的末尾调用next方法,以便将应用的执行传递给下一个中间件,特别是用于关闭 RethinkDB 连接的close连接中间件。在执行任何 CRUD 操作之前,我们应该对来自客户端的传入参数和数据进行检查。然后,我们应该将我们的代码包装在try-catch块中,以捕获和抛出任何潜在的错误。

请注意,在接下来的步骤中,我们将跳过编写参数验证和 try-catch 语句的代码,以避免冗长和重复的代码行和代码块,但你应该在实际代码中包含它们。

  1. user模块的/_routes/文件夹中创建一个名为create-user.js的文件,其中包含以下代码,用于将新用户注入到数据库中的user表中:
// modules/user/_routes/create-user.js
router.post('/user', async (ctx, next) => {
  let result = await rdb.table('user')
    .insert(document, {returnChanges: true})
    .run(ctx._rdbConn)

  if (result.inserted !== 1) {
    ctx.throw(404, 'insert user failed')
  }

  ctx.type = 'json'
  ctx.body = result
  await next()
})

如果插入失败,我们应该抛出错误,并将错误消息传递给 Koa 的throw方法,以便我们可以在前端使用try-catch块捕获它们并显示出来。

  1. user模块的/_routes/文件夹中创建一个名为fetch-user.js的文件,用于通过使用slug键从user表中获取特定用户,如下所示:
// modules/user/_routes/fetch-user.js
router.get('/:slug', async (ctx, next) => {
  const slug = ctx.params.slug
  let user = await rdb.table('user')
    .filter(searchQuery)
    .nth(0)
    .default(null)
    .run(ctx._rdbConn)

  if (!user) {
    ctx.throw(404, 'user not found')
  }

  ctx.type = 'json'
  ctx.body = user
  await next()
})

我们在查询中添加了nth命令,以显示文档的位置。在我们的情况下,我们只想获取第一个文档,因此我们将一个0整数传递给此方法。我们还添加了default命令,以便在user表中找不到用户时返回一个null异常。

  1. user模块的/_routes/文件夹中创建一个名为update-user.js的文件,用于通过使用文档 ID 更新user表中的现有用户,如下所示:
// modules/user/_routes/update-user.js
router.put('/user', async (ctx, next) => {
  let body = ctx.request.body || {}
  let objectId = body.id

  let timestamp = Date.now()
  let updateQuery = {
    name: body.name,
    slug: body.slug,
    updatedAt: timestamp
  }

  let result = await rdb.table('user')
    .get(objectId)
    .update(updateQuery, {returnChanges: true})
    .run(ctx._rdbConn)

  if (result.replaced !== 1) {
    ctx.throw(404, 'update user failed')
  }

  ctx.type = 'json'
  ctx.body = result
  await next()
})

我们在查询中添加了get命令,首先通过其 ID 获取特定文档,然后再运行更新。

  1. user模块的/_routes/文件夹中创建一个名为delete-user.js的文件,用于通过使用文档 ID 从user表中删除现有用户,如下所示:
// modules/user/_routes/delete-user.js
router.del('/user', async (ctx, next) => {
  let body = ctx.request.body || {}
  let objectId = body.id

  let result = await rdb.table('user')
    .get(objectId)
    .delete()
    .run(ctx._rdbConn)

  if (result.deleted !== 1) {
    ctx.throw(404, 'delete user failed')
  }

  ctx.type = 'json'
  ctx.body = result
  await next()
})
  1. 最后,在index.js文件中对刚刚在步骤 7中创建的用于列出user表中所有用户的 CRUD 操作进行重构,通过在查询中添加orderBy命令,该文件位于/_routes/文件夹中,如下所示:
// modules/user/_routes/index.js
router.get('/', async (ctx, next) => {
  let cursor = await rdb.table('user')
    .orderBy(rdb.desc('createdAt'))
    .run(ctx._rdbConn)

  let users = await cursor.toArray()

  ctx.type = 'json'
  ctx.body = users
  await next()
})

我们在查询中添加了orderBy命令,以便我们可以按创建日期降序(最新的在前)对文档进行排序。此外,RethinkDB 数据库返回的文档始终作为 CRUD 操作的回调包含在一个游标对象中,因此我们必须使用toArray命令来遍历游标并将对象转换为数组。

如果您想了解更多关于orderBytoArray命令,请访问rethinkdb.com/api/javascript/order_by/rethinkdb.com/api/javascript/to_array/

通过这样,您已成功在 API 中使用 RethinkDB 实现了 CRUD 操作。再次强调,这很容易且有趣,不是吗?但是我们仍然可以通过在 RethinkDB 数据库中强制执行模式来提高我们存储的文档的“质量”。我们将在下一部分学习如何做到这一点。

在 RethinkDB 中强制执行模式

就像 MongoDB 中的 BSON 数据库一样,RethinkDB 中的 JSON 数据库也是无模式的。这意味着数据库上没有蓝图,也没有强加在数据库上的公式或完整性约束。数据库的构造方式没有组织规则可能会引发数据库完整性的问题。同一张表(或 MongoDB 中的“集合”)中的某些文档可能包含不同和不需要的键,以及具有正确键的文档。您可能会错误地注入一些键,或者忘记注入所需的键和值。因此,如果您希望保持文档中的数据有组织,强制执行 JSON 或 BSON 数据库中的某种模式可能是一个好主意。RethinkDB(或 MongoDB)没有内部功能来强制执行模式,但我们可以使用 Node.js Lodash 模块创建自定义函数来强制执行一些基本模式。让我们探讨如何做到这一点:

  1. 通过 npm 安装 Lodash 模块:
$ npm i lodash
  1. /core/目录中创建一个utils.js文件,并导入lodash以创建一个名为sanitise的函数,如下所示:
// core/utils.js
import lodash from 'lodash'

function sanitise (options, schema) {
  let data = options || {}

  if (schema === undefined) {
    const err = new Error('Schema is required.')
    err.status = 400
    err.expose = true
    throw err
  }

  let keys = lodash.keys(schema)
  let defaults = lodash.defaults(data, schema)
  let picked = lodash.pick(defaults, keys)

  return picked
}
export { sanitise }

这个函数简单地选择您设置的默认键,并忽略任何不在“模式”中的额外键。

我们正在使用 Lodash 中的以下方法。有关每种方法的更多信息,请访问以下链接:lodash.com/docs/4.17.15#keys 获取keys方法的信息 lodash.com/docs/4.17.15#defaults 获取defaults方法的信息 lodash.com/docs/4.17.15#pick 获取pick方法的信息

  1. user模块中创建一个user模式,只接受以下键:
// modules/user/schema.js
export default {
  slug: null,
  name: null,
  createdAt: null,
  updatedAt: null
}
  1. 在要强制执行模式的路由中导入sanitise方法和前面的模式;例如,在create-user.js文件中:
// modules/user/_routes/create-user.js
let timestamp = Date.now()
let options = {
  name: body.name,
  slug: body.slug,
  createdAt: timestamp,
  username: 'marymoe',
  password: '123123'
}

let document = sanitise(options, schema)
let result = await rdb.table('user')
  .insert(document, {returnChanges: true})
  .run(ctx._rdbConn)

在上述代码中,示例字段usernamepassword在插入数据之前对数据进行清理时不会被注入到user表中的文档中。

您可以看到这个sanitise函数只执行简单的验证。如果您需要更复杂和高级的数据验证,可以使用 hapi web 框架的 Node.js joi 模块。

如果您想了解更多关于这个模块的信息,请访问hapi.dev/module/joi/

您接下来必须探索 RethinkDB 中的changefeeds。这是本章的主要目的 - 展示如何利用 RethinkDB 的实时功能创建实时应用程序。因此,让我们探索并玩转 RethinkDB 中的 changefeeds!

介绍 RethinkDB 中的 changefeeds

在使用 RethinkDB 客户端驱动程序在我们的应用程序中应用 changefeeds 之前,让我们再次在localhost:8080/#dataexplorer的管理 UI 中使用数据浏览器,实时在屏幕上查看实时 feeds:

  1. 粘贴以下的 ReQL 查询,并单击“运行”按钮:
r.db('nuxtdb').table('user').changes()

您应该在浏览器屏幕上看到以下信息:

Listening for events...
Waiting for more results
  1. 在浏览器上打开另一个标签,并将其指向localhost:8080/#dataexplorer。现在,您有两个数据浏览器。将其中一个从浏览器标签中拖出来,以便您可以将它们并排放置。然后,从其中一个数据浏览器中将新文档插入user表中:
r.db('nuxtdb').table('user').insert([
  { name: "Richard Roe", slug: "richard" },
  { name: "Marry Moe", slug: "marry" }
])

您应该得到以下结果:

{
  "deleted": 0,
  "errors": 0,
  "generated_keys": [
    "f7305c97-2bc9-4694-81ec-c5acaed1e757",
    "5862e1fa-e51c-4878-a16b-cb8c1f1d91de"
  ],
  "inserted": 2,
  "replaced": 0,
  "skipped": 0,
  "unchanged": 0
}

与此同时,您应该立即在另一个数据浏览器中看到以下 feeds 的实时显示:

{
  "new_val": {
    "id": "f7305c97-2bc9-4694-81ec-c5acaed1e757",
    "name": "Richard Roe",
    "slug": "richard"
  },
  "old_val": null
}

{
  "new_val": {
    "id": "5862e1fa-e51c-4878-a16b-cb8c1f1d91de",
    "name": "Marry Moe",
    "slug": "marry"
  },
  "old_val": null
}

万岁!您刚刚轻松地使用 RethinkDB 创建了实时 feeds!请注意,您将始终在每个实时 feed 中获得这两个键,new_valold_val。它们具有以下含义:

  • 如果您在new_val中获取数据,但在old_val中获取的是null,这意味着新文档被注入到数据库中。

  • 如果您在new_valold_val中都获取到数据,这意味着现有文档在数据库中已更新。

  • 如果您在old_val中获取数据,但在new_val中获取的是null,这意味着现有文档已从数据库中删除。

当我们在本章的最后一节中在 Nuxt 应用程序中使用它们时,您将可以使用这些键。因此,现在不要太担心它们。相反,下一个挑战是在 API 和 Nuxt 应用程序中实现它。为此,我们将需要另一个 Node.js 模块 - Socket.IO。因此,让我们探索一下这个模块如何帮助您实现这一目标。

介绍 Socket.IO

就像 HTTP 一样,WebSocket 是一种通信协议,但它提供了客户端和服务器之间的全双工(双向)通信。与 HTTP 不同,WebSocket 连接始终保持开放状态,用于实时数据传输。因此,在 WebSocket 应用程序中,服务器可以在没有客户端发起请求的情况下向客户端发送数据。

另外,与以 HTTP 或 HTTPS 开头的 HTTP 模式不同,WebSocket 协议模式以wswss开头,例如:

ws://example.com:4000

Socket.IO 是一个使用 WebSocket 协议和轮询作为创建实时 Web 应用的备用选项的 JavaScript 库。它支持任何平台、浏览器或设备,并处理服务器和客户端的所有降级,以实现实时的全双工通信。大多数浏览器现在都支持 WebSocket 协议,包括 Google Chrome、Microsoft Edge、Firefox、Safari 和 Opera。但是在使用 Socket.IO 时,我们必须同时使用其客户端和服务器端库。客户端库在浏览器内运行,而服务器端库在服务器端的 Node.js 应用程序上运行。因此,让我们在我们的应用程序中让这两个库一起工作。

如果您想了解更多关于 Socket.IO 的信息,请访问socket.io/

添加和使用 Socket.IO 服务器和客户端

我们将把 Socket.IO 服务器添加到我们在最近几节中构建的 API 中,然后最终将 Socket.IO 客户端添加到 Nuxt 应用程序中。但在将其添加到 Nuxt 应用程序之前,我们将其添加到一个简单的 HTML 页面中,以便我们可以全面了解 Socket.IO 服务器和 Socket.IO 客户端是如何一起工作的。让我们学习如何做到这一点:

  1. 通过 npm 安装 Socket.IO 服务器:
$ npm i socket.io
  1. 如果您还没有这样做,可以在/configs/目录中创建一个index.js文件来存储服务器设置:
// configs/index.js
export default {
  server: {
    port: 4000
  },
}

从这个简单的设置中,我们将在端口 4000 上提供我们的 API。

  1. 导入socket.io并将其绑定到 Node.js HTTP 对象,使用 Koa 的新实例创建一个新的 Socket.IO 实例,如下所示:
// backend/koa/public/index.js
import Koa from 'koa'
import socket from 'socket.io'
import http from 'http'
import config from 'Configs'
import middlewares from '../middlewares'

const app = new Koa()
const host = process.env.HOST || '127.0.0.1'
const port = process.env.PORT || config.server.port
middlewares(app)

const server = http.createServer(app.callback())
const io = socket(server)

io.sockets.on('connection', socket => {
  console.log('a user connected: ' + socket.id)
  socket.on('disconnect', () => {
    console.log('user disconnected: ' + socket.id)
  })
})
server.listen(port, host)

创建 Socket.IO 的新实例后,我们可以开始监听 Socket.IO 的connection事件,以接收来自socket回调的传入 socket。我们将传入的 socket 记录到控制台并附带其 ID。当 socket 断开连接时,我们还会记录传入 socket 的disconnect事件。最后,请注意,我们使用原生 Node.js HTTP 来启动和提供应用程序在localhost:4000上,而不是使用我们以前使用的 Koa 内部的 HTTP:

app.listen(4000)
  1. 创建一个socket-client.html页面,并通过 CDN 导入 Socket.IO 客户端。通过将localhost:4000作为特定 URL 传递来创建一个新的实例,如下所示:
// frontend/html/socket-client.html
<script src="https://cdn.jsdelivr.net/npm/socket.io-
 client@2/dist/socket.io.js"></script>

<script>
  var socket = io('http://localhost:4000/')
</script>

现在,如果你在浏览器上浏览这个 HTML 页面,或者当你刷新页面时,你应该看到控制台打印出带有 socket ID 的日志,如下所示:

a user connected: abeGnarBnELo33vQAAAB

当你关闭 HTML 页面时,你也应该看到控制台打印出带有 socket ID 的日志,如下所示:

user disconnected: abeGnarBnELo33vQAAAB

这就是连接 Socket.IO 服务器和客户端的全部操作。这非常简单易懂,不是吗?但我们在这里所做的只是连接和断开服务器和客户端。我们需要更多的功能——我们希望能够同时传输数据。为了做到这一点,我们只需要相互发射和接收事件,这将在接下来的步骤中完成。

如果你想使用 Socket.IO 客户端的本地版本,你可以将脚本标签的 URL 源指向/node_modules/socket.io-client/dist/socket.io.js

  1. 通过使用 Socket.IO 服务器的emit方法,从服务器创建一个发射事件,如下所示:
// backend/koa/public/index.js
io.sockets.on('connection', socket => {
  io.emit('emit.onserver', 'Hi client, what you up to?')
  console.log('Message to client: ' + socket.id)
})

在这里,你可以看到我们通过名为emit.onserver的自定义事件发射了一个简单的消息,并将活动记录到控制台中。请注意,我们只能在连接建立时发射事件。然后,我们可以在客户端监听这个自定义事件,并记录来自服务器的消息,如下所示:

// frontend/html/socket-client.html
socket.on('emit.onserver', function (message) {
  console.log('Message from server: ' + message)
})
  1. 所以,现在,如果你再次在浏览器上刷新页面,你应该看到控制台打印出带有 socket ID 的日志,如下所示:
Message to client: abeGnarBnELo33vQAAAB // server side
Message from server: Hi client, what you up to? // client side
  1. 通过使用 Socket.IO 客户端的emit方法,从客户端创建一个发射事件,如下所示:
// frontend/html/socket-client.html
<script
  src="https://code.jquery.com/jquery-3.4.1.slim.min.js"
  integrity="sha256-pasqAKBDmFT4eHoN2ndd6lN370kFiGUFyTiUHWhU7k8="
  crossorigin="anonymous"></script>

<button class="button-sent">Send</button>

$('.button-sent').click(function(e){
  e.preventDefault()

  var message = 'Hi server, how are you holding up?'
  socket.emit('emit.onclient', message)
  console.log('Message sent to server.')

  return false
})

在这里,你可以看到,首先,我们通过 CDN 安装 jQuery,并创建一个带有 jQuery click事件的<button>。其次,当按钮被点击时,我们发射名为emit.onclient的 Socket.IO 自定义事件,并附带一个简单的消息。最后,我们将活动记录到控制台中。

  1. 之后,我们可以在服务器端监听 Socket.IO 自定义事件,并记录来自客户端的消息,如下所示:
// backend/koa/public/index.js
socket.on('emit.onclient', (message) => {
  console.log('Message from client, '+ socket.id + ' :' + message);
})
  1. 如果你再次在浏览器上刷新页面,你应该看到控制台打印出日志,以及 socket ID,如下所示:
Message sent to server. // client side
Message from client, abeGnarBnELo33vQAAAB: Hi server, 
how are you holding up? // server side

现在你知道如何通过 Socket.IO 实时传输数据——只需发射自定义事件并监听它们。你接下来应该了解的是如何将 Socket.IO 与 RethinkDB 的 changefeeds 集成,以便将实时数据从数据库传输到客户端。所以,请继续阅读!

集成 Socket.IO 服务器和 RethinkDB changefeeds

请记住,您之前曾在localhost:8080/#dataexplorer的管理 UI 中再次使用 Data Explorer 调整 RethinkDB changefeeds。要订阅 changefeed,您只需将 ReQL 的changes命令链接到查询,如下所示:

r.db('nuxtdb').table('user').changes()

RethinkDB changefeeds 包含从 RethinkDB 数据库发出的实时数据,这意味着我们需要在服务器端使用 Socket.IO 服务器捕获这些 feed,并将它们发出到客户端。因此,让我们学习如何通过重构我们在本章中一直在开发的 API 来捕获它们:

  1. 通过 npm 将 Socket.IO 服务器安装到您的 API 中:
$ npm i socket.io
  1. /core/目录中的changefeeds.js文件中创建一个异步匿名箭头函数,代码如下:
// core/database/rethinkdb/changefeeds.js
import rdb from 'rethinkdb'
import rdbConnection from './connection'

export default async (io, tableName, eventName) => {
  try {
    const connection = await rdbConnection()
    var cursor = await rdb.table(tableName)
      .changes()
      .run(connection)

    cursor.each(function (err, row) {
      if (err) {
        throw err
      }
      io.emit(eventName, row)
    })
  } catch( err ) {
    console.error(err);
  }
}

在此函数中,我们将rethinkdb导入为rdb,将我们的 RethinkDB 数据库连接导入为rdbConnection,然后将以下项目用作此函数的参数:

  • Socket.IO 服务器的实例

  • 您将要使用的 Socket.IO 发出的自定义事件名称

  • 您要订阅其 changefeed 的 RethinkDB 表名

changefeed 将以回调的形式将文档返回为游标对象,因此我们通过游标对象进行迭代,并使用自定义事件名称发出每个文档的行。

  1. 在应用程序根目录的/public/目录中将changefeeds函数导入为rdbChangeFeeds,并将其与index.js文件中的其余现有代码集成,如下所示:
// public/index.js
import Koa from 'koa'
import socket from 'socket.io'
import http from 'http'
import config from 'Configs'
import middlewares from '../middlewares'
import rdbChangeFeeds from 'Core/database/rethinkdb/changefeeds'

const app = new Koa()
const host = process.env.HOST || '127.0.0.1'
const port = process.env.PORT || config.server.port
middlewares(app)

const server = http.createServer(app.callback())
const io = socket(server)
io.sockets.on('connection', socket => {
  //...
})

rdbChangeFeeds(io, 'user', 'user.changefeeds')
server.listen(port, host)

在上述代码中,我们要订阅的表名是user,我们要调用的发出事件名称是user.changefeeds。因此,我们将它们传递给rdbChangeFeeds函数,并使用socket.io实例。这就是您一次性全局集成 Socket.IO 和 RethinkDB 所需做的一切。

干得好!您已成功在服务器端集成了 Koa、RethinkDB 和 Socket.IO,并创建了一个实时 API。但是客户端怎么样,我们如何监听从 API 发出的事件?我们将在下一节中找出答案。

将 Socket.IO 与 Nuxt 集成

我们要构建的 Nuxt 应用程序与上一章中的应用程序非常相似,在那里我们有一个包含以下 CRUD 页面的/users/目录,该目录位于/pages/目录中,用于添加、更新、列出和删除用户:

users
├── index.vue
├── _slug.vue
├── add
│ └── index.vue
├── update
│ └── _slug.vue
└── delete
    └── _slug.vue

您可以从上一章复制这些文件。这个应用程序中唯一的主要变化和不同之处在于<script>块,我们将通过监听来自 Socket.IO 服务器的 emit 事件实时列出用户。为此,我们需要使用 Socket.IO 客户端,这是您在添加和使用 Socket.IO 服务器和客户端部分学到的,该部分使用简单的 HTML 页面。因此,让我们看看如何将我们已经知道的内容实现到 Nuxt 应用中:

  1. 通过 npm 将 Socket.IO 客户端安装到您的 Nuxt 项目中:
$ npm i socket.io-client
  1. 在 Nuxt 配置文件中创建以下变量,以便稍后重用应用的协议、主机名和跨域端口:
// nuxt.config.js
const protocol = 'http'
const host = process.env.NODE_ENV === 'production' ? 'a-cool-domain-name.com' : 'localhost'

const ports = {
  local: '8000',
  remote: '4000'
}

const remoteUrl = protocol + '://' + host + ':' + ports.remote + '/'

这些变量适用于以下情况:

  • host变量用于在 Nuxt 应用处于生产环境时获取a-cool-domain-name.com的值;也就是说,当您使用npm run start运行应用时。否则,它只会将localhost作为默认值。

  • ports变量中的local键用于为 Nuxt 应用设置服务器端口,设置为8000。请记住,Nuxt 提供应用的默认端口是3000

  • ports变量中的remote键用于告诉 Nuxt 应用 API 所在的服务器端口,即4000

  • remoteUrl变量用于将 API 与前面的变量连接起来。

  1. 将前述变量应用于 Nuxt 配置文件中的envserver选项,如下所示:
// nuxt.config.js
export default {
  env: {
    remoteUrl
  },
  server: {
    port: ports.local,
    host: host
  }
}

因此,通过这种配置,我们可以通过以下方法再次访问remoteUrl变量:

  • process.env.remoteUrl

  • context.env.remoteUrl

此外,在这个配置中,我们已将 Nuxt 应用的默认服务器端口更改为8000,在server选项中。默认端口是3000,默认主机是localhost。但是您可能出于某种原因想要使用不同的端口。这就是为什么我们在这里看如何更改它们。

如果您想了解更多关于server配置和其他选项(如timinghttps)的信息,请访问nuxtjs.org/api/configuration-server

如果您想了解更多关于env配置的信息,请访问nuxtjs.org/api/configuration-envthe-env-property

  1. 安装 Nuxt Axios 和 Proxy 模块,并在 Nuxt 配置文件中进行配置,如下所示:
// nuxt.config.js
export default {
  modules: [
    '@nuxtjs/axios'
  ],

  axios: {
    proxy: true
  },

  proxy: {
    '/api/': {
      target: remoteUrl,
      pathRewrite: {'^/api/': ''}
    }
  }
}

请注意,我们在proxy选项中重用了remoteUrl变量。因此,我们发出的每个以/api/开头的 API 请求都将转换为http://localhost:4000/api/。但由于我们在 API 的路由中没有/api/,在将其发送到 API 之前,我们使用pathRewrite选项从请求 URL 中删除这个/api/部分。

  1. /plugin/目录中创建一个插件,用于抽象 Socket.IO 客户端的实例,以便我们可以在任何地方重用它:
// plugins/socket.io.js
import io from 'socket.io-client'

const remoteUrl = process.env.remoteUrl
const socket = io(remoteUrl)

export default socket

请注意,我们通过process.env.remoteUrl重用了remoteUrl变量在 Socket.IO 客户端实例中。这意味着 Socket.IO 客户端将在localhost:4000调用 Socket.IO 服务器。

  1. socket.io客户端插件导入到<script>块中,并使用@nuxtjs/axios模块在index文件中获取用户列表。此索引文件位于/users/目录下的pages中:
// pages/users/index.vue
import socket from '~/plugins/socket.io'

export default {
  async asyncData ({ error, $axios }) {
    try {
      let { data } = await $axios.get('/api/users')
      return { users: data.data }
    } catch (err) {
      // Handle the error.
    }
  }
}
  1. 使用asyncData方法获取并设置用户后,使用 Socket.IO 插件在mounted方法中监听user.changefeeds事件,以获取来自服务器的任何新的实时反馈,如下所示:
// pages/users/index.vue
export default {
  async asyncData ({ error, $axios }) {
    //...
  },
  mounted () {
    socket.on('user.changefeeds', data => {
      if (data.new_val === undefined && data.old_val === undefined) {
        return
      }
      //...
    })
  }
}

在这里,您可以看到我们始终检查data回调,以确保传入的反馈中new_valold_val被定义。换句话说,在继续下一行之前,我们希望确保这两个键始终出现在反馈中。

  1. 检查后,如果我们在new_val键中收到数据,但old_val键为空,这意味着新用户已添加到服务器。如果我们从服务器端获取新的反馈,我们将使用 JavaScript 的unshift函数将新用户数据添加到user数组的顶部,如下所示:
// pages/users/index.vue
mounted () {
  //...
  if(data.old_val === null && data.new_val !== null) {
    this.users.unshift(data.new_val)
  }
}

然后,如果我们在old_val键中收到数据,但new_val键为空,这意味着现有用户已从服务器中删除。因此,要从数组中弹出现有用户,我们可以使用 JavaScript 的splice函数,通过其索引(在数组中的位置/位置)来删除。但首先,我们必须使用 JavaScript 的map函数按其 ID 找到用户的索引,如下所示:

// pages/users/index.vue
mounted () {
  //...
  if(data.new_val === null && data.old_val !== null) {
    var id = data.old_val.id
    var index = this.users.map(el => {
      return el.id
    }).indexOf(id)
    this.users.splice(index, 1)
  }
}

最后,如果我们在new_valold_val键中都收到数据,这意味着当前用户已更新。因此,如果用户已更新,我们必须首先找到数组中用户的索引,然后使用 JavaScript 的splice函数替换它,如下所示:

// pages/users/index.vue
mounted () {
  //...
  if(data.new_val !== null && data.old_val !== null) {
    var id = data.new_val.id
    var index = this.users.findIndex(item => item.id === id)
    this.users.splice(index, 1, data.new_val)
  }
}

请注意,我们使用 JavaScript 的findIndex函数作为map函数的另一种替代方法。

如果您想了解更多关于我们在这里使用的用于操作 JavaScript 数组的 JavaScript 标准内置函数的信息,请访问以下链接:

  1. 将以下模板添加到<template>块中以显示用户,如下所示:
// pages/users/index.vue
<div>
  <h1>Users</h1>
  <ul>
    <li v-for="user in users" v-bind:key="user.uuid">
      <nuxt-link :to="'/users/' + user.slug">
        {{ user.name }}
      </nuxt-link>
    </li>
  </ul>
  <nuxt-link to="/users/add">
    Add New
  </nuxt-link>
</div>

在此模板中,您可以看到我们只是简单地使用v-forasyncData方法获取的用户数据,并将用户uuid绑定到每个循环的元素上。之后,发生在mounted方法中的任何实时反馈都将以响应式方式更新用户数据和模板。

  1. 使用npm run dev运行 Nuxt 应用程序。您应该在终端上看到以下信息:
Listening on: http://localhost:8000/
  1. 在浏览器中打开两个标签并排,或者在两个不同的浏览器中并排,并将它们指向localhost:8000/users。从其中一个标签(或浏览器)在localhost:8000/users/add添加一个新用户。您应该看到新添加的用户立即并同时显示在所有标签(或浏览器)上,而无需刷新它们。

您可以在本书的 GitHub 存储库中的/chapter-17/frontend//chapter-17/backend/中找到本章中的所有代码和应用程序。

干得好 - 您成功了!我们希望您觉得这个应用程序有趣且易于操作,并且它能激励您进一步探索您迄今为止所学到的知识。让我们总结一下本章学到的内容。

摘要

在本章中,您成功安装并使用 RethinkDB 和 Socket.IO,将普通的后端 API 和前端 Nuxt 应用程序转变为实时应用程序。您学会了如何通过 RethinkDB 管理 UI 在服务器端创建、读取、更新和删除 JSON 数据,然后使用 Koa 与 RethinkDB 客户端驱动程序。最重要的是,您学会了如何通过 RethinkDB 管理 UI 操作 RethinkDB 中的实时订阅,称为 changefeeds,然后在服务器端将其与 Socket.IO 服务器和 Koa 集成。此外,您使用 Socket.IO 服务器发出自定义事件的数据,并使用 Socket.IO 客户端在 Nuxt 应用程序的客户端端实时监听事件并捕获数据。这难道不是一次有趣的旅程吗?

在下一章中,我们将通过第三方 API、内容管理系统(CMS)和 GraphQL 进一步了解 Nuxt。您将了解 WordPress API、Keystone 和 GraphQL。然后,您将学习如何创建自定义内容类型和自定义路由,以扩展 WordPress API,以便将其与 Nuxt 集成,并从 WordPress 项目中流式传输远程图像。您将使用 Keystone 开发自定义 CMS,安装和保护用于 Keystone 应用程序开发的 PostgreSQL,以及保护 MongoDB,您将学习如何在第九章中安装的内容,即添加服务器端数据库。最重要且令人兴奋的是,您将学习 REST API 和 GraphQL API 之间的区别;使用 GraphQL.js、Express 和 Apollo Server 构建 GraphQL API;了解 GraphQL 模式及其解析器;使用 Keystone GraphQL API;然后将它们与 Nuxt 集成。这绝对会是另一次有趣的旅程,所以系好安全带,准备好!

  • 4
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值