base

package
v0.0.0-...-3f607ea Latest Latest
Warning

This package is not in the latest version of its module.

Go to latest
Published: Dec 31, 2021 License: MIT Imports: 21 Imported by: 0

Documentation

Index

Constants

View Source
const (
	ErrorCodeSucc            = 0
	DespSucc                 = "succ"
	ErrorCodeGroupNotFound   = 1001
	DespGroupNotFound        = "group not found"
	ErrorCodeParamMissing    = 1002
	DespParamMissing         = "param missing"
	ErrorCodeSessionNotFound = 1003
	DespSessionNotFound      = "session not found"
)
View Source
const (
	// spec-rtmp_specification_1.0.pdf
	// 7.1. Types of Messages
	RtmpTypeIdAudio              uint8 = 8
	RtmpTypeIdVideo              uint8 = 9
	RtmpTypeIdMetadata           uint8 = 18 // RtmpTypeIdDataMessageAmf0
	RtmpTypeIdSetChunkSize       uint8 = 1
	RtmpTypeIdAck                uint8 = 3
	RtmpTypeIdUserControl        uint8 = 4
	RtmpTypeIdWinAckSize         uint8 = 5
	RtmpTypeIdBandwidth          uint8 = 6
	RtmpTypeIdCommandMessageAmf3 uint8 = 17
	RtmpTypeIdCommandMessageAmf0 uint8 = 20
	RtmpTypeIdAggregateMessage   uint8 = 22

	// user control message type
	RtmpUserControlStreamBegin uint8 = 0
	RtmpUserControlRecorded    uint8 = 4

	// spec-video_file_format_spec_v10.pdf
	// Video tags
	//   VIDEODATA
	//     FrameType UB[4]
	//     CodecId   UB[4]
	//   AVCVIDEOPACKET
	//     AVCPacketType   UI8
	//     CompositionTime SI24
	//     Data            UI8[n]
	RtmpFrameTypeKey   uint8 = 1
	RtmpFrameTypeInter uint8 = 2

	RtmpCodecIdAvc  uint8 = 7
	RtmpCodecIdHevc uint8 = 12

	RtmpAvcPacketTypeSeqHeader  uint8 = 0
	RtmpAvcPacketTypeNalu       uint8 = 1
	RtmpHevcPacketTypeSeqHeader       = RtmpAvcPacketTypeSeqHeader
	RtmpHevcPacketTypeNalu            = RtmpAvcPacketTypeNalu

	RtmpAvcKeyFrame    = RtmpFrameTypeKey<<4 | RtmpCodecIdAvc
	RtmpHevcKeyFrame   = RtmpFrameTypeKey<<4 | RtmpCodecIdHevc
	RtmpAvcInterFrame  = RtmpFrameTypeInter<<4 | RtmpCodecIdAvc
	RtmpHevcInterFrame = RtmpFrameTypeInter<<4 | RtmpCodecIdHevc

	// spec-video_file_format_spec_v10.pdf
	// Audio tags
	//   AUDIODATA
	//     SoundFormat UB[4]
	//     SoundRate   UB[2]
	//     SoundSize   UB[1]
	//     SoundType   UB[1]
	//   AACAUDIODATA
	//     AACPacketType UI8
	//     Data          UI8[n]
	RtmpSoundFormatAac         uint8 = 10 // 注意,视频的CodecId是后4位,音频是前4位
	RtmpAacPacketTypeSeqHeader       = 0
	RtmpAacPacketTypeRaw             = 1
)
View Source
const (
	// 注意,一般情况下,AVC使用96,AAC使用97,HEVC使用98
	// 但是我还遇到过:
	// HEVC使用96
	// AVC使用105
	RtpPacketTypeAvcOrHevc = 96
	RtpPacketTypeAac       = 97
	RtpPacketTypeHevc      = 98
)
View Source
const (
	// StatGroup.AudioCodec
	AudioCodecAac = "AAC"

	// StatGroup.VideoCodec
	VideoCodecAvc  = "H264"
	VideoCodecHevc = "H265"

	// StatSession.Protocol
	ProtocolRtmp    = "RTMP"
	ProtocolRtsp    = "RTSP"
	ProtocolHttpflv = "HTTP-FLV"
	ProtocolHttpts  = "HTTP-TS"
)
View Source
const (
	UkPreRtmpServerSession        = "RTMPPUBSUB"
	UkPreRtmpPushSession          = "RTMPPUSH"
	UkPreRtmpPullSession          = "RTMPPULL"
	UkPreRtspServerCommandSession = "RTSPSRVCMD"
	UkPreRtspPubSession           = "RTSPPUB"
	UkPreRtspSubSession           = "RTSPSUB"
	UkPreRtspPushSession          = "RTSPPUSH"
	UkPreRtspPullSession          = "RTSPPULL"
	UkPreFlvSubSession            = "FLVSUB"
	UkPreTsSubSession             = "TSSUB"
	UkPreFlvPullSession           = "FLVPULL"

	UkPreGroup    = "GROUP"
	UkPreHlsMuxer = "HLSMUXER"
	UkPreStreamer = "STREAMER"
)
View Source
const (
	DefaultRtmpPort  = 1935
	DefaultHttpPort  = 80
	DefaultHttpsPort = 443
	DefaultRtspPort  = 554
)
View Source
const HttpApiVersion = "v0.1.4"
View Source
const HttpNotifyVersion = "v0.1.0"
View Source
const LalVersion = "v0.22.0"

版本,该变量由外部脚本修改维护

View Source
const (
	NetworkTcp = "tcp"
)
View Source
const WsMagicStr = "258EAFA5-E914-47DA-95CA-C5AB0DC85B11"

Variables

View Source
var (
	ErrAddrEmpty             = errors.New("lal.base: http server addr empty")
	ErrMultiRegistForPattern = errors.New("lal.base: http server multiple registrations for pattern")
)
View Source
var (
	LalLibraryName = "lal"
	LalGithubRepo  = "github.com/yangjing0630/go-stream"
	LalGithubSite  = "https://github.com/yangjing0630/go-stream"
	LalDocSite     = "https://pengrl.com/lal"

	// e.g. lal v0.12.3 (github.com/yangjing0630/go-stream)
	LalFullInfo = LalLibraryName + " " + LalVersion + " (" + LalGithubRepo + ")"

	// e.g. 0.12.3
	LalVersionDot string

	// e.g. 0,12,3
	LalVersionComma string
)
View Source
var (
	// 植入rtmp握手随机字符串中
	// e.g. lal v0.12.3 (github.com/yangjing0630/go-stream)
	LalRtmpHandshakeWaterMark string

	// 植入rtmp server中的connect result信令中
	// 注意,有两个object,第一个object中的fmsVer我们保持通用公认的值,在第二个object中植入
	// e.g. 0,12,3
	LalRtmpConnectResultVersion string

	// e.g. lal0.12.3
	LalRtmpPushSessionConnectVersion string

	// e.g. lal0.12.3
	LalRtmpBuildMetadataEncoder string

	// e.g. lal/0.12.3
	LalHttpflvPullSessionUa string

	// e.g. lal0.12.3
	LalHttpflvSubSessionServer string

	// e.g. lal0.12.3
	LalHlsM3u8Server string

	// e.g. lal0.12.3
	LalHlsTsServer string

	// e.g. lal0.12.3
	LalRtspOptionsResponseServer string

	// e.g. lal0.12.3
	LalHttptsSubSessionServer string

	// e.g. lal0.12.3
	LalHttpApiServer string

	// e.g. lal/0.12.3
	LalRtspPullSessionUa string
)
View Source
var (
	ErrSessionNotStarted = errors.New("lal.base: session has not been started yet")
)
View Source
var ErrUrl = errors.New("lal.url: fxxk")
View Source
var (
	Logger = nazalog.GetGlobalLogger()
)

Functions

func GenUkFlvPullSession

func GenUkFlvPullSession() string

func GenUkFlvSubSession

func GenUkFlvSubSession() string

func GenUkGroup

func GenUkGroup() string

func GenUkHlsMuxer

func GenUkHlsMuxer() string

func GenUkRtmpPullSession

func GenUkRtmpPullSession() string

func GenUkRtmpPushSession

func GenUkRtmpPushSession() string

func GenUkRtmpServerSession

func GenUkRtmpServerSession() string

func GenUkRtspPubSession

func GenUkRtspPubSession() string

func GenUkRtspPullSession

func GenUkRtspPullSession() string

func GenUkRtspPushSession

func GenUkRtspPushSession() string

func GenUkRtspServerCommandSession

func GenUkRtspServerCommandSession() string

func GenUkRtspSubSession

func GenUkRtspSubSession() string

func GenUkStreamer

func GenUkStreamer() string

func GenUkTsSubSession

func GenUkTsSubSession() string

func MakeWsFrameHeader

func MakeWsFrameHeader(wsHeader WsHeader) (buf []byte)

func OsExitAndWaitPressIfWindows

func OsExitAndWaitPressIfWindows(code int)

func UpdateWebSocketHeader

func UpdateWebSocketHeader(secWebSocketKey string) []byte

Types

type ApiCtrlKickOutSession

type ApiCtrlKickOutSession struct {
	StreamName string `json:"stream_name"`
	SessionId  string `json:"session_id"`
}

type ApiCtrlStartPullReq

type ApiCtrlStartPullReq struct {
	Protocol   string `json:"protocol"`
	Addr       string `json:"addr"`
	AppName    string `json:"app_name"`
	StreamName string `json:"stream_name"`
	UrlParam   string `json:"url_param"`
}

type ApiStatAllGroup

type ApiStatAllGroup struct {
	HttpResponseBasic
	Data struct {
		Groups []StatGroup `json:"groups"`
	} `json:"data"`
}

type ApiStatGroup

type ApiStatGroup struct {
	HttpResponseBasic
	Data *StatGroup `json:"data"`
}

type ApiStatLalInfo

type ApiStatLalInfo struct {
	HttpResponseBasic
	Data LalInfo `json:"data"`
}

type AvPacket

type AvPacket struct {
	Timestamp   uint32
	PayloadType AvPacketPt
	Payload     []byte
}

不同场景使用时,字段含义可能不同。 使用AvPacket的地方,应注明各字段的含义。

type AvPacketPt

type AvPacketPt int
const (
	AvPacketPtUnknown AvPacketPt = -1
	AvPacketPtAvc     AvPacketPt = RtpPacketTypeAvcOrHevc
	AvPacketPtHevc    AvPacketPt = RtpPacketTypeHevc
	AvPacketPtAac     AvPacketPt = RtpPacketTypeAac
)

func (AvPacketPt) ReadableString

func (a AvPacketPt) ReadableString() string

type Handler

type Handler func(http.ResponseWriter, *http.Request)

type HttpResponseBasic

type HttpResponseBasic struct {
	ErrorCode int    `json:"error_code"`
	Desp      string `json:"desp"`
}

type HttpServerManager

type HttpServerManager struct {
	// contains filtered or unexported fields
}

func NewHttpServerManager

func NewHttpServerManager() *HttpServerManager

func (*HttpServerManager) AddListen

func (s *HttpServerManager) AddListen(addrCtx LocalAddrCtx, pattern string, handler Handler) error

@param pattern 必须以`/`开始,并以`/`结束

func (*HttpServerManager) Dispose

func (s *HttpServerManager) Dispose() error

func (*HttpServerManager) RunLoop

func (s *HttpServerManager) RunLoop() error

type HttpSubSession

type HttpSubSession struct {
	HttpSubSessionOption
	// contains filtered or unexported fields
}

func NewHttpSubSession

func NewHttpSubSession(option HttpSubSessionOption) *HttpSubSession

func (*HttpSubSession) AppName

func (session *HttpSubSession) AppName() string

func (*HttpSubSession) Dispose

func (session *HttpSubSession) Dispose() error

func (*HttpSubSession) GetStat

func (session *HttpSubSession) GetStat() StatSession

func (*HttpSubSession) IsAlive

func (session *HttpSubSession) IsAlive() (readAlive, writeAlive bool)

func (*HttpSubSession) RawQuery

func (session *HttpSubSession) RawQuery() string

func (*HttpSubSession) RunLoop

func (session *HttpSubSession) RunLoop() error

func (*HttpSubSession) StreamName

func (session *HttpSubSession) StreamName() string

func (*HttpSubSession) UniqueKey

func (session *HttpSubSession) UniqueKey() string

func (*HttpSubSession) UpdateStat

func (session *HttpSubSession) UpdateStat(intervalSec uint32)

func (*HttpSubSession) Url

func (session *HttpSubSession) Url() string

func (*HttpSubSession) Write

func (session *HttpSubSession) Write(b []byte)

func (*HttpSubSession) WriteHttpResponseHeader

func (session *HttpSubSession) WriteHttpResponseHeader(b []byte)

type HttpSubSessionOption

type HttpSubSessionOption struct {
	Conn          net.Conn
	ConnModOption connection.ModOption
	Uk            string // unique key
	Protocol      string
	UrlCtx        UrlContext
	IsWebSocket   bool
	WebSocketKey  string
}

type IBufWriter

type IBufWriter interface {
	Write(p []byte)
	Flush()
}

func NewWriterFuncSize

func NewWriterFuncSize(wr WriterFunc, size int) IBufWriter

type IClientSessionLifecycle

type IClientSessionLifecycle interface {
	// 关闭session
	// 业务方想主动关闭session时调用
	// 注意,Start成功后的session,必须显示调用Dispose释放资源(即使是被动接收到了WaitChan信号)
	Dispose() error

	// Start成功后,可使用这个channel来接收session结束的信号
	WaitChan() <-chan error
}

调用约束:对于Client类型的Session,调用Start函数并返回成功后才能调用,否则行为未定义

type IObject

type IObject interface {
	// 对象的全局唯一标识
	UniqueKey() string
}

type IServerSessionLifecycle

type IServerSessionLifecycle interface {
	// 开启session的事件循环,阻塞直到session结束
	RunLoop() error

	// 主动关闭session时调用
	Dispose() error
}

type ISessionStat

type ISessionStat interface {
	// 周期性调用该函数,用于计算bitrate
	//
	// @param intervalSec 距离上次调用的时间间隔,单位毫秒
	UpdateStat(intervalSec uint32)

	// 获取session状态
	//
	// @return 注意,结构体中的`Bitrate`的值由最近一次`func UpdateStat`调用计算决定,其他值为当前最新值
	GetStat() StatSession

	// 周期性调用该函数,判断是否有读取、写入数据
	// 注意,判断的依据是,距离上次调用该函数的时间间隔内,是否有读取、写入数据
	// 注意,不活跃,并一定是链路或网络有问题,也可能是业务层没有写入数据
	//
	// @return readAlive  读取是否获取
	// @return writeAlive 写入是否活跃
	IsAlive() (readAlive, writeAlive bool)
}

调用约束:对于Client类型的Session,调用Start函数并返回成功后才能调用,否则行为未定义

type ISessionUrlContext

type ISessionUrlContext interface {
	Url() string
	AppName() string
	StreamName() string
	RawQuery() string
}

获取和流地址相关的信息

调用约束:对于Client类型的Session,调用Start函数并返回成功后才能调用,否则行为未定义

type LalInfo

type LalInfo struct {
	ServerId      string `json:"server_id"`
	BinInfo       string `json:"bin_info"`
	LalVersion    string `json:"lal_version"`
	ApiVersion    string `json:"api_version"`
	NotifyVersion string `json:"notify_version"`
	StartTime     string `json:"start_time"`
}

type LocalAddrCtx

type LocalAddrCtx struct {
	IsHttps  bool
	Addr     string
	CertFile string
	KeyFile  string

	Network string // 默认为NetworkTcp
}

type PubStartInfo

type PubStartInfo struct {
	SessionEventCommonInfo
}

type PubStopInfo

type PubStopInfo struct {
	SessionEventCommonInfo
}

type RtmpConnectInfo

type RtmpConnectInfo struct {
	ServerId   string `json:"server_id"`
	SessionId  string `json:"session_id"`
	RemoteAddr string `json:"remote_addr"`
	App        string `json:"app"`
	FlashVer   string `json:"flashVer"`
	TcUrl      string `json:"tcUrl"`
}

type RtmpHeader

type RtmpHeader struct {
	Csid         int
	MsgLen       uint32 // 不包含header的大小
	MsgTypeId    uint8  // 8 audio 9 video 18 metadata
	MsgStreamId  int
	TimestampAbs uint32 // 经过计算得到的流上的绝对时间戳,单位毫秒
}

type RtmpMsg

type RtmpMsg struct {
	Header  RtmpHeader
	Payload []byte // Payload不包含Header内容。如果需要将RtmpMsg序列化成RTMP chunk,可调用rtmp.ChunkDivider相关的函数
}

func (RtmpMsg) Clone

func (msg RtmpMsg) Clone() (ret RtmpMsg)

func (RtmpMsg) IsAacSeqHeader

func (msg RtmpMsg) IsAacSeqHeader() bool

func (RtmpMsg) IsAvcKeyNalu

func (msg RtmpMsg) IsAvcKeyNalu() bool

func (RtmpMsg) IsAvcKeySeqHeader

func (msg RtmpMsg) IsAvcKeySeqHeader() bool

func (RtmpMsg) IsHevcKeyNalu

func (msg RtmpMsg) IsHevcKeyNalu() bool

func (RtmpMsg) IsHevcKeySeqHeader

func (msg RtmpMsg) IsHevcKeySeqHeader() bool

func (RtmpMsg) IsVideoKeyNalu

func (msg RtmpMsg) IsVideoKeyNalu() bool

func (RtmpMsg) IsVideoKeySeqHeader

func (msg RtmpMsg) IsVideoKeySeqHeader() bool

type ServerCtx

type ServerCtx struct {
	// contains filtered or unexported fields
}

type SessionEventCommonInfo

type SessionEventCommonInfo struct {
	Protocol      string `json:"protocol"`
	SessionId     string `json:"session_id"`
	RemoteAddr    string `json:"remote_addr"`
	ServerId      string `json:"server_id"`
	Url           string `json:"url"`
	AppName       string `json:"app_name"`
	StreamName    string `json:"stream_name"`
	UrlParam      string `json:"url_param"`
	HasInSession  bool   `json:"has_in_session"`
	HasOutSession bool   `json:"has_out_session"`
}

type StatGroup

type StatGroup struct {
	StreamName  string    `json:"stream_name"`
	AudioCodec  string    `json:"audio_codec"`
	VideoCodec  string    `json:"video_codec"`
	VideoWidth  int       `json:"video_width"`
	VideoHeight int       `json:"video_height"`
	StatPub     StatPub   `json:"pub"`
	StatSubs    []StatSub `json:"subs"`
	StatPull    StatPull  `json:"pull"`
}

type StatPub

type StatPub struct {
	StatSession
}

func StatSession2Pub

func StatSession2Pub(ss StatSession) (ret StatPub)

type StatPull

type StatPull struct {
	StatSession
}

func StatSession2Pull

func StatSession2Pull(ss StatSession) (ret StatPull)

type StatSession

type StatSession struct {
	Protocol   string `json:"protocol"`
	SessionId  string `json:"session_id"`
	RemoteAddr string `json:"remote_addr"`
	StartTime  string `json:"start_time"`

	ReadBytesSum  uint64 `json:"read_bytes_sum"`
	WroteBytesSum uint64 `json:"wrote_bytes_sum"`
	Bitrate       int    `json:"bitrate"`
	ReadBitrate   int    `json:"read_bitrate"`
	WriteBitrate  int    `json:"write_bitrate"`
}

type StatSub

type StatSub struct {
	StatSession
}

func StatSession2Sub

func StatSession2Sub(ss StatSession) (ret StatSub)

type SubStartInfo

type SubStartInfo struct {
	SessionEventCommonInfo
}

type SubStopInfo

type SubStopInfo struct {
	SessionEventCommonInfo
}

type UpdateInfo

type UpdateInfo struct {
	ServerId string      `json:"server_id"`
	Groups   []StatGroup `json:"groups"`
}

type UrlContext

type UrlContext struct {
	Url string

	Scheme       string
	Username     string
	Password     string
	StdHost      string // host or host:port
	HostWithPort string
	Host         string
	Port         int

	//UrlPathContext
	PathWithRawQuery    string
	Path                string
	PathWithoutLastItem string // 注意,没有前面的'/',也没有后面的'/'
	LastItemOfPath      string // 注意,没有前面的'/'
	RawQuery            string

	RawUrlWithoutUserInfo string
}

TODO chef: 考虑把rawUrl也放入其中

func ParseHttpUrl

func ParseHttpUrl(rawUrl string, isHttps bool, suffix string) (ctx UrlContext, err error)

func ParseHttpflvUrl

func ParseHttpflvUrl(rawUrl string, isHttps bool) (ctx UrlContext, err error)

func ParseHttptsUrl

func ParseHttptsUrl(rawUrl string, isHttps bool) (ctx UrlContext, err error)

func ParseRtmpUrl

func ParseRtmpUrl(rawUrl string) (ctx UrlContext, err error)

func ParseRtspUrl

func ParseRtspUrl(rawUrl string) (ctx UrlContext, err error)

func ParseUrl

func ParseUrl(rawUrl string, defaultPort int) (ctx UrlContext, err error)

type UrlPathContext

type UrlPathContext struct {
	PathWithRawQuery    string
	Path                string
	PathWithoutLastItem string // 注意,没有前面的'/',也没有后面的'/'
	LastItemOfPath      string // 注意,没有前面的'/'
	RawQuery            string
}

type WriterFunc

type WriterFunc func(p []byte)

type WsHeader

type WsHeader struct {
	Fin    bool
	Rsv1   bool
	Rsv2   bool
	Rsv3   bool
	Opcode WsOpcode

	PayloadLength uint64

	Masked  bool
	MaskKey uint32
}

type WsOpcode

type WsOpcode = uint8

The WebSocket Protocol https://tools.ietf.org/html/rfc6455

0 1 2 3 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 2 3 4 5 6 7 8 9 0 1 +-+-+-+-+-------+-+-------------+-------------------------------+ |F|R|R|R| opcode|M| Payload len | Extended payload length | |I|S|S|S| (4) |A| (7) | (16/64) | |N|V|V|V| |S| | (if payload len==126/127) | | |1|2|3| |K| | | +-+-+-+-+-------+-+-------------+ - - - - - - - - - - - - - - - + | Extended payload length continued, if payload len == 127 | + - - - - - - - - - - - - - - - +-------------------------------+ | |Masking-key, if MASK set to 1 | +-------------------------------+-------------------------------+ | Masking-key (continued) | Payload Data | +-------------------------------- - - - - - - - - - - - - - - - + : Payload Data continued ... : + - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - - + | Payload Data continued ... | +---------------------------------------------------------------+ opcode: * %x0 denotes a continuation frame * %x1 denotes a text frame * %x2 denotes a binary frame * %x3-7 are reserved for further non-control frames * %x8 denotes a connection close * %x9 denotes a ping * %xA denotes a pong * %xB-F are reserved for further control frames Payload length: 7 bits, 7+16 bits, or 7+64 bits Masking-key: 0 or 4 bytes mark 加密

for i := 0; i < datalen; i {
    m := markingkeys[i%4]
    data[i] = msg[i] ^ m
}
const (
	Wso_Continuous WsOpcode = iota //连续消息片断
	Wso_Text                       //文本消息片断,
	Wso_Binary                     //二进制消息片断,
	//非控制消息片断保留的操作码,
	Wso_Rsv3
	Wso_Rsv4
	Wso_Rsv5
	Wso_Rsv6
	Wso_Rsv7
	Wso_Close //连接关闭,
	Wso_Ping  //心跳检查的ping,
	Wso_Pong  //心跳检查的pong,
	//为将来的控制消息片断的保留操作码
	Wso_RsvB
	Wso_RsvC
	Wso_RsvD
	Wso_RsvE
	Wso_RsvF
)

Jump to

Keyboard shortcuts

? : This menu
/ : Search site
f or F : Jump to
y or Y : Canonical URL