// Copyright 2019, Chef. All rights reserved. // https://github.com/q191201771/lal // // Use of this source code is governed by a MIT-style license // that can be found in the License file. // // Author: Chef (191201771@qq.com) package httpflv import ( "crypto/tls" "net" "sync" log "github.com/q191201771/naza/pkg/nazalog" ) type ServerObserver interface { // 通知上层有新的拉流者 // 返回值: true则允许拉流,false则关闭连接 OnNewHTTPFLVSubSession(session *SubSession) bool OnDelHTTPFLVSubSession(session *SubSession) } type ServerConfig struct { Enable bool `json:"enable"` SubListenAddr string `json:"sub_listen_addr"` EnableHTTPS bool `json:"enable_https"` HTTPSAddr string `json:"https_addr"` HTTPSCertFile string `json:"https_cert_file"` HTTPSKeyFile string `json:"https_key_file"` } type Server struct { obs ServerObserver config ServerConfig ln net.Listener httpsLn net.Listener } // TODO chef: 监听太难看了,考虑直接传入Listener对象,或直接路由进来,使得不同server可以共用端口 func NewServer(obs ServerObserver, config ServerConfig) *Server { return &Server{ obs: obs, config: config, } } func (server *Server) Listen() (err error) { if server.config.Enable { if server.ln, err = net.Listen("tcp", server.config.SubListenAddr); err != nil { return } log.Infof("start httpflv server listen. addr=%s", server.config.SubListenAddr) } if server.config.EnableHTTPS { var cert tls.Certificate cert, err = tls.LoadX509KeyPair(server.config.HTTPSCertFile, server.config.HTTPSKeyFile) if err != nil { return err } tlsConfig := &tls.Config{Certificates: []tls.Certificate{cert}} if server.httpsLn, err = tls.Listen("tcp", server.config.HTTPSAddr, tlsConfig); err != nil { return } log.Infof("start httpsflv server listen. addr=%s", server.config.HTTPSAddr) } return } func (server *Server) RunLoop() error { var wg sync.WaitGroup // TODO chef: 临时这么搞,错误值丢失了,重构一下 if server.ln != nil { wg.Add(1) go func() { for { conn, err := server.ln.Accept() if err != nil { break } go server.handleConnect(conn) } wg.Done() }() } if server.httpsLn != nil { wg.Add(1) go func() { for { conn, err := server.httpsLn.Accept() if err != nil { break } go server.handleConnect(conn) } wg.Done() }() } wg.Wait() return nil } func (server *Server) Dispose() { if server.ln != nil { if err := server.ln.Close(); err != nil { log.Error(err) } } if server.httpsLn != nil { if err := server.httpsLn.Close(); err != nil { log.Error(err) } } } func (server *Server) handleConnect(conn net.Conn) { log.Infof("accept a httpflv connection. remoteAddr=%s", conn.RemoteAddr().String()) session := NewSubSession(conn) if err := session.ReadRequest(); err != nil { log.Errorf("[%s] read httpflv SubSession request error. err=%v", session.UniqueKey, err) return } log.Debugf("[%s] < read http request. uri=%s", session.UniqueKey, session.URI) if !server.obs.OnNewHTTPFLVSubSession(session) { session.Dispose() } err := session.RunLoop() log.Debugf("[%s] httpflv sub session loop done. err=%v", session.UniqueKey, err) server.obs.OnDelHTTPFLVSubSession(session) }