五月天激情丁香,国产精品2019,国产成人精品亚洲2020,国产精品免费视频一区二区三区,开心久久婷婷综合中文字幕,天堂视频在线观看免费完整版

基礎開始示例

demo 地址

基礎開發示例已經開源,源碼地址:https://github.com/XueSiLf/demo-3.7.x

注意事項,請先看完這里,再往下繼續瀏覽。因為下面的 demo 中使用到了 php8 的注解特性,所以您需要先學習注解如何使用,可查看 php 官方文檔的 注解 文檔。如果您已經對注解用法非常熟悉了,可直接往下繼續瀏覽。

安裝

框架安裝

Linux(Centos/Ubuntu/MacOS) 下安裝

  • 我們先把當前的 php 環境安裝好 swoole 拓展,安裝 swoole 擴展 步驟可查看 安裝 Swoole 章節,然后執行 php --ri swoole 確保可以看到 swoole 拓展版本為 4.8.13
  • 建立一個目錄,名為 Test ,執行 composer require easyswoole/easyswoole=3.7.x 引入 easyswoole
  • 執行 php vendor/bin/easyswoole.php install 進行安裝,然后輸入 YY

Docker 下安裝

docker run --name easyswoole \
-v /tmp/easyswoole:/var/www \
-p 9501:9501 -it \
--privileged -u root \
--entrypoint /bin/sh \
easyswoolexuesi2021/easyswoole:php8.1.22-alpine3.16-swoole4.8.13

cd /var/www
mkdir Test
cd Test
composer require easyswoole/easyswoole=3.7.x
php vendor/bin/easyswoole.php install # 然后輸入 `Y`、`Y`

組件引入

// 引入 IDE 代碼提示組件
composer require swoole/ide-helper

命名空間注冊

編輯 Test 根目錄下的 composer.json 文件,如果自動加載中沒有 App 命名空間,請在 autoload.psr-4 中加入 "App\\": "App/",然后執行 composer dumpautoload -o 進行名稱空間的更新。composer.json 文件大體結構如下:

{
    "require": {
        "easyswoole/easyswoole": "3.7.x",
        "swoole/ide-helper": "^5.1"
    },
    "autoload": {
        "psr-4": {
            "App\\": "App/"
        }
    }
}

安裝后目錄結構

Test                    項目部署目錄
├─App                     應用目錄
│  ├─HttpController      控制器目錄(如果沒有,請自行創建)
├─Log                     日志文件目錄(啟動后創建)
├─Temp                    臨時文件目錄(啟動后創建)
├─vendor                  第三方類庫目錄
├─bootstrap.php           框架 bootstrap 事件
├─composer.json           Composer 架構
├─composer.lock           Composer 鎖定
├─EasySwooleEvent.php     框架全局事件
├─easyswoole.php          框架管理腳本
├─dev.php                 開發配置文件
├─produce.php             生產配置文件

連接池實現

配置項

創建配置文件 Test/Config/DATABASE.php,加入以下配置信息,注意:請根據自己的 mysql 服務器信息填寫賬戶密碼

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link    http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact http://m.fe88.cn/Preface/contact.html
 * @license https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);
return [
    'DATABASE' => [
        // 添加 MySQL 及對應的連接池配置
        /*################ MYSQL CONFIG ##################*/
        'MYSQL' => [
            [
                'name'              => 'default', // 數據庫連接池名稱
                'useMysqli'         => false, // 是否是使用php-mysqli擴展
                'host'              => '127.0.0.1', // 數據庫地址
                'port'              => 3306, // 數據庫端口
                'user'              => 'easyswoole', // 數據庫用戶名
                'password'          => 'easyswoole', // 數據庫用戶密碼
                'timeout'           => 45, // 數據庫連接超時時間
                'charset'           => 'utf8', // 數據庫字符編碼
                'database'          => 'easyswoole_demo', // 數據庫名
                'autoPing'          => 5, // 自動 ping 客戶端鏈接的間隔
                'strict_type'       => false, // 不開啟嚴格模式
                'fetch_mode'        => false,
                'returnCollection'  => false, // 設置返回結果為 數組
                // 配置 數據庫 連接池配置,配置詳細說明請看連接池組件 http://m.fe88.cn/Components/Pool/introduction.html
                'intervalCheckTime' => 15 * 1000, // 設置 連接池定時器執行頻率
                'maxIdleTime'       => 10, // 設置 連接池對象最大閑置時間 (秒)
                'maxObjectNum'      => 20, // 設置 連接池最大數量
                'minObjectNum'      => 5, // 設置 連接池最小數量
                'getObjectTimeout'  => 3.0, // 設置 獲取連接池的超時時間
                'loadAverageTime'   => 0.001, // 設置負載閾值
            ]
        ]
    ]
];

修改 Test/EasySwooleEvent.php 文件,在 initialize 方法中添加如下內容,加載配置文件,

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link    http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact http://m.fe88.cn/Preface/contact.html
 * @license https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */

namespace EasySwoole\EasySwoole;

use EasySwoole\EasySwoole\AbstractInterface\Event;

class EasySwooleEvent implements Event
{
    public static function initialize()
    {
        date_default_timezone_set('Asia/Shanghai');

        // 加載配置文件
        Config::getInstance()->loadDir(EASYSWOOLE_ROOT . '/Config');
    }

    // ...
} 

進行如上配置之后,我們需要在 MySQL 服務端創建一個名為 easyswoole_demo 的數據庫,選擇字符串編碼為 utf8mb4,字符排序規則為 utf8mb4_general_ci

引入數據庫連接池組件 FastDb

執行以下命令用于實現數據庫連接池組件 FastDb 庫的引入。

composer require easyswoole/fast-db=2.x

注冊數據庫連接池

編輯 Test 項目根目錄下的 EasySwooleEvent.php 文件,在 initializemainServerCreate 事件函數中進行 FastDb 的連接池的注冊,內容如下:

<?php

namespace EasySwoole\EasySwoole;

use EasySwoole\EasySwoole\AbstractInterface\Event;
use EasySwoole\EasySwoole\Swoole\EventRegister;
use EasySwoole\FastDb\FastDb;

class EasySwooleEvent implements Event
{
    public static function initialize()
    {
        date_default_timezone_set('Asia/Shanghai');

        // 加載配置文件
        Config::getInstance()->loadDir(EASYSWOOLE_ROOT . '/Config');

        ###### 注冊 mysql orm 連接池 ######
        $mysqlConfigs = Config::getInstance()->getConf('DATABASE.MYSQL');
        foreach ($mysqlConfigs as $mysqlConfig) {
            $configObj = new \EasySwoole\FastDb\Config($mysqlConfig);
            // 【可選操作】我們已經在 DATABASE.php 中進行了配置
            # $configObj->setMaxObjectNum(20); // 配置連接池最大數量
            FastDb::getInstance()->addDb($configObj);
        }
    }

    public static function mainServerCreate(EventRegister $register)
    {
        // 或者 在此函數中注冊 和上面等價
        ###### 注冊 mysql orm 連接池 ######
        // $mysqlConfigs = Config::getInstance()->getConf('DATABASE.MYSQL');
        // foreach ($mysqlConfigs as $mysqlConfig) {
        //    $configObj = new \EasySwoole\FastDb\Config($mysqlConfig);
            // 【可選操作】我們已經在 DATABASE.php 中進行了配置
            # $configObj->setMaxObjectNum(20); // 配置連接池最大數量
        //    FastDb::getInstance()->addDb($configObj);
        // }
    }
}

initialize 事件中注冊數據庫連接池,使用這個 $config 可同時配置連接池大小等。 具體查看 FastDb 組件章節 的使用。

模型定義

管理員模型

新增管理員用戶表

easyswoole_demo 數據庫中執行如下 sql 腳本,創建管理員用戶表 admin_list

DROP TABLE IF EXISTS `admin_list`;
CREATE TABLE `admin_list` (
  `adminId` int NOT NULL AUTO_INCREMENT,
  `adminName` varchar(15) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `adminAccount` varchar(18) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `adminPassword` varchar(32) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `adminSession` varchar(32) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `adminLastLoginTime` int DEFAULT NULL,
  `adminLastLoginIp` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  PRIMARY KEY (`adminId`) USING BTREE,
  UNIQUE KEY `adminAccount` (`adminAccount`) USING BTREE,
  KEY `adminSession` (`adminSession`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

INSERT INTO `admin_list` VALUES (1, 'EasySwoole', 'easyswoole', 'e10adc3949ba59abbe56e057f20f883e', '', 1700891404, '127.0.0.1');

新增 model 文件

新建 App/Model/Admin/AdminModel.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\Model\Admin;

use App\Model\BaseModel;
use EasySwoole\FastDb\Attributes\Property;
use EasySwoole\FastDb\Beans\Query;

/**
 * Class AdminModel
 *
 * @property int    $adminId
 * @property string $adminName
 * @property string $adminAccount
 * @property string $adminPassword
 * @property string $adminSession
 * @property int    $adminLastLoginTime
 * @property string $adminLastLoginIp
 */
class AdminModel extends BaseModel
{
    #[Property(isPrimaryKey: true)]
    public int $adminId;
    #[Property]
    public string $adminName;
    #[Property]
    public string $adminAccount;
    #[Property]
    public string $adminPassword;
    #[Property]
    public string $adminSession;
    #[Property]
    public int $adminLastLoginTime;
    #[Property]
    public string $adminLastLoginIp;

    protected string $primaryKey = 'adminId';
    protected string $table = 'admin_list';

    /**
     * @getAll
     *
     * @param int         $page
     * @param null|string $keyword
     * @param int         $pageSize
     *
     * @return array[$total, $list]
     */
    public function getAll(int $page = 1, ?string $keyword = null, int $pageSize = 10): array
    {
        $where = [];
        if (!empty($keyword)) {
            $where['adminAccount'] = ['%' . $keyword . '%', 'like'];
        }

        $this->queryLimit()->page($page, true, $pageSize)
            ->orderBy($this->primaryKey, 'DESC');

        /** \EasySwoole\FastDb\Beans\ListResult $resultList */
        $resultList = $this->where($where)->all();

        $total = $resultList->totalCount();
        $list = $resultList->list();

        return ['total' => $total, 'list' => $list];
    }

    /**
     * 登錄成功后請返回更新后的bean
     */
    public function login(): ?AdminModel
    {
        $where = [
            'adminAccount'  => $this->adminAccount,
            'adminPassword' => $this->adminPassword
        ];
        return self::findRecord($where);
    }

    /**
     * 以account進行查詢
     */
    public function accountExist(array $field = ['*']): ?AdminModel
    {
        return self::findRecord(function (Query $query) use ($field) {
            $query->fields($field)
                ->where('adminAccount', $this->adminAccount);
        });
    }

    public function getOneBySession(array $field = ['*']): ?AdminModel
    {
        $this->queryLimit()->fields($field);
        $this->where(['adminSession' => $this->adminSession]);
        return $this->find();
    }

    public function logout()
    {
        $where = [$this->primaryKey => $this->adminId];
        $update = ['adminSession' => ''];
        return self::fastUpdate($where, $update);
    }
}

針對上述類似 : ?AdminModel,不懂這種函數返回值類型聲明的同學,請查看 函數返回值類型聲明,屬于 PHP 7 的新特性。

關于 Model 的定義可查看 FastDb 模型定義章節

關于 IDE 自動提示,只要你在類上面注釋中加上 @property $adminIdIDE 就可以自動提示類的這個屬性。

普通用戶模型

普通用戶模型和管理員模型同理。

建表

在數據庫中執行如下 sql 腳本,創建普通用戶表 user_list

DROP TABLE IF EXISTS `user_list`;
CREATE TABLE `user_list` (
  `userId` int NOT NULL AUTO_INCREMENT,
  `userName` varchar(32) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL,
  `userAccount` varchar(18) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL,
  `userPassword` varchar(32) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL,
  `phone` varchar(18) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL,
  `addTime` int unsigned DEFAULT '0',
  `lastLoginIp` varchar(20) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `lastLoginTime` int unsigned DEFAULT '0',
  `userSession` varchar(32) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `state` tinyint unsigned DEFAULT '0',
  `money` int unsigned NOT NULL DEFAULT '0' COMMENT '用戶余額',
  `frozenMoney` int unsigned NOT NULL DEFAULT '0' COMMENT '凍結余額',
  PRIMARY KEY (`userId`) USING BTREE,
  UNIQUE KEY `pk_userAccount` (`userAccount`) USING BTREE,
  KEY `userSession` (`userSession`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

INSERT INTO `user_list` VALUES (1, 'easyswoole', 'easyswoole', 'e10adc3949ba59abbe56e057f20f883e', '18888888888', 0, '127.0.0.1', 1700892578, '', 0, 0, 0);

新增 model 文件

新建 App/Model/User/UserModel.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\Model\User;

use App\Model\BaseModel;
use EasySwoole\FastDb\Attributes\Property;

/**
 * Class UserModel
 *
 * @property int    $userId
 * @property string $userName
 * @property string $userAccount
 * @property string $userPassword
 * @property string $phone
 * @property int    $addTime
 * @property string $lastLoginIp
 * @property int    $lastLoginTime
 * @property string $userSession
 * @property int    $state
 * @property int    $money
 * @property int    $frozenMoney
 */
class UserModel extends BaseModel
{
    protected string $table = 'user_list';
    protected string $primaryKey = 'userId';

    public const STATE_PROHIBIT = 0; // 禁用狀態
    public const STATE_NORMAL = 1; // 正常狀態

    #[Property(isPrimaryKey: true)]
    public int $userId;
    #[Property]
    public string $userName;
    #[Property]
    public string $userAccount;
    #[Property]
    public string $userPassword;
    #[Property]
    public string $phone;
    #[Property]
    public int $addTime;
    #[Property]
    public ?string $lastLoginIp;
    #[Property]
    public ?int $lastLoginTime;
    #[Property]
    public ?string $userSession;
    #[Property]
    public int $state;
    #[Property]
    public int $money;
    #[Property]
    public int $frozenMoney;

    /**
     * @getAll
     *
     * @param int         $page
     * @param string|null $keyword
     * @param int         $pageSize
     *
     * @return array[total,list]
     */
    public function getAll(int $page = 1, ?string $keyword = null, int $pageSize = 10): array
    {
        $where = [];
        if (!empty($keyword)) {
            $where['userAccount'] = ['%' . $keyword . '%', 'like'];
        }

        $this->queryLimit()->page($page, withTotalCount: true, pageSize: $pageSize)
            ->orderBy($this->primaryKey, 'DESC');
        /** \EasySwoole\FastDb\Beans\ListResult $resultList */
        $resultList = $this
            ->where($where)
            ->all();

        $total = $resultList->totalCount();
        $list = $resultList->list();

        return ['total' => $total, 'list' => $list];
    }

    public function getOneByPhone(array $field = ['*']): ?UserModel
    {
        $this->queryLimit()->fields($field);
        return $this->find(['phone' => $this->phone]);
    }

    /*
    * 登錄成功后請返回更新后的bean
    */
    public function login(): ?UserModel
    {
        return $this->find([
            'userAccount'  => $this->userAccount,
            'userPassword' => $this->userPassword
        ]);
    }

    public function getOneBySession(array $field = ['*']): ?UserModel
    {
        $this->queryLimit()->fields($field);
        return $this->find(['userSession' => $this->userSession]);
    }

    public function logout()
    {
        return $this->where([$this->primaryKey => $this->userId])->updateWithLimit(['userSession' => '']);
    }
}

banner 模型

建表

在數據中執行如下 sql 腳本,創建 bannerbanner_list

DROP TABLE IF EXISTS `banner_list`;
CREATE TABLE `banner_list` (
  `bannerId` int NOT NULL AUTO_INCREMENT,
  `bannerName` varchar(32) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `bannerImg` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci NOT NULL COMMENT 'banner圖片',
  `bannerDescription` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL,
  `bannerUrl` varchar(255) CHARACTER SET utf8mb4 COLLATE utf8mb4_general_ci DEFAULT NULL COMMENT '跳轉地址',
  `state` tinyint DEFAULT NULL COMMENT '狀態0隱藏 1正常',
  PRIMARY KEY (`bannerId`) USING BTREE
) ENGINE=InnoDB DEFAULT CHARSET=utf8mb4;

INSERT INTO `banner_list` VALUES (1, '測試banner', 'asdadsasdasd.jpg', '測試的banner數據', 'm.fe88.cn', 1);

新增 model 文件

新建 App/Model/Admin/BannerModel.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\Model\Admin;

use App\Model\BaseModel;
use EasySwoole\FastDb\Attributes\Property;

/**
 * Class BannerModel
 *
 * @property int    $bannerId
 * @property string $bannerName
 * @property string $bannerImg
 * @property string $bannerDescription
 * @property string $bannerUrl
 * @property int    $state
 */
class BannerModel extends BaseModel
{
    protected string $table = 'banner_list';
    protected string $primaryKey = 'bannerId';

    #[Property(isPrimaryKey: true)]
    public int $bannerId;
    #[Property]
    public string $bannerName;
    #[Property]
    public string $bannerImg;
    #[Property]
    public string $bannerDescription;
    #[Property]
    public string $bannerUrl;
    #[Property]
    public int $state;

    public function getAll(int $page = 1, int $state = 1, ?string $keyword = null, int $pageSize = 10): array
    {
        $where = [];
        if (!empty($keyword)) {
            $where['bannerUrl'] = ['%' . $keyword . '%', 'like'];
        }

        $where['state'] = $state;

        $this->queryLimit()->page($page, withTotalCount: true, pageSize: $pageSize)
            ->orderBy($this->primaryKey, 'DESC');
        /** \EasySwoole\FastDb\Beans\ListResult $resultList */
        $listResult = $this->where($where)->all();
        $total = $listResult->totalCount();
        $list = $listResult->list();

        return ['total' => $total, 'list' => $list];
    }
}

控制器定義

全局基礎控制器定義

新建 App/Httpcontroller/Base.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController;

use EasySwoole\EasySwoole\ServerManager;
use EasySwoole\HttpAnnotation\AnnotationController;

class Base extends AnnotationController
{
    public function index(): void
    {
        $this->actionNotFound('index');
    }

    /**
     * 獲取用戶的真實IP
     *
     * @param string $headerName 代理服務器傳遞的標頭名稱
     *
     * @return string|null
     */
    protected function clientRealIP(string $headerName = 'x-real-ip'): ?string
    {
        $server = ServerManager::getInstance()->getSwooleServer();
        $client = $server->getClientInfo($this->request()->getSwooleRequest()->fd);
        $clientAddress = $client['remote_ip'];
        $xri = $this->request()->getHeaderLine($headerName);
        $xff = $this->request()->getHeaderLine('x-forwarded-for');
        if ($clientAddress === '127.0.0.1') {
            if (!empty($xri)) {  // 如果有 xri 則判定為前端有 NGINX 等代理
                $clientAddress = $xri;
            } elseif (!empty($xff)) {  // 如果不存在 xri 則繼續判斷 xff
                $clientAddress = $xff;
            }
        }

        return $clientAddress;
    }

    protected function input(string $name, mixed $default = null)
    {
        $value = $this->request()->getRequestParam($name);
        return $value ?? $default;
    }
}

上述新增的基礎控制器 (Base.php) 里面的方法用于獲取用戶 ip,以及獲取 api 參數。

上述新增的基礎控制器 (Base.php) 繼承了 \EasySwoole\HttpAnnotation\AnnotationController ,這個是注解支持控制器,具體使用可查看 注解控制器章節

api 基礎控制器定義

新建 App/Httpcontroller/Api/ApiBase.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api;

use App\HttpController\Base;
use EasySwoole\EasySwoole\Core;
use EasySwoole\EasySwoole\Trigger;
use EasySwoole\Http\Message\Status;
use EasySwoole\HttpAnnotation\Exception\ValidateFail;

abstract class ApiBase extends Base
{
    public function index(): void
    {
        $this->actionNotFound('index');
    }

    protected function actionNotFound(?string $action): void
    {
        $this->writeJson(Status::CODE_NOT_FOUND);
    }

    public function onRequest(?string $action): ?bool
    {
        if (!parent::onRequest($action)) {
            return false;
        }
        return true;
    }

    protected function onException(\Throwable $throwable): void
    {
        if ($throwable instanceof ValidateFail) {
            $this->writeJson(400, null, $throwable->getMessage());
        } else {
            if (Core::getInstance()->runMode() === 'dev') {
                $this->writeJson(500, null, $throwable->getMessage());
            } else {
                Trigger::getInstance()->throwable($throwable);
                $this->writeJson(500, null, '系統內部錯誤,請稍后重試');
            }
        }
    }
}

上述 api 基類控制器 (ApiBase.php),用于攔截注解異常,以及 api 異常時給用戶返回一個 json 格式錯誤信息。

公共基礎控制器定義

新建 App/Httpcontroller/Api/Common/CommonBase.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\Common;

use App\HttpController\Api\ApiBase;

class CommonBase extends ApiBase
{

}

公共控制器

公共控制器放不需要登錄即可查看的控制器,例如 banner 列表查看:

新增 App/HttpController/Api/Common/Banner.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\Common;

use App\Model\Admin\BannerModel;
use EasySwoole\Http\Message\Status;
use EasySwoole\HttpAnnotation\Attributes\Api;
use EasySwoole\HttpAnnotation\Attributes\Description;
use EasySwoole\HttpAnnotation\Attributes\Param;
use EasySwoole\HttpAnnotation\Enum\HttpMethod;
use EasySwoole\HttpAnnotation\Enum\ParamFrom;
use EasySwoole\HttpAnnotation\Validator\Integer;
use EasySwoole\HttpAnnotation\Validator\MaxLength;
use EasySwoole\HttpAnnotation\Validator\Optional;
use EasySwoole\HttpAnnotation\Validator\Required;

class Banner extends CommonBase
{
    #[Api(
        apiName: 'bannerGetOne',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/common/banner/getOne',
        requestParam: [
        new Param(name: 'bannerId', from: ParamFrom::GET, validate: [
            new Required(),
            new Integer(),
        ], description: new Description('主鍵id')),
    ],
        description: 'getOne'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:20
     */
    public function getOne()
    {
        $param = $this->request()->getRequestParam();
        $model = new BannerModel();
        $bean = $model->find((int)$param['bannerId']);
        if ($bean) {
            $this->writeJson(Status::CODE_OK, $bean, 'success');
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, [], 'fail');
        }
    }

    #[Api(
        apiName: 'bannerGetAll',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/common/banner/getAll',
        requestParam: [
        new Param(name: 'page', from: ParamFrom::GET, validate: [
            new Optional(),
            new Integer(),
        ], description: new Description('頁數')),
        new Param(name: 'limit', from: ParamFrom::GET, validate: [
            new Optional(),
            new Integer(),
        ], description: new Description('每頁總數')),
        new Param(name: 'keyword', from: ParamFrom::GET, validate: [
            new Optional(),
            new MaxLength(32),
        ], description: new Description('關鍵字')),
    ],
        description: 'getAll'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:20
     */
    public function getAll()
    {
        $param = $this->request()->getRequestParam();
        $page = (int)$this->input('page', 1);
        $limit = (int)$this->input('limit', 20);
        $model = new BannerModel();
        $data = $model->getAll($page, 1, $param['keyword'] ?? null, $limit);
        $this->writeJson(Status::CODE_OK, $data, 'success');
    }
}

注意:可以看到,在上文 getAll 方法中,有個特殊的寫法 requestParam: [new Param(name: 'page', from: ParamFrom::GET, validate: [new Optional(), new Integer(),], description: new Description('頁數')), 這是 php8 支持的注解寫法,類似 Java 語言的注解,當使用這個注解之后,將會約束 page 參數必須是 int,具體的驗證機制可查看 validate 驗證器 章節。框架中如何使用注解請查看 注解控制器章節

使用 php easyswoole.php server start 命令啟動框架服務之后,訪問鏈接:http://localhost:9501/api/common/banner/getAll (示例訪問地址) 即可看到如下結果:{"code":200,"result":{"total":1,"list":[{"bannerId":1,"bannerName":"測試banner","bannerImg":"asdadsasdasd.jpg","bannerDescription":"測試的banner數據","bannerUrl":"m.fe88.cn","state":1}]},"msg":"success"} (需要有數據才能看到具體輸出)。

管理員基礎控制器定義

新建 App/HttpController/Api/Admin/AdminBase.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\Admin;

use App\Model\Admin\AdminModel;
use App\HttpController\Api\ApiBase;
use EasySwoole\Http\Message\Status;

class AdminBase extends ApiBase
{
    // public 才會根據協程清除
    public ?AdminModel $who;

    // session 的 cookie頭
    protected string $sessionKey = 'adminSession';

    // 白名單
    protected array $whiteList = [];

    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:28
     */
    public function onRequest(?string $action): ?bool
    {
        if (parent::onRequest($action)) {
            // 白名單判斷
            if (in_array($action, $this->whiteList)) {
                return true;
            }

            // 獲取登錄信息
            if (!$this->getWho()) {
                $this->writeJson(Status::CODE_UNAUTHORIZED, '', '登錄已過期');
                return false;
            }

            return true;
        }

        return false;
    }

    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:28
     */
    protected function getWho(): ?AdminModel
    {
        if (isset($this->who) && $this->who instanceof AdminModel) {
            return $this->who;
        }

        $sessionKey = $this->request()->getRequestParam($this->sessionKey);
        if (empty($sessionKey)) {
            $sessionKey = $this->request()->getCookieParams($this->sessionKey);
        }

        if (empty($sessionKey)) {
            return null;
        }

        $adminModel = new AdminModel();
        $adminModel->adminSession = $sessionKey;
        $this->who = $adminModel->getOneBySession();

        return $this->who;
    }
}

管理員登錄控制器

新建 App/HttpController/Api/Admin/Auth.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\Admin;

use App\Model\Admin\AdminModel;
use EasySwoole\FastDb\Exception\RuntimeError;
use EasySwoole\Http\Message\Status;
use EasySwoole\HttpAnnotation\Attributes\Api;
use EasySwoole\HttpAnnotation\Attributes\Description;
use EasySwoole\HttpAnnotation\Attributes\Param;
use EasySwoole\HttpAnnotation\Enum\HttpMethod;
use EasySwoole\HttpAnnotation\Enum\ParamFrom;
use EasySwoole\HttpAnnotation\Exception\Annotation;
use EasySwoole\HttpAnnotation\Validator\MaxLength;
use EasySwoole\HttpAnnotation\Validator\MinLength;
use EasySwoole\HttpAnnotation\Validator\Required;

class Auth extends AdminBase
{
    protected array $whiteList = ['login'];

    /**
     * @return void
     * @throws RuntimeError
     * @throws Annotation
     */
    #[Api(
        apiName: 'login',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/auth/login',
        requestParam: [
        new Param(name: 'account', from: ParamFrom::GET, validate: [
            new Required(),
            new MaxLength(20)
        ], description: new Description('帳號')),
        new Param(name: 'password', from: ParamFrom::GET, validate: [
            new Required(),
            new MinLength(6),
            new MaxLength(16),
        ], description: new Description('密碼')),
    ],
        description: '登陸,參數驗證注解寫法'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:31
     */
    public function login()
    {
        $param = $this->request()->getRequestParam();
        $model = new AdminModel();
        $model->adminAccount = $param['account'];
        $model->adminPassword = md5($param['password']);

        if ($user = $model->login()) {
            $sessionHash = md5(time() . $user->adminId);
            $user->updateWithLimit([
                'adminLastLoginTime' => time(),
                'adminLastLoginIp'   => $this->clientRealIP(),
                'adminSession'       => $sessionHash
            ]);

            $rs = $user->toArray();
            unset($rs['adminPassword']);
            $rs['adminSession'] = $sessionHash;
            $this->response()->setCookie('adminSession', $sessionHash, time() + 3600, '/');
            $this->writeJson(Status::CODE_OK, $rs, 'success');
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, '', '密碼錯誤');
        }
    }

    #[Api(
        apiName: 'logout',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/auth/logout',
        requestParam: [
        new Param(name: 'adminSession', from: ParamFrom::COOKIE, validate: [
            new Required(),
        ], description: new Description('帳號')),
    ],
        description: '退出登錄,參數注解寫法'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:32
     *
     * @throws Annotation
     */
    public function logout()
    {
        $sessionKey = $this->request()->getRequestParam($this->sessionKey);
        if (empty($sessionKey)) {
            $sessionKey = $this->request()->getCookieParams('adminSession');
        }

        if (empty($sessionKey)) {
            $this->writeJson(Status::CODE_UNAUTHORIZED, '', '尚未登錄');
            return false;
        }

        $result = $this->getWho()->logout();
        if ($result) {
            $this->writeJson(Status::CODE_OK, '', '退出登錄成功');
        } else {
            $this->writeJson(Status::CODE_UNAUTHORIZED, '', 'fail');
        }
    }

    #[Api(
        apiName: 'getInfo',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/auth/getInfo',
        description: '獲取管理員信息'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:34
     */
    public function getInfo()
    {
        $this->writeJson(200, $this->getWho()->toArray(), 'success');
    }
}

使用 php easyswoole.php server start 命令啟動框架服務之后,訪問鏈接:http://localhost:9501/api/admin/auth/login?account=easyswoole&password=123456 (示例訪問地址) 即可返回如下結果:``

{
  "code": 200,
  "result": {
    "adminId": 1,
    "adminName": "EasySwoole",
    "adminAccount": "easyswoole",
    "adminSession": "7262e8188ae9885e27e092538c08ca16",
    "adminLastLoginTime": 1706271125,
    "adminLastLoginIp": "127.0.0.1"
  },
  "msg": "success"
}

管理員用戶管理控制器

新增 App/httpController/Api/Admin/User.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\Admin;

use App\Model\User\UserModel;
use EasySwoole\Http\Message\Status;
use EasySwoole\HttpAnnotation\Attributes\Api;
use EasySwoole\HttpAnnotation\Attributes\Description;
use EasySwoole\HttpAnnotation\Attributes\Param;
use EasySwoole\HttpAnnotation\Enum\HttpMethod;
use EasySwoole\HttpAnnotation\Enum\ParamFrom;
use EasySwoole\HttpAnnotation\Validator\InArray;
use EasySwoole\HttpAnnotation\Validator\Integer;
use EasySwoole\HttpAnnotation\Validator\IsPhoneNumber;
use EasySwoole\HttpAnnotation\Validator\MaxLength;
use EasySwoole\HttpAnnotation\Validator\MinLength;
use EasySwoole\HttpAnnotation\Validator\Optional;
use EasySwoole\HttpAnnotation\Validator\Required;

class User extends AdminBase
{
    #[Api(
        apiName: 'userGetAll',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/user/getAll',
        requestParam: [
        new Param(name: 'page', from: ParamFrom::GET, validate: [
            new Optional(),
            new Integer(),
        ], description: new Description('頁數')),
        new Param(name: 'limit', from: ParamFrom::GET, validate: [
            new Optional(),
            new Integer(),
        ], description: new Description('每頁總數')),
        new Param(name: 'keyword', from: ParamFrom::GET, validate: [
            new Optional(),
            new MaxLength(32),
        ], description: new Description('關鍵字')),
    ],
        description: 'getAll'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:38
     */
    public function getAll()
    {
        $page = (int)$this->input('page', 1);
        $limit = (int)$this->input('limit', 20);
        $model = new UserModel();
        $data = $model->getAll($page, $this->input('keyword'), $limit);
        $this->writeJson(Status::CODE_OK, $data, 'success');
    }

    #[Api(
        apiName: 'userGetOne',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/user/getOne',
        requestParam: [
        new Param(name: 'userId', from: ParamFrom::GET, validate: [
            new Required(),
            new Integer(),
        ], description: new Description('戶id')),
    ],
        description: 'getAll'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:40
     */
    public function getOne()
    {
        $param = $this->request()->getRequestParam();
        $model = new UserModel();
        $rs = $model->find((int)$param['userId']);
        if ($rs) {
            $this->writeJson(Status::CODE_OK, $rs, 'success');
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, [], 'fail');
        }
    }

    #[Api(
        apiName: 'addUser',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/user/add',
        requestParam: [
        new Param(name: 'userName', from: ParamFrom::GET, validate: [
            new Optional(),
            new MaxLength(32),
        ], description: new Description('用戶昵稱')),
        new Param(name: 'userAccount', from: ParamFrom::GET, validate: [
            new Required(),
            new MaxLength(32),
        ], description: new Description('用戶名')),
        new Param(name: 'userPassword', from: ParamFrom::GET, validate: [
            new Required(),
            new MinLength(6),
            new MaxLength(18),
        ], description: new Description('用戶密碼')),
        new Param(name: 'phone', from: ParamFrom::GET, validate: [
            new Optional(),
            new IsPhoneNumber(),
        ], description: new Description('手機號碼')),
        new Param(name: 'state', from: ParamFrom::GET, validate: [
            new Optional(),
            new InArray([0, 1]),
        ], description: new Description('用戶狀態')),
    ],
        description: 'add'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:44
     */
    public function add()
    {
        $param = $this->request()->getRequestParam();
        $model = new UserModel($param);
        $model->userPassword = md5($param['userPassword']);
        $rs = $model->insert();
        if ($rs) {
            $this->writeJson(Status::CODE_OK, $rs, 'success');
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, [], 'add fail');
        }
    }

    #[Api(
        apiName: 'updateUser',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/user/update',
        requestParam: [
        new Param(name: 'userId', from: ParamFrom::GET, validate: [
            new Required(),
            new Integer(),
        ], description: new Description('用戶id')),
        new Param(name: 'userPassword', from: ParamFrom::GET, validate: [
            new Optional(),
            new MinLength(6),
            new MaxLength(18),
        ], description: new Description('會員密碼')),
        new Param(name: 'userName', from: ParamFrom::GET, validate: [
            new Optional(),
            new MaxLength(32),
        ], description: new Description('會員名')),
        new Param(name: 'state', from: ParamFrom::GET, validate: [
            new Optional(),
            new InArray([0, 1]),
        ], description: new Description('狀態')),
        new Param(name: 'phone', from: ParamFrom::GET, validate: [
            new Optional(),
            new IsPhoneNumber(),
        ], description: new Description('手機號')),
    ],
        description: 'update'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:50
     */
    public function update()
    {
        $model = new UserModel();
        $userId = $this->input('userId');
        /**
         * @var $userInfo UserModel
         */
        $userInfo = $model->find((int)$userId);
        if (!$userInfo) {
            $this->writeJson(Status::CODE_BAD_REQUEST, [], '未找到該會員');
            return false;
        }

        $password = $this->input('userPassword');
        $update = [
            'userName'     => $this->input('userName', $userInfo->userName),
            'userPassword' => $password ? md5($password) : $userInfo->userPassword,
            'state'        => $this->input('state', $userInfo->state),
            'phone'        => $this->input('phone', $userInfo->phone),
        ];

        $rs = $userInfo->updateWithLimit($update);
        if ($rs === 0 || $rs === 1) {
            $this->writeJson(Status::CODE_OK, $rs, 'success');
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, [], 'update fail');
        }
    }

    #[Api(
        apiName: 'deleteUser',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/admin/user/delete',
        requestParam: [
        new Param(name: 'userId', from: ParamFrom::GET, validate: [
            new Required(),
            new Integer(),
        ], description: new Description('用戶id')),
    ],
        description: 'delete'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:55
     */
    public function delete()
    {
        $param = $this->request()->getRequestParam();
        $model = new UserModel(['userId' => $param['userId']]);
        $rs = $model->delete();
        if ($rs) {
            $this->writeJson(Status::CODE_OK, $rs, 'success');
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, [], '刪除失敗');
        }
    }
}

后臺管理員登錄之后,可通過此文件的接口,去進行會員的增刪改查操作 (即 CURD)。

請求地址為:(示例訪問地址) http://127.0.0.1:9501/Api/Admin/User/getAll (等方法)

普通用戶基礎控制器定義

新增 App/HttpController/Api/User/UserBase.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\User;

use App\Model\User\UserModel;
use App\HttpController\Api\ApiBase;
use EasySwoole\Http\Message\Status;

class UserBase extends ApiBase
{
    protected ?UserModel $who;

    // session 的 cookie 頭
    protected string $sessionKey = 'userSession';

    // 白名單
    protected array $whiteList = ['login', 'register'];

    public function onRequest(?string $action): ?bool
    {
        if (parent::onRequest($action)) {
            // 白名單判斷
            if (in_array($action, $this->whiteList)) {
                return true;
            }

            // 獲取登錄信息
            if (!$data = $this->getWho()) {
                $this->writeJson(Status::CODE_UNAUTHORIZED, '', '登錄已過期');
                return false;
            }

            // 刷新 cookie 存活
            $this->response()->setCookie($this->sessionKey, $data->userSession, time() + 3600, '/');

            return true;
        }

        return false;
    }

    public function getWho(): ?UserModel
    {
        if (isset($this->who) && $this->who instanceof UserModel) {
            return $this->who;
        }

        $sessionKey = $this->request()->getRequestParam($this->sessionKey);
        if (empty($sessionKey)) {
            $sessionKey = $this->request()->getCookieParams($this->sessionKey);
        }

        if (empty($sessionKey)) {
            return null;
        }

        $userModel = new UserModel();
        $userModel->userSession = $sessionKey;
        $this->who = $userModel->getOneBySession();

        return $this->who;
    }
}

普通用戶登錄控制器

新增 App/HttpController/Api/User/Auth.php 文件,編輯內容如下:

<?php
/**
 * This file is part of EasySwoole.
 *
 * @link     http://m.fe88.cn
 * @document http://m.fe88.cn
 * @contact  http://m.fe88.cn/Preface/contact.html
 * @license  https://github.com/easy-swoole/easyswoole/blob/3.x/LICENSE
 */
declare(strict_types=1);

namespace App\HttpController\Api\User;

use App\Model\User\UserModel;
use EasySwoole\Http\Message\Status;
use EasySwoole\HttpAnnotation\Attributes\Api;
use EasySwoole\HttpAnnotation\Attributes\Description;
use EasySwoole\HttpAnnotation\Attributes\Param;
use EasySwoole\HttpAnnotation\Enum\HttpMethod;
use EasySwoole\HttpAnnotation\Enum\ParamFrom;
use EasySwoole\HttpAnnotation\Validator\MaxLength;
use EasySwoole\HttpAnnotation\Validator\MinLength;
use EasySwoole\HttpAnnotation\Validator\Optional;
use EasySwoole\HttpAnnotation\Validator\Required;

class Auth extends UserBase
{
    protected array $whiteList = ['login'];

    #[Api(
        apiName: 'login',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/user/auth/login',
        requestParam: [
        new Param(name: 'userAccount', from: ParamFrom::GET, validate: [
            new Required(),
            new MaxLength(32)
        ], description: new Description('用戶名')),
        new Param(name: 'userPassword', from: ParamFrom::GET, validate: [
            new Required(),
            new MinLength(6),
            new MaxLength(18),
        ], description: new Description('密碼')),
    ],
        description: 'login'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 00:31
     */
    public function login()
    {
        $param = $this->request()->getRequestParam();
        $model = new UserModel();
        $model->userAccount = $param['userAccount'];
        $model->userPassword = md5($param['userPassword']);

        if ($userInfo = $model->login()) {
            $sessionHash = md5(time() . $userInfo->userId);
            $userInfo->updateWithLimit([
                'lastLoginIp'   => $this->clientRealIP(),
                'lastLoginTime' => time(),
                'userSession'   => $sessionHash
            ]);
            $rs = $userInfo->toArray();
            unset($rs['userPassword']);
            $rs['userSession'] = $sessionHash;
            $this->response()->setCookie('userSession', $sessionHash, time() + 3600, '/');
            $this->writeJson(Status::CODE_OK, $rs);
        } else {
            $this->writeJson(Status::CODE_BAD_REQUEST, '', '密碼錯誤');
        }
    }

    #[Api(
        apiName: 'logout',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/user/auth/logout',
        requestParam: [
        new Param(name: 'userSession', from: ParamFrom::GET, validate: [
            new Optional(),
        ], description: new Description('用戶會話')),
    ],
        description: 'logout'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 01:07
     */
    public function logout()
    {
        $sessionKey = $this->request()->getRequestParam('userSession');
        if (empty($sessionKey)) {
            $sessionKey = $this->request()->getCookieParams('userSession');
        }

        if (empty($sessionKey)) {
            $this->writeJson(Status::CODE_UNAUTHORIZED, '', '尚未登錄');
            return false;
        }

        $result = $this->getWho()->logout();
        if ($result) {
            $this->writeJson(Status::CODE_OK, '', '退出登錄成功');
        } else {
            $this->writeJson(Status::CODE_UNAUTHORIZED, '', 'fail');
        }
    }

    #[Api(
        apiName: 'getInfo',
        allowMethod: HttpMethod::GET,
        requestPath: '/api/user/auth/getInfo',
        description: 'getInfo'
    )]
    /**
     * Author: XueSi <hui.huang8540@gmail.com>
     * Time: 01:10
     */
    public function getInfo()
    {
        $this->writeJson(200, $this->getWho(), 'success');
    }
}

訪問 http://localhost:9501/api/user/auth/login?userAccount=easyswoole&userPassword=456789 即可登錄成功。

接口訪問

上述 demo 中的所有接口均使用 GET 請求,可以在啟動 easyswoole 服務后可使用瀏覽器訪問如下 URL 進行體驗:

# Admin 管理員模塊
## auth 模塊
- login 登錄
  - http://localhost:9501/api/admin/auth/login?account=easyswoole&password=123456

- logout
  - http://localhost:9501/api/admin/auth/logout

- getInfo
  - http://localhost:9501/api/admin/auth/getInfo

## user manager 會員管理模塊
- get all user
  - http://localhost:9501/api/admin/user/getAll
  - http://localhost:9501/api/admin/user/getAll?page=1&limit=2
  - http://localhost:9501/api/admin/user/getAll?keyword=easyswoole

- get one user
  - http://localhost:9501/api/admin/user/getOne?userId=1

- add user
  - http://localhost:9501/api/admin/user/add?userName=EasySwoole1&userAccount=easyswoole1&userPassword=123456

- update user
  - http://localhost:9501/api/admin/user/update?userId=1&userPassword=456789&userName=easyswoole&state=0&phone=18888888889

- delete user
  - http://localhost:9501/api/admin/user/delete?userId=2

# Common 公共模塊
## banner 模塊
- get one banner 讀取一條banner
  - http://localhost:9501/api/common/banner/getOne?bannerId=1

- get all banner
  - http://localhost:9501/api/common/banner/getAll
  - http://localhost:9501/api/common/banner/getAll?page=1&limit=2
  - http://localhost:9501/api/common/banner/getAll?keyword=easyswoole

# User 會員模塊
- user login
  - http://localhost:9501/api/user/auth/login?userAccount=easyswoole&userPassword=456789

- get user info
  - http://localhost:9501/api/user/auth/getInfo

- logout
  - http://localhost:9501/api/user/auth/logout
主站蜘蛛池模板: 综合激情在线 | 午夜免费国产体验区免费的 | 欧美日韩欧美日韩在线观看视频 | 国产普通话bbwbbwbbw | 亚洲一区二区三区精品视频 | 国产伦精品一区二区三区免费 | 亚洲中文字幕无码一区 | 久久av综合 | 92久久精品一区二区 | 天天爽夜夜爽夜夜爽 | 成年人在线免费观看 | 久久久av亚洲男天堂 | 五月婷婷六月天 | 国产太嫩了在线观看 | 久久久网| 国产精品-区区久久久狼 | 九九99靖品| 99久久精品免费看国产 | 国产精品4p| 久久不见久久见免费影院www日本 | 久草免费av | 亚洲欧美一级久久精品国产特黄 | 国产午夜精品久久久久久 | 日韩在线免费视频 | 天天草天天爱 | 九七影院在线观看免费观看电视 | 国产v亚洲v天堂无码久久久 | 少妇视频在线播放 | 免费国产又色又爽又黄的软件 | 91丨九色丨蝌蚪丰满 | 午夜三级做爰视频在线看 | 日本人又黄又爽又大又色 | 欧美午夜精品久久久久久蜜 | 免费看一级特黄a大片 | 神宫寺奈绪一区二区三区 | 一本精品999爽爽久久久 | 男女日批免费视频 | 精品国产鲁一鲁一区二区张丽 | 亚洲国产精品成人无久久精品 | 欧美精品偷自拍另类在线观看 | 国产精品色片 | 懂色一区二区三区av片 | 免费精品午夜 | 欧美男女交配 | 久久网站免费看 | а√天堂资源官网在线资源 | 制中文字幕音影 | 国产猛烈尖叫高潮视频免费 | 国模妙妙超大尺度啪啪人体 | 久久久wwww | wwwxxxxx日本| 在线欧美a | jizz国产视频| 久久精品国产中国久久 | 亚洲色图清纯唯美 | 天天综合天天做 | 欧美日韩亚洲高清 | 麻豆秘密入口a毛片 | 我想看一级黄色毛片 | 日日噜噜夜夜狠狠久久波多野 | 欧美乱大交xxxxx疯狂俱乐部 | 一级作爱视频 | 午夜免费视频 | 少妇久久久久久被弄到高潮 | 久久精品99国产精品酒店日本 | 韩日三级视频 | 爱爱免费视频网站 | www国产无套内射com | 欧美性生交xxxxx久久久 | 日韩a∨精品日韩在线观看 免费特级黄毛片 | 特级大胆西西4444人体 | 污污的视频在线免费观看 | 国产乱色 | 久久久午夜爽爽一区二区三区三州 | 天天干视频网站 | 久久999视频 | 午夜剧场福利社 | 色av性av丰满av国产 | 丰满少妇夜夜爽爽高潮水网站 | 99精品欧美一区二区三区视频 | 日韩精品中字 | 中国挤奶哺乳午夜片 | 农民人伦一区二区三区剧情简介 | 蜜桃传媒 | 国产在线麻豆精品入口 | 伊人av影院 | 日本欧美韩国国产精品 | 久久一区 | 影音先锋久久 | 亚洲色图av在线 | 成人毛片免费网站 | 精品国产依人香蕉在线精品 | 91九色高潮 | 蜜桃av噜噜一区二区三 | 99久久99这里只有免费费精品 | 99自拍偷拍| 人妻激情文学 | 黄色大片网站 | 中文字幕无线码 | 中文字幕在线亚洲日韩6页 av无码精品一区二区三区 | 国产女人乱人伦精品一区二区 | 精品国产乱码久久久久久牛牛 | 欧美成人ⅴideosxxxxx | 蜜臀91精品国产免费观看 | 国产精品对白久久久久粗 | 777米奇影视第四色 韩产日产国产欧产 | 99综合视频| 欧美日韩成人在线视频 | 亚洲成成品牛牛 | 午夜久久久久久久久 | 国产精品永久免费 | 国产在线拍偷自揄拍无码 | 插插插综合视频 | 亚洲国产成人va在线观看天堂 | 中文字幕视频免费观看 | 亚洲地区一二三色 | 日韩视频在线观看一区二区三区 | 欧美一区二区三区网站 | 影音先锋人妻啪啪av资源网站 | 国产成人高清精品免费 | 操操操干干干 | 小龙女娇喘呻吟啊快点 | 亚洲国产婷婷香蕉久久久久久 | 亚洲老女人 | 日韩在线观看精品 | 成人影院免费 | 亚洲成人精品av | 亚洲国产精品美女久久久久 | 中国极品少妇xxxxx | 利智三级露全乳 | 人妻大战黑人白浆狂泄 | 国产97自拍 | 精品国产一区二区三区日日嗨 | 久久久久久久久久久久久久 | 欧美性生交xxxxx久久久缅北 | 人人澡 人人澡 人人看 | 久久婷婷五月综合97色 | 夜夜夜网站 | 亚洲影视中文字幕 | 国产高清在线精品一区不卡 | 网站一区二区 | 日韩一区二区精品葵司在线 | 天堂视频免费看 | 91久久国产成人精品 | 99精品视频免费版的特色功能 | 污污的视频在线免费观看 | 天堂а√8在线最新版在线 天堂а√在线地址 | 久久波多野 | av在线地址 | 6~12呦孩精品xxxx视频视频 | 内射白嫩少妇超碰 | 女同av久久中文字幕字 | jizz18国产| 日本一区二区在线视频 | 91麻豆精品国产 | 中国美女囗交视频免费看 | 特级黄色毛片视频 | 白嫩嫩翘臀美女在线视频 | 波多野结衣在线视频免费观看 | 中文字幕在线日本 | 亚洲丁香网 | 国精产品一区一区三区免费完 | 久久久91精品国产一区二区精品 | 无码国内精品久久人妻 | 老司机深夜福利在线观看 | 亚洲精品少妇 | 欧美日韩视频 | 国产在线精品一区二区夜色 | 色偷偷偷久久伊人大杳蕉 | 九九热视频免费观看 | 久一区二区三区 | 黄色录像大片 | 午夜视频久久久 | 翘臀后进娇喘呻吟的少妇91 | 国产二级一片内射视频插放 | ga∨成人网 | 久久久中文字幕日本无吗 | 久久综合导航 | 少妇高潮毛片高清免费播放 | 亚洲中文字幕av在天堂 | 自拍偷拍免费 | 91在线观 | 一边捏奶头一边高潮视频 | 亚洲最大成人综合网 | 亚洲精品萌白酱一区 | 青娱乐欧美 | missav | 免费高清av在线看 | 富二代成人短视频 | 亚洲国产视频一区二区三区 | 欧美人与牲动xxxx | 亚洲天堂一区二区三区 | 特黄一级片 | 粉色午夜视频 | 强行挺进熟睡少妇av | 国产露脸国语对白在线 | 少妇饥渴难耐 | 亚洲成人伊人 | 91ts人妖另类精品系列 | 伊人中文字幕在线观看 | 久久国产avjust麻豆 | 18国产一二三精品国产 | 91欧美精品午夜性色福利在线 | 精品夜夜爽欧美毛片视频 | 国产午夜性爽视频男人的天堂 | 富婆找两个黑人3p在线视频 | 国产日韩av在线播放 | 天天做天天爱夜夜爽 | 亚洲女人天堂网 | 日批的视频 | 香蕉视频网址 | 农村妇女毛片 | 夜夜爱夜夜操 | 精品女同一区二区三区在线 | 另类国产ts人妖高潮系列视频 | 国产午夜亚洲精品不卡下载 | 6080日韩午夜伦伦午夜伦 | 亚洲国产精品久久久久婷蜜芽 | 国产69久久精品成人看动漫 | 亚洲 欧美 日韩 综合 | 久久久久久久久久国产精品 | 在线天堂资源www在线污 | 日韩人妻中文无码一区二区七区 | 日本一级二级三级aⅴ网站 日本一级二级三级久久久 日本一级二级视频 | 黄色一级视频 | 中文字幕乱码一区av久久不卡 | 欧美a级在线免费观看 | 中文字幕日本最新乱码视频 | 真实国产老熟女粗口对白 | 久热精品视频在线播放 | 国产精品视频专区 | 国内毛片精品av一二三 | 久久久久se色偷偷亚洲精品av | 国产 日韩 欧美 制服 另类 | 九九九在线观看 | 高清二区| 国产精品一区不卡 | 欧美15一16性娇小高清 | 亚洲成人在线播放视频 | 亚洲国产无线乱码在线观看 | 日韩精品一区二区三区在线观看视频网站 | 色噜噜狠狠狠综合曰曰曰 | 高清视频在线播放 | 国产在线色| 天啪 | 国产刺激出水片 | 国产suv精品一区二区60 | 蜜乳av一区二区 | 久久久99精品 | 国产极品福利 | 老子影院午夜精品无码 | 久久综合另类激情人妖 | 日韩蜜桃视频 | а√资源新版在线天堂 | 国产精品久久久久一区二区三区 | 国产精品96久久久久久 | 国产免费又色又爽粗视频 | 国产美女精品人人做人人爽 | 欧洲av一区二区 | 精品久久久久久中文字幕2017 | 色综合另类小说图片区 | 国内精品国产三级国产a久久 | 区二区三区玖玖玖 | 色综合欧美在线视频区 | 国产精品片aa在线观看 | 精品国模一区二区三区 | 美欧一级片 | 一本综合久久 | 草青青视频 | 亚洲女同志亚洲女同女播放 | 超碰在线免费公开 | 波多野结衣一区二区三区 | 欧美日韩一区二区三区69堂 | 中文字幕日韩精品在线观看 | 一区二区日韩 | 久久国产精品视频一区 | 国产女人爽的流水毛片 | 亚洲精品一区二区三区四区乱码 | 祥仔av大片av免费看 | 麻豆久久久9性大片 | 色婷婷狠 | 综合无码一区二区三区 | 在线理论片 | 欧美久操| 国产性猛交xx乱 | 亚洲人成电影网站色mp4 | 久久久亚洲欧洲 | 波多野吉衣在线观看视频 | 欧美日韩国产传媒 | 男女一进一出超猛烈的视频不遮挡在线观看 | 精品无码国产av一区二区三区 | 狠狠躁夜夜躁人人爽天天5 中国china露脸自拍性hd | 国产精品国产馆在线真实露脸 | 性欧美视频在线观看 | 午夜影院色 | 日本成人在线播放 | 奇米影视第4色 | 免费av手机在线观看 | 1515hh成人免费看 | 两个女人互相吃奶摸下面 | 91精品国产91久久久久 | 人妻少妇乱子伦精品 | 福利在线一区 | 日韩欧美精品在线观看 | 中文字幕久久av | 日韩欧美一级视频 | 欧美肥老太牲交大战 | 女人被狂躁60分钟视频 | 中日韩精品视频在线观看 | 日日噜噜噜噜人人爽亚洲精品 | 亚洲一区二区三区免费视频 | 午夜爱爱免费视频体验区 | 最近中文字幕在线观看视频 | 少妇人妻大乳在线视频 | 99成人在线视频 | 人妻饥渴偷公乱中文字幕 | 久久精品色| 少妇人妻偷人精品视蜜桃 | 成人黄色激情小说 | 国产二级视频 | 少妇高潮太爽了在线视频 | 日韩av激情在线观看 | 色噜噜狠狠一区二区 | 久久成人久久爱 | 久久人人爽人人爽人人片av高请 | 亚洲高清av一区二区三区 | 久久这里只有精品视频9 | 揉捏奶头高潮呻吟视频 | 久久国产精品偷 | 鲁一鲁av2019在线 | 国自产拍偷拍福利精品免费一 | 久久精品人人做人人爽97 | 播播激情网 | 婷婷五月花| 国产视频观看 | 国产精品激情 | 爱情岛论坛线路一区二区 | 污视频在线免费观看 | 免费毛片看片 | 国产女人18水真多18精品一级做 | 亚洲精品一区av在线播放 | 国产精品久久久久久久蜜臀 | 天堂网av2014| 91av视频在线免费观看 | 国产手机在线αⅴ片无码观看 | 少妇呻吟白浆高潮啪啪69 | 亚洲第一视频在线观看 | 色噜噜日韩精品欧美一区二区 | 青娱乐超碰 | 乱色精品无码一区二区国产盗 | 成人综合网址 | 日本少妇色视频 | 日韩成人av网址 | 成人无码视频在线观看网站 | 久热国产精品视频一区二区三区 | 无套内射极品少妇chinese | 上原亚衣av一区二区三区 | 国产91在线观 | 正在播放凉森玲梦88av | 性色欲情网站iwww | 好看的av在线 | 国产伦精品一区二区三区免费 | 大地资源网第二页免费观看 | 狠狠操伊人| 天天狠天天插天天透 | 久久免费在线观看视频 | 男人的天堂日韩 | 国产无遮挡裸体免费直播 | 色哟哟在线免费观看 | 足疗店女技师按摩毛片 | 日韩人妻ol丝袜av一二区 | 综合在线一区 | 特黄三级又爽又粗又大 | 中文字幕天堂在线 | 欧美日韩亚洲系列 | 欧美日韩精品一区二区三区四区 | 日韩毛片在线看 | 四虎国产精品免费观看视频优播 | 91成人精品一区在线播放69 | 二级特黄绝大片免费视频大片 | 橘梨纱av一区二区三区在线观看 | 亚洲男人精品 | 男人的天堂手机在线 | 亚洲va欧美va | 一级片aaaaa 一级片av | 日本一区二区三区在线免费观看 | 久久久久亚洲国产av麻豆 | 天堂va蜜桃 | 国产免费观看久久黄av片 | 中文字幕二十三页2 | 久久人人爱 | 羞羞视频在线观看免费观看 | 毛片网站网址 | 香蕉久久福利院 | 日韩欧美精品免费 | ass艳妇猛性bbwbbw1 | 国产第3页 | 欧美高清日韩 | 爽妇网av| 美女撒尿无遮挡网站 | 日本xxxx18高清hd | 国产69久久精品成人看动漫 | 91社区在线观看高清 | 国产特级av| 污视频网站在线看 | 久久久人体 | 美女的尿囗网站免费 | 不卡三区 | 每日更新在线观看av | 国产精品69午夜妇大片 | 黄色a一级视频 | 日韩视频免费大全中文字幕 | www一起操 | 18pao国产成人免费视频 | 色屁屁xxxxⅹ免费视频 | 欧美小视频在线观看 | 亚洲久久视频 | 一级二级三级毛片 | 国语对白一区 | 手机av观看| a级黄色录像片 | 久久午夜夜伦鲁鲁片无码免费 | 99国产在线观看 | 国产网站免费在线观看 | 国产av天堂无码一区二区三区 | 夜夜嗨av一区二区三区 | 欧美性猛交久久久乱大交小说 | 久久久久久久久久久99 | 亚洲精品毛片av | 久久99综合 | 躁躁躁日日躁 | 色综合视频二区偷拍在线 | jizzxxxx18高清喷水 | 男女作爱免费网站 | 和尚与寡妇在线三级 | 永久免费的av在线电影网 | 亚洲激情自拍偷拍 | 欧美交换国产一区内射 | 久久99久久精品 | 成人做爰在线观看 | 美女调教网站18+ | 国产aⅴ夜夜欢一区二区三区 | 精品国产乱码久久久久久芒果 | av在线免费网站 | 草草黑森林av导航 | 亚洲地区天堂网 | 欧美精品一区二区蜜臀亚洲 | 四虎国产精品成人免费影视 | 毛片入口| 亚洲码国产精品高潮在线 | 少妇性l交大片7724com | 蜜桃视频在线观看www | 久久久久久九九九九九 | 亚洲国产成人欧美激情 | 中文字幕在线观看日本 | 少妇激情一区二区三区视频小说 | 日本在线h | 狠狠色香婷婷久久亚洲精品 | 樱桃成人精品视频在线播放 | 欧美一级特黄aa大片 | 精产国品一二三产区区别在线观看 | 成人免费xxxxx在线观看 | 男人爽女人下面动态图 | 丰满少妇人妻无码专区 | 极品销魂美女少妇尤物优美 | 中文字幕ipx696希岛あい | 欧美大片高清 | 天天看片天天操 | 麻豆疯狂做受xxxx高潮视频 | 亚洲视频在线观看一区 | 99热在| 日批视频在线免费看 | 久久精品视频在线免费观看 | www成年人 | 亚洲福利精品视频 | 国产精品ssss在线亚洲 | 女人裸体特黄做爰的视频 | 99久久精品日本一区二区免费 | 色八区人妻在线视频 | 欧洲综合色 | 免费欧美 | 97精品国产97久久久久久春色 | 亚洲精品一区二区三区98年 | aa成人免费视频 | 久久亚洲国产成人精品性色 | 激情五月婷婷丁香 | 国产淫语对白粗口video | 人人妻久久人人澡人人爽人人精品 | 亚洲精品人成无码中文毛片 | 日韩精品 欧美 | 欧美美女性高潮 | 中文无码日韩欧 | 啦啦啦www播放日本观看 | 中文在线观看免费高清 | 天天干欧美 | 国产午夜精品一区理论片飘花 | 国产女人18毛片水18精品 | 国产精品入口牛牛影视 | 清清草视频 | 国产精品久久久久久久久夜色 | 亚洲视频在线免费 | 无码人妻精品一区二区 | 男女天堂av| 亚洲黄色网址大全 | 五月婷婷视频 | 人成午夜 | 亚洲人色婷婷成人网站在线观看 | 成人区人妻精品一区二区不卡视频 | 国产极品美女高潮无套久久久 | 九九热最新网址 | 99999精品视频 | 欧美激情一区二区三区在线 | 在线观看污视频网站 | 久久熟妇人妻午夜寂寞影院 | 国产盗摄精品一区二区酒店 | 国产足控在线网站 | 欧美精品在线观看视频 | 国产精品夜夜春夜夜爽久久小 | 亚洲一二区 | 国产人人看 | 欧美国产成人精品一区二区三区 | 天堂网成人| 色婷婷一区二区三区四区成人网 | 97久久综合亚洲色hezyo | 性生交大片免费看网站 | 妺妺窝人体色www在线观看 | 国产精品日日摸夜夜添夜夜av | 欧洲精品码一区二区三区免费看 | 少妇愉情理伦片高潮日本 | 密桃成熟时在线观看 | 野外偷拍做爰全过程 | 中国美女毛茸茸撒尿 | 成片免费观看视频大全 | 最近中文字幕在线观看 | 亚洲国产真实交换 | 精品国产_亚洲人成在线 | 夜夜爽日日澡人人添 | 男人天堂一区 | 偷看农村妇女牲交 | 牲欲强的熟妇农村老妇女 | 亚洲看片lutube在线入口 | 免费观看性生交大片3区 | 精品极品三大极久久久久 | 中文字幕日韩精 | 午夜视频在线瓜伦 | 公的~yin之手筱田优中文字幕 | 国产又粗又猛又爽视频 | 国产乱码一区二区三区咪爱 | 老女人性视频 | 免费看美女隐私网站 | 亚洲精品国产精品自产a区红杏吧 | 成av免费大片黄在线观看 | 午夜精品视频一区二区三区在线看 | 亚洲欧洲日产av | 无遮挡边摸边吃奶边做视频 | 中文天堂资源在线www | 狠狠色丁香婷婷综合潮喷 | 极品尤物魔鬼身材啪啪仙踪林 | 玖玖爱资源站 | 中文字幕乱码在线观看 | 一级二级黄色大片 | 中日韩精品视频 | 亚洲aⅴ片 | 67194成是人免费无码 | 成人深夜福利 | 日韩高清片 | 中国免费黄色片 | 非洲黑人狂躁日本妞 | 欧美激情第三页 | www.日本在线视频 | 久久不卡免费视频 | 国产精品亚洲成在人线 | 51久久久 | 亚洲天堂av网站 | 久久综合久 | 亚洲一区二区欧美 | 免费又黄又爽又色的视频 | 精品国产91乱码一区二区三区 | 九九精品99久久久香蕉 | 舌奴调教日记 | 日韩三级黄色 | 午夜精品久久久久久久96蜜桃 | 午夜福利在线永久视频 | 野外少妇愉情中文字幕 | 少妇性bbb搡bbb爽爽爽 | 久久天天干 | 亚洲乱人伦中文字幕无码 | 中文字幕亚洲综合久久菠萝蜜 | 乱淫a裸体xxxⅹ | 日躁夜躁狠狠躁2001 | 美女黄色在线观看 | 欧美无极品 | 青草青草久热精品视频国产4 | 日本午夜三级 | 噜啦噜色姑娘综合网 | 免费视频久久 | 国产精品一二三级 | 日本熟妇人妻xxxxx人hd | 尤果网福利视频在线观看 | 男女羞羞羞视频午夜视频 | 亚洲精品久久久久久中文传媒 | 欧美在线一区二区三区四区 | 欧美成人精品一区二区三区在线观看 | 好看的91视频 | 国产熟女内射oooo | 深夜福利视频在线播放 | 欧美性猛交99久久久久99按摩 | 国产免国产免‘费 | 国产农村妇女aaaaa视频 | a级毛片蜜桃成熟时2在线播放 | 你懂的网址在线观看 | 激情戏网站 | 搡老女人老妇女老熟妇 | 色婷婷久久一区二区三区麻豆 | 狠狠色综合网站久久久久久久高清 | 99爱精品|