待机功能基本实现
This commit is contained in:
@@ -1,19 +1,51 @@
|
|||||||
package middleware
|
package middleware
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"game-driver/internal/common"
|
|
||||||
"game-driver/internal/schema"
|
"game-driver/internal/schema"
|
||||||
"game-driver/leaf"
|
"game-driver/leaf"
|
||||||
"game-driver/pkg/audio"
|
"game-driver/pkg/audio"
|
||||||
|
"github.com/gopxl/beep/v2/speaker"
|
||||||
|
"log"
|
||||||
|
"sync"
|
||||||
)
|
)
|
||||||
|
|
||||||
func PlayBgm() leaf.HandlerFunc {
|
func PlayBgm() 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)
|
||||||
|
|
||||||
bgm := common.LinkAudio(pm.BGM)
|
bgm := audio.LinkAudio(pm.BGM)
|
||||||
if bgm != nil {
|
if bgm != nil {
|
||||||
go audio.PlayMP3(c, bgm)
|
// 等待组
|
||||||
|
var wait sync.WaitGroup
|
||||||
|
defer wait.Wait()
|
||||||
|
|
||||||
|
// 结束信号通道
|
||||||
|
a := make(chan struct{})
|
||||||
|
// 发送结束信号
|
||||||
|
defer close(a)
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
// 等待结束
|
||||||
|
wait.Add(1)
|
||||||
|
defer wait.Done()
|
||||||
|
|
||||||
|
ctrl, closer := audio.PlayBgmMP3(bgm)
|
||||||
|
defer closer()
|
||||||
|
if ctrl == nil {
|
||||||
|
log.Println("播放背景音乐失败")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-a:
|
||||||
|
{
|
||||||
|
speaker.Lock()
|
||||||
|
ctrl.Streamer = nil
|
||||||
|
speaker.Unlock()
|
||||||
|
return
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
c.Next()
|
c.Next()
|
||||||
|
|||||||
@@ -7,17 +7,17 @@ import (
|
|||||||
)
|
)
|
||||||
|
|
||||||
// SoundStart 开始词播报
|
// SoundStart 开始词播报
|
||||||
func SoundStart(t *tts.AliTTS) leaf.HandlerFunc {
|
func SoundStart() 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)
|
||||||
t.Sound(pm.TTS.Start)
|
tts.DefaultTTS.Sound(pm.TTS.Start)
|
||||||
|
|
||||||
defer func() {
|
defer func() {
|
||||||
switch leaf.Value[leaf.EndType](c, leaf.EndKey) {
|
switch leaf.Value[leaf.EndType](c, leaf.EndKey) {
|
||||||
case leaf.EndTimer:
|
case leaf.EndTimer:
|
||||||
t.Sound(pm.TTS.End)
|
tts.DefaultTTS.Sound(pm.TTS.End)
|
||||||
case leaf.EndStop:
|
case leaf.EndStop:
|
||||||
t.Sound(pm.TTS.Stop)
|
tts.DefaultTTS.Sound(pm.TTS.Stop)
|
||||||
}
|
}
|
||||||
}()
|
}()
|
||||||
|
|
||||||
|
|||||||
@@ -8,7 +8,7 @@ import (
|
|||||||
"time"
|
"time"
|
||||||
)
|
)
|
||||||
|
|
||||||
func TickerAction(t *tts.AliTTS) 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)
|
||||||
|
|
||||||
@@ -50,7 +50,7 @@ func TickerAction(t *tts.AliTTS) leaf.HandlerFunc {
|
|||||||
//TODO: 屏幕打印
|
//TODO: 屏幕打印
|
||||||
}
|
}
|
||||||
if to, ok := ttsMap[s]; ok {
|
if to, ok := ttsMap[s]; ok {
|
||||||
t.Sound(to.Value)
|
tts.DefaultTTS.Sound(to.Value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,93 +0,0 @@
|
|||||||
package routes
|
|
||||||
|
|
||||||
import (
|
|
||||||
"game-driver/internal/common"
|
|
||||||
"game-driver/internal/middleware"
|
|
||||||
"game-driver/internal/schema"
|
|
||||||
"game-driver/leaf"
|
|
||||||
"game-driver/pkg/audio"
|
|
||||||
"github.com/gopxl/beep/v2"
|
|
||||||
"github.com/gopxl/beep/v2/mp3"
|
|
||||||
"github.com/gopxl/beep/v2/speaker"
|
|
||||||
"log"
|
|
||||||
"time"
|
|
||||||
)
|
|
||||||
|
|
||||||
func timerAction(timestamp int64) <-chan struct{} {
|
|
||||||
a := make(chan struct{})
|
|
||||||
|
|
||||||
go func() {
|
|
||||||
if timestamp == 0 {
|
|
||||||
close(a)
|
|
||||||
} else {
|
|
||||||
<-time.After(time.Until(time.Unix(timestamp, 0)))
|
|
||||||
close(a)
|
|
||||||
}
|
|
||||||
}()
|
|
||||||
|
|
||||||
return a
|
|
||||||
}
|
|
||||||
|
|
||||||
func BackgroundAction(c *leaf.Context) {
|
|
||||||
payload := leaf.Value[*schema.BackgroundModel](c, middleware.PayloadJSONKey)
|
|
||||||
|
|
||||||
if payload.Start != 0 && payload.End != 0 && time.Unix(payload.Start, 0).After(time.Unix(payload.End, 0)) {
|
|
||||||
log.Println("开始时间大于结束时间")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
if payload.End != 0 {
|
|
||||||
cancel := leaf.WithDeadline(c, time.Unix(payload.End, 0))
|
|
||||||
defer cancel()
|
|
||||||
}
|
|
||||||
|
|
||||||
select {
|
|
||||||
case <-c.Done():
|
|
||||||
case <-timerAction(payload.Start):
|
|
||||||
go audioAction(c, payload.Items[0], payload.TimeModel)
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
func audioAction(c *leaf.Context, item schema.BackgroundItemModel, root schema.TimeModel) {
|
|
||||||
if item.Start != 0 && time.Unix(item.Start, 0).Before(time.Unix(root.Start, 0)) {
|
|
||||||
log.Println("开始时间小于根任务开始时间")
|
|
||||||
return
|
|
||||||
}
|
|
||||||
|
|
||||||
select {
|
|
||||||
case <-c.Done():
|
|
||||||
case <-timerAction(item.Start):
|
|
||||||
{
|
|
||||||
log.Println("开始执行后台任务")
|
|
||||||
data := common.LinkAudio(item.Data)
|
|
||||||
|
|
||||||
streamer, format, err := mp3.Decode(data)
|
|
||||||
if err != nil {
|
|
||||||
return
|
|
||||||
}
|
|
||||||
defer streamer.Close()
|
|
||||||
|
|
||||||
s := beep.Resample(4, format.SampleRate, audio.DefaultSampleRate, streamer)
|
|
||||||
|
|
||||||
ctrl := &beep.Ctrl{Streamer: s}
|
|
||||||
done := make(chan struct{})
|
|
||||||
speaker.Play(beep.Seq(ctrl, beep.Callback(func() {
|
|
||||||
close(done)
|
|
||||||
})))
|
|
||||||
|
|
||||||
for {
|
|
||||||
select {
|
|
||||||
case <-done:
|
|
||||||
return
|
|
||||||
case <-c.Done():
|
|
||||||
{
|
|
||||||
speaker.Lock()
|
|
||||||
ctrl.Streamer = nil
|
|
||||||
speaker.Unlock()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
}
|
|
||||||
}
|
|
||||||
}
|
|
||||||
179
internal/routes/wait.go
Normal file
179
internal/routes/wait.go
Normal file
@@ -0,0 +1,179 @@
|
|||||||
|
package routes
|
||||||
|
|
||||||
|
import (
|
||||||
|
"game-driver/internal/middleware"
|
||||||
|
"game-driver/internal/schema"
|
||||||
|
"game-driver/leaf"
|
||||||
|
"game-driver/pkg/audio"
|
||||||
|
"game-driver/pkg/relay"
|
||||||
|
"game-driver/pkg/tts"
|
||||||
|
"github.com/gopxl/beep/v2/speaker"
|
||||||
|
"log"
|
||||||
|
"sync"
|
||||||
|
"time"
|
||||||
|
)
|
||||||
|
|
||||||
|
func timerAction(timestamp int64) <-chan struct{} {
|
||||||
|
a := make(chan struct{})
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
if timestamp == 0 {
|
||||||
|
close(a)
|
||||||
|
} else {
|
||||||
|
<-time.After(time.Until(time.Unix(timestamp, 0)))
|
||||||
|
close(a)
|
||||||
|
}
|
||||||
|
}()
|
||||||
|
|
||||||
|
return a
|
||||||
|
}
|
||||||
|
|
||||||
|
func WaitAction(c *leaf.Context) {
|
||||||
|
payload := leaf.Value[*schema.WaitModel](c, middleware.PayloadJSONKey)
|
||||||
|
|
||||||
|
if payload.Start != 0 && payload.End != 0 && time.Unix(payload.Start, 0).After(time.Unix(payload.End, 0)) {
|
||||||
|
log.Println("开始时间大于结束时间")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if payload.End != 0 {
|
||||||
|
cancel := leaf.WithDeadline(c, time.Unix(payload.End, 0))
|
||||||
|
defer cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
case <-timerAction(payload.Start):
|
||||||
|
// 等待组
|
||||||
|
var wait sync.WaitGroup
|
||||||
|
defer wait.Wait()
|
||||||
|
for _, item := range payload.Items {
|
||||||
|
switch item.Type {
|
||||||
|
case schema.WaitAudio:
|
||||||
|
// 执行音乐播放
|
||||||
|
go func() {
|
||||||
|
wait.Add(1)
|
||||||
|
defer wait.Done()
|
||||||
|
audioAction(c, item, payload.TimeModel)
|
||||||
|
}()
|
||||||
|
case schema.WaitTTS:
|
||||||
|
// 执行TTS播放
|
||||||
|
go func() {
|
||||||
|
wait.Add(1)
|
||||||
|
defer wait.Done()
|
||||||
|
ttsAction(c, item, payload.TimeModel)
|
||||||
|
}()
|
||||||
|
case schema.WaitRelay:
|
||||||
|
// 执行继电器供电
|
||||||
|
go func() {
|
||||||
|
wait.Add(1)
|
||||||
|
defer wait.Done()
|
||||||
|
relayAction(c, item, payload.TimeModel)
|
||||||
|
}()
|
||||||
|
case schema.WaitVideo:
|
||||||
|
case schema.WaitWeb:
|
||||||
|
default:
|
||||||
|
log.Printf("不支持的类型: %d\n", item.Type)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func audioAction(c *leaf.Context, item schema.WaitItemModel, root schema.TimeModel) {
|
||||||
|
if item.Start != 0 && time.Unix(item.Start, 0).Before(time.Unix(root.Start, 0)) {
|
||||||
|
log.Println("开始时间小于根任务开始时间")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.End != 0 {
|
||||||
|
cancel := leaf.WithDeadline(c, time.Unix(item.End, 0))
|
||||||
|
defer cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
case <-timerAction(item.Start):
|
||||||
|
{
|
||||||
|
log.Println("开始执行后台任务")
|
||||||
|
data := audio.LinkAudio(item.Data)
|
||||||
|
|
||||||
|
ctrl, closer := audio.PlayBgmMP3(data)
|
||||||
|
defer closer()
|
||||||
|
if ctrl == nil {
|
||||||
|
log.Println("播放背景音乐失败")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
{
|
||||||
|
speaker.Lock()
|
||||||
|
ctrl.Streamer = nil
|
||||||
|
speaker.Unlock()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func ttsAction(c *leaf.Context, item schema.WaitItemModel, root schema.TimeModel) {
|
||||||
|
if item.Start != 0 && time.Unix(item.Start, 0).Before(time.Unix(root.Start, 0)) {
|
||||||
|
log.Println("开始时间小于根任务开始时间")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.End != 0 {
|
||||||
|
cancel := leaf.WithDeadline(c, time.Unix(item.End, 0))
|
||||||
|
defer cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
reader, err := tts.DefaultTTS.Get(item.Data)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("语音合成异常: ", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
case <-timerAction(item.Start):
|
||||||
|
{
|
||||||
|
for {
|
||||||
|
audio.PlayWav(c, reader)
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
return
|
||||||
|
case <-time.After(time.Duration(item.Interval) * time.Second):
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func relayAction(c *leaf.Context, item schema.WaitItemModel, root schema.TimeModel) {
|
||||||
|
if item.Start != 0 && time.Unix(item.Start, 0).Before(time.Unix(root.Start, 0)) {
|
||||||
|
log.Println("开始时间小于根任务开始时间")
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
if item.End != 0 {
|
||||||
|
cancel := leaf.WithDeadline(c, time.Unix(item.End, 0))
|
||||||
|
defer cancel()
|
||||||
|
}
|
||||||
|
|
||||||
|
device, err := relay.New(item.Data, nil)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("继电器初始化异常: ", err)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
defer device.Close()
|
||||||
|
|
||||||
|
select {
|
||||||
|
case <-c.Done():
|
||||||
|
case <-timerAction(item.Start):
|
||||||
|
{
|
||||||
|
device.On(1)
|
||||||
|
<-c.Done()
|
||||||
|
device.Off(1)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
@@ -1,29 +0,0 @@
|
|||||||
package schema
|
|
||||||
|
|
||||||
type ItemType int
|
|
||||||
|
|
||||||
const (
|
|
||||||
TYPE_AUDIO ItemType = iota
|
|
||||||
TYPE_VIDEO
|
|
||||||
TYPE_WEB
|
|
||||||
TYPE_TTS
|
|
||||||
TYPE_RELAY
|
|
||||||
)
|
|
||||||
|
|
||||||
type TimeModel struct {
|
|
||||||
Start int64
|
|
||||||
End int64
|
|
||||||
}
|
|
||||||
|
|
||||||
type BackgroundItemModel struct {
|
|
||||||
TimeModel
|
|
||||||
Interval int64
|
|
||||||
Type ItemType
|
|
||||||
Data string
|
|
||||||
}
|
|
||||||
|
|
||||||
type BackgroundModel struct {
|
|
||||||
JsonModel
|
|
||||||
TimeModel
|
|
||||||
Items []BackgroundItemModel
|
|
||||||
}
|
|
||||||
29
internal/schema/wait.go
Normal file
29
internal/schema/wait.go
Normal file
@@ -0,0 +1,29 @@
|
|||||||
|
package schema
|
||||||
|
|
||||||
|
type WaitType int
|
||||||
|
|
||||||
|
const (
|
||||||
|
WaitAudio WaitType = iota
|
||||||
|
WaitVideo
|
||||||
|
WaitWeb
|
||||||
|
WaitTTS
|
||||||
|
WaitRelay
|
||||||
|
)
|
||||||
|
|
||||||
|
type TimeModel struct {
|
||||||
|
Start int64
|
||||||
|
End int64
|
||||||
|
}
|
||||||
|
|
||||||
|
type WaitItemModel struct {
|
||||||
|
TimeModel
|
||||||
|
Type WaitType
|
||||||
|
Data string
|
||||||
|
Interval int64
|
||||||
|
}
|
||||||
|
|
||||||
|
type WaitModel struct {
|
||||||
|
JsonModel
|
||||||
|
TimeModel
|
||||||
|
Items []WaitItemModel
|
||||||
|
}
|
||||||
@@ -96,7 +96,7 @@ func Run() {
|
|||||||
}
|
}
|
||||||
|
|
||||||
// 构建语音合成对象
|
// 构建语音合成对象
|
||||||
t := tts.New(ctx, config.C.Aliyun)
|
tts.DefaultTTS = tts.New(ctx, config.C.Aliyun)
|
||||||
|
|
||||||
// 构建继电器对象
|
// 构建继电器对象
|
||||||
//r, err := relay.New(config.C.Relay, func(msg string) {
|
//r, err := relay.New(config.C.Relay, func(msg string) {
|
||||||
@@ -117,10 +117,10 @@ func Run() {
|
|||||||
middleware.PayloadJSON[schema.PlayModal](),
|
middleware.PayloadJSON[schema.PlayModal](),
|
||||||
middleware.DeviceLock(device),
|
middleware.DeviceLock(device),
|
||||||
middleware.EmergencyStop(common.GlobalStopper),
|
middleware.EmergencyStop(common.GlobalStopper),
|
||||||
middleware.SoundStart(t),
|
middleware.SoundStart(),
|
||||||
middleware.RelayMaster(nil),
|
middleware.RelayMaster(nil),
|
||||||
middleware.TimeoutOver(config.C.Game.MaxTimeout),
|
middleware.TimeoutOver(config.C.Game.MaxTimeout),
|
||||||
middleware.TickerAction(t),
|
middleware.TickerAction(),
|
||||||
middleware.PlayBgm(),
|
middleware.PlayBgm(),
|
||||||
func(c *leaf.Context) {
|
func(c *leaf.Context) {
|
||||||
log.Println("接收到启动消息: ", string(c.Payload))
|
log.Println("接收到启动消息: ", string(c.Payload))
|
||||||
@@ -130,11 +130,11 @@ func Run() {
|
|||||||
}
|
}
|
||||||
},
|
},
|
||||||
)
|
)
|
||||||
// 处理后台线程报文
|
// 处理待机线程报文
|
||||||
router.RegisterHandler(topicPrefix+"bg",
|
router.RegisterHandler(topicPrefix+"wait",
|
||||||
middleware.PayloadJSON[schema.BackgroundModel](),
|
middleware.PayloadJSON[schema.WaitModel](),
|
||||||
middleware.EmergencyStop(common.GlobalBgStopper),
|
middleware.EmergencyStop(common.GlobalBgStopper),
|
||||||
routes.BackgroundAction,
|
routes.WaitAction,
|
||||||
)
|
)
|
||||||
// 处理指令
|
// 处理指令
|
||||||
router.RegisterHandler(topicPrefix+"command", routes.Command(device))
|
router.RegisterHandler(topicPrefix+"command", routes.Command(device))
|
||||||
|
|||||||
31
pkg/audio/bgm.go
Normal file
31
pkg/audio/bgm.go
Normal file
@@ -0,0 +1,31 @@
|
|||||||
|
package audio
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/gopxl/beep/v2"
|
||||||
|
"github.com/gopxl/beep/v2/mp3"
|
||||||
|
"github.com/gopxl/beep/v2/speaker"
|
||||||
|
"io"
|
||||||
|
"log"
|
||||||
|
)
|
||||||
|
|
||||||
|
func PlayBgmMP3(r io.ReadCloser, opts ...beep.LoopOption) (*beep.Ctrl, func() error) {
|
||||||
|
streamer, format, err := mp3.Decode(r)
|
||||||
|
if err != nil {
|
||||||
|
return nil, func() error { return nil }
|
||||||
|
}
|
||||||
|
|
||||||
|
loop2, err := beep.Loop2(streamer, opts...)
|
||||||
|
if err != nil {
|
||||||
|
log.Println("循环播放异常: ", err)
|
||||||
|
return nil, streamer.Close
|
||||||
|
}
|
||||||
|
|
||||||
|
s := beep.Resample(4, format.SampleRate, DefaultSampleRate, loop2)
|
||||||
|
|
||||||
|
ctrl := &beep.Ctrl{Streamer: s}
|
||||||
|
speaker.Play(beep.Seq(ctrl, beep.Callback(func() {
|
||||||
|
streamer.Close()
|
||||||
|
})))
|
||||||
|
|
||||||
|
return ctrl, streamer.Close
|
||||||
|
}
|
||||||
@@ -1,4 +1,4 @@
|
|||||||
package common
|
package audio
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bytes"
|
"bytes"
|
||||||
@@ -10,6 +10,24 @@ import (
|
|||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
type reader struct {
|
||||||
|
bytes.Reader
|
||||||
|
io.Closer
|
||||||
|
}
|
||||||
|
|
||||||
|
func (a *reader) Seek(offset int64, whence int) (int64, error) {
|
||||||
|
return a.Reader.Seek(offset, whence)
|
||||||
|
}
|
||||||
|
|
||||||
|
func toSeeker(src io.ReadCloser) io.ReadCloser {
|
||||||
|
buf := &bytes.Buffer{}
|
||||||
|
_, _ = io.Copy(buf, src)
|
||||||
|
return &reader{
|
||||||
|
Reader: *bytes.NewReader(buf.Bytes()),
|
||||||
|
Closer: src,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
func open(u string) io.ReadCloser {
|
func open(u string) io.ReadCloser {
|
||||||
p, _ := strings.CutPrefix(u, "file://")
|
p, _ := strings.CutPrefix(u, "file://")
|
||||||
f, e := os.Open(p)
|
f, e := os.Open(p)
|
||||||
@@ -29,13 +47,6 @@ func get(u string) io.ReadCloser {
|
|||||||
return resp.Body
|
return resp.Body
|
||||||
}
|
}
|
||||||
|
|
||||||
func toBuffer(b io.ReadCloser) io.ReadCloser {
|
|
||||||
data := &bytes.Buffer{}
|
|
||||||
_, _ = data.ReadFrom(b)
|
|
||||||
defer b.Close()
|
|
||||||
return io.NopCloser(data)
|
|
||||||
}
|
|
||||||
|
|
||||||
func LinkAudio(link string) (bgm io.ReadCloser) {
|
func LinkAudio(link string) (bgm io.ReadCloser) {
|
||||||
u, err := url.Parse(link)
|
u, err := url.Parse(link)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
@@ -47,8 +58,9 @@ func LinkAudio(link string) (bgm io.ReadCloser) {
|
|||||||
bgm = get(u.String())
|
bgm = get(u.String())
|
||||||
} else {
|
} else {
|
||||||
log.Printf("不支持的音频文件协议: %v\n", u.String())
|
log.Printf("不支持的音频文件协议: %v\n", u.String())
|
||||||
|
return
|
||||||
}
|
}
|
||||||
bgm = toBuffer(bgm)
|
bgm = toSeeker(bgm)
|
||||||
}
|
}
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
@@ -37,8 +37,10 @@ func New(portName string, reader func(msg string)) (*Device, error) {
|
|||||||
for {
|
for {
|
||||||
r := bufio.NewReader(port)
|
r := bufio.NewReader(port)
|
||||||
line, _, _ := r.ReadLine()
|
line, _, _ := r.ReadLine()
|
||||||
|
if reader != nil {
|
||||||
reader(string(line))
|
reader(string(line))
|
||||||
}
|
}
|
||||||
|
}
|
||||||
}()
|
}()
|
||||||
return &Device{port: port}, nil
|
return &Device{port: port}, nil
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -26,6 +26,8 @@ type result struct {
|
|||||||
Error error
|
Error error
|
||||||
}
|
}
|
||||||
|
|
||||||
|
var DefaultTTS = &AliTTS{}
|
||||||
|
|
||||||
// onTaskFailed 识别过程中的错误处理回调参数
|
// onTaskFailed 识别过程中的错误处理回调参数
|
||||||
func (tts *AliTTS) onTaskFailed(text string, param interface{}) {
|
func (tts *AliTTS) onTaskFailed(text string, param interface{}) {
|
||||||
p, _ := param.(*result)
|
p, _ := param.(*result)
|
||||||
|
|||||||
1
pkg/video/paly.go
Normal file
1
pkg/video/paly.go
Normal file
@@ -0,0 +1 @@
|
|||||||
|
package video
|
||||||
@@ -117,9 +117,9 @@ status
|
|||||||
|
|
||||||
> 设备接收到该指令会立即向 `device/${location}/${point}/status` 发送一次当前状态
|
> 设备接收到该指令会立即向 `device/${location}/${point}/status` 发送一次当前状态
|
||||||
|
|
||||||
## 4. 后台执行
|
## 4. 待机执行
|
||||||
|
|
||||||
Topic: `server/${location}/${point}/bg`
|
Topic: `server/${location}/${point}/wait`
|
||||||
|
|
||||||
Payload:
|
Payload:
|
||||||
|
|
||||||
@@ -148,4 +148,4 @@ Payload:
|
|||||||
}
|
}
|
||||||
```
|
```
|
||||||
|
|
||||||
> 同一个类型的后台任务只能有一个,当有新的任务到达时会覆盖之前的任务
|
> 同一个类型的待机任务只能有一个,当有新的任务到达时会覆盖之前的任务
|
||||||
|
|||||||
Reference in New Issue
Block a user