Files
engine/http.go

364 lines
8.8 KiB
Go

package engine
import (
"encoding/json"
"net/http"
"os"
"strings"
"time"
"gopkg.in/yaml.v3"
"m7s.live/engine/v4/codec"
"m7s.live/engine/v4/config"
"m7s.live/engine/v4/util"
)
const (
NO_SUCH_CONIFG = "no such config"
NO_SUCH_STREAM = "no such stream"
)
type GlobalConfig struct {
config.Engine
}
func ShouldYaml(r *http.Request) bool {
format := r.URL.Query().Get("format")
return r.URL.Query().Get("yaml") != "" || format == "yaml"
}
func (conf *GlobalConfig) ServeHTTP(rw http.ResponseWriter, r *http.Request) {
rw.Write([]byte("Monibuca API Server\n"))
for _, api := range apiList {
rw.Write([]byte(api + "\n"))
}
}
func fetchSummary() *Summary {
return &summary
}
func (conf *GlobalConfig) API_summary(rw http.ResponseWriter, r *http.Request) {
y := ShouldYaml(r)
if r.Header.Get("Accept") == "text/event-stream" {
summary.Add()
defer summary.Done()
if y {
util.ReturnYaml(fetchSummary, time.Second, rw, r)
} else {
util.ReturnJson(fetchSummary, time.Second, rw, r)
}
} else {
if !summary.Running() {
summary.collect()
}
if y {
if err := yaml.NewEncoder(rw).Encode(&summary); err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
} else if err := json.NewEncoder(rw).Encode(&summary); err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
}
}
func (conf *GlobalConfig) API_plugins(rw http.ResponseWriter, r *http.Request) {
format := r.URL.Query().Get("format")
if format == "yaml" {
if err := yaml.NewEncoder(rw).Encode(Plugins); err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
} else if err := json.NewEncoder(rw).Encode(Plugins); err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
}
func (conf *GlobalConfig) API_stream(rw http.ResponseWriter, r *http.Request) {
if streamPath := r.URL.Query().Get("streamPath"); streamPath != "" {
if s := Streams.Get(streamPath); s != nil {
if ShouldYaml(r) {
util.ReturnYaml(func() *Stream { return s }, time.Second, rw, r)
} else {
util.ReturnJson(func() *Stream { return s }, time.Second, rw, r)
}
} else {
http.Error(rw, NO_SUCH_STREAM, http.StatusNotFound)
}
} else {
http.Error(rw, "no streamPath", http.StatusBadRequest)
}
}
func (conf *GlobalConfig) API_sysInfo(rw http.ResponseWriter, r *http.Request) {
if err := json.NewEncoder(rw).Encode(&SysInfo); err != nil {
http.Error(rw, err.Error(), http.StatusInternalServerError)
}
}
func (conf *GlobalConfig) API_closeStream(w http.ResponseWriter, r *http.Request) {
if streamPath := r.URL.Query().Get("streamPath"); streamPath != "" {
if s := Streams.Get(streamPath); s != nil {
s.Close()
w.Write([]byte("ok"))
} else {
http.Error(w, NO_SUCH_STREAM, http.StatusNotFound)
}
} else {
http.Error(w, "no streamPath", http.StatusBadRequest)
}
}
// API_getConfig 获取指定的配置信息
func (conf *GlobalConfig) API_getConfig(w http.ResponseWriter, r *http.Request) {
var p *Plugin
var q = r.URL.Query()
if configName := q.Get("name"); configName != "" {
if c, ok := Plugins[configName]; ok {
p = c
} else {
http.Error(w, NO_SUCH_CONIFG, http.StatusNotFound)
return
}
} else {
p = Engine
}
if ShouldYaml(r) {
mm, err := yaml.Marshal(p.RawConfig)
if err != nil {
mm = []byte("")
}
json.NewEncoder(w).Encode(struct {
File string
Modified string
Merged string
}{
p.Yaml, p.modifiedYaml, string(mm),
})
} else if err := json.NewEncoder(w).Encode(p.RawConfig); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
// API_modifyConfig 修改并保存配置
func (conf *GlobalConfig) API_modifyConfig(w http.ResponseWriter, r *http.Request) {
var p *Plugin
var q = r.URL.Query()
var err error
if configName := q.Get("name"); configName != "" {
if c, ok := Plugins[configName]; ok {
p = c
} else {
http.Error(w, NO_SUCH_CONIFG, http.StatusNotFound)
return
}
} else {
p = Engine
}
if ShouldYaml(r) {
err = yaml.NewDecoder(r.Body).Decode(&p.Modified)
} else {
err = json.NewDecoder(r.Body).Decode(&p.Modified)
}
if err != nil {
http.Error(w, err.Error(), http.StatusBadRequest)
} else if err = p.Save(); err == nil {
p.RawConfig.Assign(p.Modified)
out, err := yaml.Marshal(p.Modified)
if err == nil {
p.modifiedYaml = string(out)
}
w.Write([]byte("ok"))
} else {
w.Write([]byte(err.Error()))
}
}
// API_updateConfig 热更新配置
func (conf *GlobalConfig) API_updateConfig(w http.ResponseWriter, r *http.Request) {
var p *Plugin
var q = r.URL.Query()
if configName := q.Get("name"); configName != "" {
if c, ok := Plugins[configName]; ok {
p = c
} else {
http.Error(w, NO_SUCH_CONIFG, http.StatusNotFound)
return
}
} else {
p = Engine
}
p.Update(p.Modified)
w.Write([]byte("ok"))
}
func (conf *GlobalConfig) API_list_pull(w http.ResponseWriter, r *http.Request) {
util.ReturnJson(func() (result []any) {
Pullers.Range(func(key, value any) bool {
result = append(result, key)
return true
})
return
}, time.Second, w, r)
}
func (conf *GlobalConfig) API_list_push(w http.ResponseWriter, r *http.Request) {
util.ReturnJson(func() (result []any) {
Pushers.Range(func(key, value any) bool {
result = append(result, value)
return true
})
return
}, time.Second, w, r)
}
func (conf *GlobalConfig) API_stopPush(w http.ResponseWriter, r *http.Request) {
q := r.URL.Query()
pusher, ok := Pushers.Load(q.Get("url"))
if ok {
pusher.(IPusher).Stop()
w.Write([]byte("ok"))
} else {
http.Error(w, "no such pusher", http.StatusNotFound)
}
}
func (conf *GlobalConfig) API_replay_rtpdump(w http.ResponseWriter, r *http.Request) {
q := r.URL.Query()
streamPath := q.Get("streamPath")
if streamPath == "" {
streamPath = "dump/rtsp"
}
dumpFile := q.Get("dump")
if dumpFile == "" {
dumpFile = streamPath + ".rtpdump"
}
cv := q.Get("vcodec")
ca := q.Get("acodec")
var pub RTPDumpPublisher
switch cv {
case "h264":
pub.VCodec = codec.CodecID_H264
case "h265":
pub.VCodec = codec.CodecID_H265
default:
pub.VCodec = codec.CodecID_H264
}
switch ca {
case "aac":
pub.ACodec = codec.CodecID_AAC
case "pcma":
pub.ACodec = codec.CodecID_PCMA
case "pcmu":
pub.ACodec = codec.CodecID_PCMU
default:
pub.ACodec = codec.CodecID_AAC
}
ss := strings.Split(dumpFile, ",")
if len(ss) > 1 {
if err := Engine.Publish(streamPath, &pub); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
for _, s := range ss {
f, err := os.Open(s)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
go pub.Feed(f)
}
w.Write([]byte("ok"))
}
} else {
f, err := os.Open(dumpFile)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
if err := Engine.Publish(streamPath, &pub); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
pub.SetIO(f)
w.Write([]byte("ok"))
go pub.Feed(f)
}
}
}
func (conf *GlobalConfig) API_replay_ts(w http.ResponseWriter, r *http.Request) {
q := r.URL.Query()
streamPath := q.Get("streamPath")
if streamPath == "" {
streamPath = "dump/ts"
}
dumpFile := q.Get("dump")
if dumpFile == "" {
dumpFile = streamPath + ".ts"
}
f, err := os.Open(dumpFile)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
var pub TSPublisher
if err := Engine.Publish(streamPath, &pub); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
pub.SetIO(f)
go pub.Feed(f)
w.Write([]byte("ok"))
}
}
func (conf *GlobalConfig) API_replay_mp4(w http.ResponseWriter, r *http.Request) {
q := r.URL.Query()
streamPath := q.Get("streamPath")
if streamPath == "" {
streamPath = "dump/mp4"
}
dumpFile := q.Get("dump")
if dumpFile == "" {
dumpFile = streamPath + ".mp4"
}
var pub MP4Publisher
f, err := os.Open(dumpFile)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
return
}
if err := Engine.Publish(streamPath, &pub); err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
pub.SetIO(f)
w.Write([]byte("ok"))
go pub.ReadMP4Data(f)
}
}
func (c *GlobalConfig) API_replay_ps(w http.ResponseWriter, r *http.Request) {
dump := r.URL.Query().Get("dump")
streamPath := r.URL.Query().Get("streamPath")
if dump == "" {
dump = "dump/ps"
}
f, err := os.OpenFile(dump, os.O_RDONLY, 0644)
if err != nil {
http.Error(w, err.Error(), http.StatusInternalServerError)
} else {
if streamPath == "" {
if strings.HasPrefix(dump, "/") {
streamPath = "replay" + dump
} else {
streamPath = "replay/" + dump
}
}
var pub PSPublisher
pub.SetIO(f)
if err = Engine.Publish(streamPath, &pub); err == nil {
go pub.Replay(f)
w.Write([]byte("ok"))
} else {
http.Error(w, err.Error(), http.StatusInternalServerError)
}
}
}