8 Commits

Author SHA1 Message Date
Yu Sun
d4385e6471 Merge branch 'master' of ssh://118.195.187.246:10022/ysun/EdgeManager 2022-08-11 16:04:41 +08:00
Yu Sun
fd7e58c02c 修复以code为插入依据可能导致的bug 2022-08-11 16:02:25 +08:00
wuyanwei
47004e2377 增加set_node_data接口根据code查询working_subclass 2022-08-10 17:52:44 +08:00
Yu Sun
f262e55698 修正无默认值参数位置问题 2022-08-10 16:04:35 +08:00
Yu Sun
fc42a5b1f0 紧急修复拼写错误 2022-08-10 15:22:18 +08:00
Yu Sun
54eae08472 增加端口号参数;对是否允许code在working_subclass间复用设定两种情况 2022-08-10 14:56:42 +08:00
Yu Sun
b54c1650cd 把EdgeManager做成daemon 2022-08-09 16:39:05 +08:00
Yu Sun
2540bdd9b7 增加调试环境的说明 2022-08-09 11:50:14 +08:00
7 changed files with 119 additions and 43 deletions

1
.dockerignore Normal file
View File

@@ -0,0 +1 @@
*

View File

@@ -8,9 +8,24 @@ require_once __DIR__ . '/vendor/autoload.php';
use EdgeManager\EDataCapture\{ EDataCapture, ENodeConfigure }; use EdgeManager\EDataCapture\{ EDataCapture, ENodeConfigure };
$options = getopt('h::', ['server_name:', 'user:', 'password:', 'help::']); $options = getopt('h::', ['no_dup_code', 'server_name:', 'port::', 'user:', 'password:', 'help::']);
init_db($options['server_name'], $options['user'], $options['password']); if (array_key_exists('h', $options) or array_key_exists('help', $options)) {
print_r(
"EdgeManager使用说明
--no_dup_code 禁止code在不同的working subclass间复用
--server_name pg实例的FQDN
--user pg实例的用户名
--port pg实例的端口号
--password pg实例的密码
-h, --help 显示此帮助信息
"
);
exit;
}
init_db($options['server_name'], $options['port'] ?? 5432, $options['user'], $options['password']);
$worker = new Worker('http://0.0.0.0:8888'); $worker = new Worker('http://0.0.0.0:8888');
$worker -> name = 'EntryPoint'; $worker -> name = 'EntryPoint';
@@ -20,8 +35,8 @@ $worker -> onWorkerStart = function(Worker $worker) {
global $options, $dbconn; global $options, $dbconn;
$dbconn = pg_connect(sprintf( $dbconn = pg_connect(sprintf(
"host=%s dbname=scada user=%s password=%s", "host=%s port=%s dbname=scada user=%s password=%s",
$options['server_name'], $options['user'], $options['password'] $options['server_name'], $options['port'] ?? 5432, $options['user'], $options['password']
)); ));
}; };
@@ -79,7 +94,7 @@ $worker -> onMessage = function(TcpConnection $connection, Request $request) {
// 执行异步连接 // 执行异步连接
$task_connection->connect(); $task_connection->connect();
} else { } else {
$enode_configure = new ENodeConfigure($dbconn, post: $post); $enode_configure = new ENodeConfigure($dbconn, no_dup_code: $options['no_dup_code'] ?? true, post: $post);
$res = $enode_configure -> $action(); $res = $enode_configure -> $action();
if ($res === true) if ($res === true)
@@ -95,7 +110,7 @@ $worker -> onMessage = function(TcpConnection $connection, Request $request) {
} }
} }
} else if ($post -> action === 'set_node_data') { } else if ($post -> action === 'set_node_data') {
$data_capture = new EDataCapture($dbconn, post: $post); $data_capture = new EDataCapture($dbconn, no_dup_code: $options['no_dup_code'] ?? true, post: $post);
if ($data_capture -> check_res === 'WRONG_WORKING_SUBCLASS') { if ($data_capture -> check_res === 'WRONG_WORKING_SUBCLASS') {
$response = new Response(200, [ $response = new Response(200, [
'Content-Type' => 'application/json;charset=utf-8', 'Content-Type' => 'application/json;charset=utf-8',
@@ -156,7 +171,7 @@ $worker -> onMessage = function(TcpConnection $connection, Request $request) {
$get = $request -> get(); $get = $request -> get();
if (isset($get['query'])) { if (isset($get['query'])) {
if ($get['query'] == 'nodes') { if ($get['query'] == 'nodes') {
$enode_configure = new ENodeConfigure($dbconn, get: $get); $enode_configure = new ENodeConfigure($dbconn, no_dup_code: $options['no_dup_code'] ?? true, get: $get);
$nodes = $enode_configure -> get_nodes(); $nodes = $enode_configure -> get_nodes();
if (is_null($nodes)) if (is_null($nodes))
$connection -> send(json_encode(array( $connection -> send(json_encode(array(
@@ -169,7 +184,7 @@ $worker -> onMessage = function(TcpConnection $connection, Request $request) {
'data' => $nodes 'data' => $nodes
))); )));
} else if ($get['query'] == 'working_subclasses') { } else if ($get['query'] == 'working_subclasses') {
$enode_configure = new ENodeConfigure($dbconn, get: $get); $enode_configure = new ENodeConfigure($dbconn, no_dup_code: $options['no_dup_code'] ?? true, get: $get);
$working_subclasses = $enode_configure -> get_working_subclasses($dbconn); $working_subclasses = $enode_configure -> get_working_subclasses($dbconn);
if (is_null($working_subclasses)) if (is_null($working_subclasses))
$connection -> send(json_encode(array( $connection -> send(json_encode(array(
@@ -182,7 +197,7 @@ $worker -> onMessage = function(TcpConnection $connection, Request $request) {
'data' => $working_subclasses 'data' => $working_subclasses
))); )));
} else if ($get['query'] == 'codes') { } else if ($get['query'] == 'codes') {
$enode_configure = new ENodeConfigure($dbconn, get: $get); $enode_configure = new ENodeConfigure($dbconn, no_dup_code: $options['no_dup_code'] ?? true, get: $get);
$codes = $enode_configure -> get_codes_by_working_subclasses(); $codes = $enode_configure -> get_codes_by_working_subclasses();
if (is_null($codes)) if (is_null($codes))
$connection -> send(json_encode(array( $connection -> send(json_encode(array(
@@ -195,7 +210,7 @@ $worker -> onMessage = function(TcpConnection $connection, Request $request) {
'data' => $codes 'data' => $codes
))); )));
} else if ($get['query'] == 'node_data') { } else if ($get['query'] == 'node_data') {
$data_capture = new EDataCapture($dbconn, get: $get); $data_capture = new EDataCapture($dbconn, no_dup_code: $options['no_dup_code'] ?? true, get: $get);
$data = $data_capture -> get_node_data(); $data = $data_capture -> get_node_data();
if (is_null($data)) if (is_null($data))
$connection -> send(json_encode(array( $connection -> send(json_encode(array(
@@ -228,8 +243,8 @@ $consumer -> onWorkerStart = function(Worker $consumer) {
global $options, $consumer_dbconn; global $options, $consumer_dbconn;
$consumer_dbconn = pg_connect(sprintf( $consumer_dbconn = pg_connect(sprintf(
"host=%s dbname=scada user=%s password=%s", "host=%s port=%s dbname=scada user=%s password=%s",
$options['server_name'], $options['user'], $options['password'] $options['server_name'], $options['port'] ?? 5432, $options['user'], $options['password']
)); ));
}; };
@@ -239,7 +254,7 @@ $consumer -> onMessage = function(TcpConnection $connection, $task_data) {
$task_data = json_decode($task_data); $task_data = json_decode($task_data);
if ($task_data -> action === 'add_node') { if ($task_data -> action === 'add_node') {
$enode_configure = new ENodeConfigure($consumer_dbconn, post: $task_data -> data); $enode_configure = new ENodeConfigure($consumer_dbconn, no_dup_code: $options['no_dup_code'] ?? true, post: $task_data -> data);
$res = $enode_configure -> add_node(); $res = $enode_configure -> add_node();
if ($res === true) if ($res === true)
@@ -250,7 +265,7 @@ $consumer -> onMessage = function(TcpConnection $connection, $task_data) {
else if ($res === "REPLICATED") else if ($res === "REPLICATED")
$connection -> send(json_encode(array( $connection -> send(json_encode(array(
'code' => 1, 'code' => 1,
'msg' => '同一工序单元内的节点编码不可重复!' 'msg' => isset($options['no_dup_code']) ? '节点编码不可重复!' : '同一工序单元内的节点编码不可重复!'
))); )));
else if ($res === false) { else if ($res === false) {
$connection -> send(json_encode(array( $connection -> send(json_encode(array(

View File

@@ -4,30 +4,44 @@ namespace EdgeManager\EDataCapture;
class EDataCapture { class EDataCapture {
function __construct( function __construct(
protected $dbconn, protected $dbconn,
protected $no_dup_code,
protected $post = NULL, protected $post = NULL,
protected $get = NULL, protected $get = NULL,
public $check_res = NULL, public $check_res = NULL,
protected $working_subclass = NULL, protected $working_subclass = NULL,
protected $code_type = [], protected $code_type = [],
protected $data = [] protected $data = [],
) { ) {
if (!is_null($this -> post)) { if (!is_null($this -> post)) {
if (!in_array( if(isset($this -> post -> param -> working_subclass)){
$this -> post -> param -> working_subclass, if (!in_array(
ENodeConfigure::get_working_subclasses($this -> dbconn) $this -> post -> param -> working_subclass,
)) { ENodeConfigure::get_working_subclasses($this -> dbconn)
$this -> check_res = 'WRONG_WORKING_SUBCLASS'; )) {
return; $this -> check_res = 'WRONG_WORKING_SUBCLASS';
} else { return;
$this -> working_subclass = $this -> post -> param -> working_subclass; } else {
$this -> working_subclass = $this -> post -> param -> working_subclass;
}
}else{
//根据code查询working_subclasses
$working_subclass = ENodeConfigure::get_working_subclasses_by_codes($this -> dbconn, $this -> post -> param -> data[0] -> code);
$this -> working_subclass = $working_subclass[0];
} }
$res = pg_fetch_all(pg_query($this -> dbconn, sprintf( if ($this -> no_dup_code) {
"SELECT code, type $res = pg_fetch_all(pg_query($this -> dbconn,
FROM hf_mes_scada_data_capture_node_configure "SELECT code, type
WHERE working_subclass = '%s'", FROM hf_mes_scada_data_capture_node_configure"
$this -> post -> param -> working_subclass ));
))); } else {
$res = pg_fetch_all(pg_query($this -> dbconn, sprintf(
"SELECT code, type
FROM hf_mes_scada_data_capture_node_configure
WHERE working_subclass = '%s'",
$this -> working_subclass
)));
}
$code_type = &$this -> code_type; $code_type = &$this -> code_type;
array_walk($res, function(&$v, $k) use (&$code_type) { array_walk($res, function(&$v, $k) use (&$code_type) {

View File

@@ -4,19 +4,30 @@ namespace EdgeManager\EDataCapture;
class ENodeConfigure { class ENodeConfigure {
function __construct( function __construct(
protected $dbconn, protected $dbconn,
protected $no_dup_code,
protected $post = NULL, protected $post = NULL,
protected $get = NULL protected $get = NULL,
) {} ) {}
function add_node() { function add_node() {
pg_query($this -> dbconn, "BEGIN"); pg_query($this -> dbconn, "BEGIN");
$exists = pg_query($this -> dbconn, sprintf( if ($this -> no_dup_code) {
"SELECT EXISTS( $exists = pg_query($this -> dbconn, sprintf(
SELECT 1 FROM hf_mes_scada_data_capture_node_configure "SELECT EXISTS(
WHERE code = '%s' SELECT 1 FROM hf_mes_scada_data_capture_node_configure
AND working_subclass = '%s' WHERE code = '%s'
)", $this -> post -> code, $this -> post -> working_subclass )", $this -> post -> code
)); ));
} else {
$exists = pg_query($this -> dbconn, sprintf(
"SELECT EXISTS(
SELECT 1 FROM hf_mes_scada_data_capture_node_configure
WHERE code = '%s'
AND working_subclass = '%s'
)", $this -> post -> code, $this -> post -> working_subclass
));
}
if (pg_fetch_assoc($exists)['exists'] === 't') { if (pg_fetch_assoc($exists)['exists'] === 't') {
pg_query($this -> dbconn, "ROLLBACK"); pg_query($this -> dbconn, "ROLLBACK");
return "REPLICATED"; return "REPLICATED";
@@ -100,7 +111,7 @@ class ENodeConfigure {
$this -> dbconn, $this -> dbconn,
'hf_mes_scada_data_capture_node_configure', 'hf_mes_scada_data_capture_node_configure',
(array) $this -> post, (array) $this -> post,
['code' => $this -> post -> code] ['id' => $this -> post -> id]
); );
} }
@@ -114,6 +125,16 @@ class ENodeConfigure {
return pg_fetch_all_columns($res, 0); return pg_fetch_all_columns($res, 0);
} }
static function get_working_subclasses_by_codes($dbconn, $code) {
$res = pg_query($dbconn, sprintf(
"SELECT working_subclass
FROM hf_mes_scada_data_capture_node_configure
WHERE code = '%s'",
$code
));
return pg_fetch_all_columns($res, 0);
}
function get_codes_by_working_subclasses() { function get_codes_by_working_subclasses() {
$res = pg_query($this -> dbconn, sprintf( $res = pg_query($this -> dbconn, sprintf(
"SELECT code "SELECT code

View File

@@ -1,13 +1,13 @@
<?php <?php
function init_db($server_name, $user, $password) { function init_db($server_name, $port, $user, $password) {
$dbconn = pg_connect(sprintf("host=%s user=%s password=%s", $server_name, $user, $password)); $dbconn = pg_connect(sprintf("host=%s port=%s user=%s password=%s", $server_name, $port, $user, $password));
$res = pg_query($dbconn, "select datname from pg_database"); $res = pg_query($dbconn, "select datname from pg_database");
if (!in_array('scada', pg_fetch_all_columns($res, 0))) if (!in_array('scada', pg_fetch_all_columns($res, 0)))
pg_query($dbconn, "CREATE DATABASE scada"); pg_query($dbconn, "CREATE DATABASE scada");
pg_close($dbconn); pg_close($dbconn);
$dbconn = pg_connect(sprintf("host=%s dbname=scada user=%s password=%s", $server_name, $user, $password)); $dbconn = pg_connect(sprintf("host=%s port=%s dbname=scada user=%s password=%s", $server_name, $port, $user, $password));
// 全局未启用TSDB的时候在这里对scada数据库启用就好 // 全局未启用TSDB的时候在这里对scada数据库启用就好
pg_query($dbconn, "CREATE EXTENSION IF NOT EXISTS timescaledb"); pg_query($dbconn, "CREATE EXTENSION IF NOT EXISTS timescaledb");
pg_query($dbconn, "CREATE TABLE IF NOT EXISTS hf_mes_scada_data_capture_node_configure ( pg_query($dbconn, "CREATE TABLE IF NOT EXISTS hf_mes_scada_data_capture_node_configure (

View File

@@ -8,6 +8,7 @@
- [后端](#后端) - [后端](#后端)
- [前端](#前端) - [前端](#前端)
- [开发环境](#开发环境) - [开发环境](#开发环境)
- [调试环境](#调试环境)
- [技术细节](#技术细节) - [技术细节](#技术细节)
- [0. `EdgeManager\EDataCapture\EDataCapture -> set_data()`为什么是以`6710885`为大小chunked的](#0-edgemanageredatacaptureedatacapture---set_data为什么是以6710885为大小chunked的) - [0. `EdgeManager\EDataCapture\EDataCapture -> set_data()`为什么是以`6710885`为大小chunked的](#0-edgemanageredatacaptureedatacapture---set_data为什么是以6710885为大小chunked的)
@@ -257,6 +258,7 @@ cd EdgeManager
一键部署PHP workerman和TimescaleDB环境 一键部署PHP workerman和TimescaleDB环境
```bash ```bash
# docker build --network host -t edge_manager .
docker compose up -d docker compose up -d
``` ```
@@ -270,7 +272,7 @@ docker exec -it edge_manager bash
```bash ```bash
# In container # In container
php EdgeManager.php --server_name=GPU-server-01 --user=postgres --password=big_dick start php EdgeManager.php --no_dup_code --server_name=GPU-server-01 --user=postgres --password=big_dick start
``` ```
前端调试: 前端调试:
@@ -292,6 +294,29 @@ psql -h localhost -U postgres
\l \l
``` ```
## 调试环境
不使用`docker compose`创建两个container分别运行EdgeManager和pg
```bash
# cd EdgeManager # 先定位到项目目录方便创建image和挂载
# 创建EdgeManager的image
docker build --network host -t edge_manager .
# 创建container运行EdgeManager
docker run -d --name edge_manager_test -v $PWD:/EdgeManager --network host -it edge_manager
# 创建container运行pg将端口映射到host的55432
docker run -d --name pg_test -v $PWD/config/postgresql.conf:/etc/postgresql/postgresql.conf -p 55432:5432 -e POSTGRES_PASSWORD=big_dick -it timescale/timescaledb-ha:pg14-latest postgres -c 'config_file=/etc/postgresql/postgresql.conf'
# 进入交互式Prompt
docker exec -it edge_manager_test bash
# 启动EdgeManagerworkerman命令省略...
# 常用命令
# 查看全部container
docker ps -a
# 启动已停止的container
docker start [container]
```
## 技术细节 ## 技术细节
### 0. `EdgeManager\EDataCapture\EDataCapture -> set_data()`为什么是以`6710885`为大小chunked的 ### 0. `EdgeManager\EDataCapture\EDataCapture -> set_data()`为什么是以`6710885`为大小chunked的

View File

@@ -1,5 +1,5 @@
services: services:
scada: edge_manager:
build: build:
context: . context: .
container_name: edge_manager container_name: edge_manager