______ _ _ _ _ _
/ ____| | | | _ _ _| _ _
| (__ __ __ ___ ___ | | ___ | | | | | |
\___ \ \ \ /\ / / / _ \ / _ \ | | / _ \ | |_ _ _ | | | |
____) | \ V V / | (_) | | (_) | | | | ___/ | _ _ _| | |_| |
|_____ / \_/\_/ \___/ \___/ |_| \___| | | \__, |
|_| | |
__ / |
|_ _ /
swoolefy是一个基于swoole实现的轻量级高性能的常驻内存型的协程级应用服务框架, 高度支持httpApi,websocket,udp服务器,以及基于tcp实现可扩展的rpc服务, 同时支持composer包方式安装部署项目。基于实用,swoolefy抽象Event事件处理类, 实现与底层的回调的解耦,支持协程调度,同步|异步调用,全局事件注册,心跳检查,异步任务,多进程(池),连接池等, 内置view、log、session、mysql、redis、mongodb等常用组件等。
swoolefy-5.0+ 版本:
目前主分支,最低要求php8.0+,swoole5.0+(或者swoole-cli-v5.0+), 或者也可以使用swoole-cli4.8+, 因为其内置php8.1+
swoolefy-4.8-lts 版本:
长期维护分支,最低要求php >= php7.2 && php < php8.0, 推荐直接swoole-v4.8+,需要通过源码编译安装swoole
选择哪个版本?
1、如果确定项目是使用php8+的,那么直接选择 swoole-v5.0+ 以上版本来编译安装或者直接使用swoole-cli-v5.0,然后选择 bingcool/swoolefy:~5.0.2
作为项目分支
2、如果确定项目是使用php7.2-php7.4的,那么选择 swoole-v4.8+ 版本来进行编译安装(不能直接使用 swoole-cli-v4.8+ 了, 因为其内置的是php8.1,与你的项目的php7不符合)
所有只能通过编译方式方式来生成swoole扩展,然后选择 bingcool/swoolefy:^4.8.2
作为项目分支
基础特性
- 支持架手脚一键创建项目
- 支持路由映射与调度,前置路由组件,后置路由组件,MVC三层,多级配置
- 支持composer的PSR-4规范,实现PSR-3的日志接口
- 支持自定义注册不同根命名空间,快速多项目部署
- 支持httpServer,实用轻量Api接口开发
- 支持多协议websocketServer、udpServer、mqttServer
- 支持基于tcp实现的rpc服务,开放式的系统接口,可自定义协议数据格式,并提供rpc-client协程组件
- 支持DI容器,组件IOC、配置化,Channel公共组件池
- 支持协程单例注册,协程上下文变量寄存
- 支持mysql、postgreSql协程组件、redis协程组件、mongodb组件
- 支持mysql的协程连接池,redis协程池
- 支持protobuf buffer的数据接口结构验证,压缩传输等
- 支持异步务管理TaskManager,定时器管理TickManager,内存表管理TableManager
- 支持自定义进程管理ProcessManager,进程池管理PoolsManger
- 支持底层异常错误的所有日志捕捉,支持全局日志,包括debug、info、notice、warning、error等级
- 支持自定义进程的redis,rabbitmq,kafka的订阅发布,消息队列等
- 支持热更新reload worker 监控以及更新
- 支持定时的系统信息采集,并以订阅发布,udp等方式收集至存贮端
- 支持命令行形式高度封装启动|停止控制的脚本,简单命令即可管理整个框架
高级特性
- 支持crontab的local调用和fork独立进程的计划任务
- 支持worker下后台daemon模式的多进程协程消费模型,包括进程自动拉起,进程数动态调整,进程健康状态监控
- 支持console终端脚本模式,跑完脚本自动退出,可用于修复数据、数据迁移等临时脚本功能
- 支持分布式服务注册(zk,etcd)
组件名称 | 安装 | 说明 |
---|---|---|
predis | composer require predis/predis:~1.1.7 | predis组件、或者Phpredis扩展 |
mongodb | composer require mongodb/mongodb:~1.3 | mongodb组件,需要使用mongodb必须安装此组件 |
rpc-client | composer require bingcool/rpc-client:dev-master | swoolefy的rpc客户端组件,当与rpc服务端通信时,需要安装此组件,支持在php-fpm中使用 |
cron-expression | composer require dragonmantank/cron-expression:~3.3.0 | crontab计划任务组件,类似Linux的crobtab |
redis lock | composer require malkusch/lock | Redis锁组件 |
amqp | composer require php-amqplib/php-amqplib:~3.5.0 | amqp php原生实现amqp协议客户端 |
ffmpeg | composer require php-ffmpeg/php-ffmpeg:~1.1.0 | php proc-open 调用ffmpeg处理音视频 |
validate | composer require vlucas/valitron | validate数据校验组件 |
bingcool/library | composer require bingcool/library | library组件库 |
实现了包括:
- Db Mysql Model组件
- PostgreSql Model组件
- Kafka Producer Consumer组件
- Rabbitmq Queue组件
- Rabbitmq Delay Queue 死信延迟队列组件
- Redis Cache组件
- Redis Queue队列组件
- Redis Delay Queue延迟队列组件
- RedisLock锁组件
- RateLimit限流组件
- Redis Public Subscribe组件
- Db 、Redis、Curl协程连接池组件
- UUid 分布式自增id组件
- Curl基础组件
- Jwt 组件
github: https://github.com/bingcool/library
1、先配置环境变量
// 独立物理机或者云主机配置系统环境变量
vi /etc/profile
在/etc/profile末尾添加一行标识环境,下面是支持的4个环境,框架将通过这个环境变量区分环境,加载不同的配置
export SWOOLEFY_CLI_ENV='dev' // 开发环境
export SWOOLEFY_CLI_ENV='test' // 测试环境
export SWOOLEFY_CLI_ENV='gra' // 灰度环境
export SWOOLEFY_CLI_ENV='prd' // 生产环境
// 最后是配置生效
source /etc/profile
// 如果是通过dockerfile 创建容器的, 可以根据不同环境生成的内置环境变量不同镜像,每个不同的环境镜像可以用在不同环境,代码将通过这个环境变量区分环境,加载不同的配置
ENV SWOOLEFY_CLI_ENV dev
2、创建项目
// 下载代码到到你的自定义目录,这里定义为myproject
composer create-project bingcool/swoolefy:^4.8.2 myproject
<?php
// 在myproject目录下添加cli.php, 这个是启动项目的入口文件
include './vendor/autoload.php';
define('IS_WORKER_SERVICE', 0);
define('IS_CLI_SCRIPT', 0);
date_default_timezone_set('Asia/Shanghai');
define('APP_NAMES', [
// 你的项目命名为App,对应协议为http协议服务器,支持多个项目的,只需要在这里添加好项目名称与对应的协议即可
'App' => 'http'
]);
include './swoolefy';
// 你定义的项目目录是App, 在myproject目录下执行下面命令行
swoole-cli cli.php create App
// 执行完上面命令行后,将会自动生成App项目目录以及内部子目录
App // 项目目录
├── Config // 应用配置
│ ├── config-dev.php
│ ├── config-gra.php
│ ├── config-prd.php
│ ├── config-test.php
│ └── defines.php
├── Controller
│ └── IndexController.php // 控制器层
├── Model
│ └── ClientModel.php
├── Module // 模块层
├── Protocol // 协议配置
│ ├── config-dev.php
│ ├── config-gra.php
│ ├── config-prd.php
│ └── config-test.php
├── Runtime
├── Router
│ └── Api.php // 路由文件,不同模块定义不同文件即可
|—— Storage
| |—— Logs // 日志文件目录
| |—— Sql // sql日志目录
│—— autoloader.php // 自定义项目自动加载
|—— Event.php // 事件实现类
|—— HttpServer.php // http server
// 终端启动 ctl+c 停止进程
swoole-cli cli.php start App
// 守护进程方式启动,添加-D参数控制
swooole-cli cli.php start App -D
// 停止进程
swooole-cli cli.php stop App
// 查看进程状态
swooole-cli cli.php status App
默认端口是9502,可以通过 http://localhost:9502 访问默认控制器
<?php
namespace App\Controller;
use Swoolefy\Core\Application;
use Swoolefy\Core\Controller\BController;
// 默认生成的IndexController
class IndexController extends BController {
public function index() {
Application::getApp()->response->write('<h1>Hello, Welcome to Swoolefy Framework! <h1>');
}
}
至此一个最简单的http的服务就创建完成了,更多例子请参考项目下Test的demo
应用层配置文件: Config/config-dev.php
开放式组件接口,闭包回调实现创建组件过程,return对象即可
<?php
return [
// db|redis连接池
'enable_component_pools' => [
// 取components的`DB`组件名称相对应
'db' => [
'pools_num' => 5, // db实例数
'push_timeout' => 2, // db实例进入channel池最长等待时间,单位s
'pop_timeout' => 1, // db实例出channel池最长等待时间,单位s.在规定时间内获取不到db对象,将降级为实时创建db实例
'live_time' => 10 // db实例的有效期,单位s.过期后将被掉弃,重新创建新DB实例
],
// 取components的`redis`组件名称相对应
'redis' => [
'pools_num' => 5,
'push_timeout' => 2,
'pop_timeout' => 1,
'live_time' => 10
]
],
// 在应用层配置文件中,例如下面使用library的Redis、Db组件
components => [
// 例如创建phpredis扩展连接实例
'redis' => function($name) { // 定义组件名,闭包回调实现创建组件过程,return对象即可
$redis = new \Common\Library\Cache\Redis();
$redis->connect('127.0.0.1', 6379);
$redis->auth('123456789');
return $redis;
},
// predis组件的redis实例
'predis' => function($name) {
$parameters = [
'scheme' => 'tcp',
'host' => '127.0.0.1',
'port' => 6379,
'password' => '123456789'
];
$predis = new \Common\Library\Cache\Predis();
$predis->setConfig($parameters);
return $predis;
},
// 适配swoole的mysql客户端组件
'db' => function() {
$config = [
// 地址
'hostname' => '127.0.0.1',
// 数据库
'database' => 'bingcool',
// 用户名
'username' => 'bingcool',
// 密码
'password' => '123456789',
// 端口
'hostport' => '3306',
// dsn
'dsn' => '',
// 数据库连接参数
'params' => [],
// 数据库编码,默认采用utf8
'charset' => 'utf8',
// 数据库表前缀
'prefix' => '',
// 是否断线重连
'break_reconnect' => true,
// 是否支持事务嵌套
'support_savepoint' => false
];
$db = new \Common\Library\Db\Mysql($config);
return $db;
},
// 适配swoole的postgreSql客户端组件
'pg' => function() {
$config = [
// 地址
'hostname' => '127.0.0.1',
// 数据库
'database' => 'dbtest',
// 用户
'username' => 'bingcool',
// 密码
'password' => '123456789',
// 端口
'hostport' => '5432',
// dsn
'dsn' => '',
// 数据库连接参数
'params' => [],
// 数据库编码,默认采用utf8
'charset' => 'utf8',
// 数据库表前缀
'prefix' => '',
// 是否断线重连
'break_reconnect' => true,
// 是否支持事务嵌套
'support_savepoint' => false
];
$pg = new \Common\Library\Db\Pgsql($config);
return $pg;
},
// 其他的组件都可以通过闭包回调创建
// 数组配置型log组件
'log' => [
'class' => \Swoolefy\Util\Log::class,
'channel' => 'application',
'logFilePath' => rtrim(LOG_PATH,'/').'/runtime.log'
],
// 或者log组件利用闭包回调创建
'log' => function($name) {
$channel= 'application';
$logFilePath = rtrim(LOG_PATH,'/').'/runtime.log';
$log = new \Swoolefy\Util\Log($channel, $logFilePath);
return $log;
},
]
// 其他配置
......
]
use Swoolefy\Core\Application;
class TestController extends BController {
/**
* 控制器
*/
public function test() {
// 获取组件,组件就是配置回调中定义的组件
$redis = Application::getApp()->redis;
//或者通过get指明组件名获取(推荐)
// $redis = Application::getApp()->get('redis');
// swoole hook 特性,这个过程会发生协程调度
$redis->set('name', swoolefy);
// predis组件
$predis = Application::getApp()->predis;
//或者通过get指明组件名获取(推荐)
// $predis = Application::getApp()->get('predis');
// 这个过程会发生协程调度
$predis->set('predis','this is a predis instance');
$predis->get('predis');
// PDO的mysql实例,这个过程会发生协程调度
$db = Application::getApp()->db;
// 或者
// $mysql = Application::getApp()->get('db');
// 添加一条数据
$sql = "INSERT INTO `user` (`username` ,`sex`) VALUES (:username, :sex)";
$numRows = $db->createCommand($sql)->insert([
':username'=>'bingcool-test',
':sex' => 1
]);
var_dump($numRows)
// 查询
$result = $db->createCommand('select * from user where id>:id')->queryOne([':id'=>100]);
var_dump($result);
// pg实例
$pg = Application::getApp()->get('pg');
// 添加一条数据
$sql = "INSERT INTO `user` (username ,sex) VALUES (:username, :sex)";
$pg->createCommand($sql)->insert([
':username'=>'bingcool-test',
':sex' => 1
]);
}
}
开发者可以根据实际使用适当调整配置项
<?php
// 加载应用层常量定义
include START_DIR_ROOT.'/'.APP_NAME.'/Config/defines.php';
// 加载应用层协议
$appConf = include START_DIR_ROOT.'/'.APP_NAME.'/Config/config-'.SWOOLEFY_ENV.'.php';
return [
// 应用层配置
'app_conf' => $appConf, // 应用层配置
'application_index' => '',
'event_handler' => \Test\Event::class,
'exception_handler' => \Test\Exception\ExceptionHandle::class,
'response_formatter' => \Swoolefy\Core\ResponseFormatter::class,
'master_process_name' => 'php-swoolefy-http-master',
'manager_process_name' => 'php-swoolefy-http-manager',
'worker_process_name' => 'php-swoolefy-http-worker',
'www_user' => '',
'host' => '0.0.0.0',
'port' => '9501',
'time_zone' => 'PRC',
'swoole_process_mode' => SWOOLE_PROCESS,
'include_files' => [],
'runtime_enable_coroutine' => true,
// swoole setting
'setting' => [
'admin_server' => '0.0.0.0:9503',
'reactor_num' => 1,
'worker_num' => 4,
'max_request' => 10000,
'task_worker_num' => 2,
'task_tmpdir' => '/dev/shm',
'daemonize' => 0,
'dispatch_mode' => 3,
'reload_async' => true,
'enable_coroutine' => 1,
'task_enable_coroutine' => 1,
// 压缩
'http_compression' => true,
// $level 压缩等级,范围是 1-9,等级越高压缩后的尺寸越小,但 CPU 消耗更多。默认为 1, 最高为 9
'http_compression_level' => 1,
'log_file' => '/tmp/' . APP_NAME . '/swoole_log.txt',
'pid_file' => '/data/' . APP_NAME . '/log/server.pid',
],
'coroutine_setting' => [
'max_coroutine' => 50000
],
// 是否内存化线上实时任务
'enable_table_tick_task' => true,
// 内存表定义
'table' => [
'table_process' => [
// 内存表建立的行数,取决于建立的process进程数,最小值64
'size' => 64,
// 定义字段
'fields'=> [
['pid','int', 10],
['process_name','string', 56],
]
]
],
// 依赖于EnableSysCollector = true,否则设置没有意义,不生效
'enable_pv_collector' => false,
'enable_sys_collector' => true,
'sys_collector_conf' => [
'type' => SWOOLEFY_SYS_COLLECTOR_UDP,
'host' => '127.0.0.1',
'port' => 9504,
'from_service' => 'http-app',
'target_service' => 'collectorService/system',
'event' => 'collect',
'tick_time' => 2,
'callback' => function () {
$sysCollector = new \Swoolefy\Core\SysCollector\SysCollector();
return $sysCollector->test();
}
],
// 热更新
'reload_conf'=> [
'enable_reload' => false, // 是否启用热文件更新功能
'after_seconds' => 3, // 检测到只要有文件更新,3s内不在检测,等待重启既可
'monitor_path' => APP_PATH, // 开发者自己定义目录
'reload_file_types' => ['.php', '.html', '.js'],
'ignore_dirs' => [],
'callback' => function () {}
]
];
MIT
Copyright (c) 2017-2023 zengbing huang