解决部分已知bug

This commit is contained in:
2025-02-27 11:36:40 +08:00
parent 593d7758bf
commit 8780f8555e
14 changed files with 96 additions and 59 deletions

View File

@@ -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

View File

@@ -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
} }

View File

@@ -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
View File

@@ -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

View File

@@ -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 {

View File

@@ -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)

View File

@@ -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)

View File

@@ -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

View File

@@ -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
} }

View File

@@ -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"
} }
} }
``` ```

View File

@@ -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

View File

@@ -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())
} }
if bgm != nil {
bgm = toSeeker(bgm) bgm = toSeeker(bgm)
} }
}
return return
} }

View File

@@ -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 {

View File

@@ -28,12 +28,11 @@
---- tts播报恭喜通关词 ---- tts播报恭喜通关词
-- 青龙云台(5) -- 青龙云台(5)
--- 待机 --- 待机
---- 网页默认页面
--- Game --- Game
---- 供电 ---- 供电
---- 播放语音 ---- 播放语音
-- 俱乐部(6) -- 俱乐部(6)
--- 待机 --- 待机
---- 启动屏幕 ---- 启动屏幕
--- Game ---- 网页默认页面
@endmindmap @endmindmap