解决部分已知bug
This commit is contained in:
49
config.yml
49
config.yml
@@ -1,5 +1,5 @@
|
|||||||
location: wushan
|
location: wushan # 项目名称
|
||||||
point: 2
|
point: 2 # 点位
|
||||||
relay:
|
relay:
|
||||||
maxTimeout: 60 # 单位 s,必须大于 0
|
maxTimeout: 60 # 单位 s,必须大于 0
|
||||||
log:
|
log:
|
||||||
@@ -17,26 +17,27 @@ aliyun:
|
|||||||
accessKeySecret:
|
accessKeySecret:
|
||||||
appKey:
|
appKey:
|
||||||
timeout: 10 # 单位 s
|
timeout: 10 # 单位 s
|
||||||
voice: zhifeng_emo
|
volume: 200 # 音量
|
||||||
|
voice: zhifeng_emo # 发音人
|
||||||
speechRate: 50 # 语速
|
speechRate: 50 # 语速
|
||||||
game:
|
#game:
|
||||||
# addr: /dev/ttyUSB0 # 点位 5 的串口地址
|
# addr: /dev/ttyUSB0 # 点位 11 的串口地址
|
||||||
pushGroups: # 点位 4 的发卡器配置
|
# pushGroups: # 点位 10 的发卡器配置
|
||||||
- name: gpiochip0
|
# - name: gpiochip0
|
||||||
read: /dev/ttyUSB0
|
# read: /dev/ttyUSB0
|
||||||
outOK: 21
|
# outOK: 21
|
||||||
lower: 20
|
# lower: 20
|
||||||
error: 16
|
# error: 16
|
||||||
empty: 12
|
# empty: 12
|
||||||
push: 13
|
# push: 13
|
||||||
reset: 19
|
# reset: 19
|
||||||
pull: 26
|
# pull: 26
|
||||||
- name: gpiochip0
|
# - name: gpiochip0
|
||||||
read: /dev/ttyUSB0
|
# read: /dev/ttyUSB0
|
||||||
outOK: 7
|
# outOK: 7
|
||||||
lower: 8
|
# lower: 8
|
||||||
error: 25
|
# error: 25
|
||||||
empty: 24
|
# empty: 24
|
||||||
push: 10
|
# push: 10
|
||||||
reset: 9
|
# reset: 9
|
||||||
pull: 11
|
# pull: 11
|
||||||
@@ -13,6 +13,7 @@ type AliyunConfig struct {
|
|||||||
AccessKeySecret string
|
AccessKeySecret string
|
||||||
AppKey string
|
AppKey string
|
||||||
Timeout int
|
Timeout int
|
||||||
|
Volume int
|
||||||
Voice string
|
Voice string
|
||||||
SpeechRate int
|
SpeechRate int
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -4,9 +4,9 @@ type Config any
|
|||||||
|
|
||||||
func NewConfig(point int) Config {
|
func NewConfig(point int) Config {
|
||||||
switch point {
|
switch point {
|
||||||
case 4:
|
case 10:
|
||||||
return ConfigPush{}
|
return ConfigPush{}
|
||||||
case 5:
|
case 11:
|
||||||
return ConfigWait{}
|
return ConfigWait{}
|
||||||
default:
|
default:
|
||||||
return nil
|
return nil
|
||||||
|
|||||||
2
go.mod
2
go.mod
@@ -5,6 +5,7 @@ go 1.23.2
|
|||||||
require (
|
require (
|
||||||
github.com/aliyun/alibabacloud-nls-go-sdk v1.1.1
|
github.com/aliyun/alibabacloud-nls-go-sdk v1.1.1
|
||||||
github.com/eclipse/paho.golang v0.22.0
|
github.com/eclipse/paho.golang v0.22.0
|
||||||
|
github.com/go-pkgz/cronrange v0.2.0
|
||||||
github.com/go-rod/rod v0.116.2
|
github.com/go-rod/rod v0.116.2
|
||||||
github.com/gopxl/beep/v2 v2.1.0
|
github.com/gopxl/beep/v2 v2.1.0
|
||||||
github.com/grid-x/modbus v0.0.0-20241004123532-f6c6fb5201b3
|
github.com/grid-x/modbus v0.0.0-20241004123532-f6c6fb5201b3
|
||||||
@@ -20,7 +21,6 @@ require (
|
|||||||
github.com/ebitengine/oto/v3 v3.3.2 // indirect
|
github.com/ebitengine/oto/v3 v3.3.2 // indirect
|
||||||
github.com/ebitengine/purego v0.8.1 // indirect
|
github.com/ebitengine/purego v0.8.1 // indirect
|
||||||
github.com/fsnotify/fsnotify v1.8.0 // indirect
|
github.com/fsnotify/fsnotify v1.8.0 // indirect
|
||||||
github.com/go-pkgz/cronrange v0.2.0 // indirect
|
|
||||||
github.com/gorilla/websocket v1.5.3 // indirect
|
github.com/gorilla/websocket v1.5.3 // indirect
|
||||||
github.com/grid-x/serial v0.0.0-20211107191517-583c7356b3aa // indirect
|
github.com/grid-x/serial v0.0.0-20211107191517-583c7356b3aa // indirect
|
||||||
github.com/hajimehoshi/go-mp3 v0.3.4 // indirect
|
github.com/hajimehoshi/go-mp3 v0.3.4 // indirect
|
||||||
|
|||||||
@@ -4,6 +4,7 @@ import (
|
|||||||
"game-driver/internal/common"
|
"game-driver/internal/common"
|
||||||
"game-driver/leaf"
|
"game-driver/leaf"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
|
"sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
// EmergencyStop 紧急停止中间件
|
// EmergencyStop 紧急停止中间件
|
||||||
@@ -12,13 +13,19 @@ func EmergencyStop(stopper common.Stopper) leaf.HandlerFunc {
|
|||||||
cancel := leaf.WithCancel(c)
|
cancel := leaf.WithCancel(c)
|
||||||
defer stopper.Reset()
|
defer stopper.Reset()
|
||||||
|
|
||||||
|
// 等待组
|
||||||
|
var wait sync.WaitGroup
|
||||||
|
defer wait.Wait()
|
||||||
|
|
||||||
// 结束信号通道
|
// 结束信号通道
|
||||||
a := make(chan struct{})
|
a := make(chan struct{})
|
||||||
// 发送结束信号
|
// 发送结束信号
|
||||||
defer close(a)
|
defer close(a)
|
||||||
|
|
||||||
zap.S().Infoln("监听停止信号")
|
zap.S().Infoln("监听停止信号")
|
||||||
|
wait.Add(1)
|
||||||
go func() {
|
go func() {
|
||||||
|
defer wait.Done()
|
||||||
defer zap.S().Infoln("结束停止信号监听")
|
defer zap.S().Infoln("结束停止信号监听")
|
||||||
|
|
||||||
select {
|
select {
|
||||||
|
|||||||
@@ -9,6 +9,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
// TickerAction 定时器动作,用于在指定时间点执行打印和语音播报
|
||||||
func TickerAction() leaf.HandlerFunc {
|
func TickerAction() leaf.HandlerFunc {
|
||||||
return func(c *leaf.Context) {
|
return func(c *leaf.Context) {
|
||||||
pm := leaf.Value[*schema.PlayModal](c, PayloadJSONKey)
|
pm := leaf.Value[*schema.PlayModal](c, PayloadJSONKey)
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
// TimeoutOver 定时器中间件,用于定时触发屏幕打印和语音播报。 t 是语音播报实例
|
// TimeoutOver 超时停止
|
||||||
func TimeoutOver(maxTimeout int) leaf.HandlerFunc {
|
func TimeoutOver(maxTimeout int) leaf.HandlerFunc {
|
||||||
return func(c *leaf.Context) {
|
return func(c *leaf.Context) {
|
||||||
pm := leaf.Value[*schema.PlayModal](c, PayloadJSONKey)
|
pm := leaf.Value[*schema.PlayModal](c, PayloadJSONKey)
|
||||||
|
|||||||
@@ -19,11 +19,11 @@ func switchPoint(ctx context.Context, point int) leaf.HandlerFunc {
|
|||||||
switch point {
|
switch point {
|
||||||
case 2: // 镇水塔点位
|
case 2: // 镇水塔点位
|
||||||
return play.OnlyVideo
|
return play.OnlyVideo
|
||||||
case 4:
|
case 10:
|
||||||
// 4号点位(发卡机)
|
// 10号点位(发卡机)
|
||||||
return play.PushCard(ctx)
|
return play.PushCard(ctx)
|
||||||
case 5:
|
case 11:
|
||||||
// 5号点位(等待插卡)
|
// 11号点位(等待插卡)
|
||||||
return play.WaitCard(ctx)
|
return play.WaitCard(ctx)
|
||||||
default:
|
default:
|
||||||
return play.Default
|
return play.Default
|
||||||
|
|||||||
@@ -2,6 +2,7 @@ package routes
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"fmt"
|
||||||
"game-driver/internal/middleware"
|
"game-driver/internal/middleware"
|
||||||
"game-driver/internal/schema"
|
"game-driver/internal/schema"
|
||||||
"game-driver/leaf"
|
"game-driver/leaf"
|
||||||
@@ -18,10 +19,12 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
func runAction(c *leaf.Context, item schema.WaitItemModel, rootRules []cronrange.Rule, play func(c context.Context, item schema.WaitItemModel)) {
|
func runAction(c *leaf.Context, item schema.WaitItemModel, rootRules []cronrange.Rule, play func(c context.Context, item schema.WaitItemModel) error) {
|
||||||
|
// 设定默认时间规则
|
||||||
if item.Cron == "" {
|
if item.Cron == "" {
|
||||||
item.Cron = "* * * *"
|
item.Cron = "* * * *"
|
||||||
}
|
}
|
||||||
|
|
||||||
rules, err := cronrange.Parse(item.Cron)
|
rules, err := cronrange.Parse(item.Cron)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Errorln("解析时间规则异常: ", err)
|
zap.S().Errorln("解析时间规则异常: ", err)
|
||||||
@@ -64,7 +67,17 @@ func runAction(c *leaf.Context, item schema.WaitItemModel, rootRules []cronrange
|
|||||||
return
|
return
|
||||||
default:
|
default:
|
||||||
if run {
|
if run {
|
||||||
play(ctx, item)
|
err := play(ctx, item)
|
||||||
|
if err != nil {
|
||||||
|
zap.S().Errorln("执行动作异常: ", err)
|
||||||
|
<-time.After(time.Minute)
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
return
|
||||||
|
case <-time.After(time.Second):
|
||||||
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
@@ -73,6 +86,11 @@ func runAction(c *leaf.Context, item schema.WaitItemModel, rootRules []cronrange
|
|||||||
func WaitAction(c *leaf.Context) {
|
func WaitAction(c *leaf.Context) {
|
||||||
payload := leaf.Value[*schema.WaitModel](c, middleware.PayloadJSONKey)
|
payload := leaf.Value[*schema.WaitModel](c, middleware.PayloadJSONKey)
|
||||||
|
|
||||||
|
// 设定默认时间规则
|
||||||
|
if payload.Cron == "" {
|
||||||
|
payload.Cron = "* * * *"
|
||||||
|
}
|
||||||
|
|
||||||
rules, err := cronrange.Parse(payload.Cron)
|
rules, err := cronrange.Parse(payload.Cron)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Errorln("解析时间规则异常: ", err)
|
zap.S().Errorln("解析时间规则异常: ", err)
|
||||||
@@ -125,11 +143,13 @@ func WaitAction(c *leaf.Context) {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func audioAction(c context.Context, item schema.WaitItemModel) {
|
func audioAction(c context.Context, item schema.WaitItemModel) error {
|
||||||
data, err := utils.LinkAudio(item.Data)
|
data, err := utils.LinkAudio(item.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Errorln("音频数据获取异常: ", err)
|
return fmt.Errorf("音频数据获取异常: %w", err)
|
||||||
return
|
}
|
||||||
|
if data == nil {
|
||||||
|
return fmt.Errorf("音频数据获取为空")
|
||||||
}
|
}
|
||||||
|
|
||||||
zap.S().Infoln("播放待机音乐")
|
zap.S().Infoln("播放待机音乐")
|
||||||
@@ -138,8 +158,7 @@ func audioAction(c context.Context, item schema.WaitItemModel) {
|
|||||||
ctrl, closer, e := audio.PlayBgmMP3(data)
|
ctrl, closer, e := audio.PlayBgmMP3(data)
|
||||||
defer closer()
|
defer closer()
|
||||||
if e != nil {
|
if e != nil {
|
||||||
zap.S().Errorln("播放待机音乐异常", e)
|
return fmt.Errorf("播放待机音乐异常: %w", e)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
<-c.Done()
|
<-c.Done()
|
||||||
@@ -147,13 +166,14 @@ func audioAction(c context.Context, item schema.WaitItemModel) {
|
|||||||
speaker.Lock()
|
speaker.Lock()
|
||||||
ctrl.Streamer = nil
|
ctrl.Streamer = nil
|
||||||
speaker.Unlock()
|
speaker.Unlock()
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func ttsAction(c context.Context, item schema.WaitItemModel) {
|
func ttsAction(c context.Context, item schema.WaitItemModel) error {
|
||||||
reader, err := tts.DefaultTTS.Get(item.Data)
|
reader, err := tts.DefaultTTS.Get(item.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Errorln("语音合成异常: ", err)
|
return fmt.Errorf("语音合成异常: %w", err)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
zap.S().Infoln("循环播放待机 TTS 语音")
|
zap.S().Infoln("循环播放待机 TTS 语音")
|
||||||
@@ -163,17 +183,16 @@ func ttsAction(c context.Context, item schema.WaitItemModel) {
|
|||||||
audio.PlayWav(c, reader)
|
audio.PlayWav(c, reader)
|
||||||
select {
|
select {
|
||||||
case <-c.Done():
|
case <-c.Done():
|
||||||
return
|
return nil
|
||||||
case <-time.After(time.Duration(item.Interval) * time.Second):
|
case <-time.After(time.Duration(item.Interval) * time.Second):
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func relayAction(c context.Context, item schema.WaitItemModel) {
|
func relayAction(c context.Context, item schema.WaitItemModel) error {
|
||||||
r, err := relay.New(item.Data)
|
r, err := relay.New(item.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Errorln("继电器初始化异常: ", err)
|
return fmt.Errorf("继电器初始化异常: %w", err)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
defer r.Close()
|
defer r.Close()
|
||||||
|
|
||||||
@@ -183,13 +202,14 @@ func relayAction(c context.Context, item schema.WaitItemModel) {
|
|||||||
_ = r.On(0)
|
_ = r.On(0)
|
||||||
<-c.Done()
|
<-c.Done()
|
||||||
_ = r.Off(0)
|
_ = r.Off(0)
|
||||||
|
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|
||||||
func videoAction(c context.Context, item schema.WaitItemModel) {
|
func videoAction(c context.Context, item schema.WaitItemModel) error {
|
||||||
local, err := utils.LinkVideo(item.Data)
|
local, err := utils.LinkVideo(item.Data)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Errorln("视频文件获取异常: ", err)
|
return fmt.Errorf("视频文件获取异常: %w", err)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
|
|
||||||
zap.S().Infoln("循环播放待机视频")
|
zap.S().Infoln("循环播放待机视频")
|
||||||
@@ -201,18 +221,17 @@ func videoAction(c context.Context, item schema.WaitItemModel) {
|
|||||||
for {
|
for {
|
||||||
err := video.Play(c, local)
|
err := video.Play(c, local)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
zap.S().Infof("视频播放异常: %s", err)
|
return fmt.Errorf("视频播放异常: %w", err)
|
||||||
return
|
|
||||||
}
|
}
|
||||||
select {
|
select {
|
||||||
case <-c.Done():
|
case <-c.Done():
|
||||||
return
|
return nil
|
||||||
case <-time.After(time.Duration(item.Interval) * time.Second):
|
case <-time.After(time.Duration(item.Interval) * time.Second):
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func webAction(c context.Context, item schema.WaitItemModel) {
|
func webAction(c context.Context, item schema.WaitItemModel) error {
|
||||||
zap.S().Infoln("打开待机网页")
|
zap.S().Infoln("打开待机网页")
|
||||||
|
|
||||||
// 控制背光
|
// 控制背光
|
||||||
@@ -220,4 +239,5 @@ func webAction(c context.Context, item schema.WaitItemModel) {
|
|||||||
defer utils.BlankClose()
|
defer utils.BlankClose()
|
||||||
|
|
||||||
browser.OpenApp(c, item.Data)
|
browser.OpenApp(c, item.Data)
|
||||||
|
return nil
|
||||||
}
|
}
|
||||||
|
|||||||
4
json.md
4
json.md
@@ -8,7 +8,7 @@ url: `server/wushan/2/wait`
|
|||||||
"cron": "* * * *",
|
"cron": "* * * *",
|
||||||
"items": [
|
"items": [
|
||||||
{
|
{
|
||||||
"data": "file://./三峡龙脊BGM.mp3"
|
"data": "file:///opt/game/三峡龙脊BGM.mp3"
|
||||||
}
|
}
|
||||||
]
|
]
|
||||||
}
|
}
|
||||||
@@ -30,7 +30,7 @@ url: `server/wushan/2/play`
|
|||||||
]
|
]
|
||||||
},
|
},
|
||||||
"game": {
|
"game": {
|
||||||
"video": "file://./镇水塔法阵.mp4"
|
"video": "file:///opt/game/镇水塔法阵.mp4"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|||||||
@@ -71,7 +71,7 @@ func (tts *AliTTS) getToken() error {
|
|||||||
|
|
||||||
func (tts *AliTTS) Get(text string) (io.Reader, error) {
|
func (tts *AliTTS) Get(text string) (io.Reader, error) {
|
||||||
param := nls.DefaultSpeechSynthesisParam()
|
param := nls.DefaultSpeechSynthesisParam()
|
||||||
param.Volume = 200
|
param.Volume = tts.Volume
|
||||||
param.Voice = tts.Voice
|
param.Voice = tts.Voice
|
||||||
param.SpeechRate = tts.SpeechRate
|
param.SpeechRate = tts.SpeechRate
|
||||||
|
|
||||||
|
|||||||
@@ -64,7 +64,9 @@ func LinkAudio(link string) (bgm io.ReadCloser, err error) {
|
|||||||
} else {
|
} else {
|
||||||
err = fmt.Errorf("不支持的链接协议: %v", u.String())
|
err = fmt.Errorf("不支持的链接协议: %v", u.String())
|
||||||
}
|
}
|
||||||
bgm = toSeeker(bgm)
|
if bgm != nil {
|
||||||
|
bgm = toSeeker(bgm)
|
||||||
|
}
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -5,6 +5,7 @@ import (
|
|||||||
"context"
|
"context"
|
||||||
"go.uber.org/zap"
|
"go.uber.org/zap"
|
||||||
"io"
|
"io"
|
||||||
|
"os"
|
||||||
"os/exec"
|
"os/exec"
|
||||||
"sync"
|
"sync"
|
||||||
)
|
)
|
||||||
@@ -14,6 +15,11 @@ func Play(ctx context.Context, file string) error {
|
|||||||
zap.S().Infoln("video file is empty")
|
zap.S().Infoln("video file is empty")
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
// 判断文件是否存在
|
||||||
|
if _, err := os.Stat(file); err != nil {
|
||||||
|
zap.S().Errorf("视频文件不存在: %v", err)
|
||||||
|
return err
|
||||||
|
}
|
||||||
cmd := exec.CommandContext(ctx, "ffplay", "-autoexit", "-fs", file)
|
cmd := exec.CommandContext(ctx, "ffplay", "-autoexit", "-fs", file)
|
||||||
pipe, err := cmd.StderrPipe()
|
pipe, err := cmd.StderrPipe()
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
|||||||
@@ -28,12 +28,11 @@
|
|||||||
---- tts播报恭喜通关词
|
---- tts播报恭喜通关词
|
||||||
-- 青龙云台(5)
|
-- 青龙云台(5)
|
||||||
--- 待机
|
--- 待机
|
||||||
---- 网页默认页面
|
|
||||||
--- Game
|
--- Game
|
||||||
---- 供电
|
---- 供电
|
||||||
---- 播放语音
|
---- 播放语音
|
||||||
-- 俱乐部(6)
|
-- 俱乐部(6)
|
||||||
--- 待机
|
--- 待机
|
||||||
---- 启动屏幕
|
---- 启动屏幕
|
||||||
--- Game
|
---- 网页默认页面
|
||||||
@endmindmap
|
@endmindmap
|
||||||
|
|||||||
Reference in New Issue
Block a user