diff --git a/README.md b/README.md index 7a385aa..64a6d46 100644 --- a/README.md +++ b/README.md @@ -7,8 +7,9 @@ SIP UA library for client/b2bua using golang - [x] Transports UDP/TCP/TLS/WS/WSS. - [x] Simple pure Go SIP Client. - [x] Simple pure Go B2BUA, support RFC8599, Google FCM/Apple PushKit. -- [ ] RTP relay (UDP<-->UDP, WebRTC/ICE<->UDP) -- [ ] WebRTC2SIP Gateway. +- [x] RTP relay (UDP<-->UDP, WebRTC/ICE<->UDP) +- [x] WebRTC2SIP Gateway. +- [ ] SRTP/SDES for SIP Client. ## Running the examples diff --git a/examples/b2bua/b2bua/b2bua.go b/examples/b2bua/b2bua/b2bua.go index 4942ae1..55ae1ac 100644 --- a/examples/b2bua/b2bua/b2bua.go +++ b/examples/b2bua/b2bua/b2bua.go @@ -6,29 +6,18 @@ import ( "github.com/cloudwebrtc/go-sip-ua/examples/b2bua/fcm" "github.com/cloudwebrtc/go-sip-ua/examples/b2bua/pushkit" "github.com/cloudwebrtc/go-sip-ua/examples/b2bua/registry" + "github.com/pion/webrtc/v3" "github.com/cloudwebrtc/go-sip-ua/pkg/account" "github.com/cloudwebrtc/go-sip-ua/pkg/auth" - "github.com/cloudwebrtc/go-sip-ua/pkg/session" "github.com/cloudwebrtc/go-sip-ua/pkg/stack" "github.com/cloudwebrtc/go-sip-ua/pkg/ua" "github.com/cloudwebrtc/go-sip-ua/pkg/utils" "github.com/ghettovoice/gosip/log" "github.com/ghettovoice/gosip/sip" - "github.com/ghettovoice/gosip/sip/parser" "github.com/ghettovoice/gosip/transport" ) -type B2BCall struct { - src *session.Session - //TODO: Add support for forked calls - dest *session.Session -} - -func (b *B2BCall) ToString() string { - return b.src.Contact() + " => " + b.dest.Contact() -} - func pushCallback(pn *registry.PNParams, payload map[string]string) error { fmt.Printf("Handle Push Request:\nprovider=%v\nparam=%v\nprid=%v\npayload=%v", pn.Provider, pn.Param, pn.PRID, payload) switch pn.Provider { @@ -42,43 +31,95 @@ func pushCallback(pn *registry.PNParams, payload map[string]string) error { return fmt.Errorf("%v provider not found", pn.Provider) } +type B2BUAConfig struct { + UserAgent string + DisableAuth bool + EnableTLS bool + EnableWebSocket bool + EnableRFC8599 bool + Dns string + ListenAddress string + UdpPort int + TcpPort int + TlsPort int + WSPort int + WSSPort int + SSLCert string + SSLKey string + UdpPortRange []int + UaMediaConfig UserAgentMediaConfig +} + // B2BUA . type B2BUA struct { - stack *stack.SipStack - ua *ua.UserAgent accounts map[string]string registry registry.Registry - domains []string - calls []*B2BCall + service *CallService rfc8599 *registry.RFC8599 } var ( - logger log.Logger + logger log.Logger + b2buaConfig *B2BUAConfig ) func init() { - logger = utils.NewLogrusLogger(log.InfoLevel, "B2BUA", nil) + logger = utils.NewLogrusLogger(utils.DefaultLogLevel, "B2BUA", nil) + b2buaConfig = &B2BUAConfig{ + UserAgent: "Go B2BUA/1.0.0", + DisableAuth: false, + EnableWebSocket: false, + EnableTLS: false, + Dns: "8.8.8.8", + ListenAddress: "0.0.0.0", + UdpPort: 5060, + TcpPort: 5060, + TlsPort: 5061, + WSPort: 8088, + WSSPort: 8089, + SSLCert: "certs/cert.pem", + SSLKey: "certs/key.pem", + UdpPortRange: []int{60000, 65535}, + UaMediaConfig: UserAgentMediaConfig{ + Codecs: []string{"PCMU", "PCMA", "H264"}, + ExternalRtpAddress: "0.0.0.0", + RtcpFeedback: []webrtc.RTCPFeedback{{"goog-remb", ""}, {"transport-cc", ""}, {"ccm", "fir"}, {"nack", ""}, {"nack", "pli"}}, + }, + } } -//NewB2BUA . -func NewB2BUA(disableAuth bool, enableTLS bool) *B2BUA { +// NewB2BUA . +func NewB2BUA(disableAuth bool, enableTLS bool, enableWebSocket bool, enalbeRFC8599 bool) *B2BUA { + + b2buaConfig.DisableAuth = disableAuth + b2buaConfig.EnableTLS = enableTLS + b2buaConfig.EnableWebSocket = enableWebSocket + b2buaConfig.EnableRFC8599 = enalbeRFC8599 + + memRegistry := registry.Registry(registry.NewMemoryRegistry()) + b := &B2BUA{ - registry: registry.Registry(registry.NewMemoryRegistry()), + registry: memRegistry, accounts: make(map[string]string), - rfc8599: registry.NewRFC8599(pushCallback), + } + + if b2buaConfig.EnableRFC8599 { + b.rfc8599 = registry.NewRFC8599(pushCallback) + logger.Infof("RFC8599 enabled") } var authenticator *auth.ServerAuthorizer = nil - if !disableAuth { + if !b2buaConfig.DisableAuth { authenticator = auth.NewServerAuthorizer(b.requestCredential, "b2bua", false) + } else { + logger.Warn("Auth disabled") } stack := stack.NewSipStack(&stack.SipStackConfig{ - UserAgent: "Go B2BUA/1.0.0", + UserAgent: b2buaConfig.UserAgent, Extensions: []string{"replaces", "outbound"}, - Dns: "8.8.8.8", + Dns: b2buaConfig.Dns, ServerAuthManager: stack.ServerAuthManager{ Authenticator: authenticator, RequiresChallenge: b.requiresChallenge, @@ -87,179 +128,76 @@ func NewB2BUA(disableAuth bool, enableTLS bool) *B2BUA { stack.OnConnectionError(b.handleConnectionError) - if err := stack.Listen("udp", "0.0.0.0:5060"); err != nil { + var listenAddress = fmt.Sprintf("%s:%d", b2buaConfig.ListenAddress, b2buaConfig.UdpPort) + + if err := stack.Listen("udp", listenAddress); err != nil { logger.Panic(err) } + logger.Infof("listening on: udp://%s", listenAddress) - if err := stack.Listen("tcp", "0.0.0.0:5060"); err != nil { + listenAddress = fmt.Sprintf("%s:%d", b2buaConfig.ListenAddress, b2buaConfig.TcpPort) + if err := stack.Listen("tcp", listenAddress); err != nil { logger.Panic(err) } - if enableTLS { - tlsOptions := &transport.TLSConfig{Cert: "certs/cert.pem", Key: "certs/key.pem"} - - if err := stack.ListenTLS("tls", "0.0.0.0:5061", tlsOptions); err != nil { - logger.Panic(err) - } + logger.Infof("listening on: tcp://%s", listenAddress) - if err := stack.ListenTLS("wss", "0.0.0.0:5081", tlsOptions); err != nil { + if b2buaConfig.EnableWebSocket { + listenAddress = fmt.Sprintf("%s:%d", b2buaConfig.ListenAddress, b2buaConfig.WSPort) + if err := stack.Listen("ws", listenAddress); err != nil { logger.Panic(err) } + logger.Infof("listening on: ws://%s", listenAddress) } - ua := ua.NewUserAgent(&ua.UserAgentConfig{ + if b2buaConfig.EnableTLS { - SipStack: stack, - }) - - ua.InviteStateHandler = func(sess *session.Session, req *sip.Request, resp *sip.Response, state session.Status) { - logger.Infof("InviteStateHandler: state => %v, type => %s", state, sess.Direction()) - - switch state { - // Handle incoming call. - case session.InviteReceived: - to, _ := (*req).To() - from, _ := (*req).From() - caller := from.Address - called := to.Address - - doInvite := func(instance *registry.ContactInstance) { - displayName := "" - if from.DisplayName != nil { - displayName = from.DisplayName.String() - } - - // Create a temporary profile. In the future, it will support reading profiles from files or data - // For example: use a specific ip or sip account as outbound trunk - profile := account.NewProfile(caller, displayName, nil, 0, stack) - - recipient, err2 := parser.ParseSipUri("sip:" + called.User().String() + "@" + instance.Source + ";transport=" + instance.Transport) - if err2 != nil { - logger.Error(err2) - } - - offer := sess.RemoteSdp() - dest, err := ua.Invite(profile, called, recipient, &offer) - if err != nil { - logger.Errorf("B-Leg session error: %v", err) - return - } - b.calls = append(b.calls, &B2BCall{src: sess, dest: dest}) - } + logger.Infof("TLS enabled: %s, %s", b2buaConfig.SSLCert, b2buaConfig.SSLKey) + tlsOptions := &transport.TLSConfig{Cert: b2buaConfig.SSLCert, Key: b2buaConfig.SSLKey} - // Try to find online contact records. - if contacts, found := b.registry.GetContacts(called); found { - sess.Provisional(100, "Trying") - for _, instance := range *contacts { - doInvite(instance) - } - return - } - - // Pushable: try to find pn-params in contact records. - // Try to push the UA and wait for it to wake up. - pusher, ok := b.rfc8599.TryPush(called, from) - if ok { - sess.Provisional(100, "Trying") - instance, err := pusher.WaitContactOnline() - if err != nil { - logger.Errorf("Push failed, error: %v", err) - sess.Reject(500, fmt.Sprint("Push failed")) - return - } - doInvite(instance) - return - } - - // Could not found any records - sess.Reject(404, fmt.Sprintf("%v Not found", called)) - - // Handle re-INVITE or UPDATE. - case session.ReInviteReceived: - logger.Infof("re-INVITE") - switch sess.Direction() { - case session.Incoming: - sess.Accept(200) - case session.Outgoing: - //TODO: Need to provide correct answer. - } - - // Handle 1XX - case session.EarlyMedia: - fallthrough - case session.Provisional: - call := b.findCall(sess) - if call != nil && call.dest == sess { - answer := call.dest.RemoteSdp() - call.src.ProvideAnswer(answer) - call.src.Provisional((*resp).StatusCode(), (*resp).Reason()) - } - - // Handle 200OK or ACK - case session.Confirmed: - //TODO: Add support for forked calls - call := b.findCall(sess) - if call != nil && call.dest == sess { - answer := call.dest.RemoteSdp() - call.src.ProvideAnswer(answer) - call.src.Accept(200) - } + listenAddress = fmt.Sprintf("%s:%d", b2buaConfig.ListenAddress, b2buaConfig.TlsPort) + if err := stack.ListenTLS("tls", listenAddress, tlsOptions); err != nil { + logger.Panic(err) + } - // Handle 4XX+ - case session.Failure: - fallthrough - case session.Canceled: - fallthrough - case session.Terminated: - //TODO: Add support for forked calls - call := b.findCall(sess) - if call != nil { - if call.src == sess { - call.dest.End() - } else if call.dest == sess { - call.src.End() - } + logger.Infof("listening on: tls://%s", listenAddress) + if b2buaConfig.EnableWebSocket { + listenAddress = fmt.Sprintf("%s:%d", b2buaConfig.ListenAddress, b2buaConfig.WSSPort) + if err := stack.ListenTLS("wss", "0.0.0.0:8089", tlsOptions); err != nil { + logger.Panic(err) } - b.removeCall(sess) - + logger.Infof("listening on: wss://%s", listenAddress) } } - ua.RegisterStateHandler = func(state account.RegisterState) { - logger.Infof("RegisterStateHandler: state => %v", state) - } + ua := ua.NewUserAgent(&ua.UserAgentConfig{ + SipStack: stack, + }) + ua.RegisterStateHandler = b.registerStateHandler stack.OnRequest(sip.REGISTER, b.handleRegister) - b.stack = stack - b.ua = ua + + b.service = NewCallService(stack, ua, memRegistry, b.rfc8599) + ua.InviteStateHandler = b.service.inviteStateHandler return b } -func (b *B2BUA) Calls() []*B2BCall { - return b.calls +func (b *B2BUA) registerStateHandler(state account.RegisterState) { + logger.Infof("RegisterStateHandler: state => %v", state) } -func (b *B2BUA) findCall(sess *session.Session) *B2BCall { - for _, call := range b.calls { - if call.src == sess || call.dest == sess { - return call - } - } - return nil +func (b *B2BUA) BridgedCalls() []*CallBridge { + return b.service.callBridges } -func (b *B2BUA) removeCall(sess *session.Session) { - for idx, call := range b.calls { - if call.src == sess || call.dest == sess { - b.calls = append(b.calls[:idx], b.calls[idx+1:]...) - return - } - } +// Originate . +func (b *B2BUA) Originate(source string, destination string) { + b.service.Originate(source, destination) } -//Shutdown . +// Shutdown . func (b *B2BUA) Shutdown() { - b.ua.Shutdown() + b.service.Shutdown() } func (b *B2BUA) requiresChallenge(req sip.Request) bool { @@ -287,22 +225,22 @@ func (b *B2BUA) requiresChallenge(req sip.Request) bool { return false } -//AddAccount . +// AddAccount . func (b *B2BUA) AddAccount(username string, password string) { b.accounts[username] = password } -//GetAccounts . +// GetAccounts . func (b *B2BUA) GetAccounts() map[string]string { return b.accounts } -//GetRegistry . +// GetRegistry . func (b *B2BUA) GetRegistry() registry.Registry { return b.registry } -//GetRFC8599 . +// GetRFC8599 . func (b *B2BUA) GetRFC8599() *registry.RFC8599 { return b.rfc8599 } @@ -343,7 +281,6 @@ func (b *B2BUA) handleRegister(request sip.Request, tx sip.ServerTransaction) { sip.CopyHeaders("Expires", request, resp) utils.BuildContactHeader("Contact", request, resp, &expires) tx.Respond(resp) - } func (b *B2BUA) handleConnectionError(connError *transport.ConnectionError) { diff --git a/examples/b2bua/b2bua/buffer/bucket.go b/examples/b2bua/b2bua/buffer/bucket.go new file mode 100644 index 0000000..8344486 --- /dev/null +++ b/examples/b2bua/b2bua/buffer/bucket.go @@ -0,0 +1,114 @@ +package buffer + +import ( + "encoding/binary" + "math" +) + +const maxPktSize = 1500 + +type Bucket struct { + buf []byte + src *[]byte + + init bool + step int + headSN uint16 + maxSteps int +} + +func NewBucket(buf *[]byte) *Bucket { + return &Bucket{ + src: buf, + buf: *buf, + maxSteps: int(math.Floor(float64(len(*buf))/float64(maxPktSize))) - 1, + } +} + +func (b *Bucket) AddPacket(pkt []byte, sn uint16, latest bool) ([]byte, error) { + if !b.init { + b.headSN = sn - 1 + b.init = true + } + if !latest { + return b.set(sn, pkt) + } + diff := sn - b.headSN + b.headSN = sn + for i := uint16(1); i < diff; i++ { + b.step++ + if b.step >= b.maxSteps { + b.step = 0 + } + } + return b.push(pkt), nil +} + +func (b *Bucket) GetPacket(buf []byte, sn uint16) (i int, err error) { + p := b.get(sn) + if p == nil { + err = errPacketNotFound + return + } + i = len(p) + if cap(buf) < i { + err = errBufferTooSmall + return + } + if len(buf) < i { + buf = buf[:i] + } + copy(buf, p) + return +} + +func (b *Bucket) push(pkt []byte) []byte { + binary.BigEndian.PutUint16(b.buf[b.step*maxPktSize:], uint16(len(pkt))) + off := b.step*maxPktSize + 2 + copy(b.buf[off:], pkt) + b.step++ + if b.step > b.maxSteps { + b.step = 0 + } + return b.buf[off : off+len(pkt)] +} + +func (b *Bucket) get(sn uint16) []byte { + pos := b.step - int(b.headSN-sn+1) + if pos < 0 { + if pos*-1 > b.maxSteps+1 { + return nil + } + pos = b.maxSteps + pos + 1 + } + off := pos * maxPktSize + if off > len(b.buf) { + return nil + } + if binary.BigEndian.Uint16(b.buf[off+4:off+6]) != sn { + return nil + } + sz := int(binary.BigEndian.Uint16(b.buf[off : off+2])) + return b.buf[off+2 : off+2+sz] +} + +func (b *Bucket) set(sn uint16, pkt []byte) ([]byte, error) { + if b.headSN-sn >= uint16(b.maxSteps+1) { + return nil, errPacketTooOld + } + pos := b.step - int(b.headSN-sn+1) + if pos < 0 { + pos = b.maxSteps + pos + 1 + } + off := pos * maxPktSize + if off > len(b.buf) || off < 0 { + return nil, errPacketTooOld + } + // Do not overwrite if packet exist + if binary.BigEndian.Uint16(b.buf[off+4:off+6]) == sn { + return nil, errRTXPacket + } + binary.BigEndian.PutUint16(b.buf[off:], uint16(len(pkt))) + copy(b.buf[off+2:], pkt) + return b.buf[off+2 : off+2+len(pkt)], nil +} diff --git a/examples/b2bua/b2bua/buffer/bucket_test.go b/examples/b2bua/b2bua/buffer/bucket_test.go new file mode 100644 index 0000000..706387f --- /dev/null +++ b/examples/b2bua/b2bua/buffer/bucket_test.go @@ -0,0 +1,139 @@ +package buffer + +import ( + "testing" + + "github.com/pion/rtp" + "github.com/stretchr/testify/assert" +) + +var TestPackets = []*rtp.Packet{ + { + Header: rtp.Header{ + SequenceNumber: 1, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 3, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 4, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 6, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 7, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 10, + }, + }, +} + +func Test_queue(t *testing.T) { + b := make([]byte, 25000) + q := NewBucket(&b) + + for _, p := range TestPackets { + p := p + buf, err := p.Marshal() + assert.NoError(t, err) + assert.NotPanics(t, func() { + q.AddPacket(buf, p.SequenceNumber, true) + }) + } + var expectedSN uint16 + expectedSN = 6 + np := rtp.Packet{} + buff := make([]byte, maxPktSize) + i, err := q.GetPacket(buff, 6) + assert.NoError(t, err) + err = np.Unmarshal(buff[:i]) + assert.NoError(t, err) + assert.Equal(t, expectedSN, np.SequenceNumber) + + np2 := &rtp.Packet{ + Header: rtp.Header{ + SequenceNumber: 8, + }, + } + buf, err := np2.Marshal() + assert.NoError(t, err) + expectedSN = 8 + q.AddPacket(buf, 8, false) + i, err = q.GetPacket(buff, expectedSN) + assert.NoError(t, err) + err = np.Unmarshal(buff[:i]) + assert.NoError(t, err) + assert.Equal(t, expectedSN, np.SequenceNumber) + + _, err = q.AddPacket(buf, 8, false) + assert.ErrorIs(t, err, errRTXPacket) +} + +func Test_queue_edges(t *testing.T) { + var TestPackets = []*rtp.Packet{ + { + Header: rtp.Header{ + SequenceNumber: 65533, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 65534, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 2, + }, + }, + } + b := make([]byte, 25000) + q := NewBucket(&b) + for _, p := range TestPackets { + p := p + assert.NotNil(t, p) + assert.NotPanics(t, func() { + p := p + buf, err := p.Marshal() + assert.NoError(t, err) + assert.NotPanics(t, func() { + q.AddPacket(buf, p.SequenceNumber, true) + }) + }) + } + var expectedSN uint16 + expectedSN = 65534 + np := rtp.Packet{} + buff := make([]byte, maxPktSize) + i, err := q.GetPacket(buff, expectedSN) + assert.NoError(t, err) + err = np.Unmarshal(buff[:i]) + assert.NoError(t, err) + assert.Equal(t, expectedSN, np.SequenceNumber) + + np2 := rtp.Packet{ + Header: rtp.Header{ + SequenceNumber: 65535, + }, + } + buf, err := np2.Marshal() + assert.NoError(t, err) + q.AddPacket(buf, np2.SequenceNumber, false) + i, err = q.GetPacket(buff, expectedSN+1) + assert.NoError(t, err) + err = np.Unmarshal(buff[:i]) + assert.NoError(t, err) + assert.Equal(t, expectedSN+1, np.SequenceNumber) +} diff --git a/examples/b2bua/b2bua/buffer/buffer.go b/examples/b2bua/b2bua/buffer/buffer.go new file mode 100644 index 0000000..b7bef0e --- /dev/null +++ b/examples/b2bua/b2bua/buffer/buffer.go @@ -0,0 +1,596 @@ +package buffer + +import ( + "encoding/binary" + "io" + "strings" + "sync" + "sync/atomic" + "time" + + "github.com/gammazero/deque" + "github.com/go-logr/logr" + "github.com/pion/rtcp" + "github.com/pion/rtp" + "github.com/pion/sdp/v3" + "github.com/pion/webrtc/v3" +) + +const ( + maxSN = 1 << 16 + + reportDelta = 1e9 +) + +// Logger is an implementation of logr.Logger. If is not provided - will be turned off. +var Logger logr.Logger = logr.Discard() + +type pendingPackets struct { + arrivalTime int64 + packet []byte +} + +type ExtPacket struct { + Head bool + Cycle uint32 + Arrival int64 + Packet rtp.Packet + Payload interface{} + KeyFrame bool +} + +// Buffer contains all packets +type Buffer struct { + sync.Mutex + bucket *Bucket + nacker *nackQueue + videoPool *sync.Pool + audioPool *sync.Pool + codecType webrtc.RTPCodecType + extPackets deque.Deque + pPackets []pendingPackets + closeOnce sync.Once + mediaSSRC uint32 + clockRate uint32 + maxBitrate uint64 + lastReport int64 + twccExt uint8 + audioExt uint8 + bound bool + closed atomicBool + mime string + + // supported feedbacks + remb bool + nack bool + twcc bool + audioLevel bool + + minPacketProbe int + lastPacketRead int + maxTemporalLayer int32 + bitrate uint64 + bitrateHelper uint64 + lastSRNTPTime uint64 + lastSRRTPTime uint32 + lastSRRecv int64 // Represents wall clock of the most recent sender report arrival + baseSN uint16 + cycles uint32 + lastRtcpPacketTime int64 // Time the last RTCP packet was received. + lastRtcpSrTime int64 // Time the last RTCP SR was received. Required for DLSR computation. + lastTransit uint32 + maxSeqNo uint16 // The highest sequence number received in an RTP data packet + + stats Stats + + latestTimestamp uint32 // latest received RTP timestamp on packet + latestTimestampTime int64 // Time of the latest timestamp (in nanos since unix epoch) + + // callbacks + onClose func() + onAudioLevel func(level uint8) + feedbackCB func([]rtcp.Packet) + feedbackTWCC func(sn uint16, timeNS int64, marker bool) + + // logger + logger logr.Logger +} + +type Stats struct { + LastExpected uint32 + LastReceived uint32 + LostRate float32 + PacketCount uint32 // Number of packets received from this source. + Jitter float64 // An estimate of the statistical variance of the RTP data packet inter-arrival time. + TotalByte uint64 +} + +// BufferOptions provides configuration options for the buffer +type Options struct { + MaxBitRate uint64 +} + +// NewBuffer constructs a new Buffer +func NewBuffer(ssrc uint32, vp, ap *sync.Pool, logger logr.Logger) *Buffer { + b := &Buffer{ + mediaSSRC: ssrc, + videoPool: vp, + audioPool: ap, + logger: logger, + } + b.extPackets.SetMinCapacity(7) + return b +} + +func (b *Buffer) Bind(params webrtc.RTPParameters, o Options) { + b.Lock() + defer b.Unlock() + + if len(params.Codecs) == 0 { + return + } + + codec := params.Codecs[0] + b.clockRate = codec.ClockRate + b.maxBitrate = o.MaxBitRate + b.mime = strings.ToLower(codec.MimeType) + + switch { + case strings.HasPrefix(b.mime, "audio/"): + b.codecType = webrtc.RTPCodecTypeAudio + b.bucket = NewBucket(b.audioPool.Get().(*[]byte)) + case strings.HasPrefix(b.mime, "video/"): + b.codecType = webrtc.RTPCodecTypeVideo + b.bucket = NewBucket(b.videoPool.Get().(*[]byte)) + default: + b.codecType = webrtc.RTPCodecType(0) + } + + for _, ext := range params.HeaderExtensions { + if ext.URI == sdp.TransportCCURI { + b.twccExt = uint8(ext.ID) + break + } + } + + if b.codecType == webrtc.RTPCodecTypeVideo { + for _, fb := range codec.RTCPFeedback { + switch fb.Type { + case webrtc.TypeRTCPFBGoogREMB: + b.logger.V(1).Info("Setting feedback", "type", "webrtc.TypeRTCPFBGoogREMB") + b.remb = true + case webrtc.TypeRTCPFBTransportCC: + b.logger.V(1).Info("Setting feedback", "type", webrtc.TypeRTCPFBTransportCC) + b.twcc = true + case webrtc.TypeRTCPFBNACK: + b.logger.V(1).Info("Setting feedback", "type", webrtc.TypeRTCPFBNACK) + b.nacker = newNACKQueue() + b.nack = true + } + } + } else if b.codecType == webrtc.RTPCodecTypeAudio { + for _, h := range params.HeaderExtensions { + if h.URI == sdp.AudioLevelURI { + b.audioLevel = true + b.audioExt = uint8(h.ID) + } + } + } + + for _, pp := range b.pPackets { + b.calc(pp.packet, pp.arrivalTime) + } + b.pPackets = nil + b.bound = true + + b.logger.V(1).Info("NewBuffer", "MaxBitRate", o.MaxBitRate) +} + +// Write adds a RTP Packet, out of order, new packet may be arrived later +func (b *Buffer) Write(pkt []byte) (n int, err error) { + b.Lock() + defer b.Unlock() + + if b.closed.get() { + err = io.EOF + return + } + + if !b.bound { + packet := make([]byte, len(pkt)) + copy(packet, pkt) + b.pPackets = append(b.pPackets, pendingPackets{ + packet: packet, + arrivalTime: time.Now().UnixNano(), + }) + return + } + + b.calc(pkt, time.Now().UnixNano()) + + return +} + +func (b *Buffer) Read(buff []byte) (n int, err error) { + for { + if b.closed.get() { + err = io.EOF + return + } + b.Lock() + if b.pPackets != nil && len(b.pPackets) > b.lastPacketRead { + if len(buff) < len(b.pPackets[b.lastPacketRead].packet) { + err = errBufferTooSmall + b.Unlock() + return + } + n = len(b.pPackets[b.lastPacketRead].packet) + copy(buff, b.pPackets[b.lastPacketRead].packet) + b.lastPacketRead++ + b.Unlock() + return + } + b.Unlock() + time.Sleep(25 * time.Millisecond) + } +} + +func (b *Buffer) ReadExtended() (*ExtPacket, error) { + for { + if b.closed.get() { + return nil, io.EOF + } + b.Lock() + if b.extPackets.Len() > 0 { + extPkt := b.extPackets.PopFront().(*ExtPacket) + b.Unlock() + return extPkt, nil + } + b.Unlock() + time.Sleep(10 * time.Millisecond) + } +} + +func (b *Buffer) Close() error { + b.Lock() + defer b.Unlock() + + b.closeOnce.Do(func() { + if b.bucket != nil && b.codecType == webrtc.RTPCodecTypeVideo { + b.videoPool.Put(b.bucket.src) + } + if b.bucket != nil && b.codecType == webrtc.RTPCodecTypeAudio { + b.audioPool.Put(b.bucket.src) + } + b.closed.set(true) + b.onClose() + }) + return nil +} + +func (b *Buffer) OnClose(fn func()) { + b.onClose = fn +} + +func (b *Buffer) calc(pkt []byte, arrivalTime int64) { + sn := binary.BigEndian.Uint16(pkt[2:4]) + + if b.stats.PacketCount == 0 { + b.baseSN = sn + b.maxSeqNo = sn + b.lastReport = arrivalTime + } else if (sn-b.maxSeqNo)&0x8000 == 0 { + if sn < b.maxSeqNo { + b.cycles += maxSN + } + if b.nack { + diff := sn - b.maxSeqNo + for i := uint16(1); i < diff; i++ { + var extSN uint32 + msn := sn - i + if msn > b.maxSeqNo && msn&0x8000 > 0 && b.maxSeqNo&0x8000 == 0 { + extSN = (b.cycles - maxSN) | uint32(msn) + } else { + extSN = b.cycles | uint32(msn) + } + b.nacker.push(extSN) + } + } + b.maxSeqNo = sn + } else if b.nack && (sn-b.maxSeqNo)&0x8000 > 0 { + var extSN uint32 + if sn > b.maxSeqNo && sn&0x8000 > 0 && b.maxSeqNo&0x8000 == 0 { + extSN = (b.cycles - maxSN) | uint32(sn) + } else { + extSN = b.cycles | uint32(sn) + } + b.nacker.remove(extSN) + } + + var p rtp.Packet + pb, err := b.bucket.AddPacket(pkt, sn, sn == b.maxSeqNo) + if err != nil { + if err == errRTXPacket { + return + } + return + } + if err = p.Unmarshal(pb); err != nil { + return + } + + b.stats.TotalByte += uint64(len(pkt)) + b.bitrateHelper += uint64(len(pkt)) + b.stats.PacketCount++ + + ep := ExtPacket{ + Head: sn == b.maxSeqNo, + Cycle: b.cycles, + Packet: p, + Arrival: arrivalTime, + } + + switch b.mime { + case "video/vp8": + vp8Packet := VP8{} + if err := vp8Packet.Unmarshal(p.Payload); err != nil { + return + } + ep.Payload = vp8Packet + ep.KeyFrame = vp8Packet.IsKeyFrame + case "video/h264": + ep.KeyFrame = isH264Keyframe(p.Payload) + } + + if b.minPacketProbe < 25 { + if sn < b.baseSN { + b.baseSN = sn + } + + if b.mime == "video/vp8" { + pld := ep.Payload.(VP8) + mtl := atomic.LoadInt32(&b.maxTemporalLayer) + if mtl < int32(pld.TID) { + atomic.StoreInt32(&b.maxTemporalLayer, int32(pld.TID)) + } + } + + b.minPacketProbe++ + } + + b.extPackets.PushBack(&ep) + + // if first time update or the timestamp is later (factoring timestamp wrap around) + latestTimestamp := atomic.LoadUint32(&b.latestTimestamp) + latestTimestampTimeInNanosSinceEpoch := atomic.LoadInt64(&b.latestTimestampTime) + if (latestTimestampTimeInNanosSinceEpoch == 0) || IsLaterTimestamp(p.Timestamp, latestTimestamp) { + atomic.StoreUint32(&b.latestTimestamp, p.Timestamp) + atomic.StoreInt64(&b.latestTimestampTime, arrivalTime) + } + + arrival := uint32(arrivalTime / 1e6 * int64(b.clockRate/1e3)) + transit := arrival - p.Timestamp + if b.lastTransit != 0 { + d := int32(transit - b.lastTransit) + if d < 0 { + d = -d + } + b.stats.Jitter += (float64(d) - b.stats.Jitter) / 16 + } + b.lastTransit = transit + + if b.twcc { + if ext := p.GetExtension(b.twccExt); len(ext) > 1 { + b.feedbackTWCC(binary.BigEndian.Uint16(ext[0:2]), arrivalTime, p.Marker) + } + } + + if b.audioLevel { + if e := p.GetExtension(b.audioExt); e != nil && b.onAudioLevel != nil { + ext := rtp.AudioLevelExtension{} + if err := ext.Unmarshal(e); err == nil { + b.onAudioLevel(ext.Level) + } + } + } + + diff := arrivalTime - b.lastReport + + if b.nacker != nil { + if r := b.buildNACKPacket(); r != nil { + b.feedbackCB(r) + } + } + + if diff >= reportDelta { + br := (8 * b.bitrateHelper * uint64(reportDelta)) / uint64(diff) + atomic.StoreUint64(&b.bitrate, br) + b.feedbackCB(b.getRTCP()) + b.lastReport = arrivalTime + b.bitrateHelper = 0 + } +} + +func (b *Buffer) buildNACKPacket() []rtcp.Packet { + if nacks, askKeyframe := b.nacker.pairs(b.cycles | uint32(b.maxSeqNo)); len(nacks) > 0 || askKeyframe { + var pkts []rtcp.Packet + if len(nacks) > 0 { + pkts = []rtcp.Packet{&rtcp.TransportLayerNack{ + MediaSSRC: b.mediaSSRC, + Nacks: nacks, + }} + } + + if askKeyframe { + pkts = append(pkts, &rtcp.PictureLossIndication{ + MediaSSRC: b.mediaSSRC, + }) + } + return pkts + } + return nil +} + +func (b *Buffer) buildREMBPacket() *rtcp.ReceiverEstimatedMaximumBitrate { + br := b.bitrate + if b.stats.LostRate < 0.02 { + br = uint64(float64(br)*1.09) + 2000 + } + if b.stats.LostRate > .1 { + br = uint64(float64(br) * float64(1-0.5*b.stats.LostRate)) + } + if br > b.maxBitrate { + br = b.maxBitrate + } + if br < 100000 { + br = 100000 + } + b.stats.TotalByte = 0 + + return &rtcp.ReceiverEstimatedMaximumBitrate{ + Bitrate: float32(br), + SSRCs: []uint32{b.mediaSSRC}, + } +} + +func (b *Buffer) buildReceptionReport() rtcp.ReceptionReport { + extMaxSeq := b.cycles | uint32(b.maxSeqNo) + expected := extMaxSeq - uint32(b.baseSN) + 1 + lost := uint32(0) + if b.stats.PacketCount < expected && b.stats.PacketCount != 0 { + lost = expected - b.stats.PacketCount + } + expectedInterval := expected - b.stats.LastExpected + b.stats.LastExpected = expected + + receivedInterval := b.stats.PacketCount - b.stats.LastReceived + b.stats.LastReceived = b.stats.PacketCount + + lostInterval := expectedInterval - receivedInterval + + b.stats.LostRate = float32(lostInterval) / float32(expectedInterval) + var fracLost uint8 + if expectedInterval != 0 && lostInterval > 0 { + fracLost = uint8((lostInterval << 8) / expectedInterval) + } + var dlsr uint32 + + lastSRRecv := atomic.LoadInt64(&b.lastSRRecv) + if lastSRRecv != 0 { + delayMS := uint32((time.Now().UnixNano() - lastSRRecv) / 1e6) + dlsr = (delayMS / 1e3) << 16 + dlsr |= (delayMS % 1e3) * 65536 / 1000 + } + + rr := rtcp.ReceptionReport{ + SSRC: b.mediaSSRC, + FractionLost: fracLost, + TotalLost: lost, + LastSequenceNumber: extMaxSeq, + Jitter: uint32(b.stats.Jitter), + LastSenderReport: uint32(atomic.LoadUint64(&b.lastSRNTPTime) >> 16), + Delay: dlsr, + } + return rr +} + +func (b *Buffer) SetSenderReportData(rtpTime uint32, ntpTime uint64) { + atomic.StoreUint64(&b.lastSRNTPTime, ntpTime) + atomic.StoreUint32(&b.lastSRRTPTime, rtpTime) + atomic.StoreInt64(&b.lastSRRecv, time.Now().UnixNano()) +} + +func (b *Buffer) getRTCP() []rtcp.Packet { + var pkts []rtcp.Packet + + pkts = append(pkts, &rtcp.ReceiverReport{ + Reports: []rtcp.ReceptionReport{b.buildReceptionReport()}, + }) + + if b.remb && !b.twcc { + pkts = append(pkts, b.buildREMBPacket()) + } + + return pkts +} + +func (b *Buffer) GetPacket(buff []byte, sn uint16) (int, error) { + b.Lock() + defer b.Unlock() + if b.closed.get() { + return 0, io.EOF + } + return b.bucket.GetPacket(buff, sn) +} + +// Bitrate returns the current publisher stream bitrate. +func (b *Buffer) Bitrate() uint64 { + return atomic.LoadUint64(&b.bitrate) +} + +func (b *Buffer) MaxTemporalLayer() int32 { + return atomic.LoadInt32(&b.maxTemporalLayer) +} + +func (b *Buffer) OnTransportWideCC(fn func(sn uint16, timeNS int64, marker bool)) { + b.feedbackTWCC = fn +} + +func (b *Buffer) OnFeedback(fn func(fb []rtcp.Packet)) { + b.feedbackCB = fn +} + +func (b *Buffer) OnAudioLevel(fn func(level uint8)) { + b.onAudioLevel = fn +} + +// GetMediaSSRC returns the associated SSRC of the RTP stream +func (b *Buffer) GetMediaSSRC() uint32 { + return b.mediaSSRC +} + +// GetClockRate returns the RTP clock rate +func (b *Buffer) GetClockRate() uint32 { + return b.clockRate +} + +// GetSenderReportData returns the rtp, ntp and nanos of the last sender report +func (b *Buffer) GetSenderReportData() (rtpTime uint32, ntpTime uint64, lastReceivedTimeInNanosSinceEpoch int64) { + rtpTime = atomic.LoadUint32(&b.lastSRRTPTime) + ntpTime = atomic.LoadUint64(&b.lastSRNTPTime) + lastReceivedTimeInNanosSinceEpoch = atomic.LoadInt64(&b.lastSRRecv) + + return rtpTime, ntpTime, lastReceivedTimeInNanosSinceEpoch +} + +// GetStats returns the raw statistics about a particular buffer state +func (b *Buffer) GetStats() (stats Stats) { + b.Lock() + stats = b.stats + b.Unlock() + return +} + +// GetLatestTimestamp returns the latest RTP timestamp factoring in potential RTP timestamp wrap-around +func (b *Buffer) GetLatestTimestamp() (latestTimestamp uint32, latestTimestampTimeInNanosSinceEpoch int64) { + latestTimestamp = atomic.LoadUint32(&b.latestTimestamp) + latestTimestampTimeInNanosSinceEpoch = atomic.LoadInt64(&b.latestTimestampTime) + + return latestTimestamp, latestTimestampTimeInNanosSinceEpoch +} + +// IsTimestampWrapAround returns true if wrap around happens from timestamp1 to timestamp2 +func IsTimestampWrapAround(timestamp1 uint32, timestamp2 uint32) bool { + return (timestamp1&0xC000000 == 0) && (timestamp2&0xC000000 == 0xC000000) +} + +// IsLaterTimestamp returns true if timestamp1 is later in time than timestamp2 factoring in timestamp wrap-around +func IsLaterTimestamp(timestamp1 uint32, timestamp2 uint32) bool { + if timestamp1 > timestamp2 { + return !IsTimestampWrapAround(timestamp2, timestamp1) + } + if IsTimestampWrapAround(timestamp1, timestamp2) { + return true + } + return false +} diff --git a/examples/b2bua/b2bua/buffer/buffer_test.go b/examples/b2bua/b2bua/buffer/buffer_test.go new file mode 100644 index 0000000..7327aac --- /dev/null +++ b/examples/b2bua/b2bua/buffer/buffer_test.go @@ -0,0 +1,184 @@ +package buffer + +import ( + "sync" + "testing" + + "github.com/pion/ion-sfu/pkg/logger" + "github.com/pion/rtcp" + + "github.com/pion/rtp" + "github.com/pion/webrtc/v3" + "github.com/stretchr/testify/assert" +) + +func CreateTestPacket(pktStamp *SequenceNumberAndTimeStamp) *rtp.Packet { + if pktStamp == nil { + return &rtp.Packet{ + Header: rtp.Header{}, + Payload: []byte{1, 2, 3}, + } + } + + return &rtp.Packet{ + Header: rtp.Header{ + SequenceNumber: pktStamp.SequenceNumber, + Timestamp: pktStamp.Timestamp, + }, + Payload: []byte{1, 2, 3}, + } +} + +type SequenceNumberAndTimeStamp struct { + SequenceNumber uint16 + Timestamp uint32 +} + +func CreateTestListPackets(snsAndTSs []SequenceNumberAndTimeStamp) (packetList []*rtp.Packet) { + for _, item := range snsAndTSs { + item := item + packetList = append(packetList, CreateTestPacket(&item)) + } + + return packetList +} + +func TestNack(t *testing.T) { + pool := &sync.Pool{ + New: func() interface{} { + b := make([]byte, 1500) + return &b + }, + } + logger.SetGlobalOptions(logger.GlobalConfig{V: 1}) // 2 - TRACE + logger := logger.New() + buff := NewBuffer(123, pool, pool, logger) + buff.codecType = webrtc.RTPCodecTypeVideo + assert.NotNil(t, buff) + var wg sync.WaitGroup + // 3 nacks 1 Pli + wg.Add(4) + buff.OnFeedback(func(fb []rtcp.Packet) { + for _, pkt := range fb { + switch p := pkt.(type) { + case *rtcp.TransportLayerNack: + if p.Nacks[0].PacketList()[0] == 2 && p.MediaSSRC == 123 { + wg.Done() + } + case *rtcp.PictureLossIndication: + if p.MediaSSRC == 123 { + wg.Done() + } + } + } + }) + buff.Bind(webrtc.RTPParameters{ + HeaderExtensions: nil, + Codecs: []webrtc.RTPCodecParameters{ + { + RTPCodecCapability: webrtc.RTPCodecCapability{ + MimeType: "video/vp8", + ClockRate: 90000, + RTCPFeedback: []webrtc.RTCPFeedback{{ + Type: "nack", + }}, + }, + PayloadType: 96, + }, + }, + }, Options{}) + for i := 0; i < 15; i++ { + if i == 2 { + continue + } + pkt := rtp.Packet{ + Header: rtp.Header{SequenceNumber: uint16(i), Timestamp: uint32(i)}, + Payload: []byte{0xff, 0xff, 0xff, 0xfd, 0xb4, 0x9f, 0x94, 0x1}, + } + b, err := pkt.Marshal() + assert.NoError(t, err) + _, err = buff.Write(b) + assert.NoError(t, err) + } + wg.Wait() +} + +func TestNewBuffer(t *testing.T) { + type args struct { + options Options + ssrc uint32 + } + tests := []struct { + name string + args args + }{ + { + name: "Must not be nil and add packets in sequence", + args: args{ + options: Options{ + MaxBitRate: 1e6, + }, + }, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + var TestPackets = []*rtp.Packet{ + { + Header: rtp.Header{ + SequenceNumber: 65533, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 65534, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 2, + }, + }, + { + Header: rtp.Header{ + SequenceNumber: 65535, + }, + }, + } + pool := &sync.Pool{ + New: func() interface{} { + b := make([]byte, 1500) + return &b + }, + } + logger.SetGlobalOptions(logger.GlobalConfig{V: 2}) // 2 - TRACE + logger := logger.New() + buff := NewBuffer(123, pool, pool, logger) + buff.codecType = webrtc.RTPCodecTypeVideo + assert.NotNil(t, buff) + assert.NotNil(t, TestPackets) + buff.OnFeedback(func(_ []rtcp.Packet) { + }) + buff.Bind(webrtc.RTPParameters{ + HeaderExtensions: nil, + Codecs: []webrtc.RTPCodecParameters{{ + RTPCodecCapability: webrtc.RTPCodecCapability{ + MimeType: "video/vp8", + ClockRate: 9600, + RTCPFeedback: nil, + }, + PayloadType: 0, + }}, + }, Options{}) + + for _, p := range TestPackets { + buf, _ := p.Marshal() + buff.Write(buf) + } + // assert.Equal(t, 6, buff.PacketQueue.size) + assert.Equal(t, uint32(1<<16), buff.cycles) + assert.Equal(t, uint16(2), buff.maxSeqNo) + }) + } +} diff --git a/examples/b2bua/b2bua/buffer/errors.go b/examples/b2bua/b2bua/buffer/errors.go new file mode 100644 index 0000000..8c273fe --- /dev/null +++ b/examples/b2bua/b2bua/buffer/errors.go @@ -0,0 +1,10 @@ +package buffer + +import "errors" + +var ( + errPacketNotFound = errors.New("packet not found in cache") + errBufferTooSmall = errors.New("buffer too small") + errPacketTooOld = errors.New("received packet too old") + errRTXPacket = errors.New("packet already received") +) diff --git a/examples/b2bua/b2bua/buffer/factory.go b/examples/b2bua/b2bua/buffer/factory.go new file mode 100644 index 0000000..627c760 --- /dev/null +++ b/examples/b2bua/b2bua/buffer/factory.go @@ -0,0 +1,97 @@ +package buffer + +import ( + "io" + "sync" + + "github.com/go-logr/logr" + "github.com/pion/transport/packetio" +) + +type Factory struct { + sync.RWMutex + videoPool *sync.Pool + audioPool *sync.Pool + rtpBuffers map[uint32]*Buffer + rtcpReaders map[uint32]*RTCPReader + logger logr.Logger +} + +func NewBufferFactory(trackingPackets int, logger logr.Logger) *Factory { + // Enable package wide logging for non-method functions. + // If logger is empty - use default Logger. + // Logger is a public variable in buffer package. + if logger == (logr.Logger{}) { + logger = Logger + } else { + Logger = logger + } + + return &Factory{ + videoPool: &sync.Pool{ + New: func() interface{} { + b := make([]byte, trackingPackets*maxPktSize) + return &b + }, + }, + audioPool: &sync.Pool{ + New: func() interface{} { + b := make([]byte, maxPktSize*25) + return &b + }, + }, + rtpBuffers: make(map[uint32]*Buffer), + rtcpReaders: make(map[uint32]*RTCPReader), + logger: logger, + } +} + +func (f *Factory) GetOrNew(packetType packetio.BufferPacketType, ssrc uint32) io.ReadWriteCloser { + f.Lock() + defer f.Unlock() + switch packetType { + case packetio.RTCPBufferPacket: + if reader, ok := f.rtcpReaders[ssrc]; ok { + return reader + } + reader := NewRTCPReader(ssrc) + f.rtcpReaders[ssrc] = reader + reader.OnClose(func() { + f.Lock() + delete(f.rtcpReaders, ssrc) + f.Unlock() + }) + return reader + case packetio.RTPBufferPacket: + if reader, ok := f.rtpBuffers[ssrc]; ok { + return reader + } + buffer := NewBuffer(ssrc, f.videoPool, f.audioPool, f.logger) + f.rtpBuffers[ssrc] = buffer + buffer.OnClose(func() { + f.Lock() + delete(f.rtpBuffers, ssrc) + f.Unlock() + }) + return buffer + } + return nil +} + +func (f *Factory) GetBufferPair(ssrc uint32) (*Buffer, *RTCPReader) { + f.RLock() + defer f.RUnlock() + return f.rtpBuffers[ssrc], f.rtcpReaders[ssrc] +} + +func (f *Factory) GetBuffer(ssrc uint32) *Buffer { + f.RLock() + defer f.RUnlock() + return f.rtpBuffers[ssrc] +} + +func (f *Factory) GetRTCPReader(ssrc uint32) *RTCPReader { + f.RLock() + defer f.RUnlock() + return f.rtcpReaders[ssrc] +} diff --git a/examples/b2bua/b2bua/buffer/helpers.go b/examples/b2bua/b2bua/buffer/helpers.go new file mode 100644 index 0000000..308e966 --- /dev/null +++ b/examples/b2bua/b2bua/buffer/helpers.go @@ -0,0 +1,212 @@ +package buffer + +import ( + "encoding/binary" + "errors" + "fmt" + "sync/atomic" +) + +var ( + errShortPacket = errors.New("packet is not large enough") + errNilPacket = errors.New("invalid nil packet") +) + +type atomicBool int32 + +func (a *atomicBool) set(value bool) { + var i int32 + if value { + i = 1 + } + atomic.StoreInt32((*int32)(a), i) +} + +func (a *atomicBool) get() bool { + return atomic.LoadInt32((*int32)(a)) != 0 +} + +// VP8 is a helper to get temporal data from VP8 packet header +/* + VP8 Payload Descriptor + 0 1 2 3 4 5 6 7 0 1 2 3 4 5 6 7 + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + |X|R|N|S|R| PID | (REQUIRED) |X|R|N|S|R| PID | (REQUIRED) + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + X: |I|L|T|K| RSV | (OPTIONAL) X: |I|L|T|K| RSV | (OPTIONAL) + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + I: |M| PictureID | (OPTIONAL) I: |M| PictureID | (OPTIONAL) + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + L: | TL0PICIDX | (OPTIONAL) | PictureID | + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + T/K:|TID|Y| KEYIDX | (OPTIONAL) L: | TL0PICIDX | (OPTIONAL) + +-+-+-+-+-+-+-+-+ +-+-+-+-+-+-+-+-+ + T/K:|TID|Y| KEYIDX | (OPTIONAL) + +-+-+-+-+-+-+-+-+ +*/ +type VP8 struct { + TemporalSupported bool + // Optional Header + PictureID uint16 /* 8 or 16 bits, picture ID */ + PicIDIdx int + MBit bool + TL0PICIDX uint8 /* 8 bits temporal level zero index */ + TlzIdx int + + // Optional Header If either of the T or K bits are set to 1, + // the TID/Y/KEYIDX extension field MUST be present. + TID uint8 /* 2 bits temporal layer idx*/ + // IsKeyFrame is a helper to detect if current packet is a keyframe + IsKeyFrame bool +} + +// Unmarshal parses the passed byte slice and stores the result in the VP8 this method is called upon +func (p *VP8) Unmarshal(payload []byte) error { + if payload == nil { + return errNilPacket + } + + payloadLen := len(payload) + + if payloadLen < 1 { + return errShortPacket + } + + idx := 0 + S := payload[idx]&0x10 > 0 + // Check for extended bit control + if payload[idx]&0x80 > 0 { + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + // Check if T is present, if not, no temporal layer is available + p.TemporalSupported = payload[idx]&0x20 > 0 + K := payload[idx]&0x10 > 0 + L := payload[idx]&0x40 > 0 + // Check for PictureID + if payload[idx]&0x80 > 0 { + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + p.PicIDIdx = idx + pid := payload[idx] & 0x7f + // Check if m is 1, then Picture ID is 15 bits + if payload[idx]&0x80 > 0 { + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + p.MBit = true + p.PictureID = binary.BigEndian.Uint16([]byte{pid, payload[idx]}) + } else { + p.PictureID = uint16(pid) + } + } + // Check if TL0PICIDX is present + if L { + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + p.TlzIdx = idx + + if int(idx) >= payloadLen { + return errShortPacket + } + p.TL0PICIDX = payload[idx] + } + if p.TemporalSupported || K { + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + p.TID = (payload[idx] & 0xc0) >> 6 + } + if idx >= payloadLen { + return errShortPacket + } + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + // Check is packet is a keyframe by looking at P bit in vp8 payload + p.IsKeyFrame = payload[idx]&0x01 == 0 && S + } else { + idx++ + if payloadLen < idx+1 { + return errShortPacket + } + // Check is packet is a keyframe by looking at P bit in vp8 payload + p.IsKeyFrame = payload[idx]&0x01 == 0 && S + } + return nil +} + +// isH264Keyframe detects if h264 payload is a keyframe +// this code was taken from https://github.com/jech/galene/blob/codecs/rtpconn/rtpreader.go#L45 +// all credits belongs to Juliusz Chroboczek @jech and the awesome Galene SFU +func isH264Keyframe(payload []byte) bool { + if len(payload) < 1 { + return false + } + nalu := payload[0] & 0x1F + if nalu == 0 { + // reserved + return false + } else if nalu <= 23 { + // simple NALU + return nalu == 5 + } else if nalu == 24 || nalu == 25 || nalu == 26 || nalu == 27 { + // STAP-A, STAP-B, MTAP16 or MTAP24 + i := 1 + if nalu == 25 || nalu == 26 || nalu == 27 { + // skip DON + i += 2 + } + for i < len(payload) { + if i+2 > len(payload) { + return false + } + length := uint16(payload[i])<<8 | + uint16(payload[i+1]) + i += 2 + if i+int(length) > len(payload) { + return false + } + offset := 0 + if nalu == 26 { + offset = 3 + } else if nalu == 27 { + offset = 4 + } + if offset >= int(length) { + return false + } + n := payload[i+offset] & 0x1F + if n == 7 { + return true + } else if n >= 24 { + // is this legal? + fmt.Print("Non-simple NALU within a STAP") + } + i += int(length) + } + if i == len(payload) { + return false + } + return false + } else if nalu == 28 || nalu == 29 { + // FU-A or FU-B + if len(payload) < 2 { + return false + } + if (payload[1] & 0x80) == 0 { + // not a starting fragment + return false + } + return payload[1]&0x1F == 7 + } + return false +} diff --git a/examples/b2bua/b2bua/buffer/helpers_test.go b/examples/b2bua/b2bua/buffer/helpers_test.go new file mode 100644 index 0000000..5cdff93 --- /dev/null +++ b/examples/b2bua/b2bua/buffer/helpers_test.go @@ -0,0 +1,94 @@ +package buffer + +import ( + "testing" + + "github.com/stretchr/testify/assert" +) + +func TestVP8Helper_Unmarshal(t *testing.T) { + type args struct { + payload []byte + } + tests := []struct { + name string + args args + wantErr bool + checkTemporal bool + temporalSupport bool + checkKeyFrame bool + keyFrame bool + checkPictureID bool + pictureID uint16 + checkTlzIdx bool + tlzIdx uint8 + checkTempID bool + temporalID uint8 + }{ + { + name: "Empty or nil payload must return error", + args: args{payload: []byte{}}, + wantErr: true, + }, + { + name: "Temporal must be supported by setting T bit to 1", + args: args{payload: []byte{0xff, 0x20, 0x1, 0x2, 0x3, 0x4}}, + checkTemporal: true, + temporalSupport: true, + }, + { + name: "Picture must be ID 7 bits by setting M bit to 0 and present by I bit set to 1", + args: args{payload: []byte{0xff, 0xff, 0x11, 0x2, 0x3, 0x4}}, + checkPictureID: true, + pictureID: 17, + }, + { + name: "Picture ID must be 15 bits by setting M bit to 1 and present by I bit set to 1", + args: args{payload: []byte{0xff, 0xff, 0x92, 0x67, 0x3, 0x4, 0x5}}, + checkPictureID: true, + pictureID: 4711, + }, + { + name: "Temporal level zero index must be present if L set to 1", + args: args{payload: []byte{0xff, 0xff, 0xff, 0xfd, 0xb4, 0x4, 0x5}}, + checkTlzIdx: true, + tlzIdx: 180, + }, + { + name: "Temporal index must be present and used if T bit set to 1", + args: args{payload: []byte{0xff, 0xff, 0xff, 0xfd, 0xb4, 0x9f, 0x5, 0x6}}, + checkTempID: true, + temporalID: 2, + }, + { + name: "Check if packet is a keyframe by looking at P bit set to 0", + args: args{payload: []byte{0xff, 0xff, 0xff, 0xfd, 0xb4, 0x9f, 0x94, 0x1}}, + checkKeyFrame: true, + keyFrame: true, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + p := &VP8{} + if err := p.Unmarshal(tt.args.payload); (err != nil) != tt.wantErr { + t.Errorf("Unmarshal() error = %v, wantErr %v", err, tt.wantErr) + } + if tt.checkTemporal { + assert.Equal(t, tt.temporalSupport, p.TemporalSupported) + } + if tt.checkKeyFrame { + assert.Equal(t, tt.keyFrame, p.IsKeyFrame) + } + if tt.checkPictureID { + assert.Equal(t, tt.pictureID, p.PictureID) + } + if tt.checkTlzIdx { + assert.Equal(t, tt.tlzIdx, p.TL0PICIDX) + } + if tt.checkTempID { + assert.Equal(t, tt.temporalID, p.TID) + } + }) + } +} diff --git a/examples/b2bua/b2bua/buffer/nack.go b/examples/b2bua/b2bua/buffer/nack.go new file mode 100644 index 0000000..0050ba7 --- /dev/null +++ b/examples/b2bua/b2bua/buffer/nack.go @@ -0,0 +1,100 @@ +package buffer + +import ( + "sort" + + "github.com/pion/rtcp" +) + +const maxNackTimes = 3 // Max number of times a packet will be NACKed +const maxNackCache = 100 // Max NACK sn the sfu will keep reference + +type nack struct { + sn uint32 + nacked uint8 +} + +type nackQueue struct { + nacks []nack + kfSN uint32 +} + +func newNACKQueue() *nackQueue { + return &nackQueue{ + nacks: make([]nack, 0, maxNackCache+1), + } +} + +func (n *nackQueue) remove(extSN uint32) { + i := sort.Search(len(n.nacks), func(i int) bool { return n.nacks[i].sn >= extSN }) + if i >= len(n.nacks) || n.nacks[i].sn != extSN { + return + } + copy(n.nacks[i:], n.nacks[i+1:]) + n.nacks = n.nacks[:len(n.nacks)-1] +} + +func (n *nackQueue) push(extSN uint32) { + i := sort.Search(len(n.nacks), func(i int) bool { return n.nacks[i].sn >= extSN }) + if i < len(n.nacks) && n.nacks[i].sn == extSN { + return + } + + nck := nack{ + sn: extSN, + nacked: 0, + } + if i == len(n.nacks) { + n.nacks = append(n.nacks, nck) + } else { + n.nacks = append(n.nacks[:i+1], n.nacks[i:]...) + n.nacks[i] = nck + } + + if len(n.nacks) >= maxNackCache { + copy(n.nacks, n.nacks[1:]) + } +} + +func (n *nackQueue) pairs(headSN uint32) ([]rtcp.NackPair, bool) { + if len(n.nacks) == 0 { + return nil, false + } + i := 0 + askKF := false + var np rtcp.NackPair + var nps []rtcp.NackPair + for _, nck := range n.nacks { + if nck.nacked >= maxNackTimes { + if nck.sn > n.kfSN { + n.kfSN = nck.sn + askKF = true + } + continue + } + if nck.sn >= headSN-2 { + n.nacks[i] = nck + i++ + continue + } + n.nacks[i] = nack{ + sn: nck.sn, + nacked: nck.nacked + 1, + } + i++ + if np.PacketID == 0 || uint16(nck.sn) > np.PacketID+16 { + if np.PacketID != 0 { + nps = append(nps, np) + } + np.PacketID = uint16(nck.sn) + np.LostPackets = 0 + continue + } + np.LostPackets |= 1 << (uint16(nck.sn) - np.PacketID - 1) + } + if np.PacketID != 0 { + nps = append(nps, np) + } + n.nacks = n.nacks[:i] + return nps, askKF +} diff --git a/examples/b2bua/b2bua/buffer/nack_test.go b/examples/b2bua/b2bua/buffer/nack_test.go new file mode 100644 index 0000000..3c60eab --- /dev/null +++ b/examples/b2bua/b2bua/buffer/nack_test.go @@ -0,0 +1,181 @@ +package buffer + +import ( + "math/rand" + "reflect" + "testing" + "time" + + "github.com/pion/rtcp" + "github.com/stretchr/testify/assert" +) + +func Test_nackQueue_pairs(t *testing.T) { + type fields struct { + nacks []nack + } + tests := []struct { + name string + fields fields + args []uint32 + want []rtcp.NackPair + }{ + { + name: "Must return correct single pairs pair", + fields: fields{ + nacks: nil, + }, + args: []uint32{1, 2, 4, 5}, + want: []rtcp.NackPair{{ + PacketID: 1, + LostPackets: 13, + }}, + }, + { + name: "Must return 2 pairs pair", + fields: fields{ + nacks: nil, + }, + args: []uint32{1, 2, 4, 5, 20, 22, 24, 27}, + want: []rtcp.NackPair{ + { + PacketID: 1, + LostPackets: 13, + }, + { + PacketID: 20, + LostPackets: 74, + }, + }, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + n := &nackQueue{ + nacks: tt.fields.nacks, + } + for _, sn := range tt.args { + n.push(sn) + } + got, _ := n.pairs(30) + if !reflect.DeepEqual(got, tt.want) { + t.Errorf("pairs() = %v, want %v", got, tt.want) + } + }) + } +} + +func Test_nackQueue_push(t *testing.T) { + type fields struct { + nacks []nack + } + type args struct { + sn []uint32 + } + tests := []struct { + name string + fields fields + args args + want []uint32 + }{ + { + name: "Must keep packet order", + fields: fields{ + nacks: make([]nack, 0, 10), + }, + args: args{ + sn: []uint32{3, 4, 1, 5, 8, 7, 5}, + }, + want: []uint32{1, 3, 4, 5, 7, 8}, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + n := &nackQueue{ + nacks: tt.fields.nacks, + } + for _, sn := range tt.args.sn { + n.push(sn) + } + var newSN []uint32 + for _, sn := range n.nacks { + newSN = append(newSN, sn.sn) + } + assert.Equal(t, tt.want, newSN) + }) + } +} + +func Test_nackQueue(t *testing.T) { + type fields struct { + nacks []nack + } + type args struct { + sn []uint32 + } + tests := []struct { + name string + fields fields + args args + }{ + { + name: "Must keep packet order", + fields: fields{ + nacks: make([]nack, 0, 10), + }, + args: args{ + sn: []uint32{3, 4, 1, 5, 8, 7, 5}, + }, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + n := nackQueue{} + r := rand.New(rand.NewSource(time.Now().UnixNano())) + for i := 0; i < 100; i++ { + assert.NotPanics(t, func() { + n.push(uint32(r.Intn(60000))) + n.remove(uint32(r.Intn(60000))) + n.pairs(60001) + }) + } + }) + } +} + +func Test_nackQueue_remove(t *testing.T) { + type args struct { + sn []uint32 + } + tests := []struct { + name string + args args + want []uint32 + }{ + { + name: "Must keep packet order", + args: args{ + sn: []uint32{3, 4, 1, 5, 8, 7, 5}, + }, + want: []uint32{1, 3, 4, 7, 8}, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + n := nackQueue{} + for _, sn := range tt.args.sn { + n.push(sn) + } + n.remove(5) + var newSN []uint32 + for _, sn := range n.nacks { + newSN = append(newSN, sn.sn) + } + assert.Equal(t, tt.want, newSN) + }) + } +} diff --git a/examples/b2bua/b2bua/buffer/rtcpreader.go b/examples/b2bua/b2bua/buffer/rtcpreader.go new file mode 100644 index 0000000..7472a2e --- /dev/null +++ b/examples/b2bua/b2bua/buffer/rtcpreader.go @@ -0,0 +1,44 @@ +package buffer + +import ( + "io" + "sync/atomic" +) + +type RTCPReader struct { + ssrc uint32 + closed atomicBool + onPacket atomic.Value //func([]byte) + onClose func() +} + +func NewRTCPReader(ssrc uint32) *RTCPReader { + return &RTCPReader{ssrc: ssrc} +} + +func (r *RTCPReader) Write(p []byte) (n int, err error) { + if r.closed.get() { + err = io.EOF + return + } + if f, ok := r.onPacket.Load().(func([]byte)); ok { + f(p) + } + return +} + +func (r *RTCPReader) OnClose(fn func()) { + r.onClose = fn +} + +func (r *RTCPReader) Close() error { + r.closed.set(true) + r.onClose() + return nil +} + +func (r *RTCPReader) OnPacket(f func([]byte)) { + r.onPacket.Store(f) +} + +func (r *RTCPReader) Read(_ []byte) (n int, err error) { return } diff --git a/examples/b2bua/b2bua/call.go b/examples/b2bua/b2bua/call.go new file mode 100644 index 0000000..be7b9e6 --- /dev/null +++ b/examples/b2bua/b2bua/call.go @@ -0,0 +1,129 @@ +package b2bua + +import ( + "github.com/cloudwebrtc/go-sip-ua/pkg/session" + "github.com/ghettovoice/gosip/sip" +) + +type CallState string + +const ( + New CallState = "New" + Connecting CallState = "Connecting" + Ringing CallState = "Ringing" + EarlyMedia CallState = "EarlyMedia" + Confirmed CallState = "Confirmed" + Failure CallState = "Failure" + Terminated CallState = "Terminated" +) + +func (s CallState) String() string { + return string(s) +} + +type Call struct { + // sip session + sess *session.Session + // media transport + mediaTransport MediaTransport + + originalMediaDesc *MediaDescription +} + +func (b *Call) Init(transType MediaTransportType, md *MediaDescription) { + b.originalMediaDesc = md + + if transType == TransportTypeWebRTC { + b.mediaTransport = NewWebRTCMediaTransport(md) + } else { + b.mediaTransport = NewStandardMediaTransport(md) + } + + b.mediaTransport.Init(b2buaConfig.UaMediaConfig) +} + +func (b *Call) Id() string { + return string(b.sess.CallID()) +} + +func (b *Call) ToString() string { + return (b.sess.CallID()).String() + ", uri: " + b.sess.Contact() +} + +func (b *Call) MediaInfo() string { + info := "[" + b.mediaTransport.Type().String() + "]" + for _, trackInfo := range b.originalMediaDesc.Tracks { + info += trackInfo.String() + " " + } + return info +} + +func (b *Call) Provisional(statusCode sip.StatusCode, reason string) { + b.sess.Provisional(statusCode, reason) +} + +func (b *Call) Reject(statusCode sip.StatusCode, reason string) { + b.sess.Reject(statusCode, reason) +} + +func (b *Call) Accept(answer string) { + if aLegAnswer, err := b.mediaTransport.CreateAnswer(); err != nil { + logger.Errorf("CreateAnswer failed: %v", err) + return + } else { + // for sdp fix + replaceCodec(aLegAnswer, answer) + b.sess.ProvideAnswer(aLegAnswer.SDP) + } + b.sess.Accept(200) +} + +func (b *Call) Terminate() { + if b.sess.IsEstablished() { + b.sess.Bye() + } else if b.sess.IsInProgress() { + b.sess.End() + } + b.mediaTransport.OnRtpPacket(nil) + b.mediaTransport.OnRtcpPacket(nil) + if err := b.mediaTransport.Close(); err != nil { + logger.Errorf("Close media transport error: %v", err) + } +} + +func (b *Call) OnOffer(sdp *Desc) error { + err := b.mediaTransport.OnOffer(sdp) + if err != nil { + logger.Errorf("OnOffer error: %v", err) + return err + } + return nil +} + +func (b *Call) CreateOffer() (*Desc, error) { + offer, err := b.mediaTransport.CreateOffer() + if err != nil { + logger.Errorf("Offer error: %v", err) + return nil, err + } + return offer, nil +} + +func (b *Call) OnAnswer(sdp *Desc) error { + err := b.mediaTransport.OnAnswer(sdp) + if err != nil { + logger.Errorf("OnAnswer error: %v", err) + return err + } + + return nil +} + +func (b *Call) CreateAnswer() (*Desc, error) { + answer, err := b.mediaTransport.CreateAnswer() + if err != nil { + logger.Errorf("Answer error: %v", err) + return nil, err + } + return answer, nil +} diff --git a/examples/b2bua/b2bua/call_bridge.go b/examples/b2bua/b2bua/call_bridge.go new file mode 100644 index 0000000..5bafd56 --- /dev/null +++ b/examples/b2bua/b2bua/call_bridge.go @@ -0,0 +1,76 @@ +package b2bua + +type BridgeType string + +const ( + B2BCall BridgeType = "B2BCall" + OriginateCall BridgeType = "OriginateCall" + Conference BridgeType = "Conference" + PlayBack BridgeType = "PlayBack" + Record BridgeType = "Record" +) + +func (b BridgeType) String() string { + return string(b) +} + +type CallBridge struct { + src *Call + dest *Call + state CallState + bType BridgeType +} + +func (b *CallBridge) Init() { + b.state = New +} + +func (b *CallBridge) State() CallState { + return b.state +} + +func (b *CallBridge) SetState(state CallState) { + b.state = state +} + +func (b *CallBridge) Src() *Call { + return b.src +} + +func (b *CallBridge) Dest() *Call { + return b.dest +} + +func (b *CallBridge) Type() BridgeType { + return b.bType +} + +func (b *CallBridge) ToString() string { + str := b.Type().String() + ": [" + b.src.ToString() + " -> " + b.dest.ToString() + "]\n" + str = str + "State: " + b.State().String() + "\n" + str = str + "Src: " + b.src.MediaInfo() + "\n" + str = str + "Dest: " + b.dest.MediaInfo() + return str +} + +func (b *CallBridge) Terminate(call *Call) { + if b.bType == B2BCall || b.bType == OriginateCall { + if b.src == call { + b.dest.Terminate() + } else if b.dest == call { + b.src.Terminate() + } + b.state = Terminated + } +} + +func BridgeMediaStream(src, dest MediaTransport) error { + // 这里 + src.OnRtpPacket(dest.WriteRTP) + src.OnRtcpPacket(dest.WriteRTCP) + src.OnRequestKeyFrame(dest.RequestKeyFrame) + dest.OnRtpPacket(src.WriteRTP) + dest.OnRtcpPacket(src.WriteRTCP) + dest.OnRequestKeyFrame(src.RequestKeyFrame) + return nil +} diff --git a/examples/b2bua/b2bua/call_service.go b/examples/b2bua/b2bua/call_service.go new file mode 100644 index 0000000..2bc0ba2 --- /dev/null +++ b/examples/b2bua/b2bua/call_service.go @@ -0,0 +1,329 @@ +package b2bua + +import ( + "fmt" + + "github.com/cloudwebrtc/go-sip-ua/examples/b2bua/registry" + "github.com/cloudwebrtc/go-sip-ua/pkg/account" + "github.com/cloudwebrtc/go-sip-ua/pkg/session" + "github.com/cloudwebrtc/go-sip-ua/pkg/stack" + "github.com/cloudwebrtc/go-sip-ua/pkg/ua" + "github.com/ghettovoice/gosip/sip" + "github.com/ghettovoice/gosip/sip/parser" + "github.com/ghettovoice/gosip/util" + "github.com/pixelbender/go-sdp/sdp" +) + +type CallService struct { + registry registry.Registry + rfc8599 *registry.RFC8599 + callBridges []*CallBridge + calls map[*session.Session]*Call + stack *stack.SipStack + ua *ua.UserAgent +} + +func NewCallService(stack *stack.SipStack, ua *ua.UserAgent, registry registry.Registry, rfc8599 *registry.RFC8599) *CallService { + return &CallService{ + stack: stack, + ua: ua, + registry: registry, + rfc8599: rfc8599, + calls: make(map[*session.Session]*Call), + } +} + +func (s *CallService) Init() { +} + +func (s *CallService) inviteStateHandler(sess *session.Session, req *sip.Request, resp *sip.Response, state session.Status) { + logger.Infof("InviteStateHandler: sess %v, state => %v, type => %s", sess.CallID().String(), state, sess.Direction()) + + switch state { + // Handle incoming call. + case session.InviteReceived: + offer := &Desc{Type: "offer", SDP: sess.RemoteSdp()} + sdpSess, _ := offer.Parse() + transType := ParseTransportType(sdpSess) + + md, err := ParseMediaDescription(sdpSess) + if err != nil { + logger.Errorf("ParseTrackInfos error: %v", err) + return + } + + src := &Call{sess: sess} + src.Init(transType, md) + src.OnOffer(offer) + s.calls[sess] = src + + if code, err := s.handleIncomingCall(src, req); err != nil { + logger.Warnf("handleIncomingCall error: %v, code: %s", err, code) + src.Reject(code, fmt.Sprintf("%v", err)) + s.removeCall(sess) + return + } + // Handle re-INVITE or UPDATE. + case session.ReInviteReceived: + logger.Infof("re-INVITE") + switch sess.Direction() { + case session.Incoming: + sess.Accept(200) + case session.Outgoing: + //TODO: Need to provide correct answer. + } + + // Handle 1XX + case session.EarlyMedia: + fallthrough + case session.Provisional: + call := s.findCall(sess) + if call != nil { + //bridge.SetState(EarlyMedia) + //bridge.src.Provisional((*resp).StatusCode(), (*resp).Reason()) + } + // Handle 200OK or ACK + case session.Confirmed: + //TODO: Add support for forked calls + call := s.findCall(sess) + if call != nil && sess.Direction() == session.Outgoing { + answer := call.sess.RemoteSdp() + call.OnAnswer(&Desc{Type: "answer", SDP: answer}) + bridge := s.findBridgedCallByCall(call) + if bridge != nil && bridge.dest.sess == sess && bridge.bType == B2BCall { + bridge.dest.OnAnswer(&Desc{Type: "answer", SDP: answer}) + bridge.src.Accept(answer) + BridgeMediaStream(bridge.src.mediaTransport, bridge.dest.mediaTransport) + bridge.SetState(Confirmed) + } + } + + // Handle 4XX+ + case session.Failure: + fallthrough + case session.Canceled: + fallthrough + case session.Terminated: + //TODO: Add support for forked calls + call := s.findCall(sess) + if call != nil { + bridge := s.findBridgedCallByCall(call) + if bridge != nil { + bridge.Terminate(call) + } + s.removeCallBridgeByCall(call) + } + s.removeCall(sess) + } +} + +func (s *CallService) handleIncomingCall(src *Call, req *sip.Request) (sip.StatusCode, error) { + src.Provisional(100, "Trying") + + to, _ := (*req).To() + from, _ := (*req).From() + + caller := from.Address + called := to.Address + + displayName := "" + if from.DisplayName != nil { + displayName = from.DisplayName.String() + } + + // Try to find online contact records. + if contacts, found := s.registry.GetContacts(called); found { + for _, instance := range *contacts { + dest, err := s.makeOutgoingCall(caller, displayName, called, instance, src.originalMediaDesc) + if err != nil { + logger.Errorf("makeOutgoingCall error: %v", err) + return 500, err + } + s.StoreBridgedCall(src, dest, B2BCall) + return 0, nil + } + } + + if s.rfc8599 != nil { + // Pushable: try to find pn-params in contact records. + // Try to push the UA and wait for it to wake up. + pusher, ok := s.rfc8599.TryPush(called, from) + if ok { + instance, err := pusher.WaitContactOnline() + if err != nil { + logger.Errorf("Push failed, error: %v", err) + src.Reject(500, "Push failed") + return 500, err + } + dest, err := s.makeOutgoingCall(caller, displayName, called, instance, src.originalMediaDesc) + if err != nil { + logger.Errorf("makeOutgoingCall error: %v", err) + return 500, err + } + s.StoreBridgedCall(src, dest, B2BCall) + return 0, err + } + } + + // try make direct call + dest, err := s.makeOutgoingCall(caller, displayName, called, nil, src.originalMediaDesc) + if err != nil { + logger.Errorf("makeOutgoingCall error: %v", err) + return 500, err + } + + s.StoreBridgedCall(src, dest, B2BCall) + return 0, nil +} + +func (s *CallService) StoreBridgedCall(src, dest *Call, bType BridgeType) *CallBridge { + bridge := &CallBridge{src: src, dest: dest, bType: bType} + bridge.Init() + bridge.SetState(Connecting) + s.callBridges = append(s.callBridges, bridge) + return bridge +} + +func (s *CallService) makeOutgoingCall(caller sip.Uri, displayName string, called sip.Uri, instance *registry.ContactInstance, md *MediaDescription) (*Call, error) { + + profile := account.NewProfile(caller, displayName, nil, 0, s.stack) + + destUri := "sip:" + called.User().String() + "@" + + if instance != nil { + destUri = destUri + instance.Source + ";transport=" + instance.Transport + } else { + destUri = destUri + called.Host() + ";transport=udp" + } + + recipient, err2 := parser.ParseSipUri(destUri) + if err2 != nil { + logger.Error(err2) + } + var tpType = TransportTypeStandard + + if instance != nil && instance.SupportIce() { + tpType = TransportTypeWebRTC + } + + dest := &Call{} + dest.Init(tpType, md) + destOffer, _ := dest.CreateOffer() + + sess, err := s.ua.Invite(profile, called, recipient, &destOffer.SDP) + if err != nil { + logger.Errorf("makeOutgoingCall error: %v", err) + return nil, err + } + dest.sess = sess + s.calls[sess] = dest + return dest, nil + +} + +func (s *CallService) findBridgedCallByCall(call *Call) *CallBridge { + for _, bridge := range s.callBridges { + if bridge.src == call || bridge.dest == call { + return bridge + } + } + return nil +} + +func (s *CallService) removeCallBridgeByCall(call *Call) { + for idx, bridge := range s.callBridges { + if bridge.src == call || bridge.dest == call { + s.callBridges = append(s.callBridges[:idx], s.callBridges[idx+1:]...) + return + } + } +} + +func (s *CallService) findCall(sess *session.Session) *Call { + if call, found := s.calls[sess]; found { + return call + } + return nil +} + +func (s *CallService) removeCall(sess *session.Session) { + delete(s.calls, sess) +} + +func (s *CallService) Shutdown() { + s.ua.Shutdown() +} + +func (s *CallService) Originate(source string, destination string) error { + logger.Infof("Originate %s => %s", source, destination) + + displayName := "Originate" + + host := b2buaConfig.UaMediaConfig.ExternalRtpAddress + + if host == "" || host == "0.0.0.0" { + if v, err := util.ResolveSelfIP(); err == nil { + host = v.String() + } + } + + srcUri, err := parser.ParseUri("sip:" + source + "@" + host) + if err != nil { + logger.Error(err) + } + + destUri, err := parser.ParseUri("sip:" + destination + "@" + host) + if err != nil { + logger.Error(err) + } + + var srcCall *Call = nil + var destCall *Call = nil + + var audioCodecs []*sdp.Format + var videoCodecs []*sdp.Format + + for _, codec := range defaultAudioCodecs { + audioCodecs = append(audioCodecs, codec) + } + + for _, codec := range defaultVideoCodecs { + videoCodecs = append(videoCodecs, codec) + } + + originalTrackInfos := map[TrackType]*TrackInfo{ + TrackTypeAudio: {TrackType: TrackTypeAudio, Direction: "sendrecv", Codecs: audioCodecs}, + TrackTypeVideo: {TrackType: TrackTypeVideo, Direction: "sendrecv", Codecs: videoCodecs}, + } + + md := &MediaDescription{ + Tracks: originalTrackInfos, + Connection: &sdp.Connection{ + Address: host, + }, + } + var err2 error = nil + if contacts, found := s.registry.GetContacts(srcUri); found { + for _, instance := range *contacts { + srcCall, err2 = s.makeOutgoingCall(srcUri, displayName, destUri, instance, md) + if err != nil { + logger.Errorf("Originate error: %v", err) + return err2 + } + } + } + + if contacts, found := s.registry.GetContacts(destUri); found { + for _, instance := range *contacts { + destCall, err2 = s.makeOutgoingCall(destUri, displayName, srcUri, instance, md) + if err != nil { + logger.Errorf("Originate error: %v", err) + return err2 + } + } + } + + s.StoreBridgedCall(srcCall, destCall, OriginateCall) + BridgeMediaStream(srcCall.mediaTransport, destCall.mediaTransport) + return nil +} diff --git a/examples/b2bua/b2bua/conf.go b/examples/b2bua/b2bua/conf.go new file mode 100644 index 0000000..4768e10 --- /dev/null +++ b/examples/b2bua/b2bua/conf.go @@ -0,0 +1,9 @@ +package b2bua + +import "github.com/pion/webrtc/v3" + +type UserAgentMediaConfig struct { + Codecs []string `json:"codecs"` + ExternalRtpAddress string `json:"external_rtp_address"` + RtcpFeedback []webrtc.RTCPFeedback `json:"rtcp_feedback"` +} diff --git a/examples/b2bua/b2bua/jitterbuffer/buffer.go b/examples/b2bua/b2bua/jitterbuffer/buffer.go new file mode 100644 index 0000000..e22d833 --- /dev/null +++ b/examples/b2bua/b2bua/jitterbuffer/buffer.go @@ -0,0 +1,53 @@ +package jitterbuffer + +import ( + "github.com/pion/rtp" +) + +type JitterBufferType string + +const ( + JBOFF JitterBufferType = "OFF" + JBUF_FIXED JitterBufferType = "FIXED" + JBUF_ADAPTIVE JitterBufferType = "ADAPTIVE" +) + +var ( + JBUF_RDIFF_EMA_COEFF = 1024 + JBUF_RDIFF_UP_SPEED = 512 + JBUF_PUT_TIMEOUT = 400 +) + +/** Defines a packet frame */ +type packet struct { + hdr rtp.Header /**< RTP Header */ + buf []byte /**< RTP Payload */ +} + +type JitterBuffer struct { + Type JitterBufferType + + min int + max int + + // Wish size for adaptive mode + wish int + + payload int + + packets []packet +} + +/* + * @param min Minimum delay in [frames] + * @param max Maximum delay in [packets] + */ +func NewJitterBuffer(jbType JitterBufferType, min, max int) *JitterBuffer { + return &JitterBuffer{ + Type: jbType, + min: min, + max: max, + wish: min, + packets: make([]packet, max), + } +} diff --git a/examples/b2bua/b2bua/media_desc.go b/examples/b2bua/b2bua/media_desc.go new file mode 100644 index 0000000..a073f65 --- /dev/null +++ b/examples/b2bua/b2bua/media_desc.go @@ -0,0 +1,118 @@ +package b2bua + +import ( + "errors" + "fmt" + + "github.com/pixelbender/go-sdp/sdp" +) + +type Desc struct { + Type string `json:"type"` + SDP string `json:"sdp"` +} + +func (d *Desc) Parse() (*sdp.Session, error) { + return sdp.Parse([]byte(d.SDP)) +} + +func (d *Desc) FromSdpSession(sess *sdp.Session) error { + d.SDP = sess.String() + return nil +} + +type TransportType string + +const ( + MediaTransportUDP TransportType = "UDP" + MediaTransportTCP TransportType = "TCP" + MediaTransportTLS TransportType = "TLS" + MediaTransportDTLS TransportType = "DTLS" +) + +type MediaStreamDir string + +const ( + MediaStreamSendRecv MediaStreamDir = "sendrecv" + MediaStreamSendOnly MediaStreamDir = "sendonly" + MediaStreamRecvOnly MediaStreamDir = "recvonly" + MediaStreamInactive MediaStreamDir = "inactive" +) + +type TrackType string + +const ( + TrackTypeAudio TrackType = "audio" + TrackTypeVideo TrackType = "video" +) + +func (t TrackType) String() string { + return string(t) +} + +type TrackInfo struct { + TrackType TrackType + Codecs []*sdp.Format + Direction string + Port int + RtcpPort int + Ssrc uint32 +} + +func (t *TrackInfo) String() string { + return t.TrackType.String() + ": " + t.Codecs[0].Name + "/" + fmt.Sprintf("%d", t.Codecs[0].ClockRate) + ", pt: " + fmt.Sprintf("%d", t.Codecs[0].Payload) +} + +type MediaDescription struct { + Tracks map[TrackType]*TrackInfo + Connection *sdp.Connection +} + +func MediaDescriptionFrom(sdp *Desc) (*MediaDescription, error) { + sess, err := sdp.Parse() + if err != nil { + return nil, err + } + md, err := ParseMediaDescription(sess) + if err != nil { + return nil, err + } + return md, nil +} + +func ParseMediaDescription(sdp *sdp.Session) (*MediaDescription, error) { + if sdp == nil { + return nil, errors.New("sdp is nil") + } + mediaDesc := &MediaDescription{ + Connection: sdp.Connection, + Tracks: make(map[TrackType]*TrackInfo), + } + for _, m := range sdp.Media { + trackInfo := &TrackInfo{ + Ssrc: 0, + Direction: m.Mode, + Port: m.Port, + } + + if trackInfo.Port > 0 { + trackInfo.RtcpPort = m.Port + 1 + } + trackInfo.Codecs = fixFormatName(m.Format) + if m.Type == "audio" { + trackInfo.TrackType = TrackTypeAudio + } else if m.Type == "video" { + trackInfo.TrackType = TrackTypeVideo + } else { + continue + } + mediaDesc.Tracks[trackInfo.TrackType] = trackInfo + } + + return mediaDesc, nil +} + +func Negotiation(local *MediaDescription, remote *MediaDescription) (*MediaDescription, error) { + + return nil, nil +} diff --git a/examples/b2bua/b2bua/rtc_media_tp.go b/examples/b2bua/b2bua/rtc_media_tp.go new file mode 100644 index 0000000..a811ef9 --- /dev/null +++ b/examples/b2bua/b2bua/rtc_media_tp.go @@ -0,0 +1,656 @@ +package b2bua + +import ( + "context" + "fmt" + "io" + "math/rand" + "net" + "strings" + "sync" + + "github.com/cloudwebrtc/go-sip-ua/examples/b2bua/b2bua/buffer" + "github.com/cloudwebrtc/go-sip-ua/pkg/utils" + "github.com/pion/interceptor" + "github.com/pion/rtcp" + "github.com/pion/rtp" + "github.com/pion/webrtc/v3" +) + +const maxPktSize = 1500 + +var ( + webrtcSettings webrtc.SettingEngine + MaxPacketTrack = 500 +) + +const ( + mimeTypeH264 = "video/H264" + mimeTypeOpus = "audio/OPUS" + mimeTypeVP8 = "video/VP8" + mimeTypeVP9 = "video/VP9" + mimeTypeAV1 = "audio/AV1" + mineTypePCMA = "audio/PCMA" + mineTypePCMU = "audio/PCMU" + mimeTypeG722 = "audio/G722" + mimeTypeILBC = "audio/ILBC" +) + +func init() { + webrtcSettings = webrtc.SettingEngine{} + udpListener, err := net.ListenUDP("udp", &net.UDPAddr{ + IP: net.IP{0, 0, 0, 0}, + Port: 50160, + }) + if err != nil { + panic(err) + } + webrtcSettings.SetICEUDPMux(webrtc.NewICEUDPMux(nil, udpListener)) +} + +const DefaultPayloadTypeOpus = 111 + +var ( + setting webrtc.SettingEngine + cfg = webrtc.Configuration{ + SDPSemantics: webrtc.SDPSemanticsUnifiedPlanWithFallback, + } +) + +type WebRTCMediaTransport struct { + pc *webrtc.PeerConnection + answer webrtc.SessionDescription + offer webrtc.SessionDescription + localTracks map[TrackType]*webrtc.TrackLocalStaticRTP + remoteTracks map[TrackType]*webrtc.TrackRemote + closed utils.AtomicBool + ctx context.Context + cancel context.CancelFunc + md *MediaDescription + + videoPool *sync.Pool + audioPool *sync.Pool + + sequencer *sequencer + buff *buffer.Buffer + bmu sync.Mutex + mu sync.RWMutex + rtpHandler func(trackType TrackType, pkt rtp.Packet) (int, error) + rtcpHandler func(trackType TrackType, pkt rtcp.Packet) (int, error) + requestKeyFrameHandler func() error +} + +func NewWebRTCMediaTransport(md *MediaDescription) *WebRTCMediaTransport { + c := &WebRTCMediaTransport{ + md: md, + localTracks: make(map[TrackType]*webrtc.TrackLocalStaticRTP), + remoteTracks: make(map[TrackType]*webrtc.TrackRemote), + sequencer: newSequencer(MaxPacketTrack), + videoPool: &sync.Pool{ + New: func() interface{} { + b := make([]byte, MaxPacketTrack*maxPktSize) + return &b + }, + }, + audioPool: &sync.Pool{ + New: func() interface{} { + b := make([]byte, maxPktSize*25) + return &b + }, + }, + buff: nil, + } + c.ctx, c.cancel = context.WithCancel(context.Background()) + c.closed.Set(false) + return c +} + +func (c *WebRTCMediaTransport) Type() MediaTransportType { + return TransportTypeWebRTC +} + +func (c *WebRTCMediaTransport) Init(umc UserAgentMediaConfig) error { + // Create a MediaEngine object to configure the supported codec + m := &webrtc.MediaEngine{} + for _, codec := range []webrtc.RTPCodecParameters{ + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mineTypePCMU, ClockRate: 8000, Channels: 1, SDPFmtpLine: "", RTCPFeedback: nil}, + PayloadType: 0, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mineTypePCMA, ClockRate: 8000, Channels: 1, SDPFmtpLine: "", RTCPFeedback: nil}, + PayloadType: 8, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeG722, ClockRate: 8000, Channels: 1, SDPFmtpLine: "", RTCPFeedback: nil}, + PayloadType: 9, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeILBC, ClockRate: 8000, Channels: 1, SDPFmtpLine: "", RTCPFeedback: nil}, + PayloadType: 103, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeOpus, ClockRate: 48000, Channels: 2, SDPFmtpLine: "minptime=10;useinbandfec=1", RTCPFeedback: nil}, + PayloadType: 111, + }, + } { + for _, trackInfo := range c.md.Tracks { + if trackInfo.TrackType == TrackTypeAudio { + for _, c := range trackInfo.Codecs { + if strings.Contains(strings.ToLower(codec.RTPCodecCapability.MimeType), strings.ToLower(c.Name)) { + //codec.PayloadType = webrtc.PayloadType(c.Payload) + if err := m.RegisterCodec(codec, webrtc.RTPCodecTypeAudio); err != nil { + return err + } + } + } + } + } + } + + videoRTCPFeedback := b2buaConfig.UaMediaConfig.RtcpFeedback + + for _, codec := range []webrtc.RTPCodecParameters{ + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeVP8, ClockRate: 90000, RTCPFeedback: videoRTCPFeedback}, + PayloadType: 100, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeVP9, ClockRate: 90000, RTCPFeedback: videoRTCPFeedback}, + PayloadType: 127, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeAV1, ClockRate: 90000, RTCPFeedback: videoRTCPFeedback}, + PayloadType: 35, + }, + { + RTPCodecCapability: webrtc.RTPCodecCapability{MimeType: mimeTypeH264, ClockRate: 90000, SDPFmtpLine: "level-asymmetry-allowed=1;packetization-mode=1;profile-level-id=640c33", RTCPFeedback: videoRTCPFeedback}, + PayloadType: 96, + }, + } { + for _, trackInfo := range c.md.Tracks { + if trackInfo.TrackType == TrackTypeVideo { + for _, c := range trackInfo.Codecs { + if strings.Contains(strings.ToLower(codec.RTPCodecCapability.MimeType), strings.ToLower(c.Name)) { + //codec.PayloadType = webrtc.PayloadType(c.Payload) + if err := m.RegisterCodec(codec, webrtc.RTPCodecTypeVideo); err != nil { + return err + } + } + } + } + } + } + // Create a InterceptorRegistry. This is the user configurable RTP/RTCP Pipeline. + // This provides NACKs, RTCP Reports and other features. If you use `webrtc.NewPeerConnection` + // this is enabled by default. If you are manually managing You MUST create a InterceptorRegistry + // for each PeerConnection. + i := &interceptor.Registry{} + + // Use the default set of Interceptors + if err := webrtc.RegisterDefaultInterceptors(m, i); err != nil { + panic(err) + } + + // Create the API object with the MediaEngine + api := webrtc.NewAPI(webrtc.WithMediaEngine(m), webrtc.WithInterceptorRegistry(i)) + + // Prepare the configuration + config := webrtc.Configuration{ + ICEServers: []webrtc.ICEServer{}, + SDPSemantics: webrtc.SDPSemanticsUnifiedPlanWithFallback, + //RTCPMuxPolicy: webrtc.RTCPMuxPolicyRequire, + BundlePolicy: webrtc.BundlePolicyBalanced, + } + + // Create a new RTCPeerConnection + pc, err := api.NewPeerConnection(config) + if err != nil { + panic(err) + } + + c.pc = pc + + c.pc.OnICEConnectionStateChange(func(connectionState webrtc.ICEConnectionState) { + logger.Infof("ICE Connection State has changed: %s\n", connectionState.String()) + }) + c.pc.OnConnectionStateChange(func(state webrtc.PeerConnectionState) { + logger.Infof("PeerConnection State has changed: %s\n", state.String()) + }) + + c.pc.OnTrack(func(track *webrtc.TrackRemote, recevier *webrtc.RTPReceiver) { + if track.Kind() == webrtc.RTPCodecTypeVideo { + c.bmu.Lock() + if c.buff == nil { + c.buff = buffer.NewBuffer(uint32(track.SSRC()), c.videoPool, c.audioPool, buffer.Logger) + c.buff.Bind(recevier.GetParameters(), buffer.Options{ + MaxBitRate: 1500, + }) + + c.buff.OnFeedback(func(fb []rtcp.Packet) {}) + } + c.bmu.Unlock() + c.remoteTracks[TrackTypeVideo] = track + } else if track.Kind() == webrtc.RTPCodecTypeAudio { + c.remoteTracks[TrackTypeAudio] = track + } + buf := make([]byte, 1500) + for { + if c.closed.Get() { + logger.Infof("OnTrack: stop now!") + break + } + // Read + n, _, readErr := track.Read(buf) + if readErr != nil { + logger.Errorf("track.Read: readErr => %v", readErr) + break + } + //logger.Infof("WebRTCTransport::OnTrack: read %d bytes", n) + if track.Kind() == webrtc.RTPCodecTypeAudio { + c.onRtpPacket(TrackTypeAudio, buf[:n]) + } else if track.Kind() == webrtc.RTPCodecTypeVideo { + c.onRtpPacket(TrackTypeVideo, buf[:n]) + } + } + + }) + return nil +} + +func (c *WebRTCMediaTransport) OnRtpPacket(rtpHandler func(trackType TrackType, pkt rtp.Packet) (int, error)) { + c.mu.Lock() + defer c.mu.Unlock() + c.rtpHandler = rtpHandler +} + +func (c *WebRTCMediaTransport) OnRtcpPacket(rtcpHandler func(trackType TrackType, pkt rtcp.Packet) (int, error)) { + c.mu.Lock() + defer c.mu.Unlock() + c.rtcpHandler = rtcpHandler +} + +func (c *WebRTCMediaTransport) OnRequestKeyFrame(keyHandler func() error) { + c.mu.Lock() + defer c.mu.Unlock() + c.requestKeyFrameHandler = keyHandler +} + +func (c *WebRTCMediaTransport) onRtpPacket(trackType TrackType, packet []byte) error { + logger.Tracef("WebRTCTransport::OnRtpPacketReceived: %v read %d bytes", trackType, len(packet)) + c.mu.RLock() + defer c.mu.RUnlock() + + p := rtp.Packet{} + if err := p.Unmarshal(packet); err != nil { + return err + } + + if c.rtpHandler != nil { + if _, err := c.rtpHandler(trackType, p); err != nil { + return err + } + } + return nil +} + +func (c *WebRTCMediaTransport) onRtcpPacket(trackType TrackType, packet []byte) error { + logger.Debugf("WebRTCTransport::OnRtcpPacketReceived: %v read %d bytes", trackType, len(packet)) + c.mu.RLock() + defer c.mu.RUnlock() + + pkts, err := rtcp.Unmarshal(packet) + if err != nil { + return err + } + + if c.rtcpHandler != nil { + for _, pkt := range pkts { + if _, err := c.rtcpHandler(trackType, pkt); err != nil { + return err + } + } + } + return nil +} + +func (c *WebRTCMediaTransport) WriteRTP(trackType TrackType, pkt rtp.Packet) (int, error) { + if c.closed.Get() { + return 0, fmt.Errorf("WebRTCTransport::SendRtpPacket: closed") + } + if trackType == TrackTypeAudio { + if c.localTracks[TrackTypeAudio] != nil { + err := c.localTracks[TrackTypeAudio].WriteRTP(&pkt) + if err != nil { + return 0, fmt.Errorf("WebRTCTransport::SendRtpPacket: %v", err) + } + return len(pkt.Payload), nil + } + } else if trackType == TrackTypeVideo { + if c.localTracks[TrackTypeVideo] != nil { + if buf, err := pkt.Marshal(); err == nil { + c.bmu.Lock() + if c.buff != nil { + c.buff.Write(buf) + if err != io.EOF { + if c.sequencer != nil { + c.sequencer.push(pkt.SequenceNumber, pkt.SequenceNumber, pkt.Timestamp, 0, true) + } + } + } + c.bmu.Unlock() + } + + err := c.localTracks[TrackTypeVideo].WriteRTP(&pkt) + if err != nil { + return 0, fmt.Errorf("WebRTCTransport::SendRtpPacket: %v", err) + } + return len(pkt.Payload), nil + } + } + return 0, fmt.Errorf("WebRTCTransport::SendRtpPacket: invalid trackType %v", trackType) +} + +func (c *WebRTCMediaTransport) WriteRTCP(trackType TrackType, pkt rtcp.Packet) (n int, err error) { + if c.closed.Get() { + return 0, fmt.Errorf("WebRTCTransport::SendRtcpPacket: closed") + } + + err = c.pc.WriteRTCP([]rtcp.Packet{pkt}) + if err != nil { + return 0, fmt.Errorf("WebRTCTransport::SendRtcpPacket: pc.WriteRTCP err => %v", err) + } + + return 0, nil +} + +func (c *WebRTCMediaTransport) Close() error { + if c.closed.Get() { + return nil + } + c.closed.Set(true) + c.cancel() + return c.pc.Close() +} + +func (c *WebRTCMediaTransport) AddLocalTracks() error { + + for _, trackInfo := range c.md.Tracks { + + if trackInfo.TrackType == TrackTypeAudio { + + audioTrack, err := webrtc.NewTrackLocalStaticRTP( + webrtc.RTPCodecCapability{MimeType: webrtc.MimeTypePCMU}, + fmt.Sprintf("audio-%d", rand.Uint32()), + fmt.Sprintf("rtc-%d", rand.Uint32()), + ) + + if err != nil { + logger.Errorf("NewTrack: panic => %v", err) + return err + } + + if _, err = c.pc.AddTrack(audioTrack); err != nil { + logger.Errorf("AddTrack: panic => %v", err) + return err + } + + c.localTracks[TrackTypeAudio] = audioTrack + } else if trackInfo.TrackType == TrackTypeVideo { + + videoTrack, err := webrtc.NewTrackLocalStaticRTP( + webrtc.RTPCodecCapability{MimeType: mimeTypeH264}, + fmt.Sprintf("video-%d", rand.Uint32()), + fmt.Sprintf("rtc-%d", rand.Uint32()), + ) + + if err != nil { + logger.Errorf("NewTrack: panic => %v", err) + return err + } + + if rtpSender, err := c.pc.AddTrack(videoTrack); err == nil { + c.HandleRtcpFb(rtpSender) + } else { + logger.Errorf("AddTrack: panic => %v", err) + return err + } + + c.localTracks[TrackTypeVideo] = videoTrack + } + } + + return nil +} + +func (c *WebRTCMediaTransport) CreateOffer() (*Desc, error) { + + err := c.AddLocalTracks() + if err != nil { + logger.Errorf("AddLocalTracks: panic => %v", err) + return nil, err + } + + c.offer, err = c.pc.CreateOffer(nil) + if err != nil { + logger.Errorf("CreateOffer: panic => %v", err) + return nil, err + } + gatherComplete := webrtc.GatheringCompletePromise(c.pc) + if err = c.pc.SetLocalDescription(c.offer); err != nil { + logger.Errorf("SetLocalDescription: panic => %v", err) + return nil, err + } + <-gatherComplete + c.offer = *c.pc.LocalDescription() + return &Desc{SDP: c.offer.SDP, Type: "offer"}, nil +} + +func (c *WebRTCMediaTransport) OnAnswer(answer *Desc) error { + c.answer = webrtc.SessionDescription{ + Type: webrtc.SDPTypeAnswer, + SDP: answer.SDP, + } + if err := c.pc.SetRemoteDescription(c.answer); err != nil { + logger.Errorf("OnAnswer::WebRTCTransport::SetRemoteDescription: panic => %v", err) + return err + } + return nil +} + +func (c *WebRTCMediaTransport) OnOffer(offer *Desc) error { + + err := c.AddLocalTracks() + if err != nil { + logger.Errorf("AddLocalTracks: panic => %v", err) + return err + } + + desc := webrtc.SessionDescription{ + Type: webrtc.SDPTypeOffer, + SDP: offer.SDP, + } + + if err := c.pc.SetRemoteDescription(desc); err != nil { + logger.Errorf("WebRTCTransport::OnOffer::SetRemoteDescription: panic => %v", err) + return err + } + return nil +} + +func (c *WebRTCMediaTransport) CreateAnswer() (*Desc, error) { + var err error = nil + c.answer, err = c.pc.CreateAnswer(nil) + if err != nil { + logger.Errorf("CreateAnswer: panic => %v", err) + return nil, err + } + + gatherComplete := webrtc.GatheringCompletePromise(c.pc) + if err = c.pc.SetLocalDescription(c.answer); err != nil { + logger.Errorf("SetLocalDescription: panic => %v", err) + return nil, err + } + <-gatherComplete + c.answer = *c.pc.LocalDescription() + + return &Desc{SDP: c.answer.SDP, Type: "answer"}, nil +} + +func (c *WebRTCMediaTransport) HandleRtcpFb(rtpSender *webrtc.RTPSender) { + // Read incoming RTCP packets + // Before these packets are returned they are processed by interceptors. For things + // like NACK this needs to be called. + go func() { + rtcpBuf := make([]byte, 1500) + for { + n, _, rtcpErr := rtpSender.Read(rtcpBuf) + if rtcpErr != nil { + return + } + bytes := rtcpBuf[:n] + pkts, err := rtcp.Unmarshal(bytes) + if err != nil { + logger.Errorf("Unmarshal rtcp receiver packets err %v", err) + } + var fwdPkts []rtcp.Packet + pliOnce := true + firOnce := true + var ( + maxRatePacketLoss uint8 + expectedMinBitrate uint64 + ) + for _, pkt := range pkts { + switch p := pkt.(type) { + case *rtcp.PictureLossIndication: + if pliOnce { + fwdPkts = append(fwdPkts, p) + logger.Tracef("Picture Loss Indication") + if c.requestKeyFrameHandler != nil { + c.requestKeyFrameHandler() + } + pliOnce = false + } + case *rtcp.FullIntraRequest: + if firOnce { + fwdPkts = append(fwdPkts, p) + logger.Tracef("FullIntraRequest") + if c.requestKeyFrameHandler != nil { + c.requestKeyFrameHandler() + } + firOnce = false + } + case *rtcp.ReceiverEstimatedMaximumBitrate: + if expectedMinBitrate == 0 || expectedMinBitrate > uint64(p.Bitrate) { + expectedMinBitrate = uint64(p.Bitrate) + //hi.CameraUpdateBitrate(uint32(expectedMinBitrate / 1024)) + logger.Tracef("[%v] ReceiverEstimatedMaximumBitrate %d", rtpSender.Track().Kind(), expectedMinBitrate/1024) + } + case *rtcp.ReceiverReport: + for _, r := range p.Reports { + if maxRatePacketLoss == 0 || maxRatePacketLoss < r.FractionLost { + maxRatePacketLoss = r.FractionLost + logger.Tracef("maxRatePacketLoss %d", maxRatePacketLoss) + } + } + case *rtcp.TransportLayerNack: + logger.Infof("Nack %v", p) + if c.sequencer != nil { + var nackedPackets []packetMeta + for _, pair := range p.Nacks { + nackedPackets = append(nackedPackets, c.sequencer.getSeqNoPairs(pair.PacketList())...) + } + if len(nackedPackets) > 0 { + if err = c.RetransmitPackets(nackedPackets); err == nil { + logger.Infof("Nack pair %v", nackedPackets) + } + } else { + buf, _ := p.Marshal() + c.onRtcpPacket(TrackTypeVideo, buf) + } + } + } + } + } + }() +} + +func (c *WebRTCMediaTransport) RequestKeyFrame() error { + track := c.remoteTracks[TrackTypeVideo] + if track == nil { + return fmt.Errorf("video track is nil") + } + c.pc.WriteRTCP([]rtcp.Packet{&rtcp.PictureLossIndication{MediaSSRC: uint32(track.SSRC())}}) + return nil +} + +func (c *WebRTCMediaTransport) RetransmitPackets(nackedPackets []packetMeta) error { + c.bmu.Lock() + defer c.bmu.Unlock() + if c.buff == nil { + return fmt.Errorf("buffer is nil") + } + for _, meta := range nackedPackets { + pktBuff := make([]byte, 1500) + i, err := c.buff.GetPacket(pktBuff, meta.sourceSeqNo) + if err != nil { + if err == io.EOF { + break + } + continue + } + var pkt rtp.Packet + if err = pkt.Unmarshal(pktBuff[:i]); err != nil { + continue + } + pkt.Header.SequenceNumber = meta.targetSeqNo + //pkt.Header.Timestamp = meta.timestamp + //pkt.Header.SSRC = track.ssrc + //pkt.Header.PayloadType = track.payloadType + //if _, err = track.writeStream.WriteRTP(&pkt.Header, pkt.Payload); err != nil { + // logger.Error(err, "Writing rtx packet err") + //} + packet, _ := pkt.Marshal() + c.localTracks[TrackTypeVideo].Write(packet) + + } + return nil +} + +// InitWebRTC init WebRTCTransport setting +func InitWebRTC(nat1to1 []string, icelite bool, iceServers []webrtc.ICEServer, icePortStart, icePortEnd uint16, iceSinglePort int) error { + var err error + if icePortStart != 0 || icePortEnd != 0 { + err = setting.SetEphemeralUDPPortRange(icePortStart, icePortEnd) + if err != nil { + logger.Errorf("SetEphemeralUDPPortRange: err => %v", err) + return err + } + } + + if len(nat1to1) > 0 { + setting.SetNAT1To1IPs(nat1to1, webrtc.ICECandidateTypeHost) + } + if icelite { + setting.SetLite(icelite) + cfg.ICEServers = []webrtc.ICEServer{} + } else { + cfg.ICEServers = iceServers + } + + setting.DisableMediaEngineCopy(true) + + if iceSinglePort != 0 { + logger.Info("Listen on ", "single-port") + udpListener, err := net.ListenUDP("udp", &net.UDPAddr{ + IP: net.IP{0, 0, 0, 0}, + Port: iceSinglePort, + }) + if err != nil { + panic(err) + } + setting.SetICEUDPMux(webrtc.NewICEUDPMux(nil, udpListener)) + } + + return err +} diff --git a/examples/b2bua/b2bua/sequencer.go b/examples/b2bua/b2bua/sequencer.go new file mode 100644 index 0000000..9a3c60f --- /dev/null +++ b/examples/b2bua/b2bua/sequencer.go @@ -0,0 +1,129 @@ +package b2bua + +import ( + "sync" + "time" +) + +const ( + ignoreRetransmission = 100 // Ignore packet retransmission after ignoreRetransmission milliseconds +) + +type packetMeta struct { + // Original sequence number from stream. + // The original sequence number is used to find the original + // packet from publisher + sourceSeqNo uint16 + // Modified sequence number after offset. + // This sequence number is used for the associated + // down track, is modified according the offsets, and + // must not be shared + targetSeqNo uint16 + // Modified timestamp for current associated + // down track. + timestamp uint32 + // The last time this packet was nack requested. + // Sometimes clients request the same packet more than once, so keep + // track of the requested packets helps to avoid writing multiple times + // the same packet. + // The resolution is 1 ms counting after the sequencer start time. + lastNack uint32 + // Spatial layer of packet + layer uint8 + // Information that differs depending the codec + misc uint32 +} + +func (p *packetMeta) setVP8PayloadMeta(tlz0Idx uint8, picID uint16) { + p.misc = uint32(tlz0Idx)<<16 | uint32(picID) +} + +func (p *packetMeta) getVP8PayloadMeta() (uint8, uint16) { + return uint8(p.misc >> 16), uint16(p.misc) +} + +// Sequencer stores the packet sequence received by the down track +type sequencer struct { + sync.Mutex + init bool + max int + seq []packetMeta + step int + headSN uint16 + startTime int64 +} + +func newSequencer(maxTrack int) *sequencer { + return &sequencer{ + startTime: time.Now().UnixNano() / 1e6, + max: maxTrack, + seq: make([]packetMeta, maxTrack), + } +} + +func (n *sequencer) push(sn, offSn uint16, timeStamp uint32, layer uint8, head bool) *packetMeta { + n.Lock() + defer n.Unlock() + if !n.init { + n.headSN = offSn + n.init = true + } + + step := 0 + if head { + inc := offSn - n.headSN + for i := uint16(1); i < inc; i++ { + n.step++ + if n.step >= n.max { + n.step = 0 + } + } + step = n.step + n.headSN = offSn + } else { + step = n.step - int(n.headSN-offSn) + if step < 0 { + if step*-1 >= n.max { + logger.Info("Old packet received, can not be sequenced", "head", sn, "received", offSn) + return nil + } + step = n.max + step + } + } + n.seq[n.step] = packetMeta{ + sourceSeqNo: sn, + targetSeqNo: offSn, + timestamp: timeStamp, + layer: layer, + } + pm := &n.seq[n.step] + n.step++ + if n.step >= n.max { + n.step = 0 + } + return pm +} + +func (n *sequencer) getSeqNoPairs(seqNo []uint16) []packetMeta { + n.Lock() + meta := make([]packetMeta, 0, 17) + refTime := uint32(time.Now().UnixNano()/1e6 - n.startTime) + for _, sn := range seqNo { + step := n.step - int(n.headSN-sn) - 1 + if step < 0 { + if step*-1 >= n.max { + continue + } + step = n.max + step + } + seq := &n.seq[step] + if seq.targetSeqNo == sn { + if seq.lastNack == 0 || refTime-seq.lastNack > ignoreRetransmission { + seq.lastNack = refTime + meta = append(meta, *seq) + } + } + } + n.Unlock() + return meta +} diff --git a/examples/b2bua/b2bua/sequencer_test.go b/examples/b2bua/b2bua/sequencer_test.go new file mode 100644 index 0000000..60e8162 --- /dev/null +++ b/examples/b2bua/b2bua/sequencer_test.go @@ -0,0 +1,99 @@ +package b2bua + +import ( + "reflect" + "testing" + "time" + + "github.com/stretchr/testify/assert" +) + +func Test_sequencer(t *testing.T) { + seq := newSequencer(500) + off := uint16(15) + + for i := uint16(1); i < 520; i++ { + seq.push(i, i+off, 123, 2, true) + } + + time.Sleep(60 * time.Millisecond) + req := []uint16{57, 58, 62, 63, 513, 514, 515, 516, 517} + res := seq.getSeqNoPairs(req) + assert.Equal(t, len(req), len(res)) + for i, val := range res { + assert.Equal(t, val.targetSeqNo, req[i]) + assert.Equal(t, val.sourceSeqNo, req[i]-off) + assert.Equal(t, val.layer, uint8(2)) + } + res = seq.getSeqNoPairs(req) + assert.Equal(t, 0, len(res)) + time.Sleep(150 * time.Millisecond) + res = seq.getSeqNoPairs(req) + assert.Equal(t, len(req), len(res)) + for i, val := range res { + assert.Equal(t, val.targetSeqNo, req[i]) + assert.Equal(t, val.sourceSeqNo, req[i]-off) + assert.Equal(t, val.layer, uint8(2)) + } + + s := seq.push(521, 521+off, 123, 1, true) + var ( + tlzIdx = uint8(15) + picID = uint16(16) + ) + s.setVP8PayloadMeta(tlzIdx, picID) + s.sourceSeqNo = 12 + m := seq.getSeqNoPairs([]uint16{521 + off}) + assert.Equal(t, 1, len(m)) + tlz0, pID := m[0].getVP8PayloadMeta() + assert.Equal(t, tlzIdx, tlz0) + assert.Equal(t, picID, pID) +} + +func Test_sequencer_getNACKSeqNo(t *testing.T) { + type args struct { + seqNo []uint16 + } + type fields struct { + input []uint16 + offset uint16 + } + + tests := []struct { + name string + fields fields + args args + want []uint16 + }{ + { + name: "Should get correct seq numbers", + fields: fields{ + input: []uint16{2, 3, 4, 7, 8}, + offset: 5, + }, + args: args{ + seqNo: []uint16{4 + 5, 5 + 5, 8 + 5}, + }, + want: []uint16{4, 8}, + }, + } + for _, tt := range tests { + tt := tt + t.Run(tt.name, func(t *testing.T) { + n := newSequencer(500) + + for _, i := range tt.fields.input { + n.push(i, i+tt.fields.offset, 123, 3, true) + } + + g := n.getSeqNoPairs(tt.args.seqNo) + var got []uint16 + for _, sn := range g { + got = append(got, sn.sourceSeqNo) + } + if !reflect.DeepEqual(got, tt.want) { + t.Errorf("getSeqNoPairs() = %v, want %v", got, tt.want) + } + }) + } +} diff --git a/examples/b2bua/b2bua/standard_tp.go b/examples/b2bua/b2bua/standard_tp.go new file mode 100644 index 0000000..d064b03 --- /dev/null +++ b/examples/b2bua/b2bua/standard_tp.go @@ -0,0 +1,369 @@ +package b2bua + +import ( + "context" + "fmt" + "net" + "sync" + "time" + + "github.com/cloudwebrtc/go-sip-ua/pkg/utils" + "github.com/ghettovoice/gosip/util" + "github.com/pion/rtcp" + "github.com/pion/rtp" + "github.com/pixelbender/go-sdp/sdp" +) + +type StandardMediaTransport struct { + md *MediaDescription + ports map[TrackType]*UdpPort + localDescription *sdp.Session + remoteDescription *sdp.Session + + mu sync.RWMutex + rtpHandler func(trackType TrackType, pkt rtp.Packet) (int, error) + rtcpHandler func(trackType TrackType, pkt rtcp.Packet) (int, error) + requestKeyFrameHandler func() error + + videoSSRC uint32 + closed utils.AtomicBool + ctx context.Context + cancel context.CancelFunc +} + +func NewStandardMediaTransport(md *MediaDescription) *StandardMediaTransport { + t := &StandardMediaTransport{ + md: md, + ports: make(map[TrackType]*UdpPort), + videoSSRC: 0, + } + + t.ctx, t.cancel = context.WithCancel(context.TODO()) + t.closed.Set(false) + return t +} + +func (c *StandardMediaTransport) Init(umc UserAgentMediaConfig) error { + + host := b2buaConfig.UaMediaConfig.ExternalRtpAddress + + if host == "" || host == "0.0.0.0" { + if v, err := util.ResolveSelfIP(); err == nil { + host = v.String() + } + } + + c.localDescription = &sdp.Session{ + Origin: &sdp.Origin{ + Username: "-", + Address: host, + SessionID: time.Now().UnixNano() / 1e6, + SessionVersion: time.Now().UnixNano() / 1e6, + }, + Timing: &sdp.Timing{Start: time.Time{}, Stop: time.Time{}}, + //Name: "play", // Session Name ("s=") + Connection: &sdp.Connection{ + Address: host, + }, + //Bandwidth: []*sdp.Bandwidth{{Type: "AS", Value: 117}}, + } + + var medias []*sdp.Media + + for _, trackInfo := range c.md.Tracks { + + var rAddr *net.UDPAddr = nil + var rRtcpAddr *net.UDPAddr = nil + if c.md.Connection != nil { + rAddr, _ = net.ResolveUDPAddr("udp", fmt.Sprintf("%s:%d", c.md.Connection.Address, trackInfo.Port)) + rRtcpAddr, _ = net.ResolveUDPAddr("udp", fmt.Sprintf("%s:%d", c.md.Connection.Address, trackInfo.RtcpPort)) + } + + udpPort, err := NewUdpPort(trackInfo.TrackType, rAddr, rRtcpAddr, umc.ExternalRtpAddress) + if err != nil { + return err + } + udpPort.Init() + udpPort.OnRtpPacket(c.onRtpPacket) + udpPort.OnRtcpPacket(c.onRtcpPacket) + c.ports[trackInfo.TrackType] = udpPort + + media := &sdp.Media{} + media.Type = string(trackInfo.TrackType) + media.Port = udpPort.LocalPort() + media.Proto = "RTP/AVP" + media.Mode = trackInfo.Direction + media.Connection = []*sdp.Connection{{Address: host}} + //Bandwidth: []*sdp.Bandwidth{{Type: "TIAS", Value: 96000}}, + + var formats []*sdp.Format + for _, codec := range trackInfo.Codecs { + for _, enabledCodec := range b2buaConfig.UaMediaConfig.Codecs { + if codec.Name == enabledCodec { + formats = append(formats, codec) + } + } + } + media.Format = formats + medias = append(medias, media) + } + + c.localDescription.Media = medias + return nil +} + +func (c *StandardMediaTransport) OnRtpPacket(rtpHandler func(trackType TrackType, pkt rtp.Packet) (int, error)) { + c.mu.Lock() + defer c.mu.Unlock() + c.rtpHandler = rtpHandler +} + +func (c *StandardMediaTransport) OnRtcpPacket(rtcpHandler func(trackType TrackType, pkt rtcp.Packet) (int, error)) { + c.mu.Lock() + defer c.mu.Unlock() + c.rtcpHandler = rtcpHandler +} + +func (c *StandardMediaTransport) OnRequestKeyFrame(keyHandler func() error) { + c.mu.Lock() + defer c.mu.Unlock() + c.requestKeyFrameHandler = keyHandler +} + +func (c *StandardMediaTransport) onRtpPacket(trackType TrackType, packet []byte, raddr net.Addr) error { + logger.Tracef("UdpTansport::onRtpPacket: %v read %d bytes, raddr %v", trackType, len(packet), raddr) + + p := rtp.Packet{} + if err := p.Unmarshal(packet); err != nil { + logger.Errorf("rtp.Packet Unmarshal: e %v len %v", err, len(packet)) + } + + logger.Tracef("UdpTansport::onRtpPacket: [%v] read %d bytes, seq %d, ts %d, ssrc %v, payload %v", trackType, len(packet), p.SequenceNumber, p.Timestamp, p.SSRC, p.PayloadType) + + if trackType == TrackTypeVideo && c.videoSSRC == 0 { + c.videoSSRC = p.SSRC + //c.sendPLI(c.videoSSRC) + } + + /* + track, found := c.md.Tracks[TrackTypeVideo] + if found && len(track.Codecs) > 0 && strings.ToUpper(track.Codecs[0].Name) == "PS" { + + return nil + } + */ + + c.mu.RLock() + defer c.mu.RUnlock() + if c.rtpHandler != nil { + if _, err := c.rtpHandler(trackType, p); err != nil { + logger.Warnf("UdpTansport::onRtpPacket: panic => %v", err) + } + } + return nil +} + +func (c *StandardMediaTransport) onRtcpPacket(trackType TrackType, buf []byte, raddr net.Addr) error { + logger.Tracef("UdpTansport::OnRtcpPacketReceived: %v read %d bytes, raddr %v", trackType, len(buf), raddr) + c.mu.RLock() + defer c.mu.RUnlock() + + pkts, err := rtcp.Unmarshal(buf) + + if err != nil { + //logger.Warnf("UdpTansport::OnRtcpPacketReceived: Unmarshal rtcp receiver packets err %v", err) + return err + } + + if c.rtcpHandler != nil { + for _, pkt := range pkts { + if _, err := c.rtcpHandler(trackType, pkt); err != nil { + logger.Warnf("UdpTansport::onRtcpPacket: panic => %v", err) + } + } + } + return nil +} + +func (c *StandardMediaTransport) WriteRTP(trackType TrackType, pkt rtp.Packet) (int, error) { + logger.Tracef("UdpTansport::WriteRTP: %v, write %d bytes", trackType, len(pkt.Payload)) + + port := c.ports[trackType] + + if port == nil { + logger.Errorf("UdpTansport::WriteRTP: port is nil") + return 0, nil + } + + track, found := c.md.Tracks[trackType] + if !found { + return 0, fmt.Errorf("track %v not found", trackType) + } + + //re-write payload type + pkt.PayloadType = track.Codecs[0].Payload + pktbuf, err := pkt.Marshal() + + if err != nil { + logger.Errorf("UdpTansport::WriteRTP: Marshal rtp receiver packets err %v", err) + } + + return port.WriteRtp(pktbuf) +} + +func (c *StandardMediaTransport) WriteRTCP(trackType TrackType, pkt rtcp.Packet) (int, error) { + port := c.ports[trackType] + + if port == nil { + logger.Errorf("UdpTansport::WriteRTCP: port is nil") + return 0, nil + } + + buf, err := pkt.Marshal() + if err != nil { + logger.Errorf("UdpTansport::WriteRTCP: Marshal rtcp receiver packets err %v", err) + return 0, err + } + + logger.Tracef("UdpTansport::WriteRTCP: %v, write %d bytes", trackType, len(buf)) + return port.WriteRtcp(buf) +} + +func (c *StandardMediaTransport) Type() MediaTransportType { + return TransportTypeStandard +} + +func (c *StandardMediaTransport) Close() error { + for _, udpPort := range c.ports { + udpPort.Close() + } + + return nil +} + +func (c *StandardMediaTransport) CreateOffer() (*Desc, error) { + return &Desc{ + Type: "offer", + SDP: c.localDescription.String(), + }, nil +} + +func (c *StandardMediaTransport) OnAnswer(answer *Desc) error { + sess, err := sdp.Parse([]byte(answer.SDP)) + if err != nil { + return err + } + conn := sess.Connection + if conn != nil { + logger.Debugf("remote connection address: %s", conn.Address) + } + c.md, _ = MediaDescriptionFrom(answer) + c.remoteDescription = sess + return nil +} + +func (c *StandardMediaTransport) OnOffer(offer *Desc) error { + sess, err := sdp.Parse([]byte(offer.SDP)) + if err != nil { + return err + } + c.remoteDescription = sess + + return nil +} + +func (c *StandardMediaTransport) CreateAnswer() (*Desc, error) { + return &Desc{ + Type: "offer", + SDP: c.localDescription.String(), + }, nil +} + +func (c *StandardMediaTransport) RequestKeyFrame() error { + if c.videoSSRC == 0 { + return fmt.Errorf("video ssrc is 0") + } + return c.sendPLI(c.videoSSRC) +} + +func (c *StandardMediaTransport) sendPLI(ssrc uint32) error { + pli := rtcp.PictureLossIndication{MediaSSRC: uint32(ssrc)} + + _, errSend := c.WriteRTCP(TrackTypeVideo, &pli) + if errSend != nil { + logger.Error(errSend) + return errSend + } + logger.Infof("Sent PLI %v", pli) + return nil +} + +func (c *StandardMediaTransport) sendTntervalPlic(ssrc uint32) error { + go func() { + ticker := time.NewTicker(time.Second * 1) + for range ticker.C { + if c.closed.Get() { + logger.Infof("Terminate: stop pli loop now!") + return + } + pli := rtcp.PictureLossIndication{SenderSSRC: uint32(0), MediaSSRC: uint32(ssrc)} + _, errSend := c.WriteRTCP(TrackTypeVideo, &pli) + if errSend != nil { + logger.Error(errSend) + return + } + logger.Infof("Sent PLI %v", pli) + } + }() + return nil +} + +func (c *StandardMediaTransport) handleRtcpFeedback(packet []byte) { + pkts, err := rtcp.Unmarshal(packet) + if err != nil { + logger.Errorf("Unmarshal rtcp receiver packets err %v", err) + } + var fwdPkts []rtcp.Packet + pliOnce := true + firOnce := true + var ( + maxRatePacketLoss uint8 + expectedMinBitrate uint64 + ) + for _, pkt := range pkts { + switch p := pkt.(type) { + case *rtcp.PictureLossIndication: + if pliOnce { + fwdPkts = append(fwdPkts, p) + logger.Infof("Picture Loss Indication") + if c.requestKeyFrameHandler != nil { + c.requestKeyFrameHandler() + } + pliOnce = false + } + case *rtcp.FullIntraRequest: + if firOnce { + fwdPkts = append(fwdPkts, p) + logger.Infof("FullIntraRequest") + if c.requestKeyFrameHandler != nil { + c.requestKeyFrameHandler() + } + firOnce = false + } + case *rtcp.ReceiverEstimatedMaximumBitrate: + if expectedMinBitrate == 0 || expectedMinBitrate > uint64(p.Bitrate) { + expectedMinBitrate = uint64(p.Bitrate) + logger.Debugf(" ReceiverEstimatedMaximumBitrate %d", expectedMinBitrate/1024) + } + case *rtcp.ReceiverReport: + for _, r := range p.Reports { + if maxRatePacketLoss == 0 || maxRatePacketLoss < r.FractionLost { + maxRatePacketLoss = r.FractionLost + logger.Infof("maxRatePacketLoss %d", maxRatePacketLoss) + } + } + case *rtcp.TransportLayerNack: + logger.Infof("Nack %v", p) + } + } +} diff --git a/examples/b2bua/b2bua/transport_interface.go b/examples/b2bua/b2bua/transport_interface.go new file mode 100644 index 0000000..5d5199c --- /dev/null +++ b/examples/b2bua/b2bua/transport_interface.go @@ -0,0 +1,37 @@ +package b2bua + +import ( + "github.com/pion/rtcp" + "github.com/pion/rtp" +) + +type MediaTransportType string + +const ( + TransportTypeStandard MediaTransportType = "Standard-AVP/RTP/UDP" + TransportTypeWebRTC MediaTransportType = "WebRTC-SAVPF/DTLS/SRTP" +) + +func (t MediaTransportType) String() string { + return string(t) +} + +type MediaTransport interface { + Init(umc UserAgentMediaConfig) error + Close() error + CreateOffer() (*Desc, error) + OnAnswer(desc *Desc) error + OnOffer(sdp *Desc) error + CreateAnswer() (*Desc, error) + Type() MediaTransportType + + OnRtpPacket(rtpHandler func(trackType TrackType, pkt rtp.Packet) (int, error)) + OnRtcpPacket(rtcpHandler func(trackType TrackType, pkt rtcp.Packet) (int, error)) + + OnRequestKeyFrame(func() error) + + WriteRTP(trackType TrackType, pkt rtp.Packet) (int, error) + WriteRTCP(trackType TrackType, pkt rtcp.Packet) (int, error) + + RequestKeyFrame() error +} diff --git a/examples/b2bua/b2bua/udp_port.go b/examples/b2bua/b2bua/udp_port.go new file mode 100644 index 0000000..3351e40 --- /dev/null +++ b/examples/b2bua/b2bua/udp_port.go @@ -0,0 +1,189 @@ +package b2bua + +import ( + "context" + "fmt" + "net" + "sync" + + "github.com/cloudwebrtc/go-sip-ua/pkg/utils" + "github.com/ghettovoice/gosip/util" +) + +type UdpPort struct { + udpConns []*net.UDPConn + closed utils.AtomicBool + ctx context.Context + cancel context.CancelFunc + handleRtpPacket func(trackType TrackType, packet []byte, raddr net.Addr) error + handleRtcpPacket func(trackType TrackType, packet []byte, raddr net.Addr) error + mutex sync.Mutex + trackType TrackType + externalRtpAddress string + rAddr *net.UDPAddr + rRtcpAddr *net.UDPAddr +} + +func NewUdpPort(trackType TrackType, rAddr, rRtcpAddr *net.UDPAddr, externalRtpAddress string) (*UdpPort, error) { + c := &UdpPort{ + trackType: trackType, + externalRtpAddress: externalRtpAddress, + rAddr: rAddr, + rRtcpAddr: rRtcpAddr, + } + c.ctx, c.cancel = context.WithCancel(context.TODO()) + c.closed.Set(false) + return c, nil +} + +func (c *UdpPort) Init() error { + lAddr := &net.UDPAddr{IP: net.ParseIP("0.0.0.0"), Port: 0} + lRtcpAddr := &net.UDPAddr{IP: net.ParseIP("0.0.0.0"), Port: 0} + + // TODO: set port range from config + udpPortRangeMin := 4000 + udpPortRangeMax := 5000 + + if b2buaConfig.UdpPortRange[0] != 0 { + udpPortRangeMin = b2buaConfig.UdpPortRange[0] + } + + if b2buaConfig.UdpPortRange[1] != 0 { + udpPortRangeMax = b2buaConfig.UdpPortRange[1] + } + + rtpConns, err := ListenRTPInPortRange(udpPortRangeMin, udpPortRangeMax, "udp", lAddr, lRtcpAddr) + if err != nil { + logger.Errorf("ListenUDP: err => %v", err) + return err + } + + host := b2buaConfig.UaMediaConfig.ExternalRtpAddress + if host == "" || host == "0.0.0.0" { + if v, err := util.ResolveSelfIP(); err == nil { + host = v.String() + } + } + + logger.Infof("[%s] ListenUDP: RTP => udp://%s:%v, RTCP => udp://%s:%v", c.trackType, host, rtpConns[0].LocalAddr().(*net.UDPAddr).Port, host, rtpConns[1].LocalAddr().(*net.UDPAddr).Port) + + go c.loop(rtpConns[0], func(packet []byte, raddr net.Addr) { + c.mutex.Lock() + defer c.mutex.Unlock() + c.rAddr = raddr.(*net.UDPAddr) + if c.handleRtpPacket != nil { + c.handleRtpPacket(c.trackType, packet, raddr) + } + }) + + go c.loop(rtpConns[1], func(packet []byte, raddr net.Addr) { + c.mutex.Lock() + defer c.mutex.Unlock() + c.rRtcpAddr = raddr.(*net.UDPAddr) + if c.handleRtcpPacket != nil { + c.handleRtcpPacket(c.trackType, packet, raddr) + } + }) + + c.udpConns = rtpConns + return nil +} + +func (c *UdpPort) GetTrackType() TrackType { + return c.trackType +} + +func (c *UdpPort) LocalPort() int { + return c.udpConns[0].LocalAddr().(*net.UDPAddr).Port +} + +func (c *UdpPort) SetRemoteAddress(raddr *net.UDPAddr) { + c.rAddr = raddr +} + +func (c *UdpPort) SetRemoteRtcpAddress(raddr *net.UDPAddr) { + c.rRtcpAddr = raddr +} + +func (c *UdpPort) Close() { + c.mutex.Lock() + defer c.mutex.Unlock() + + if !c.closed.Get() { + c.closed.Set(true) + } + c.cancel() + + for _, conn := range c.udpConns { + conn.Close() + } + c.udpConns = nil + +} + +func (c *UdpPort) OnRtpPacket(callback func(trackType TrackType, packet []byte, raddr net.Addr) error) { + c.mutex.Lock() + defer c.mutex.Unlock() + c.handleRtpPacket = callback +} + +func (c *UdpPort) OnRtcpPacket(callback func(trackType TrackType, packet []byte, raddr net.Addr) error) { + c.mutex.Lock() + defer c.mutex.Unlock() + c.handleRtcpPacket = callback +} + +func (c *UdpPort) WriteRtp(data []byte) (int, error) { + if c.closed.Get() { + return 0, fmt.Errorf("closed") + } + + if c.rAddr == nil { + return 0, fmt.Errorf("rAddr is nil") + } + + if c.udpConns == nil { + return 0, fmt.Errorf("udpConns is nil") + } + + logger.Tracef("UdpPort::WriteRTP: raddr %v", c.rAddr) + return c.udpConns[0].WriteToUDP(data, c.rAddr) +} + +func (c *UdpPort) WriteRtcp(data []byte) (int, error) { + if c.closed.Get() { + return 0, fmt.Errorf("closed") + } + var addr *net.UDPAddr = c.rRtcpAddr + + if addr == nil { + addr = c.rRtcpAddr + if addr == nil { + return 0, fmt.Errorf("rRtcpAddr is nil") + } + } + + if c.udpConns == nil { + return 0, fmt.Errorf("udpConns is nil") + } + + logger.Tracef("UdpPort::WriteRTCP: %d packets, raddr %v", len(data), addr) + return c.udpConns[1].WriteToUDP(data, addr) +} + +func (c *UdpPort) loop(conn *net.UDPConn, onPacketReceived func(data []byte, raddr net.Addr)) { + buf := make([]byte, 1500) + for { + if c.closed.Get() { + logger.Infof("Terminate: stop rtp conn now!") + return + } + n, raddr, err := conn.ReadFromUDP(buf) + if err != nil { + logger.Debugf("RTP Conn [%v] refused, stop now!", raddr) + return + } + //logger.Infof("raddr: %v, size %d", raddr, n) + onPacketReceived(buf[:n], raddr) + } +} diff --git a/examples/b2bua/b2bua/util.go b/examples/b2bua/b2bua/util.go new file mode 100644 index 0000000..ca3259c --- /dev/null +++ b/examples/b2bua/b2bua/util.go @@ -0,0 +1,444 @@ +package b2bua + +import ( + "bytes" + "encoding/binary" + "encoding/json" + "errors" + "fmt" + "io/ioutil" + "math/rand" + "net" + "os" + "runtime" + "runtime/debug" + "strconv" + "strings" + "time" + "unicode/utf8" + + "github.com/pixelbender/go-sdp/sdp" +) + +type Map map[string]interface{} + +var ( + localIPPrefix = [...]string{"192.168", "10.0", "169.254", "172.16"} + ErrPort = errors.New("invalid port") +) + +func IsLocalIP(ip string) bool { + for i := 0; i < len(localIPPrefix); i++ { + if strings.HasPrefix(ip, localIPPrefix[i]) { + return true + } + } + return false +} + +func GetIntefaceIP() string { + addrs, _ := net.InterfaceAddrs() + + // get internet ip first + for _, a := range addrs { + if ipnet, ok := a.(*net.IPNet); ok && !ipnet.IP.IsLoopback() && ipnet.IP.To4() != nil { + if !IsLocalIP(ipnet.IP.String()) { + return ipnet.IP.String() + } + } + } + + // get internat ip + for _, a := range addrs { + if ipnet, ok := a.(*net.IPNet); ok && !ipnet.IP.IsLoopback() && ipnet.IP.To4() != nil { + return ipnet.IP.String() + } + } + + return "" +} + +func ResolveSelfIP() (net.IP, error) { + ifaces, err := net.Interfaces() + if err != nil { + return nil, err + } + for _, iface := range ifaces { + if iface.Flags&net.FlagUp == 0 { + continue // interface down + } + if iface.Flags&net.FlagLoopback != 0 { + continue // loopback interface + } + addrs, err := iface.Addrs() + if err != nil { + return nil, err + } + for _, addr := range addrs { + var ip net.IP + switch v := addr.(type) { + case *net.IPNet: + ip = v.IP + case *net.IPAddr: + ip = v.IP + } + if ip == nil || ip.IsLoopback() { + continue + } + ip = ip.To4() + if ip == nil { + continue // not an ipv4 address + } + return ip, nil + } + } + return nil, errors.New("server not connected to any network") +} + +func Recover(flag string) { + _, _, l, _ := runtime.Caller(1) + if err := recover(); err != nil { + logger.Errorf("[%s] Recover panic line => %v", flag, l) + logger.Errorf("[%s] Recover err => %v", flag, err) + debug.PrintStack() + } +} + +func Marshal(m map[string]interface{}) string { + if byt, err := json.Marshal(m); err != nil { + logger.Errorf("Marshal: err ===> %v", err) + return "" + } else { + return string(byt) + } +} + +// get value from map +func Val(msg map[string]interface{}, key string) string { + if msg == nil { + return "" + } + val := msg[key] + if val == nil { + return "" + } + switch val.(type) { + case string: + return val.(string) + case map[string]interface{}: + return Marshal(val.(map[string]interface{})) + default: + return fmt.Sprint(val) + } +} + +func StrToInt(str string) int { + i, _ := strconv.ParseInt(str, 10, 32) + //logger.Infof("StrToUint32 str=%v i=%v err=%v", str, i, err) + return int(i) +} + +func StrToInt64(str string) int64 { + i, _ := strconv.ParseInt(str, 10, 64) + //logger.Infof("StrToUint32 str=%v i=%v err=%v", str, i, err) + return int64(i) +} + +// fileExists checks if a file exists and is not a directory before we +// try using it to prevent further errors. +func FileExists(filename string) bool { + info, err := os.Stat(filename) + if os.IsNotExist(err) { + return false + } + return !info.IsDir() +} + +func DialUDPInPortRange(portMin, portMax int, network string, laddr *net.UDPAddr, raddr *net.UDPAddr) (*net.UDPConn, error) { + if (laddr.Port != 0) || ((portMin == 0) && (portMax == 0)) { + return net.DialUDP(network, laddr, raddr) + } + var i, j int + i = portMin + if i == 0 { + i = 1 + } + j = portMax + if j == 0 { + j = 0xFFFF + } + if i > j { + return nil, ErrPort + } + + portStart := rand.Intn(j-i+1) + i + portCurrent := portStart + for { + *laddr = net.UDPAddr{IP: laddr.IP, Port: portCurrent} + c, e := net.DialUDP(network, laddr, raddr) + if e == nil { + return c, e + } + logger.Debugf("failed to listen %s: %v", laddr.String(), e) + portCurrent++ + if portCurrent > j { + portCurrent = i + } + if portCurrent == portStart { + break + } + } + return nil, ErrPort +} + +func ListenRTPInPortRange(portMin, portMax int, network string, lRtpAddr *net.UDPAddr, lRtcpAddr *net.UDPAddr) ([]*net.UDPConn, error) { + var i, j int + i = portMin + if i == 0 { + i = 1 + } + j = portMax + if j == 0 { + j = 0xFFFF + } + if i > j { + return nil, ErrPort + } + + portStart := rand.Intn(j-i+1) + i + portCurrent := portStart + + conns := make([]*net.UDPConn, 2) + for { + *lRtpAddr = net.UDPAddr{IP: lRtpAddr.IP, Port: portCurrent} + c, e := net.ListenUDP(network, lRtpAddr) + if e == nil { + c.SetReadBuffer(321024) + c.SetWriteBuffer(321024) + if conns[0] == nil { + conns[0] = c + } else { + conns[1] = c + return conns, e + } + *lRtcpAddr = net.UDPAddr{IP: lRtcpAddr.IP, Port: portCurrent + 1} + c, e = net.ListenUDP(network, lRtcpAddr) + if e == nil { + c.SetReadBuffer(321024) + c.SetWriteBuffer(321024) + conns[1] = c + return conns, e + } + } + logger.Errorf("failed to listen %s: %v", lRtpAddr.String(), e) + portCurrent++ + if portCurrent > j { + portCurrent = i + } + if portCurrent == portStart { + break + } + } + + return conns, ErrPort +} + +func PathExists(path string) (bool, error) { + _, err := os.Stat(path) + if err == nil { + return true, nil + } + if os.IsNotExist(err) { + return false, nil + } + return false, err +} + +func GetAllFiles(pathname string) ([]string, error) { + files := make([]string, 0) + rd, err := ioutil.ReadDir(pathname) + for _, fi := range rd { + if fi.IsDir() { + //fmt.Printf("[%s]\n", pathname+"\\"+fi.Name()) + fls, err := GetAllFiles(pathname + fi.Name() + "\\") + if err != nil { + continue + } + files = append(files, fls...) + } else { + files = append(files, fi.Name()) + } + } + return files, err +} + +func GetUtf8Length(str string) int { + return utf8.RuneCountInString(str) +} + +func JsonEncode(str string) map[string]interface{} { + var data map[string]interface{} + if err := json.Unmarshal([]byte(str), &data); err != nil { + panic(err) + } + return data +} + +func BytesCombine(pBytes ...[]byte) []byte { + len := len(pBytes) + s := make([][]byte, len) + for index := 0; index < len; index++ { + s[index] = pBytes[index] + } + sep := []byte("") + return bytes.Join(s, sep) +} + +func IntToBytes(n int) []byte { + x := int32(n) + bytesBuffer := bytes.NewBuffer([]byte{}) + binary.Write(bytesBuffer, binary.BigEndian, x) + return bytesBuffer.Bytes() +} + +func BytesToInt(b []byte) int { + bytesBuffer := bytes.NewBuffer(b) + var x int32 + binary.Read(bytesBuffer, binary.BigEndian, &x) + return int(x) +} + +func RandInt64(min, max int64) int64 { + if min >= max || min == 0 || max == 0 { + return max + } + return rand.Int63n(max-min) + min +} + +var letterRunes = []rune("abcdefghijklmnopqrstuvwxyzABCDEFGHIJKLMNOPQRSTUVWXYZ1234567890") + +func RandomString(n int) string { + rand.Seed(time.Now().UnixNano()) + b := make([]rune, n) + for i := range b { + b[i] = letterRunes[rand.Intn(len(letterRunes))] + } + return string(b) +} + +// make kv to map, args should be multiple of 2 +func NewMap(args ...interface{}) map[string]interface{} { + if len(args)%2 != 0 { + return nil + } + msg := make(map[string]interface{}) + for i := 0; i < len(args)/2; i++ { + msg[args[2*i].(string)] = args[2*i+1] + } + return msg +} + +func HasWebRTCAttributes(attributes []*sdp.Attr) bool { + hasIce := false + hasDtls := false + for _, a := range attributes { + if a.Name == "ice-ufrag" { + hasIce = true + } + if a.Name == "fingerprint" { + hasDtls = true + } + } + return hasIce && hasDtls +} + +func ParseTransportType(sdp *sdp.Session) MediaTransportType { + for _, m := range sdp.Media { + // Proto: "UDP/TLS/RTP/SAVPF" + if strings.Contains(m.Proto, "SAVPF") && HasWebRTCAttributes(m.Attributes) { + return TransportTypeWebRTC + } + } + return TransportTypeStandard +} + +/* + +rtpmap:111 opus/48000/2 +a=rtcp-fb:111 transport-cc +a=fmtp:111 minptime=10;useinbandfec=1 +a=rtpmap:63 red/48000/2 +a=fmtp:63 111/111 +a=rtpmap:9 G722/8000 +a=rtpmap:102 ILBC/8000 +a=rtpmap:0 PCMU/8000 +a=rtpmap:8 PCMA/8000 +a=rtpmap:13 CN/8000 +a=rtpmap:110 telephone-event/48000 +a=rtpmap:126 telephone-event/8000 +*/ + +var defaultAudioCodecs = map[uint8]*sdp.Format{ + 0: {Payload: 0, Name: "PCMU", ClockRate: 8000, Channels: 1}, + 8: {Payload: 8, Name: "PCMA", ClockRate: 8000, Channels: 1}, + 9: {Payload: 9, Name: "G722", ClockRate: 8000, Channels: 1}, + 13: {Payload: 13, Name: "CN", ClockRate: 8000, Channels: 1}, + 110: {Payload: 110, Name: "telephone-event", ClockRate: 48000, Channels: 1}, + 111: {Payload: 111, Name: "opus", ClockRate: 48000, Channels: 1}, + 126: {Payload: 126, Name: "telephone-event", ClockRate: 8000, Channels: 1}, +} + +var defaultVideoCodecs = map[uint8]*sdp.Format{ + 96: {Payload: 96, Name: "VP8", ClockRate: 90000}, + 97: {Payload: 97, Name: "H264", ClockRate: 90000}, +} + +func fixFormatName(fmts []*sdp.Format) []*sdp.Format { + for _, f := range fmts { + if ff, ok := defaultAudioCodecs[f.Payload]; ok && f.Name == "" { + f.Name = ff.Name + if f.ClockRate == 0 { + f.ClockRate = ff.ClockRate + } + if f.Channels == 0 { + f.Channels = ff.Channels + } + } + } + return fmts +} + +func replaceCodec(src *Desc, answer string) error { + srcSess, _ := sdp.Parse([]byte(src.SDP)) + sdpSess, _ := sdp.Parse([]byte(answer)) + for idx, m := range sdpSess.Media { + srcSess.Media[idx].Format = fixFormatName(m.Format) + } + src.SDP = srcSess.String() + return nil +} + +func ParseTrackInfos(sdp *sdp.Session) ([]*TrackInfo, error) { + if sdp == nil { + return nil, errors.New("sdp is nil") + } + trackInfos := make([]*TrackInfo, 0) + for _, m := range sdp.Media { + trackInfo := &TrackInfo{} + trackInfo.Direction = m.Mode + trackInfo.Port = m.Port + if trackInfo.Port > 0 { + trackInfo.RtcpPort = m.Port + 1 + } + trackInfo.Codecs = fixFormatName(m.Format) + if m.Type == "audio" { + trackInfo.TrackType = TrackTypeAudio + } else if m.Type == "video" { + trackInfo.TrackType = TrackTypeVideo + } else { + continue + } + trackInfos = append(trackInfos, trackInfo) + } + return trackInfos, nil +} diff --git a/examples/b2bua/main.go b/examples/b2bua/main.go index 36c0d12..7321662 100644 --- a/examples/b2bua/main.go +++ b/examples/b2bua/main.go @@ -18,11 +18,12 @@ import ( func completer(d prompt.Document) []prompt.Suggest { s := []prompt.Suggest{ {Text: "users", Description: "Show sip accounts"}, - {Text: "onlines", Description: "Show online sip devices"}, + {Text: "show onlines", Description: "Show online sip devices"}, {Text: "calls", Description: "Show active calls"}, + {Text: "originate", Description: "Originate a call and bridge to another call"}, {Text: "set debug on", Description: "Show debug msg in console"}, {Text: "set debug off", Description: "Turn off debug msg in console"}, - {Text: "show loggers", Description: "Print Loggers"}, + {Text: "loggers level", Description: "Print Loggers"}, {Text: "exit", Description: "Exit"}, } return prompt.FilterHasPrefix(s, d.GetWordBeforeCursor(), true) @@ -38,8 +39,21 @@ Options: } func consoleLoop(b2bua *b2bua.B2BUA) { - - fmt.Println("Please select command.") + usersCompleter := func(d prompt.Document) []prompt.Suggest { + accounts := b2bua.GetAccounts() + s := make([]prompt.Suggest, 0, len(accounts)) + for user := range accounts { + s = append(s, prompt.Suggest{Text: user, Description: "User"}) + } + aors := b2bua.GetRegistry().GetAllContacts() + for aor := range aors { + for _, instance := range aors[aor] { + s = append(s, prompt.Suggest{Text: instance.Contact.Address.String(), Description: "online device"}) + } + } + return prompt.FilterHasPrefix(s, d.GetWordBeforeCursor(), true) + } + fmt.Println("Please select command, type ? or h or help for help") for { t := prompt.Input("CLI> ", completer, prompt.OptionTitle("GO B2BUA 1.0.0"), @@ -74,14 +88,20 @@ func consoleLoop(b2bua *b2bua.B2BUA) { } else { fmt.Printf("No users\n") } + case "originate": + fmt.Printf("Enter the source user: ") + source := prompt.Input("Source> ", usersCompleter) + fmt.Printf("Enter the destination user: ") + destination := prompt.Input("Destination> ", usersCompleter) + b2bua.Originate(source, destination) case "calls": fallthrough case "cl": /* call list*/ - calls := b2bua.Calls() - if len(calls) > 0 { - fmt.Printf("Calls:\n") - for _, call := range calls { - fmt.Printf("%v:\n", call.ToString()) + bridges := b2bua.BridgedCalls() + if len(bridges) > 0 { + fmt.Printf("Bridged Calls:\n") + for _, call := range bridges { + fmt.Printf("%v\n", call.ToString()) } } else { fmt.Printf("No active calls\n") @@ -114,6 +134,20 @@ func consoleLoop(b2bua *b2bua.B2BUA) { } else { fmt.Printf("No pn records\n") } + case "?": + fallthrough + case "h": + fallthrough + case "help": + fmt.Println("Commands:") + fmt.Println(" users: Show sip accounts") + fmt.Println(" onlines: Show online sip devices") + fmt.Println(" calls: Show active calls") + fmt.Println(" originate: Originate a call and bridge to another call") + fmt.Println(" set debug on: Show debug msg in console") + fmt.Println(" set debug off: Turn off debug msg in console") + fmt.Println(" show loggers: Print Loggers") + fmt.Println(" exit: Exit") case "exit": fmt.Println("Exit now.") b2bua.Shutdown() @@ -122,15 +156,28 @@ func consoleLoop(b2bua *b2bua.B2BUA) { } } +var ( + logger log.Logger +) + +func init() { + logger = utils.NewLogrusLogger(utils.DefaultLogLevel, "main", nil) +} + func main() { noconsole := false disableAuth := false enableTLS := false + enableWebsocket := true + enalbeRFC8599 := false + h := false flag.BoolVar(&h, "h", false, "this help") flag.BoolVar(&noconsole, "nc", false, "no console mode") flag.BoolVar(&disableAuth, "da", false, "disable auth mode") - flag.BoolVar(&enableTLS, "tls", false, "enable TLS") + flag.BoolVar(&enableTLS, "tls", false, "enable tls") + flag.BoolVar(&enableWebsocket, "ws", false, "enable websocket") + flag.BoolVar(&enalbeRFC8599, "rfc8599", false, "enable rfc8599 push notification") flag.Usage = usage flag.Parse() @@ -144,11 +191,11 @@ func main() { signal.Notify(stop, syscall.SIGTERM, syscall.SIGINT) go func() { - fmt.Print("Start pprof on :6658\n") + logger.Info("Start pprof on :6658") http.ListenAndServe(":6658", nil) }() - b2bua := b2bua.NewB2BUA(disableAuth, enableTLS) + b2bua := b2bua.NewB2BUA(disableAuth, enableTLS, enableWebsocket, enalbeRFC8599) // Add sample accounts. b2bua.AddAccount("100", "100") @@ -159,6 +206,8 @@ func main() { if !noconsole { consoleLoop(b2bua) return + } else { + logger.Info("No console mode") } <-stop diff --git a/examples/b2bua/registry/registry.go b/examples/b2bua/registry/registry.go index 92ce1ee..65da1ff 100644 --- a/examples/b2bua/registry/registry.go +++ b/examples/b2bua/registry/registry.go @@ -14,6 +14,10 @@ type ContactInstance struct { Transport string } +func (c ContactInstance) SupportIce() bool { + return c.Contact.Params.Has("+sip.ice") +} + func (c *ContactInstance) GetPNParams() *PNParams { params := c.Contact.Address.UriParams() if provider, ok := params.Get("pn-provider"); ok { diff --git a/examples/client/main.go b/examples/client/main.go index 81353fa..d08863c 100644 --- a/examples/client/main.go +++ b/examples/client/main.go @@ -25,7 +25,7 @@ var ( ) func init() { - logger = utils.NewLogrusLogger(log.DebugLevel, "Client", nil) + logger = utils.NewLogrusLogger(utils.DefaultLogLevel, "Client", nil) } func createUdp() *rtp.RtpUDPStream { diff --git a/examples/register/main.go b/examples/register/main.go index 503d77a..5942771 100644 --- a/examples/register/main.go +++ b/examples/register/main.go @@ -21,7 +21,7 @@ var ( ) func init() { - logger = utils.NewLogrusLogger(log.DebugLevel, "Register", nil) + logger = utils.NewLogrusLogger(utils.DefaultLogLevel, "Register", nil) } func main() { diff --git a/go.mod b/go.mod index 25f1a56..1f6f054 100644 --- a/go.mod +++ b/go.mod @@ -5,10 +5,20 @@ go 1.13 require ( firebase.google.com/go v3.13.0+incompatible github.com/c-bata/go-prompt v0.2.6 - github.com/ghettovoice/gosip v0.0.0-20230322091832-d77de1c97f89 + github.com/gammazero/deque v0.1.0 + github.com/ghettovoice/gosip v0.0.0-20240619135023-afc3f006312f + github.com/go-logr/logr v1.2.4 github.com/google/uuid v1.3.0 - github.com/pixelbender/go-sdp v1.1.0 + github.com/pion/interceptor v0.1.12 + github.com/pion/ion-sfu v1.11.0 + github.com/pion/rtcp v1.2.10 + github.com/pion/rtp v1.7.13 + github.com/pion/sdp/v3 v3.0.6 + github.com/pion/transport v0.14.1 + github.com/pion/webrtc/v3 v3.1.59 + github.com/pixelbender/go-sdp v1.1.1-0.20240403132153-9683c2a0405f github.com/sirupsen/logrus v1.9.0 + github.com/stretchr/testify v1.8.2 github.com/tevino/abool v1.2.0 github.com/x-cray/logrus-prefixed-formatter v0.5.2 golang.org/x/crypto v0.7.0 diff --git a/go.sum b/go.sum index 3d5cc72..a96ffbb 100644 --- a/go.sum +++ b/go.sum @@ -1,6 +1,8 @@ +bazil.org/fuse v0.0.0-20180421153158-65cc252bf669/go.mod h1:Xbm+BRKSBEpa4q4hTSxohYNQpsxXPbPry4JJWOB3LB8= cloud.google.com/go v0.26.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= cloud.google.com/go v0.34.0/go.mod h1:aQUYkXzVsufM+DwF1aE+0xfcU+56JwCaLick0ClmMTw= cloud.google.com/go v0.38.0/go.mod h1:990N+gfupTy94rShfmMCWGDn0LpTmnzTp2qbd1dvSRU= +cloud.google.com/go v0.39.0/go.mod h1:rVLT6fkc8chs9sfPtFc1SBH6em7n+ZoXaG+87tDISts= cloud.google.com/go v0.44.1/go.mod h1:iSa0KzasP4Uvy3f1mN/7PiObzGgflwredwwASm/v6AU= cloud.google.com/go v0.44.2/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= cloud.google.com/go v0.44.3/go.mod h1:60680Gw3Yr4ikxnPRS/oxxkBccT6SA1yMk63TGekxKY= @@ -238,6 +240,7 @@ cloud.google.com/go/eventarc v1.10.0/go.mod h1:u3R35tmZ9HvswGRBnF48IlYgYeBcPUCjk cloud.google.com/go/filestore v1.3.0/go.mod h1:+qbvHGvXU1HaKX2nD0WEPo92TP/8AQuCVEBXNY9z0+w= cloud.google.com/go/filestore v1.4.0/go.mod h1:PaG5oDfo9r224f8OYXURtAsY+Fbyq/bLYoINEK8XQAI= cloud.google.com/go/filestore v1.5.0/go.mod h1:FqBXDWBp4YLHqRnVGveOkHDf8svj9r5+mUDLupOWEDs= +cloud.google.com/go/firestore v1.6.0/go.mod h1:afJwI0vaXwAG54kI7A//lP/lSPDkQORQuMkv56TxEPU= cloud.google.com/go/firestore v1.9.0 h1:IBlRyxgGySXu5VuW0RgGFlTtLukSnNkpDiEOMkQkmpA= cloud.google.com/go/firestore v1.9.0/go.mod h1:HMkjKHNTtRyZNiMzu7YAsLr9K3X2udY2AMwDaMEQiiE= cloud.google.com/go/functions v1.6.0/go.mod h1:3H1UA3qiIPRWD7PeZKLvHZ9SaQhR26XIJcC0A5GbvAk= @@ -524,27 +527,87 @@ cloud.google.com/go/workflows v1.7.0/go.mod h1:JhSrZuVZWuiDfKEFxU0/F1PQjmpnpcoIS cloud.google.com/go/workflows v1.8.0/go.mod h1:ysGhmEajwZxGn1OhGOGKsTXc5PyxOc0vfKf5Af+to4M= cloud.google.com/go/workflows v1.9.0/go.mod h1:ZGkj1aFIOd9c8Gerkjjq7OW7I5+l6cSvT3ujaO/WwSA= cloud.google.com/go/workflows v1.10.0/go.mod h1:fZ8LmRmZQWacon9UCX1r/g/DfAXx5VcPALq2CxzdePw= +code.gitea.io/sdk/gitea v0.11.3/go.mod h1:z3uwDV/b9Ls47NGukYM9XhnHtqPh/J+t40lsUrR6JDY= +contrib.go.opencensus.io/exporter/aws v0.0.0-20181029163544-2befc13012d0/go.mod h1:uu1P0UCM/6RbsMrgPa98ll8ZcHM858i/AD06a9aLRCA= +contrib.go.opencensus.io/exporter/ocagent v0.5.0/go.mod h1:ImxhfLRpxoYiSq891pBrLVhN+qmP8BTVvdH2YLs7Gl0= +contrib.go.opencensus.io/exporter/stackdriver v0.12.1/go.mod h1:iwB6wGarfphGGe/e5CWqyUk/cLzKnWsOKPVW3no6OTw= +contrib.go.opencensus.io/integrations/ocsql v0.1.4/go.mod h1:8DsSdjz3F+APR+0z0WkU1aRorQCFfRxvqjUUPMbF3fE= +contrib.go.opencensus.io/resource v0.1.1/go.mod h1:F361eGI91LCmW1I/Saf+rX0+OFcigGlFvXwEGEnkRLA= dmitri.shuralyov.com/gpu/mtl v0.0.0-20190408044501-666a987793e9/go.mod h1:H6x//7gZCb22OMCxBHrMx7a5I7Hp++hsVxbQ4BYO7hU= firebase.google.com/go v3.13.0+incompatible h1:3TdYC3DDi6aHn20qoRkxwGqNgdjtblwVAyRLQwGn/+4= firebase.google.com/go v3.13.0+incompatible/go.mod h1:xlah6XbEyW6tbfSklcfe5FHJIwjt8toICdV5Wh9ptHs= gioui.org v0.0.0-20210308172011-57750fc8a0a6/go.mod h1:RSH6KIUZ0p2xy5zHDxgAM4zumjgTw83q2ge/PI+yyw8= git.sr.ht/~sbinet/gg v0.3.1/go.mod h1:KGYtlADtqsqANL9ueOFkWymvzUvLMQllU5Ixo+8v3pc= +github.com/Azure/azure-amqp-common-go/v2 v2.1.0/go.mod h1:R8rea+gJRuJR6QxTir/XuEd+YuKoUiazDC/N96FiDEU= +github.com/Azure/azure-pipeline-go v0.2.1/go.mod h1:UGSo8XybXnIGZ3epmeBw7Jdz+HiUVpqIlpz/HKHylF4= +github.com/Azure/azure-sdk-for-go v29.0.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc= +github.com/Azure/azure-sdk-for-go v30.1.0+incompatible/go.mod h1:9XXNKU+eRnpl9moKnB4QOLf1HestfXbmab5FXxiDBjc= +github.com/Azure/azure-service-bus-go v0.9.1/go.mod h1:yzBx6/BUGfjfeqbRZny9AQIbIe3AcV9WZbAdpkoXOa0= +github.com/Azure/azure-storage-blob-go v0.8.0/go.mod h1:lPI3aLPpuLTeUwh1sViKXFxwl2B6teiRqI0deQUvsw0= +github.com/Azure/go-autorest v12.0.0+incompatible/go.mod h1:r+4oMnoxhatjLLJ6zxSWATqVooLgysK6ZNox3g/xq24= github.com/BurntSushi/toml v0.3.1/go.mod h1:xHWCNGjB5oqiDr8zfno3MHue2Ht5sIBksp03qcyfWMU= github.com/BurntSushi/xgb v0.0.0-20160522181843-27f122750802/go.mod h1:IVnqGOEym/WlBOVXweHU+Q+/VP0lqqI8lqeDx9IjBqo= +github.com/GoogleCloudPlatform/cloudsql-proxy v0.0.0-20191009163259-e802c2cb94ae/go.mod h1:mjwGPas4yKduTyubHvD1Atl9r1rUq8DfVy+gkVvZ+oo= github.com/JohnCGriffin/overflow v0.0.0-20211019200055-46fa312c352c/go.mod h1:X0CRv0ky0k6m906ixxpzmDRLvX58TFUKS2eePweuyxk= +github.com/Knetic/govaluate v3.0.1-0.20171022003610-9aa49832a739+incompatible/go.mod h1:r7JcOSlj0wfOMncg0iLm8Leh48TZaKVeNIfJntJ2wa0= +github.com/Masterminds/semver/v3 v3.0.3/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs= +github.com/Masterminds/semver/v3 v3.1.0/go.mod h1:VPu/7SZ7ePZ3QOrcuXROw5FAcLl4a0cBrbBpGY/8hQs= github.com/OneOfOne/xxhash v1.2.2/go.mod h1:HSdplMjZKSmBqAxg5vPj2TmRDmfkzw+cTzAElWljhcU= +github.com/Shopify/sarama v1.19.0/go.mod h1:FVkBWblsNy7DGZRfXLU0O9RCGt5g3g3yEuWXgklEdEo= +github.com/Shopify/toxiproxy v2.1.4+incompatible/go.mod h1:OXgGpZ6Cli1/URJOF1DMxUHB2q5Ap20/P/eIdh4G0pI= +github.com/VividCortex/gohistogram v1.0.0/go.mod h1:Pf5mBqqDxYaXu3hDrrU+w6nw50o/4+TcAqDqk/vUH7g= +github.com/afex/hystrix-go v0.0.0-20180502004556-fa1af6a1f4f5/go.mod h1:SkGFH1ia65gfNATL8TAiHDNxPzPdmEL5uirI2Uyuz6c= github.com/ajstarks/deck v0.0.0-20200831202436-30c9fc6549a9/go.mod h1:JynElWSGnm/4RlzPXRlREEwqTHAN3T56Bv2ITsFT3gY= github.com/ajstarks/deck/generate v0.0.0-20210309230005-c3f852c02e19/go.mod h1:T13YZdzov6OU0A1+RfKZiZN9ca6VeKdBdyDV+BY97Tk= github.com/ajstarks/svgo v0.0.0-20180226025133-644b8db467af/go.mod h1:K08gAheRH3/J6wwsYMMT4xOr94bZjxIelGM0+d/wbFw= github.com/ajstarks/svgo v0.0.0-20211024235047-1546f124cd8b/go.mod h1:1KcenG0jGWcpt8ov532z81sp/kMMUG485J2InIOyADM= +github.com/alecthomas/kingpin v2.2.6+incompatible/go.mod h1:59OFYbFVLKQKq+mqrL6Rw5bR0c3ACQaawgXx0QYndlE= +github.com/alecthomas/template v0.0.0-20160405071501-a0175ee3bccc/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/template v0.0.0-20190718012654-fb15b899a751/go.mod h1:LOuyumcjzFXgccqObfd/Ljyb9UuFJ6TxHnclSeseNhc= +github.com/alecthomas/units v0.0.0-20151022065526-2efee857e7cf/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= +github.com/alecthomas/units v0.0.0-20190717042225-c3de453c63f4/go.mod h1:ybxpYRFXyAe+OPACYpWeL0wqObRcbAqCMya13uyzqw0= +github.com/alecthomas/units v0.0.0-20190924025748-f65c72e2690d/go.mod h1:rBZYJk541a8SKzHPHnH3zbiI+7dagKZ0cgpgrD7Fyho= github.com/andybalholm/brotli v1.0.4/go.mod h1:fO7iG3H7G2nSZ7m0zPUDn85XEX2GTukHGRSepvi9Eig= github.com/antihax/optional v1.0.0/go.mod h1:uupD/76wgC+ih3iEmQUL+0Ugr19nfwCT1kdvxnR2qWY= github.com/apache/arrow/go/v10 v10.0.1/go.mod h1:YvhnlEePVnBS4+0z3fhPfUy7W1Ikj0Ih0vcRo/gZ1M0= +github.com/apache/thrift v0.12.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= +github.com/apache/thrift v0.13.0/go.mod h1:cp2SuWMxlEZw2r+iP2GNCdIi4C1qmUzdZFSVb+bacwQ= github.com/apache/thrift v0.16.0/go.mod h1:PHK3hniurgQaNMZYaCLEqXKsYK8upmhPbmdP2FXSqgU= +github.com/apex/log v1.1.4/go.mod h1:AlpoD9aScyQfJDVHmLMEcx4oU6LqzkWp4Mg9GdAcEvQ= +github.com/apex/logs v0.0.4/go.mod h1:XzxuLZ5myVHDy9SAmYpamKKRNApGj54PfYLcFrXqDwo= +github.com/aphistic/golf v0.0.0-20180712155816-02c07f170c5a/go.mod h1:3NqKYiepwy8kCu4PNA+aP7WUV72eXWJeP9/r3/K9aLE= +github.com/aphistic/sweet v0.2.0/go.mod h1:fWDlIh/isSE9n6EPsRmC0det+whmX6dJid3stzu0Xys= +github.com/armon/circbuf v0.0.0-20150827004946-bbbad097214e/go.mod h1:3U/XgcO3hCbHZ8TKRvWD2dDTCfh9M9ya+I9JpbB7O8o= +github.com/armon/consul-api v0.0.0-20180202201655-eb2c6b5be1b6/go.mod h1:grANhF5doyWs3UAsr3K4I6qtAmlQcZDesFNEHPZAzj8= +github.com/armon/go-metrics v0.0.0-20180917152333-f0300d1749da/go.mod h1:Q73ZrmVTwzkszR9V5SSuryQ31EELlFMUz1kKyl939pY= +github.com/armon/go-radix v0.0.0-20180808171621-7fddfc383310/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= +github.com/armon/go-radix v1.0.0/go.mod h1:ufUuZ+zHj4x4TnLV4JWEpy2hxWSpsRywHrMgIH9cCH8= +github.com/aryann/difflib v0.0.0-20170710044230-e206f873d14a/go.mod h1:DAHtR1m6lCRdSC2Tm3DSWRPvIPr6xNKyeHdqDQSQT+A= +github.com/aws/aws-lambda-go v1.13.3/go.mod h1:4UKl9IzQMoD+QF79YdCuzCwp8VbmG4VAQwij/eHl5CU= +github.com/aws/aws-sdk-go v1.15.27/go.mod h1:mFuSZ37Z9YOHbQEwBWztmVzqXrEkub65tZoCYDt7FT0= +github.com/aws/aws-sdk-go v1.19.18/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= +github.com/aws/aws-sdk-go v1.19.45/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= +github.com/aws/aws-sdk-go v1.20.6/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= +github.com/aws/aws-sdk-go v1.25.11/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= +github.com/aws/aws-sdk-go v1.27.0/go.mod h1:KmX6BPdI08NWTb3/sm4ZGu5ShLoqVDhKgpiN924inxo= +github.com/aws/aws-sdk-go-v2 v0.18.0/go.mod h1:JWVYvqSMppoMJC0x5wdwiImzgXTI9FuZwxzkQq9wy+g= +github.com/aybabtme/rgbterm v0.0.0-20170906152045-cc83f3b3ce59/go.mod h1:q/89r3U2H7sSsE2t6Kca0lfwTK8JdoNGS/yzM/4iH5I= +github.com/beorn7/perks v0.0.0-20180321164747-3a771d992973/go.mod h1:Dwedo/Wpr24TaqPxmxbtue+5NUziq4I4S80YR8gNf3Q= +github.com/beorn7/perks v1.0.0/go.mod h1:KWe93zE9D1o94FZ5RNwFwVgaQK1VOXiVxmqh+CedLV8= +github.com/beorn7/perks v1.0.1/go.mod h1:G2ZrVWU2WbWT9wwq4/hrbKbnv/1ERSJQ0ibhJ6rlkpw= +github.com/bep/debounce v1.2.0/go.mod h1:H8yggRPQKLUhUoqrJC1bO2xNya7vanpDl7xR3ISbCJ0= +github.com/bgentry/speakeasy v0.1.0/go.mod h1:+zsyZBPWlz7T6j88CTgSN5bM796AkVf0kBD4zp0CCIs= +github.com/blakesmith/ar v0.0.0-20190502131153-809d4375e1fb/go.mod h1:PkYb9DJNAwrSvRx5DYA+gUcOIgTGVMNkfSCbZM8cWpI= github.com/boombuler/barcode v1.0.0/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= github.com/boombuler/barcode v1.0.1/go.mod h1:paBWMcWSl3LHKBqUq+rly7CNSldXjb2rDl3JlRe0mD8= github.com/c-bata/go-prompt v0.2.6 h1:POP+nrHE+DfLYx370bedwNhsqmpCUynWPxuHi0C5vZI= github.com/c-bata/go-prompt v0.2.6/go.mod h1:/LMAke8wD2FsNu9EXNdHxNLbd9MedkPnCdfpU9wwHfY= +github.com/caarlos0/ctrlc v1.0.0/go.mod h1:CdXpj4rmq0q/1Eb44M9zi2nKB0QraNKuRGYGrrHhcQw= +github.com/campoy/unique v0.0.0-20180121183637-88950e537e7e/go.mod h1:9IOqJGCPMSc6E5ydlp5NIonxObaeu/Iub/X03EKPVYo= +github.com/casbin/casbin/v2 v2.1.2/go.mod h1:YcPU1XXisHhLzuxH9coDNf2FbKpjGlbCg3n9yuLkIJQ= +github.com/cavaliercoder/go-cpio v0.0.0-20180626203310-925f9528c45e/go.mod h1:oDpT4efm8tSYHXV5tHSdRvBet/b/QzxZ+XyyPehvm3A= +github.com/cenkalti/backoff v2.2.1+incompatible/go.mod h1:90ReRw6GdpyfrHakVjL/QHaoyV4aDUVVkXQJJJ3NXXM= +github.com/census-instrumentation/opencensus-proto v0.2.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/census-instrumentation/opencensus-proto v0.2.1/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/census-instrumentation/opencensus-proto v0.3.0/go.mod h1:f6KPmirojxKA12rnyqOA5BBL4O983OfeGPqjHWSTneU= github.com/census-instrumentation/opencensus-proto v0.4.1/go.mod h1:4T9NM4+4Vw91VeyqjLS6ao50K5bOcLKN6Q42XnYaRYw= @@ -554,7 +617,10 @@ github.com/cespare/xxhash/v2 v2.2.0/go.mod h1:VGX0DQ3Q6kWi7AoAeZDth3/j3BFtOZR5XL github.com/chzyer/logex v1.1.10/go.mod h1:+Ywpsq7O8HXn0nuIou7OrIPyXbp3wmkHB+jjWRnGsAI= github.com/chzyer/readline v0.0.0-20180603132655-2972be24d48e/go.mod h1:nSuG5e5PlCu98SY8svDHJxuZscDgtXS6KTTbou5AhLI= github.com/chzyer/test v0.0.0-20180213035817-a1ea475d72b1/go.mod h1:Q3SI9o4m/ZMnBNeIyt5eFwwo7qiLfzFZmjNmxjkiQlU= +github.com/clbanning/x2j v0.0.0-20191024224557-825249438eec/go.mod h1:jMjuTZXRI4dUb/I5gc9Hdhagfvm9+RyrPryS/auMzxE= github.com/client9/misspell v0.3.4/go.mod h1:qj6jICC3Q7zFZvVWo7KLAzC3yx5G7kyvSDkc90ppPyw= +github.com/cloudwebrtc/nats-discovery v0.3.0/go.mod h1:xYZdYTliuLw9F7S9UrH0ygDd3lRm8/zxt+g662cCYXI= +github.com/cloudwebrtc/nats-grpc v1.0.0/go.mod h1:zLT0CZd/ilNE0C92/F+7xwbPPl/Ff3eq+LuedTp9wug= github.com/cncf/udpa/go v0.0.0-20191209042840-269d4d468f6f/go.mod h1:M8M6+tZqaGXZJjfX53e64911xZQV5JYwmTeXPW+k8Sc= github.com/cncf/udpa/go v0.0.0-20200629203442-efcf912fb354/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= github.com/cncf/udpa/go v0.0.0-20201120205902-5459f2c99403/go.mod h1:WmhPx2Nbnhtbo57+VJT5O0JRkEi1Wbu0z5j0R8u5Hbk= @@ -567,14 +633,39 @@ github.com/cncf/xds/go v0.0.0-20211001041855-01bcc9b48dfe/go.mod h1:eXthEFrGJvWH github.com/cncf/xds/go v0.0.0-20211011173535-cb28da3451f1/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= github.com/cncf/xds/go v0.0.0-20220314180256-7f1daf1720fc/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= github.com/cncf/xds/go v0.0.0-20230105202645-06c439db220b/go.mod h1:eXthEFrGJvWHgFFCl3hGmgk+/aYT6PnTQLykKQRLhEs= +github.com/cockroachdb/datadriven v0.0.0-20190809214429-80d97fb3cbaa/go.mod h1:zn76sxSg3SzpJ0PPJaLDCu+Bu0Lg3sKTORVIj19EIF8= +github.com/codahale/hdrhistogram v0.0.0-20161010025455-3a0bb77429bd/go.mod h1:sE/e/2PUdi/liOCUjSTXgM1o87ZssimdTWN964YiIeI= +github.com/coreos/bbolt v1.3.2/go.mod h1:iRUV2dpdMOn7Bo10OQBFzIJO9kkE559Wcmn+qkEiiKk= +github.com/coreos/etcd v3.3.10+incompatible/go.mod h1:uF7uidLiAD3TWHmW31ZFd/JWoc32PjwdhPthX9715RE= +github.com/coreos/go-semver v0.2.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-semver v0.3.0/go.mod h1:nnelYz7RCh+5ahJtPPxZlU+153eP4D4r3EedlOD2RNk= +github.com/coreos/go-systemd v0.0.0-20180511133405-39ca1b05acc7/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= +github.com/coreos/go-systemd v0.0.0-20190321100706-95778dfbb74e/go.mod h1:F5haX7vjVVG0kc13fIWeqUViNPyEJxv/OmvnBo0Yme4= +github.com/coreos/go-systemd/v22 v22.3.2/go.mod h1:Y58oyj3AT4RCenI/lSvhwexgC+NSVTIJ3seZv2GcEnc= +github.com/coreos/pkg v0.0.0-20160727233714-3ac0863d7acf/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= +github.com/coreos/pkg v0.0.0-20180928190104-399ea9e2e55f/go.mod h1:E3G3o1h8I7cfcXa63jLwjI0eiQQMgzzUDFVpN/nH/eA= +github.com/cpuguy83/go-md2man/v2 v2.0.0-20190314233015-f79a8a8ca69d/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= +github.com/cpuguy83/go-md2man/v2 v2.0.0/go.mod h1:maD7wRr/U5Z6m/iR4s+kqSMx2CaBsrgA7czyZG/E6dU= +github.com/creack/pty v1.1.7/go.mod h1:lj5s0c3V2DBrqTV7llrYr5NG6My20zk30Fl46Y7DoTY= github.com/creack/pty v1.1.9/go.mod h1:oKZEueFk5CKHvIhNR5MUki03XCEU+Q6VDXinZuGJ33E= github.com/davecgh/go-spew v1.1.0/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= github.com/davecgh/go-spew v1.1.1 h1:vj9j/u1bqnvCEfJOwUhtlOARqs3+rkHYY13jYWTU97c= github.com/davecgh/go-spew v1.1.1/go.mod h1:J7Y8YcW2NihsgmVo/mv3lAwl/skON4iLHjSsI+c5H38= +github.com/desertbit/timer v0.0.0-20180107155436-c41aec40b27f/go.mod h1:xH/i4TFMt8koVQZ6WFms69WAsDWr2XsYL3Hkl7jkoLE= +github.com/devigned/tab v0.1.1/go.mod h1:XG9mPq0dFghrYvoBF3xdRrJzSTX1b7IQrvaL9mzjeJY= +github.com/dgrijalva/jwt-go v3.2.0+incompatible/go.mod h1:E3ru+11k8xSBh+hMPgOLZmtrrCbhqsmaPHjLKYnJCaQ= +github.com/dgryski/go-sip13 v0.0.0-20181026042036-e10d5fee7954/go.mod h1:vAd38F8PWV+bWy6jNmig1y/TA+kYO4g3RSRF0IAv0no= +github.com/dimchansky/utfbom v1.1.0/go.mod h1:rO41eb7gLfo8SF1jd9F8HplJm1Fewwi4mQvIirEdv+8= github.com/discoviking/fsm v0.0.0-20150126104936-f4a273feecca h1:cTTdXpkQ1aVbOOmHwdwtYuwUZcQtcMrleD1UXLWhAq8= github.com/discoviking/fsm v0.0.0-20150126104936-f4a273feecca/go.mod h1:W+3LQaEkN8qAwwcw0KC546sUEnX86GIT8CcMLZC4mG0= github.com/docopt/docopt-go v0.0.0-20180111231733-ee0de3bc6815/go.mod h1:WwZ+bS3ebgob9U8Nd0kOddGdZWjyMGR8Wziv+TBNwSE= +github.com/dustin/go-humanize v0.0.0-20171111073723-bb3d318650d4/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= github.com/dustin/go-humanize v1.0.0/go.mod h1:HtrtbFcZ19U5GC7JDqmcUSB87Iq5E25KnS6fMYU6eOk= +github.com/eapache/go-resiliency v1.1.0/go.mod h1:kFI+JgMyC7bLPUVY133qvEBtVayf5mFgVsvEsIPBvNs= +github.com/eapache/go-xerial-snappy v0.0.0-20180814174437-776d5712da21/go.mod h1:+020luEh2TKB4/GOp8oxxtq0Daoen/Cii55CzbTV6DU= +github.com/eapache/queue v1.1.0/go.mod h1:6eCeP0CKFpHLu8blIFXhExK/dRa7WDZfr6jVFPTqq+I= +github.com/edsrzf/mmap-go v1.0.0/go.mod h1:YO35OhQPt3KJa3ryjFM5Bs14WD66h8eGKpfaBNrHW5M= +github.com/envoyproxy/go-control-plane v0.6.9/go.mod h1:SBwIajubJHhxtWwsL9s8ss4safvEdbitLhGGK48rN6g= github.com/envoyproxy/go-control-plane v0.9.0/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.1-0.20191026205805-5f8ba28d4473/go.mod h1:YTl/9mNaCwkRvm6d1a2C3ymFceY/DCBVvsKhRF0iEA4= github.com/envoyproxy/go-control-plane v0.9.4/go.mod h1:6rpuAdCZL397s3pYoYcLgu1mIlRU8Am5FuJP05cCM98= @@ -588,14 +679,27 @@ github.com/envoyproxy/go-control-plane v0.10.3/go.mod h1:fJJn/j26vwOu972OllsvAgJ github.com/envoyproxy/protoc-gen-validate v0.1.0/go.mod h1:iSmxcyjqTsJpI2R4NaDN7+kN2VEUnK/pcBlmesArF7c= github.com/envoyproxy/protoc-gen-validate v0.6.7/go.mod h1:dyJXwwfPK2VSqiB9Klm1J6romD608Ba7Hij42vrOBCo= github.com/envoyproxy/protoc-gen-validate v0.9.1/go.mod h1:OKNgG7TCp5pF4d6XftA0++PMirau2/yoOwVac3AbF2w= +github.com/fatih/color v1.7.0/go.mod h1:Zm6kSWBoL9eyXnKyktHP6abPY2pDugNf5KwzbycvMj4= +github.com/fatih/color v1.9.0/go.mod h1:eQcE1qtQxscV5RaZvpXrrb8Drkc3/DdQ+uUYCNjL+zU= github.com/fogleman/gg v1.2.1-0.20190220221249-0403632d5b90/go.mod h1:R/bRT+9gY/C5z7JzPU0zXsXHKM4/ayA+zqcVNZzPa1k= github.com/fogleman/gg v1.3.0/go.mod h1:R/bRT+9gY/C5z7JzPU0zXsXHKM4/ayA+zqcVNZzPa1k= +github.com/fortytw2/leaktest v1.2.0/go.mod h1:jDsjWgpAGjm2CA7WthBh/CdZYEPF31XHquHwclZch5g= +github.com/fortytw2/leaktest v1.3.0/go.mod h1:jDsjWgpAGjm2CA7WthBh/CdZYEPF31XHquHwclZch5g= +github.com/franela/goblin v0.0.0-20200105215937-c9ffbefa60db/go.mod h1:7dvUGVsVBjqR7JHJk0brhHOZYGmfBYOrK0ZhYMEtBr4= +github.com/franela/goreq v0.0.0-20171204163338-bcd34c9993f8/go.mod h1:ZhphrRTfi2rbfLwlschooIH4+wKKDR4Pdxhh+TRoA20= github.com/fsnotify/fsnotify v1.4.7/go.mod h1:jwhsz4b93w/PPRr/qN1Yymfu8t87LnFCMoQvtojpjFo= -github.com/fsnotify/fsnotify v1.4.9 h1:hsms1Qyu0jgnwNXIxa+/V/PDsU6CfLf6CNO8H7IWoS4= github.com/fsnotify/fsnotify v1.4.9/go.mod h1:znqG4EE+3YCdAaPaxE2ZRY/06pZUdp0tY4IgpuI1SZQ= -github.com/ghettovoice/gosip v0.0.0-20230322091832-d77de1c97f89 h1:bUtgAwa7cfrp0bEnlfr+k6fgsfn7/OFQ0pvQvoujOAo= -github.com/ghettovoice/gosip v0.0.0-20230322091832-d77de1c97f89/go.mod h1:rlD1yLOErWYohWTryG/2bTTpmzB79p52ntLA/uIFXeI= +github.com/fsnotify/fsnotify v1.5.1 h1:mZcQUHVQUQWoPXXtuf9yuEXKudkV2sx1E06UadKWpgI= +github.com/fsnotify/fsnotify v1.5.1/go.mod h1:T3375wBYaZdLLcVNkcVbzGHY7f1l/uK5T5Ai1i3InKU= +github.com/fullstorydev/grpcurl v1.8.0/go.mod h1:Mn2jWbdMrQGJQ8UD62uNyMumT2acsZUCkZIqFxsQf1o= +github.com/gammazero/deque v0.1.0 h1:f9LnNmq66VDeuAlSAapemq/U7hJ2jpIWa4c09q8Dlik= +github.com/gammazero/deque v0.1.0/go.mod h1:KQw7vFau1hHuM8xmI9RbgKFbAsQFWmBpqQ2KenFLk6M= +github.com/gammazero/workerpool v1.1.2/go.mod h1:UelbXcO0zCIGFcufcirHhq2/xtLXJdQ29qZNlXG9OjQ= +github.com/ghettovoice/gosip v0.0.0-20240619135023-afc3f006312f h1:U7o+nojFRcazerHWksrQ584QCA5DEqv0Pls4/cMFzuA= +github.com/ghettovoice/gosip v0.0.0-20240619135023-afc3f006312f/go.mod h1:rlD1yLOErWYohWTryG/2bTTpmzB79p52ntLA/uIFXeI= github.com/ghodss/yaml v1.0.0/go.mod h1:4dBDuWmgqj2HViK6kFavaiC9ZROes6MMH2rRYeMEF04= +github.com/gin-contrib/sse v0.1.0/go.mod h1:RHrZQHXnP2xjPF+u1gW/2HnVO7nvIa9PG3Gm+fLHvGI= +github.com/gin-gonic/gin v1.6.3/go.mod h1:75u5sXoLsGZoRN5Sgbi1eraJ4GU3++wFwWzhwvtwp4M= github.com/go-fonts/dejavu v0.1.0/go.mod h1:4Wt4I4OU2Nq9asgDCteaAaWZOV24E+0/Pwo0gppep4g= github.com/go-fonts/latin-modern v0.2.0/go.mod h1:rQVLdDMK+mK1xscDwsqM5J8U2jrRa3T0ecnM9pNujks= github.com/go-fonts/liberation v0.1.1/go.mod h1:K6qoJYypsmfVjWg8KOVDQhLc8UDgIK2HYqyqAO9z7GY= @@ -604,20 +708,54 @@ github.com/go-fonts/stix v0.1.0/go.mod h1:w/c1f0ldAUlJmLBvlbkvVXLAD+tAMqobIIQpmn github.com/go-gl/glfw v0.0.0-20190409004039-e6da0acd62b1/go.mod h1:vR7hzQXu2zJy9AVAgeJqvqgH9Q5CA+iKCZ2gyEVpxRU= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20191125211704-12ad95a8df72/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= github.com/go-gl/glfw/v3.3/glfw v0.0.0-20200222043503-6f7a984d4dc4/go.mod h1:tQ2UAYgL5IevRw8kRxooKSPJfGvJ9fJQFa0TUsXzTg8= +github.com/go-ini/ini v1.25.4/go.mod h1:ByCAeIL28uOIIG0E3PJtZPDL8WnHpFKFOtgjp+3Ies8= +github.com/go-kit/kit v0.8.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= +github.com/go-kit/kit v0.9.0/go.mod h1:xBxKIO96dXMWWy0MnWVtmwkA9/13aqxPnvrjFYMA2as= +github.com/go-kit/kit v0.10.0/go.mod h1:xUsJbQ/Fp4kEt7AFgCuvyX4a71u8h9jB8tj/ORgOZ7o= +github.com/go-kit/log v0.1.0/go.mod h1:zbhenjAZHb184qTLMA9ZjW7ThYL0H2mk7Q6pNt4vbaY= github.com/go-latex/latex v0.0.0-20210118124228-b3d85cf34e07/go.mod h1:CO1AlKB2CSIqUrmQPqA0gdRIlnLEY0gK5JGjh37zN5U= github.com/go-latex/latex v0.0.0-20210823091927-c0d11ff05a81/go.mod h1:SX0U8uGpxhq9o2S/CELCSUxEWWAuoCUcVCQWv7G2OCk= +github.com/go-logfmt/logfmt v0.3.0/go.mod h1:Qt1PoO58o5twSAckw1HlFXLmHsOX5/0LbT9GBnD5lWE= +github.com/go-logfmt/logfmt v0.4.0/go.mod h1:3RMwSq7FuexP4Kalkev3ejPJsZTpXXBr9+V4qmtdjCk= +github.com/go-logfmt/logfmt v0.5.0/go.mod h1:wCYkCAKZfumFQihp8CzCvQ3paCTfi41vtzG1KdI/P7A= +github.com/go-logr/logr v1.2.0/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/logr v1.2.4 h1:g01GSCwiDw2xSZfjJ2/T9M+S6pFdcNtFYsp+Y43HYDQ= +github.com/go-logr/logr v1.2.4/go.mod h1:jdQByPbusPIv2/zmleS9BjJVeZ6kBagPoEUsqbVz/1A= +github.com/go-logr/zerologr v1.2.0 h1:oS1fjSSEHwpv8Lam3SNmPTLTUw6V4DoB2ZzryqrkMB0= +github.com/go-logr/zerologr v1.2.0/go.mod h1:O82obOiXzyxiBNgAMRT1m+XbOvY8K18Kf6XhT52oqoc= github.com/go-pdf/fpdf v0.5.0/go.mod h1:HzcnA+A23uwogo0tp9yU+l3V+KXhiESpt1PMayhOh5M= github.com/go-pdf/fpdf v0.6.0/go.mod h1:HzcnA+A23uwogo0tp9yU+l3V+KXhiESpt1PMayhOh5M= +github.com/go-playground/assert/v2 v2.0.1/go.mod h1:VDjEfimB/XKnb+ZQfWdccd7VUvScMdVu0Titje2rxJ4= +github.com/go-playground/locales v0.13.0/go.mod h1:taPMhCMXrRLJO55olJkUXHZBHCxTMfnGwq/HNwmWNS8= +github.com/go-playground/universal-translator v0.17.0/go.mod h1:UkSxE5sNxxRwHyU+Scu5vgOQjsIJAF8j9muTVoKLVtA= +github.com/go-playground/validator/v10 v10.2.0/go.mod h1:uOYAAleCW8F/7oMFd6aG0GOhaH6EGOAJShg8Id5JGkI= +github.com/go-redis/redis/v7 v7.4.0/go.mod h1:JDNMw23GTyLNC4GZu9njt15ctBQVn7xjRfnwdHj/Dcg= +github.com/go-sql-driver/mysql v1.4.0/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= +github.com/go-sql-driver/mysql v1.4.1/go.mod h1:zAC/RDZ24gD3HViQzih4MyKcchzm+sOG5ZlKdlhCg5w= +github.com/go-stack/stack v1.8.0/go.mod h1:v0f6uXyyMGvRgIKkXu+yp6POWl0qKG85gN/melR3HDY= +github.com/go-task/slim-sprig v0.0.0-20210107165309-348f09dbbbc0/go.mod h1:fyg7847qk6SyHyPtNmDHnmrv/HOrqktSC+C9fM+CJOE= +github.com/gobwas/httphead v0.0.0-20180130184737-2c6c146eadee/go.mod h1:L0fX3K22YWvt/FAX9NnzrNzcI4wNYi9Yku4O0LKYflo= github.com/gobwas/httphead v0.1.0 h1:exrUm0f4YX0L7EBwZHuCF4GDp8aJfVeBrlLQrs6NqWU= github.com/gobwas/httphead v0.1.0/go.mod h1:O/RXo79gxV8G+RqlR/otEwx4Q36zl9rqC5u12GKvMCM= +github.com/gobwas/pool v0.2.0/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= github.com/gobwas/pool v0.2.1 h1:xfeeEhW7pwmX8nuLVlqbzVc7udMDrwetjEv+TZIz1og= github.com/gobwas/pool v0.2.1/go.mod h1:q8bcK0KcYlCgd9e7WYLm9LpyS+YeLd8JVDW6WezmKEw= +github.com/gobwas/ws v1.0.2/go.mod h1:szmBTxLgaFppYjEmNtny/v3w89xOydFnnZMcgRRu/EM= github.com/gobwas/ws v1.1.0-rc.1 h1:VK3aeRXMI8osaS6YCDKNZhU6RKtcP3B2wzqxOogNDz8= github.com/gobwas/ws v1.1.0-rc.1/go.mod h1:nzvNcVha5eUziGrbxFCo6qFIojQHjJV5cLYIbezhfL0= github.com/goccy/go-json v0.9.11/go.mod h1:6MelG93GURQebXPDq3khkgXZkazVtN9CRI+MGFi0w8I= +github.com/godbus/dbus/v5 v5.0.4/go.mod h1:xhWf0FNVPg57R7Z0UbKHbJfkEywrmjJnf7w5xrFpKfA= +github.com/gogo/googleapis v1.1.0/go.mod h1:gf4bu3Q80BeJ6H1S1vYPm8/ELATdvryBaNFGgqEef3s= +github.com/gogo/protobuf v1.1.1/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.2.0/go.mod h1:r8qH/GZQm5c6nD/R0oafs1akxWv10x8SbQlK7atdtwQ= +github.com/gogo/protobuf v1.2.1/go.mod h1:hp+jE20tsWTFYpLwKvXlhS1hjn+gTNwPg2I6zVXpSg4= +github.com/gogo/protobuf v1.3.2/go.mod h1:P1XiOD3dCwIKUDQYPy72D8LYyHL2YPYrpS2s69NZV8Q= +github.com/golang-jwt/jwt/v4 v4.1.0/go.mod h1:/xlHOz8bRuivTWchD4jCa+NbatV+wEUSzwAxVc6locg= github.com/golang/freetype v0.0.0-20170609003504-e2365dfdc4a0/go.mod h1:E/TSTwGwJL78qG/PmXZO1EjYhfJinVAhrmmHX6Z8B9k= github.com/golang/glog v0.0.0-20160126235308-23def4e6c14b/go.mod h1:SBH7ygxi8pfUlaOkMMuAQtPIUF8ecWP5IEl/CR7VP2Q= github.com/golang/glog v1.0.0/go.mod h1:EWib/APOK0SL3dFbYqvxE3UYd8E6s1ouQ7iEp/0LWV4= +github.com/golang/groupcache v0.0.0-20160516000752-02826c3e7903/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= +github.com/golang/groupcache v0.0.0-20190129154638-5b532d6fd5ef/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20190702054246-869f871628b6/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20191227052852-215e87163ea7/go.mod h1:cIg4eruTrX1D+g88fzRXU5OdNfaM+9IcxsU14FzY7Hc= github.com/golang/groupcache v0.0.0-20200121045136-8c9f03a8e57e h1:1r7pUrabqp18hOBcwBwiTsbnFeTZHV9eER/QT5JVZxY= @@ -649,6 +787,7 @@ github.com/golang/protobuf v1.5.0/go.mod h1:FsONVRAS9T7sI+LIUmWTfcYkHO4aIWwzhcaS github.com/golang/protobuf v1.5.1/go.mod h1:DopwsBzvsk0Fs44TXzsVbJyPhcCPeIwnvohx4u74HPM= github.com/golang/protobuf v1.5.2 h1:ROPKBNFfQgOUMifHyP+KYbvpjbdoFNs+aK7DXlji0Tw= github.com/golang/protobuf v1.5.2/go.mod h1:XVQd3VNwM+JqD3oG2Ue2ip4fOMUkwXdXDdiuN0vRsmY= +github.com/golang/snappy v0.0.0-20180518054509-2e65f85255db/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/golang/snappy v0.0.3/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= github.com/golang/snappy v0.0.4 h1:yAGX7huGHXlcLOEtBnF4w7FQwA26wojNCwOYAEhLjQM= github.com/golang/snappy v0.0.4/go.mod h1:/XxbfmMg8lxefKM7IXC3fBNl/7bRcc72aCRzEWrmP2Q= @@ -671,8 +810,14 @@ github.com/google/go-cmp v0.5.7/go.mod h1:n+brtR0CgQNWTVd5ZUFpTBC8YFBDLK/h/bpaJ8 github.com/google/go-cmp v0.5.8/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= github.com/google/go-cmp v0.5.9 h1:O2Tfq5qg4qc4AmwVlvv0oLiVAGB7enBSJ2x2DqQFi38= github.com/google/go-cmp v0.5.9/go.mod h1:17dUlkBOakJ0+DkrSSNjCkIjxS6bF9zb3elmeNGIjoY= -github.com/google/martian v2.1.0+incompatible h1:/CP5g8u/VJHijgedC/Legn3BAbAaWPgecwXBIDzw5no= +github.com/google/go-github/v28 v28.1.1/go.mod h1:bsqJWQX05omyWVmc00nEUql9mhQyv38lDZ8kPZcQVoM= +github.com/google/go-querystring v1.0.0/go.mod h1:odCYkC5MyYFN7vkCjXpyrEuKhc/BUO6wN/zVPAxq5ck= +github.com/google/go-replayers/grpcreplay v0.1.0/go.mod h1:8Ig2Idjpr6gifRd6pNVggX6TC1Zw6Jx74AKp7QNH2QE= +github.com/google/go-replayers/httpreplay v0.1.0/go.mod h1:YKZViNhiGgqdBlUbI2MwGpq4pXxNmhJLPHQ7cv2b5no= +github.com/google/gofuzz v1.0.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/martian v2.1.0+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= +github.com/google/martian v2.1.1-0.20190517191504-25dcb96d9e51+incompatible h1:xmapqc1AyLoB+ddYT6r04bD9lIjlOqGaREovi0SzFaE= +github.com/google/martian v2.1.1-0.20190517191504-25dcb96d9e51+incompatible/go.mod h1:9I4somxYTbIHy5NJKHRl3wXiIaQGbYVAs8BPL6v8lEs= github.com/google/martian/v3 v3.0.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= github.com/google/martian/v3 v3.1.0/go.mod h1:y5Zk1BBys9G+gd6Jrk0W3cC1+ELVxBWuIGO+w/tUAp0= github.com/google/martian/v3 v3.2.1/go.mod h1:oBOf6HBosgwRXnUGWUB05QECsc6uvmMiJ3+6W4l/CUk= @@ -694,15 +839,22 @@ github.com/google/pprof v0.0.0-20210601050228-01bbb1931b22/go.mod h1:kpwsk12EmLe github.com/google/pprof v0.0.0-20210609004039-a478d1d731e9/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/pprof v0.0.0-20210720184732-4bb14d4b1be1/go.mod h1:kpwsk12EmLew5upagYY7GY0pfYCcupk39gWOCRROcvE= github.com/google/renameio v0.1.0/go.mod h1:KWCgfxg9yswjAJkECMjeO8J8rahYeXnNhOm40UhjYkI= +github.com/google/rpmpack v0.0.0-20191226140753-aa36bfddb3a0/go.mod h1:RaTPr0KUf2K7fnZYLNDrr8rxAamWs3iNywJLtQ2AzBg= +github.com/google/subcommands v1.0.1/go.mod h1:ZjhPrFU+Olkh9WazFPsl27BQ4UPiG37m3yTrtFlrHVk= +github.com/google/uuid v1.0.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/uuid v1.1.1/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.1.2/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= github.com/google/uuid v1.3.0 h1:t6JiXgmwXMjEs8VusXIJk2BXHsn+wx8BZdTaoZ5fu7I= github.com/google/uuid v1.3.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo= +github.com/google/wire v0.3.0/go.mod h1:i1DMg/Lu8Sz5yYl25iOdmc5CT5qusaa+zmRWs16741s= github.com/googleapis/enterprise-certificate-proxy v0.0.0-20220520183353-fd19c99a87aa/go.mod h1:17drOmN3MwGY7t0e+Ei9b45FFGA3fBs3x36SsCg1hq8= github.com/googleapis/enterprise-certificate-proxy v0.1.0/go.mod h1:17drOmN3MwGY7t0e+Ei9b45FFGA3fBs3x36SsCg1hq8= github.com/googleapis/enterprise-certificate-proxy v0.2.0/go.mod h1:8C0jb7/mgJe/9KK8Lm7X9ctZC2t60YyIpYEI16jx0Qg= github.com/googleapis/enterprise-certificate-proxy v0.2.1/go.mod h1:AwSRAtLfXpU5Nm3pW+v7rGDHp09LsPtGY9MduiEsR9k= github.com/googleapis/enterprise-certificate-proxy v0.2.3 h1:yk9/cqRKtT9wXZSsRH9aurXEpJX+U6FLtpYTdC3R06k= github.com/googleapis/enterprise-certificate-proxy v0.2.3/go.mod h1:AwSRAtLfXpU5Nm3pW+v7rGDHp09LsPtGY9MduiEsR9k= +github.com/googleapis/gax-go v2.0.2+incompatible h1:silFMLAnr330+NRuag/VjIGF7TLp/LBrV2CJKFLWEww= +github.com/googleapis/gax-go v2.0.2+incompatible/go.mod h1:SFVmujtThgffbyetf+mdk2eWhX2bMyUtNHzFKcPA9HY= github.com/googleapis/gax-go/v2 v2.0.4/go.mod h1:0Wqv26UfaUD9n4G6kQubkQ+KchISgw+vpHVxEJEs9eg= github.com/googleapis/gax-go/v2 v2.0.5/go.mod h1:DWXyrwAJ9X0FpwwEdw+IPEYBICEFu5mhpdKc/us6bOk= github.com/googleapis/gax-go/v2 v2.1.0/go.mod h1:Q3nei7sK6ybPYH7twZdmQpAd1MKb7pfu6SK+H1/DsU0= @@ -717,28 +869,107 @@ github.com/googleapis/gax-go/v2 v2.7.1 h1:gF4c0zjUP2H/s/hEGyLA3I0fA2ZWjzYiONAD6c github.com/googleapis/gax-go/v2 v2.7.1/go.mod h1:4orTrqY6hXxxaUL4LHIPl6lGo8vAE38/qKbhSAKP6QI= github.com/googleapis/go-type-adapters v1.0.0/go.mod h1:zHW75FOG2aur7gAO2B+MLby+cLsWGBF62rFAi7WjWO4= github.com/googleapis/google-cloud-go-testing v0.0.0-20200911160855-bcd43fbb19e8/go.mod h1:dvDLG8qkwmyD9a/MJJN3XJcT3xFxOKAvTZGvuZmac9g= +github.com/gopherjs/gopherjs v0.0.0-20181017120253-0766667cb4d1/go.mod h1:wJfORRmW1u3UXTncJ5qlYoELFm8eSnnEO6hX4iZ3EWY= +github.com/gordonklaus/ineffassign v0.0.0-20200309095847-7953dde2c7bf/go.mod h1:cuNKsD1zp2v6XfE/orVX2QE1LC+i254ceGcVeDT3pTU= +github.com/goreleaser/goreleaser v0.134.0/go.mod h1:ZT6Y2rSYa6NxQzIsdfWWNWAlYGXGbreo66NmE+3X3WQ= +github.com/goreleaser/nfpm v1.2.1/go.mod h1:TtWrABZozuLOttX2uDlYyECfQX7x5XYkVxhjYcR6G9w= +github.com/gorilla/context v1.1.1/go.mod h1:kBGZzfjB9CEq2AlWe17Uuf7NDRt0dE0s8S51q0aT7Yg= +github.com/gorilla/mux v1.6.2/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs= +github.com/gorilla/mux v1.7.3/go.mod h1:1lud6UwP+6orDFRuTfBEV8e9/aOM/c4fVVCaMa2zaAs= +github.com/gorilla/websocket v0.0.0-20170926233335-4201258b820c/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= +github.com/gorilla/websocket v1.4.0/go.mod h1:E7qHFY5m1UJ88s3WnNqhKjPHQ0heANvMoAMk2YaljkQ= +github.com/gorilla/websocket v1.4.1/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= +github.com/gorilla/websocket v1.4.2/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= +github.com/grpc-ecosystem/go-grpc-middleware v1.0.0/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= +github.com/grpc-ecosystem/go-grpc-middleware v1.0.1-0.20190118093823-f849b5445de4/go.mod h1:FiyG127CGDf3tlThmgyCl78X/SZQqEOJBCDaAfeWzPs= +github.com/grpc-ecosystem/go-grpc-middleware v1.2.2/go.mod h1:EaizFBKfUKtMIF5iaDEhniwNedqGo9FuLFzppDr3uwI= +github.com/grpc-ecosystem/go-grpc-prometheus v1.2.0/go.mod h1:8NvIoxWQoOIhqOTXgfV/d3M/q6VIi02HzZEHgUlZvzk= +github.com/grpc-ecosystem/grpc-gateway v1.8.5/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= +github.com/grpc-ecosystem/grpc-gateway v1.9.0/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= +github.com/grpc-ecosystem/grpc-gateway v1.9.2/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= +github.com/grpc-ecosystem/grpc-gateway v1.9.5/go.mod h1:vNeuVxBJEsws4ogUvrchl83t/GYV9WGTSLVdBhOQFDY= github.com/grpc-ecosystem/grpc-gateway v1.16.0/go.mod h1:BDjrQk3hbvj6Nolgz8mAMFbcEtjT1g+wF4CSlocrBnw= github.com/grpc-ecosystem/grpc-gateway/v2 v2.7.0/go.mod h1:hgWBS7lorOAVIJEQMi4ZsPv9hVvWI6+ch50m39Pf2Ks= github.com/grpc-ecosystem/grpc-gateway/v2 v2.11.3/go.mod h1:o//XUCC/F+yRGJoPO/VU0GSB0f8Nhgmxx0VIRUvaC0w= +github.com/hashicorp/consul/api v1.3.0/go.mod h1:MmDNSzIMUjNpY/mQ398R4bk2FnqQLoPndWW5VkKPlCE= +github.com/hashicorp/consul/api v1.10.1/go.mod h1:XjsvQN+RJGWI2TWy1/kqaE16HrR2J/FWgkYjdZQsX9M= +github.com/hashicorp/consul/sdk v0.3.0/go.mod h1:VKf9jXwCTEY1QZP2MOLRhb5i/I/ssyNV1vwHyQBF0x8= +github.com/hashicorp/consul/sdk v0.8.0/go.mod h1:GBvyrGALthsZObzUGsfgHZQDXjg4lOjagTIwIR1vPms= +github.com/hashicorp/errwrap v1.0.0/go.mod h1:YH+1FKiLXxHSkmPseP+kNlulaMuP3n2brvKWEqk/Jc4= +github.com/hashicorp/go-cleanhttp v0.5.1/go.mod h1:JpRdi6/HCYpAwUzNwuwqhbovhLtngrth3wmdIIUrZ80= +github.com/hashicorp/go-hclog v0.9.2/go.mod h1:5CU+agLiy3J7N7QjHK5d05KxGsuXiQLrjA0H7acj2lQ= +github.com/hashicorp/go-hclog v0.12.0/go.mod h1:whpDNt7SSdeAju8AWKIWsul05p54N/39EeqMAyrmvFQ= +github.com/hashicorp/go-immutable-radix v1.0.0/go.mod h1:0y9vanUI8NX6FsYoO3zeMjhV/C5i9g4Q3DwcSNZ4P60= +github.com/hashicorp/go-msgpack v0.5.3/go.mod h1:ahLV/dePpqEmjfWmKiqvPkv/twdG7iPBM1vqhUKIvfM= +github.com/hashicorp/go-multierror v1.0.0/go.mod h1:dHtQlpGsu+cZNNAkkCN/P3hoUDHhCYQXV3UM06sGGrk= +github.com/hashicorp/go-multierror v1.1.0/go.mod h1:spPvp8C1qA32ftKqdAHm4hHTbPw+vmowP0z+KUhOZdA= +github.com/hashicorp/go-retryablehttp v0.6.4/go.mod h1:vAew36LZh98gCBJNLH42IQ1ER/9wtLZZ8meHqQvEYWY= +github.com/hashicorp/go-rootcerts v1.0.0/go.mod h1:K6zTfqpRlCUIjkwsN4Z+hiSfzSTQa6eBIzfwKfwNnHU= +github.com/hashicorp/go-rootcerts v1.0.2/go.mod h1:pqUvnprVnM5bf7AOirdbb01K4ccR319Vf4pU3K5EGc8= +github.com/hashicorp/go-sockaddr v1.0.0/go.mod h1:7Xibr9yA9JjQq1JpNB2Vw7kxv8xerXegt+ozgdvDeDU= +github.com/hashicorp/go-syslog v1.0.0/go.mod h1:qPfqrKkXGihmCqbJM2mZgkZGvKG1dFdvsLplgctolz4= +github.com/hashicorp/go-uuid v1.0.0/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-uuid v1.0.1/go.mod h1:6SBZvOh/SIDV7/2o3Jml5SYk/TvGqwFJ/bN7x4byOro= +github.com/hashicorp/go-version v1.2.0/go.mod h1:fltr4n8CU8Ke44wwGCBoEymUuxUHl09ZGVZPK5anwXA= +github.com/hashicorp/go.net v0.0.1/go.mod h1:hjKkEWcCURg++eb33jQU7oqQcI9XDCnUzHA0oac0k90= github.com/hashicorp/golang-lru v0.5.0/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= github.com/hashicorp/golang-lru v0.5.1/go.mod h1:/m3WP610KZHVQ1SGc6re/UDhFvYD7pJ4Ao+sR/qLZy8= +github.com/hashicorp/hcl v1.0.0/go.mod h1:E5yfLk+7swimpb2L/Alb/PJmXilQ/rhwaUYs4T20WEQ= +github.com/hashicorp/logutils v1.0.0/go.mod h1:QIAnNjmIWmVIIkWDTG1z5v++HQmx9WQRO+LraFDTW64= +github.com/hashicorp/mdns v1.0.0/go.mod h1:tL+uN++7HEJ6SQLQ2/p+z2pH24WQKWjBPkE0mNTz8vQ= +github.com/hashicorp/mdns v1.0.1/go.mod h1:4gW7WsVCke5TE7EPeYliwHlRUyBtfCwuFwuMg2DmyNY= +github.com/hashicorp/memberlist v0.1.3/go.mod h1:ajVTdAv/9Im8oMAAj5G31PhhMCZJV2pPBoIllUwCN7I= +github.com/hashicorp/memberlist v0.2.2/go.mod h1:MS2lj3INKhZjWNqd3N0m3J+Jxf3DAOnAH9VT3Sh9MUE= +github.com/hashicorp/serf v0.8.2/go.mod h1:6hOLApaqBFA1NXqRQAsxw9QxuDEvNxSQRwA/JwenrHc= +github.com/hashicorp/serf v0.9.5/go.mod h1:UWDWwZeL5cuWDJdl0C6wrvrUwEqtQ4ZKBKKENpqIUyk= github.com/hpcloud/tail v1.0.0/go.mod h1:ab1qPbhIpdTxEkNHXyeSf5vhxWSCs/tWer42PpOxQnU= +github.com/hudl/fargo v1.3.0/go.mod h1:y3CKSmjA+wD2gak7sUSXTAoopbhU08POFhmITJgmKTg= github.com/iancoleman/strcase v0.2.0/go.mod h1:iwCmte+B7n89clKwxIoIXy/HfoL7AsD47ZCWhYzw7ho= github.com/ianlancetaylor/demangle v0.0.0-20181102032728-5e5cf60278f6/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= github.com/ianlancetaylor/demangle v0.0.0-20200824232613-28f6c0f3b639/go.mod h1:aSSvb/t6k1mPoxDqO4vJh6VOCGPwU4O0C2/Eqndh1Sc= +github.com/imdario/mergo v0.3.8/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA= +github.com/imdario/mergo v0.3.9/go.mod h1:2EnlNZ0deacrJVfApfmtdGgDfMuh/nq6Ok1EcJh5FfA= +github.com/improbable-eng/grpc-web v0.14.1/go.mod h1:zEjGHa8DAlkoOXmswrNvhUGEYQA9UI7DhrGeHR1DMGU= +github.com/inconshreveable/mousetrap v1.0.0/go.mod h1:PxqpIevigyE2G7u3NXJIT2ANytuPF1OarO4DADm73n8= +github.com/influxdata/influxdb1-client v0.0.0-20191209144304-8bf82d3c094d/go.mod h1:qj24IKcXYK6Iy9ceXlo3Tc+vtHo9lIhSX5JddghvEPo= +github.com/jarcoal/httpmock v1.0.5/go.mod h1:ATjnClrvW/3tijVmpL/va5Z3aAyGvqU3gCT8nX0Txik= +github.com/jhump/protoreflect v1.6.1/go.mod h1:RZQ/lnuN+zqeRVpQigTwO6o0AJUkxbnSnpuG7toUTG4= +github.com/jhump/protoreflect v1.8.2/go.mod h1:7GcYQDdMU/O/BBrl/cX6PNHpXh6cenjd8pneu5yW7Tg= +github.com/jmespath/go-jmespath v0.0.0-20160202185014-0b12d6b521d8/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= +github.com/jmespath/go-jmespath v0.0.0-20180206201540-c2b33e8439af/go.mod h1:Nht3zPeWKUH0NzdCt2Blrr5ys8VGpn0CEB0cQHVjt7k= +github.com/joho/godotenv v1.3.0/go.mod h1:7hK45KPybAkOC6peb+G5yklZfMxEjkZhHbwpqxOKXbg= +github.com/jonboulle/clockwork v0.1.0/go.mod h1:Ii8DK3G1RaLaWxj9trq07+26W01tbo22gdxWY5EU2bo= +github.com/jpillora/backoff v0.0.0-20180909062703-3050d21c67d7/go.mod h1:2iMrUgbbvHEiQClaW2NsSzMyGHqN+rDFqY705q49KG0= +github.com/jpillora/backoff v1.0.0/go.mod h1:J/6gKK9jxlEcS3zixgDgUAsiuZ7yrSoa/FX5e0EB2j4= +github.com/json-iterator/go v1.1.6/go.mod h1:+SdeFBvtyEkXs7REEP0seUULqWtbJapLOCVDaaPEHmU= +github.com/json-iterator/go v1.1.7/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.8/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.9/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.10/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= +github.com/json-iterator/go v1.1.11/go.mod h1:KdQUCv79m/52Kvf8AW2vK1V8akMuk1QjK/uOdHXbAo4= github.com/jstemmer/go-junit-report v0.0.0-20190106144839-af01ea7f8024/go.mod h1:6v2b51hI/fHJwM22ozAgKL4VKDeJcHhJFhtBdhmNjmU= github.com/jstemmer/go-junit-report v0.9.1/go.mod h1:Brl9GWCQeLvo8nXZwPNNblvFj/XSXhF0NWZEnDohbsk= +github.com/jtolds/gls v4.20.0+incompatible/go.mod h1:QJZ7F/aHp+rZTRtaJ1ow/lLfFfVYBRgL+9YlvaHOwJU= +github.com/julienschmidt/httprouter v1.2.0/go.mod h1:SYymIcj16QtmaHHD7aYtjjsJG7VTCxuUUipMqKk8s4w= +github.com/julienschmidt/httprouter v1.3.0/go.mod h1:JR6WtHb+2LUe8TCKY3cZOxFyyO8IZAc4RVcycCCAKdM= github.com/jung-kurt/gofpdf v1.0.0/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= github.com/jung-kurt/gofpdf v1.0.3-0.20190309125859-24315acbbda5/go.mod h1:7Id9E/uU8ce6rXgefFLlgrJj/GYY22cpxn+r32jIOes= github.com/kballard/go-shellquote v0.0.0-20180428030007-95032a82bc51/go.mod h1:CzGEWj7cYgsdH8dAjBGEr58BoE7ScuLd+fwFZ44+/x8= +github.com/kisielk/errcheck v1.1.0/go.mod h1:EZBBE59ingxPouuu3KfxchcWSUPOHkagtvWXihfKN4Q= +github.com/kisielk/errcheck v1.5.0/go.mod h1:pFxgyoBC7bSaBwPgfKdkLd5X25qrDl4LWUI2bnpBCr8= github.com/kisielk/gotool v1.0.0/go.mod h1:XhKaO+MFFWcvkIS/tQcRk01m1F5IRFswLeQ+oQHNcck= github.com/klauspost/asmfmt v1.3.2/go.mod h1:AG8TuvYojzulgDAMCnYn50l/5QV3Bs/tp6j0HLHbNSE= +github.com/klauspost/compress v1.10.3/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs= +github.com/klauspost/compress v1.11.7/go.mod h1:aoV0uJVorq1K+umq18yTdKaF57EivdYsUV+/s2qKfXs= github.com/klauspost/compress v1.15.9/go.mod h1:PhcZ0MbTNciWF3rruxRgKxI5NkcHHrHUDtV4Yw2GlzU= github.com/klauspost/cpuid/v2 v2.0.9/go.mod h1:FInQzS24/EEf25PyTYn52gqo7WaD8xa0213Md/qVLRg= github.com/konsorten/go-windows-terminal-sequences v1.0.1/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/konsorten/go-windows-terminal-sequences v1.0.2/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= +github.com/konsorten/go-windows-terminal-sequences v1.0.3/go.mod h1:T0+1ngSBFLxvqU3pZ+m/2kptfBszLMUkC4ZK/EgS/cQ= github.com/kr/fs v0.1.0/go.mod h1:FFnZGqtBN9Gxj7eW1uZ42v5BccTP0vu6NEaFoC2HwRg= +github.com/kr/logfmt v0.0.0-20140226030751-b84e30acd515/go.mod h1:+0opPa2QZZtGFBFZlji/RkVcI2GknAs/DXo4wKdlNEc= github.com/kr/pretty v0.1.0/go.mod h1:dAy3ld7l9f0ibDNOQOHHMYYIIbhfbHSm3C4ZsoJORNo= +github.com/kr/pretty v0.2.0/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.2.1/go.mod h1:ipq/a2n7PKx3OHsz4KJII5eveXtPO4qwEXGdVfWzfnI= github.com/kr/pretty v0.3.0 h1:WgNl7dwNpEZ6jJ9k1snq4pZsg7DOEN8hP9Xw0Tsjwk0= github.com/kr/pretty v0.3.0/go.mod h1:640gp4NfQd8pI5XOwp5fnNeVWj67G7CFk/SaSQn7NBk= @@ -746,98 +977,363 @@ github.com/kr/pty v1.1.1/go.mod h1:pFQYn66WHrOpPYNljwOMqo10TkYh1fy3cYio2l3bCsQ= github.com/kr/text v0.1.0/go.mod h1:4Jbv+DJW3UT/LiOwJeYQe1efqtUx/iVham/4vfdArNI= github.com/kr/text v0.2.0 h1:5Nx0Ya0ZqY2ygV366QzturHI13Jq95ApcVaJBhpS+AY= github.com/kr/text v0.2.0/go.mod h1:eLer722TekiGuMkidMxC/pM04lWEeraHUUmBw8l2grE= +github.com/leodido/go-urn v1.2.0/go.mod h1:+8+nEpDfqqsY+g338gtMEUOtuK+4dEMhiQEgxpxOKII= +github.com/lib/pq v1.1.1/go.mod h1:5WUZQaWbwv1U+lTReE5YruASi9Al49XbQIvNi/34Woo= +github.com/lightstep/lightstep-tracer-common/golang/gogo v0.0.0-20190605223551-bc2310a04743/go.mod h1:qklhhLq1aX+mtWk9cPHPzaBjWImj5ULL6C7HFJtXQMM= +github.com/lightstep/lightstep-tracer-go v0.18.1/go.mod h1:jlF1pusYV4pidLvZ+XD0UBX0ZE6WURAspgAczcDHrL4= +github.com/lucsky/cuid v1.2.1/go.mod h1:QaaJqckboimOmhRSJXSx/+IT+VTfxfPGSo/6mfgUfmE= github.com/lyft/protoc-gen-star v0.6.0/go.mod h1:TGAoBVkt8w7MPG72TrKIu85MIdXwDuzJYeZuUPFPNwA= github.com/lyft/protoc-gen-star v0.6.1/go.mod h1:TGAoBVkt8w7MPG72TrKIu85MIdXwDuzJYeZuUPFPNwA= +github.com/lyft/protoc-gen-validate v0.0.13/go.mod h1:XbGvPuh87YZc5TdIa2/I4pLk0QoUACkjt2znoq26NVQ= +github.com/magiconair/properties v1.8.0/go.mod h1:PppfXfuXeibc/6YijjN8zIbojt8czPbwD3XqdrwzmxQ= +github.com/magiconair/properties v1.8.5/go.mod h1:y3VJvCyxH9uVvJTWEGAELF3aiYNyPKd5NZ3oSwXrF60= +github.com/mattn/go-colorable v0.0.9/go.mod h1:9vuHe8Xs5qXnSaW/c/ABM9alt+Vo+STaOChaDxuIBZU= +github.com/mattn/go-colorable v0.1.1/go.mod h1:FuOcm+DKB9mbwrcAfNl7/TZVBZ6rcnceauSikq3lYCQ= +github.com/mattn/go-colorable v0.1.2/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= github.com/mattn/go-colorable v0.1.4/go.mod h1:U0ppj6V5qS13XJ6of8GYAs25YV2eR4EVcfRqFIhoBtE= -github.com/mattn/go-colorable v0.1.7 h1:bQGKb3vps/j0E9GfJQ03JyhRuxsvdAanXlT9BTw3mdw= +github.com/mattn/go-colorable v0.1.6/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= github.com/mattn/go-colorable v0.1.7/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= +github.com/mattn/go-colorable v0.1.8 h1:c1ghPdyEDarC70ftn0y+A/Ee++9zz8ljHG1b13eJ0s8= +github.com/mattn/go-colorable v0.1.8/go.mod h1:u6P/XSegPjTcexA+o6vUJrdnUu04hMope9wVRipJSqc= +github.com/mattn/go-ieproxy v0.0.0-20190610004146-91bb50d98149/go.mod h1:31jz6HNzdxOmlERGGEc4v/dMssOfmp2p5bT/okiKFFc= +github.com/mattn/go-isatty v0.0.3/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.4/go.mod h1:M+lRXTBqGeGNdLjl/ufCoiOlB5xdOkqRJdNxMWT7Zi4= +github.com/mattn/go-isatty v0.0.5/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= github.com/mattn/go-isatty v0.0.8/go.mod h1:Iq45c/XA43vh69/j3iqttzPXn0bhXyGjM0Hdxcsrc5s= github.com/mattn/go-isatty v0.0.10/go.mod h1:qgIWMr58cqv1PHHyhnkY9lrL7etaEgOFcMEpPG5Rm84= +github.com/mattn/go-isatty v0.0.11/go.mod h1:PhnuNfih5lzO57/f3n+odYbM4JtupLOxQOAqxQCu2WE= github.com/mattn/go-isatty v0.0.12/go.mod h1:cbi8OIDigv2wuxKPP5vlRcQ1OAZbq2CE4Kysco4FUpU= github.com/mattn/go-isatty v0.0.16 h1:bq3VjFmv/sOjHtdEhmkEV4x1AJtvUvOJ2PFAZ5+peKQ= github.com/mattn/go-isatty v0.0.16/go.mod h1:kYGgaQfpe5nmfYZH+SKPsOc2e4SrIfOl2e/yFXSvRLM= +github.com/mattn/go-runewidth v0.0.2/go.mod h1:LwmH8dsx7+W8Uxz3IHJYH5QSwggIsqBzpuz5H//U1FU= github.com/mattn/go-runewidth v0.0.6/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= github.com/mattn/go-runewidth v0.0.9 h1:Lm995f3rfxdpd6TSmuVCHVb/QhupuXlYr8sCI/QdE+0= github.com/mattn/go-runewidth v0.0.9/go.mod h1:H031xJmbD/WCDINGzjvQ9THkh0rPKHF+m2gUSrubnMI= +github.com/mattn/go-shellwords v1.0.10/go.mod h1:EZzvwXDESEeg03EKmM+RmDnNOPKG4lLtQsUlTZDWQ8Y= github.com/mattn/go-sqlite3 v1.14.14/go.mod h1:NyWgC/yNuGj7Q9rpYnZvas74GogHl5/Z4A/KQRfk6bU= github.com/mattn/go-tty v0.0.3 h1:5OfyWorkyO7xP52Mq7tB36ajHDG5OHrmBGIS/DtakQI= github.com/mattn/go-tty v0.0.3/go.mod h1:ihxohKRERHTVzN+aSVRwACLCeqIoZAWpoICkkvrWyR0= -github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b h1:j7+1HpAFS1zy5+Q4qx1fWh90gTKwiN4QCGoY9TWyyO4= +github.com/mattn/go-zglob v0.0.1/go.mod h1:9fxibJccNxU2cnpIKLRRFA7zX7qhkJIQWBb449FYHOo= +github.com/matttproud/golang_protobuf_extensions v1.0.1/go.mod h1:D8He9yQNgCq6Z5Ld7szi9bcBfOoFv/3dc6xSMkL2PC0= github.com/mgutz/ansi v0.0.0-20170206155736-9520e82c474b/go.mod h1:01TrycV0kFyexm33Z7vhZRXopbI8J3TDReVlkTgMUxE= +github.com/mgutz/ansi v0.0.0-20200706080929-d51e80ef957d h1:5PJl274Y63IEHC+7izoQE9x6ikvDFZS2mDVS3drnohI= +github.com/mgutz/ansi v0.0.0-20200706080929-d51e80ef957d/go.mod h1:01TrycV0kFyexm33Z7vhZRXopbI8J3TDReVlkTgMUxE= +github.com/miekg/dns v1.0.14/go.mod h1:W1PPwlIAgtquWBMBEV9nkV9Cazfe8ScdGz/Lj7v3Nrg= +github.com/miekg/dns v1.1.26/go.mod h1:bPDLeHnStXmXAq1m/Ch/hvfNHr14JKNPMBo3VZKjuso= github.com/minio/asm2plan9s v0.0.0-20200509001527-cdd76441f9d8/go.mod h1:mC1jAcsrzbxHt8iiaC+zU4b1ylILSosueou12R++wfY= github.com/minio/c2goasm v0.0.0-20190812172519-36a3d3bbc4f3/go.mod h1:RagcQ7I8IeTMnF8JTXieKnO4Z6JCsikNEzj0DwauVzE= +github.com/mitchellh/cli v1.0.0/go.mod h1:hNIlj7HEI86fIcpObd7a0FcrxTWetlwJDGcceTlRvqc= +github.com/mitchellh/cli v1.1.0/go.mod h1:xcISNoH86gajksDmfB23e/pu+B+GeFRMYmoHXxx3xhI= +github.com/mitchellh/go-homedir v1.0.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= +github.com/mitchellh/go-homedir v1.1.0/go.mod h1:SfyaCUpYCn1Vlf4IUYiD9fPX4A5wJrkLzIz1N1q0pr0= +github.com/mitchellh/go-testing-interface v1.0.0/go.mod h1:kRemZodwjscx+RGhAo8eIhFbs2+BFgRtFPeD/KE+zxI= +github.com/mitchellh/gox v0.4.0/go.mod h1:Sd9lOJ0+aimLBi73mGofS1ycjY8lL3uZM3JPS42BGNg= +github.com/mitchellh/iochan v1.0.0/go.mod h1:JwYml1nuB7xOzsp52dPpHFffvOCDupsG0QubkSMEySY= +github.com/mitchellh/mapstructure v0.0.0-20160808181253-ca63d7c062ee/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/mitchellh/mapstructure v1.1.2/go.mod h1:FVVH3fgwuzCH5S8UJGiWEs2h04kUh9fWfEaFds41c1Y= +github.com/mitchellh/mapstructure v1.4.2/go.mod h1:bFUtVrKA4DC2yAKiSyO/QUcy7e+RRV2QTWOzhPopBRo= +github.com/modern-go/concurrent v0.0.0-20180228061459-e0a39a4cb421/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/concurrent v0.0.0-20180306012644-bacd9c7ef1dd/go.mod h1:6dJC0mAP4ikYIbvyc7fijjWJddQyLn8Ig3JB5CqoB9Q= +github.com/modern-go/reflect2 v0.0.0-20180701023420-4b7aa43c6742/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/modern-go/reflect2 v1.0.1/go.mod h1:bx2lNnkwVCuqBIxFjflWJWanXIb3RllmbCylyMrvgv0= +github.com/mwitkow/go-conntrack v0.0.0-20161129095857-cc309e4a2223/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/mwitkow/go-conntrack v0.0.0-20190716064945-2f068394615f/go.mod h1:qRWi+5nqEBWmkhHvq77mSJWrCKwh8bxhgT7d/eI7P4U= +github.com/mwitkow/grpc-proxy v0.0.0-20181017164139-0f1106ef9c76/go.mod h1:x5OoJHDHqxHS801UIuhqGl6QdSAEJvtausosHSdazIo= +github.com/nats-io/jwt v0.3.0/go.mod h1:fRYCDE99xlTsqUzISS1Bi75UBJ6ljOJQOAAu5VglpSg= +github.com/nats-io/jwt v0.3.2/go.mod h1:/euKqTS1ZD+zzjYrY7pseZrTtWQSjujC7xjPc8wL6eU= +github.com/nats-io/jwt v1.1.0/go.mod h1:n3cvmLfBfnpV4JJRN7lRYCyZnw48ksGsbThGXEk4w9M= +github.com/nats-io/nats-server/v2 v2.1.2/go.mod h1:Afk+wRZqkMQs/p45uXdrVLuab3gwv3Z8C4HTBu8GD/k= +github.com/nats-io/nats-server/v2 v2.1.9/go.mod h1:9qVyoewoYXzG1ME9ox0HwkkzyYvnlBDugfR4Gg/8uHU= +github.com/nats-io/nats.go v1.9.1/go.mod h1:ZjDU1L/7fJ09jvUSRVBR2e7+RnLiiIQyqyzEE/Zbp4w= +github.com/nats-io/nats.go v1.10.0/go.mod h1:AjGArbfyR50+afOUotNX2Xs5SYHf+CoOa5HH1eEl2HE= +github.com/nats-io/nats.go v1.12.0/go.mod h1:BPko4oXsySz4aSWeFgOHLZs3G4Jq4ZAyE6/zMCxRT6w= +github.com/nats-io/nkeys v0.1.0/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= +github.com/nats-io/nkeys v0.1.3/go.mod h1:xpnFELMwJABBLVhffcfd1MZx6VsNRFpEugbxziKVo7w= +github.com/nats-io/nkeys v0.1.4/go.mod h1:XdZpAbhgyyODYqjTawOnIOI7VlbKSarI9Gfy1tqEu/s= +github.com/nats-io/nkeys v0.3.0/go.mod h1:gvUNGjVcM2IPr5rCsRsC6Wb3Hr2CQAm08dsxtV6A5y4= +github.com/nats-io/nuid v1.0.1/go.mod h1:19wcPz3Ph3q0Jbyiqsd0kePYG7A95tJPxeL+1OSON2c= +github.com/nishanths/predeclared v0.0.0-20200524104333-86fad755b4d3/go.mod h1:nt3d53pc1VYcphSCIaYAJtnPYnr3Zyn8fMq2wvPGPso= github.com/nxadm/tail v1.4.4/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= -github.com/nxadm/tail v1.4.5 h1:obHEce3upls1IBn1gTw/o7bCv7OJb6Ib/o7wNO+4eKw= github.com/nxadm/tail v1.4.5/go.mod h1:kenIhsEOeOJmVchQTgglprH7qJGnHDVpk1VPCcaMI8A= +github.com/nxadm/tail v1.4.8 h1:nPr65rt6Y5JFSKQO7qToXr7pePgD6Gwiw05lkbyAQTE= +github.com/nxadm/tail v1.4.8/go.mod h1:+ncqLTQzXmGhMZNUePPaPqPvBxHAIsmXswZKocGu+AU= +github.com/oklog/oklog v0.3.2/go.mod h1:FCV+B7mhrz4o+ueLpx+KqkyXRGMWOYEvfiXtdGtbWGs= +github.com/oklog/run v1.0.0/go.mod h1:dlhp/R75TPv97u0XWUtDeV/lRKWPKSdTuV0TZvrmrQA= +github.com/oklog/ulid v1.3.1/go.mod h1:CirwcVhetQ6Lv90oh/F+FBtV6XMibvdAFo93nm5qn4U= +github.com/olekukonko/tablewriter v0.0.0-20170122224234-a0225b3f23b5/go.mod h1:vsDQFd/mU46D+Z4whnwzcISnGGzXWMclvtLoiIKAKIo= github.com/onsi/ginkgo v1.6.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.7.0/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= +github.com/onsi/ginkgo v1.10.1/go.mod h1:lLunBs/Ym6LB5Z9jYTR76FiuTmxDTDusOGeTQH+WWjE= github.com/onsi/ginkgo v1.12.1/go.mod h1:zj2OWP4+oCPe1qIXoGWkgMRwljMUYCdkwsT2108oapk= -github.com/onsi/ginkgo v1.14.2 h1:8mVmC9kjFFmA8H4pKMUhcblgifdkOIXPvbhN1T36q1M= github.com/onsi/ginkgo v1.14.2/go.mod h1:iSB4RoI2tjJc9BBv4NKIKWKya62Rps+oPG/Lv9klQyY= +github.com/onsi/ginkgo v1.16.1/go.mod h1:CObGmKUOKaSC0RjmoAK7tKyn4Azo5P2IWuoMnvwxz1E= +github.com/onsi/ginkgo v1.16.4/go.mod h1:dX+/inL/fNMqNlz0e9LfyB9TswhZpCVdJM/Z6Vvnwo0= +github.com/onsi/ginkgo v1.16.5 h1:8xi0RTUf59SOSfEtZMvwTvXYMzG4gV23XVHOZiXNtnE= +github.com/onsi/ginkgo v1.16.5/go.mod h1:+E8gABHa3K6zRBolWtd+ROzc/U5bkGt0FwiG042wbpU= +github.com/onsi/gomega v1.4.3/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= +github.com/onsi/gomega v1.5.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= +github.com/onsi/gomega v1.7.0/go.mod h1:ex+gbHU/CVuBBDIJjb2X0qEXbFg53c61hWP/1CpauHY= github.com/onsi/gomega v1.7.1/go.mod h1:XdKZgCCFLUoM/7CFJVPcG8C1xQ1AJ0vpAezJrB7JYyY= github.com/onsi/gomega v1.10.1/go.mod h1:iN09h71vgCQne3DLsj+A5owkum+a2tYe+TOCB1ybHNo= -github.com/onsi/gomega v1.10.4 h1:NiTx7EEvBzu9sFOD1zORteLSt3o8gnlvZZwSE9TnY9U= github.com/onsi/gomega v1.10.4/go.mod h1:g/HbgYopi++010VEqkFgJHKC09uJiW9UkXvMUuKHUCQ= +github.com/onsi/gomega v1.11.0/go.mod h1:azGKhqFUon9Vuj0YmTfLSmx0FUwqXYSTl5re8lQLTUg= +github.com/onsi/gomega v1.15.0/go.mod h1:cIuvLEne0aoVhAgh/O6ac0Op8WWw9H6eYCriF+tEHG0= +github.com/onsi/gomega v1.17.0 h1:9Luw4uT5HTjHTN8+aNcSThgH1vdXnmdJ8xIfZ4wyTRE= +github.com/onsi/gomega v1.17.0/go.mod h1:HnhC7FXeEQY45zxNK3PPoIUhzk/80Xly9PcubAlGdZY= +github.com/op/go-logging v0.0.0-20160315200505-970db520ece7/go.mod h1:HzydrMdWErDVzsI23lYNej1Htcns9BCg93Dk0bBINWk= +github.com/opentracing-contrib/go-observer v0.0.0-20170622124052-a52f23424492/go.mod h1:Ngi6UdF0k5OKD5t5wlmGhe/EDKPoUM3BXZSSfIuJbis= +github.com/opentracing/basictracer-go v1.0.0/go.mod h1:QfBfYuafItcjQuMwinw9GhYKwFXS9KnPs5lxoYwgW74= +github.com/opentracing/opentracing-go v1.0.2/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= +github.com/opentracing/opentracing-go v1.1.0/go.mod h1:UkNAQd3GIcIGf0SeVgPpRdFStlNbqXla1AfSYxPUl2o= +github.com/openzipkin-contrib/zipkin-go-opentracing v0.4.5/go.mod h1:/wsWhb9smxSfWAKL3wpBW7V8scJMt8N8gnaMCS9E/cA= +github.com/openzipkin/zipkin-go v0.1.6/go.mod h1:QgAqvLzwWbR/WpD4A3cGpPtJrZXNIiJc5AZX7/PBEpw= +github.com/openzipkin/zipkin-go v0.2.1/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4= +github.com/openzipkin/zipkin-go v0.2.2/go.mod h1:NaW6tEwdmWMaCDZzg8sh+IBNOxHMPnhQw8ySjnjRyN4= +github.com/pact-foundation/pact-go v1.0.4/go.mod h1:uExwJY4kCzNPcHRj+hCR/HBbOOIwwtUjcrb0b5/5kLM= +github.com/pascaldekloe/goe v0.0.0-20180627143212-57f6aae5913c/go.mod h1:lzWF7FIEvWOWxwDKqyGYQf6ZUaNfKdP144TG7ZOy1lc= +github.com/pborman/uuid v1.2.0/go.mod h1:X/NO0urCmaxf9VXbdlT7C2Yzkj2IKimNn4k+gtPdI/k= +github.com/pelletier/go-toml v1.2.0/go.mod h1:5z9KED0ma1S8pY6P1sdut58dfprrGBbd/94hg7ilaic= +github.com/pelletier/go-toml v1.9.4/go.mod h1:u1nR/EPcESfeI/szUZKdtJ0xRNbUoANCkoOuaOx1Y+c= +github.com/performancecopilot/speed v3.0.0+incompatible/go.mod h1:/CLtqpZ5gBg1M9iaPbIdPPGyKcA8hKdoy6hAWba7Yac= github.com/phpdave11/gofpdf v1.4.2/go.mod h1:zpO6xFn9yxo3YLyMvW8HcKWVdbNqgIfOOp2dXMnm1mY= github.com/phpdave11/gofpdi v1.0.12/go.mod h1:vBmVV0Do6hSBHC8uKUQ71JGW+ZGQq74llk/7bXwjDoI= github.com/phpdave11/gofpdi v1.0.13/go.mod h1:vBmVV0Do6hSBHC8uKUQ71JGW+ZGQq74llk/7bXwjDoI= +github.com/pierrec/lz4 v1.0.2-0.20190131084431-473cd7ce01a1/go.mod h1:3/3N9NVKO0jef7pBehbT1qWhCMrIgbYNnFAZCqQ5LRc= +github.com/pierrec/lz4 v2.0.5+incompatible/go.mod h1:pdkljMzZIN41W+lC3N2tnIh5sFi+IEE17M5jbnwPHcY= github.com/pierrec/lz4/v4 v4.1.15/go.mod h1:gZWDp/Ze/IJXGXf23ltt2EXimqmTUXEy0GFuRQyBid4= -github.com/pixelbender/go-sdp v1.1.0 h1:rkm9aFBNKrnB+YGfhLmAkal3pC8XYXb9h+172PlrCBU= -github.com/pixelbender/go-sdp v1.1.0/go.mod h1:6IBlz9+BrUHoFTea7gcp4S54khtOhjCW/nVDLhmZBAs= +github.com/pion/datachannel v1.5.0/go.mod h1:TVbgWP+PVM9TlwL1IkG3JqXXfjGxLvsu9QUeFdpTegI= +github.com/pion/datachannel v1.5.5 h1:10ef4kwdjije+M9d7Xm9im2Y3O6A6ccQb0zcqZcJew8= +github.com/pion/datachannel v1.5.5/go.mod h1:iMz+lECmfdCMqFRhXhcA/219B0SQlbpoR2V118yimL0= +github.com/pion/dtls/v2 v2.0.9/go.mod h1:O0Wr7si/Zj5/EBFlDzDd6UtVxx25CE1r7XM7BQKYQho= +github.com/pion/dtls/v2 v2.0.10/go.mod h1:00OxfeCRWHShcqT9jx8pKKmBWuTt0NCZoVPCaC4VKvU= +github.com/pion/dtls/v2 v2.2.6 h1:yXMxKr0Skd+Ub6A8UqXTRLSywskx93ooMRHsQUtd+Z4= +github.com/pion/dtls/v2 v2.2.6/go.mod h1:t8fWJCIquY5rlQZwA2yWxUS1+OCrAdXrhVKXB5oD/wY= +github.com/pion/ice/v2 v2.1.13/go.mod h1:ovgYHUmwYLlRvcCLI67PnQ5YGe+upXZbGgllBDG/ktU= +github.com/pion/ice/v2 v2.3.2 h1:vh+fi4RkZ8H5fB4brZ/jm3j4BqFgMmNs+aB3X52Hu7M= +github.com/pion/ice/v2 v2.3.2/go.mod h1:AMIpuJqcpe+UwloocNebmTSWhCZM1TUCo9v7nW50jX0= +github.com/pion/interceptor v0.1.0/go.mod h1:j5NIl3tJJPB3u8+Z2Xz8MZs/VV6rc+If9mXEKNuFmEM= +github.com/pion/interceptor v0.1.12 h1:CslaNriCFUItiXS5o+hh5lpL0t0ytQkFnUcbbCs2Zq8= +github.com/pion/interceptor v0.1.12/go.mod h1:bDtgAD9dRkBZpWHGKaoKb42FhDHTG2rX8Ii9LRALLVA= +github.com/pion/ion v1.10.0/go.mod h1:PiH6J8i3NheVTTOxh1wC/c5aAH68eUMzgmnboxEio3A= +github.com/pion/ion-log v1.2.0/go.mod h1:oUlvCy7LZNPzOxmCZVraaMhcS/hB9XFog4m1A8QpVgM= +github.com/pion/ion-log v1.2.1/go.mod h1:oUlvCy7LZNPzOxmCZVraaMhcS/hB9XFog4m1A8QpVgM= +github.com/pion/ion-log v1.2.2/go.mod h1:oUlvCy7LZNPzOxmCZVraaMhcS/hB9XFog4m1A8QpVgM= +github.com/pion/ion-sfu v1.10.10/go.mod h1:sP+lxPeuzUqabYC6u+YOXZI8YENtqqbnCHnHfOxA/dQ= +github.com/pion/ion-sfu v1.11.0 h1:yZcl09yylFa1iN/7MBvJ99FZhPGkazGGwHiaMINmeEw= +github.com/pion/ion-sfu v1.11.0/go.mod h1:EDq8qb5/wmFfSay2TooWHnzeQo0LGRVdkTRFi8wDCcI= +github.com/pion/logging v0.2.2 h1:M9+AIj/+pxNsDfAT64+MAVgJO0rsyLnoJKCqf//DoeY= +github.com/pion/logging v0.2.2/go.mod h1:k0/tDVsRCX2Mb2ZEmTqNa7CWsQPc+YYCB7Q+5pahoms= +github.com/pion/mdns v0.0.5/go.mod h1:UgssrvdD3mxpi8tMxAXbsppL3vJ4Jipw1mTCW+al01g= +github.com/pion/mdns v0.0.7 h1:P0UB4Sr6xDWEox0kTVxF0LmQihtCbSAdW0H2nEgkA3U= +github.com/pion/mdns v0.0.7/go.mod h1:4iP2UbeFhLI/vWju/bw6ZfwjJzk0z8DNValjGxR/dD8= +github.com/pion/randutil v0.1.0 h1:CFG1UdESneORglEsnimhUjf33Rwjubwj6xfiOXBa3mA= +github.com/pion/randutil v0.1.0/go.mod h1:XcJrSMMbbMRhASFVOlj/5hQial/Y8oH/HVo7TBZq+j8= +github.com/pion/rtcp v1.2.6/go.mod h1:52rMNPWFsjr39z9B9MhnkqhPLoeHTv1aN63o/42bWE0= +github.com/pion/rtcp v1.2.8/go.mod h1:qVPhiCzAm4D/rxb6XzKeyZiQK69yJpbUDJSF7TgrqNo= +github.com/pion/rtcp v1.2.10 h1:nkr3uj+8Sp97zyItdN60tE/S6vk4al5CPRR6Gejsdjc= +github.com/pion/rtcp v1.2.10/go.mod h1:ztfEwXZNLGyF1oQDttz/ZKIBaeeg/oWbRYqzBM9TL1I= +github.com/pion/rtp v1.7.0/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= +github.com/pion/rtp v1.7.2/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= +github.com/pion/rtp v1.7.4/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= +github.com/pion/rtp v1.7.13 h1:qcHwlmtiI50t1XivvoawdCGTP4Uiypzfrsap+bijcoA= +github.com/pion/rtp v1.7.13/go.mod h1:bDb5n+BFZxXx0Ea7E5qe+klMuqiBrP+w8XSjiWtCUko= +github.com/pion/sctp v1.7.12/go.mod h1:xFe9cLMZ5Vj6eOzpyiKjT9SwGM4KpK/8Jbw5//jc+0s= +github.com/pion/sctp v1.8.5/go.mod h1:SUFFfDpViyKejTAdwD1d/HQsCu+V/40cCs2nZIvC3s0= +github.com/pion/sctp v1.8.6 h1:CUex11Vkt9YS++VhLf8b55O3VqKrWL6W3SDwX4jAqsI= +github.com/pion/sctp v1.8.6/go.mod h1:SUFFfDpViyKejTAdwD1d/HQsCu+V/40cCs2nZIvC3s0= +github.com/pion/sdp/v3 v3.0.4/go.mod h1:bNiSknmJE0HYBprTHXKPQ3+JjacTv5uap92ueJZKsRk= +github.com/pion/sdp/v3 v3.0.6 h1:WuDLhtuFUUVpTfus9ILC4HRyHsW6TdugjEX/QY9OiUw= +github.com/pion/sdp/v3 v3.0.6/go.mod h1:iiFWFpQO8Fy3S5ldclBkpXqmWy02ns78NOKoLLL0YQw= +github.com/pion/srtp/v2 v2.0.5/go.mod h1:8k6AJlal740mrZ6WYxc4Dg6qDqqhxoRG2GSjlUhDF0A= +github.com/pion/srtp/v2 v2.0.12 h1:WrmiVCubGMOAObBU1vwWjG0H3VSyQHawKeer2PVA5rY= +github.com/pion/srtp/v2 v2.0.12/go.mod h1:C3Ep44hlOo2qEYaq4ddsmK5dL63eLehXFbHaZ9F5V9Y= +github.com/pion/stun v0.3.5/go.mod h1:gDMim+47EeEtfWogA37n6qXZS88L5V6LqFcf+DZA2UA= +github.com/pion/stun v0.4.0 h1:vgRrbBE2htWHy7l3Zsxckk7rkjnjOsSM7PHZnBwo8rk= +github.com/pion/stun v0.4.0/go.mod h1:QPsh1/SbXASntw3zkkrIk3ZJVKz4saBY2G7S10P3wCw= +github.com/pion/transport v0.10.1/go.mod h1:PBis1stIILMiis0PewDw91WJeLJkyIMcEk+DwKOzf4A= +github.com/pion/transport v0.12.2/go.mod h1:N3+vZQD9HlDP5GWkZ85LohxNsDcNgofQmyL6ojX5d8Q= +github.com/pion/transport v0.12.3/go.mod h1:OViWW9SP2peE/HbwBvARicmAVnesphkNkCVZIWJ6q9A= +github.com/pion/transport v0.14.1 h1:XSM6olwW+o8J4SCmOBb/BpwZypkHeyM0PGFCxNQBr40= +github.com/pion/transport v0.14.1/go.mod h1:4tGmbk00NeYA3rUa9+n+dzCCoKkcy3YlYb99Jn2fNnI= +github.com/pion/transport/v2 v2.0.0/go.mod h1:HS2MEBJTwD+1ZI2eSXSvHJx/HnzQqRy2/LXxt6eVMHc= +github.com/pion/transport/v2 v2.0.2 h1:St+8o+1PEzPT51O9bv+tH/KYYLMNR5Vwm5Z3Qkjsywg= +github.com/pion/transport/v2 v2.0.2/go.mod h1:vrz6bUbFr/cjdwbnxq8OdDDzHf7JJfGsIRkxfpZoTA0= +github.com/pion/turn/v2 v2.0.5/go.mod h1:APg43CFyt/14Uy7heYUOGWdkem/Wu4PhCO/bjyrTqMw= +github.com/pion/turn/v2 v2.1.0 h1:5wGHSgGhJhP/RpabkUb/T9PdsAjkGLS6toYz5HNzoSI= +github.com/pion/turn/v2 v2.1.0/go.mod h1:yrT5XbXSGX1VFSF31A3c1kCNB5bBZgk/uu5LET162qs= +github.com/pion/udp v0.1.1 h1:8UAPvyqmsxK8oOjloDk4wUt63TzFe9WEJkg5lChlj7o= +github.com/pion/udp v0.1.1/go.mod h1:6AFo+CMdKQm7UiA0eUPA8/eVCTx8jBIITLZHc9DWX5M= +github.com/pion/udp/v2 v2.0.1 h1:xP0z6WNux1zWEjhC7onRA3EwwSliXqu1ElUZAQhUP54= +github.com/pion/udp/v2 v2.0.1/go.mod h1:B7uvTMP00lzWdyMr/1PVZXtV3wpPIxBRd4Wl6AksXn8= +github.com/pion/webrtc/v3 v3.1.7/go.mod h1:SQxttydYlKo2rCQjHzkUCJFAfEO/Oh2efMWbKYXLk98= +github.com/pion/webrtc/v3 v3.1.59 h1:B3YFo8q6dwBYKA2LUjWRChP59Qtt+xvv1Ul7UPDp6Zc= +github.com/pion/webrtc/v3 v3.1.59/go.mod h1:rJGgStRoFyFOWJULHLayaimsG+jIEoenhJ5MB5gIFqw= +github.com/pixelbender/go-sdp v1.1.1-0.20240403132153-9683c2a0405f h1:2olHyEeX4G9ZN6LZ8Y6jvDn7uHJQUbsEgCwsZEh7K30= +github.com/pixelbender/go-sdp v1.1.1-0.20240403132153-9683c2a0405f/go.mod h1:6IBlz9+BrUHoFTea7gcp4S54khtOhjCW/nVDLhmZBAs= github.com/pkg/diff v0.0.0-20210226163009-20ebb0f2a09e/go.mod h1:pJLUxLENpZxwdsKMEsNbx1VGcRFpLqf3715MtcvvzbA= +github.com/pkg/errors v0.8.0/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.8.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= github.com/pkg/errors v0.9.1/go.mod h1:bwawxfHBFNV+L2hUp1rHADufV3IMtnDRdf1r5NINEl0= +github.com/pkg/profile v1.2.1/go.mod h1:hJw3o1OdXxsrSjjVksARp5W95eeEaEfptyVZyv6JUPA= github.com/pkg/sftp v1.10.1/go.mod h1:lYOWFsE0bwd1+KfKJaKeuokY15vzFx25BLbzYYoAxZI= github.com/pkg/sftp v1.13.1/go.mod h1:3HaPG6Dq1ILlpPZRO0HVMrsydcdLt6HRDccSgb87qRg= github.com/pkg/term v1.2.0-beta.2 h1:L3y/h2jkuBVFdWiJvNfYfKmzcCnILw7mJWm2JQuMppw= github.com/pkg/term v1.2.0-beta.2/go.mod h1:E25nymQcrSllhX42Ok8MRm1+hyBdHY0dCeiKZ9jpNGw= github.com/pmezard/go-difflib v1.0.0 h1:4DBwDE0NGyQoBHbLQYPwSUPoCMWR5BEzIk/f1lZbAQM= github.com/pmezard/go-difflib v1.0.0/go.mod h1:iKH77koFhYxTK1pcRnkKkqfTogsbg7gZNVY4sRDYZ/4= +github.com/posener/complete v1.1.1/go.mod h1:em0nMJCgc9GFtwrmVmEMR/ZL6WyhyjMBndrE9hABlRI= +github.com/posener/complete v1.2.3/go.mod h1:WZIdtGGp+qx0sLrYKtIRAruyNpv6hFCicSgv7Sy7s/s= +github.com/prometheus/client_golang v0.9.1/go.mod h1:7SWBe2y4D6OKWSNQJUaRYU/AaXPKyh/dDVn+NZz0KFw= +github.com/prometheus/client_golang v0.9.3-0.20190127221311-3c4408c8b829/go.mod h1:p2iRAGwDERtqlqzRXnrOVns+ignqQo//hLXqYxZYVNs= +github.com/prometheus/client_golang v0.9.3/go.mod h1:/TN21ttK/J9q6uSwhBd54HahCDft0ttaMvbicHlPoso= +github.com/prometheus/client_golang v1.0.0/go.mod h1:db9x61etRT2tGnBNRi70OPL5FsnadC4Ky3P0J6CfImo= +github.com/prometheus/client_golang v1.3.0/go.mod h1:hJaj2vgQTGQmVCsAACORcieXFeDPbaTKGT+JTgUa3og= +github.com/prometheus/client_golang v1.7.1/go.mod h1:PY5Wy2awLA44sXw4AOSfFBetzPP4j5+D6mVACh+pe2M= +github.com/prometheus/client_golang v1.11.0/go.mod h1:Z6t4BnS23TR94PD6BsDNk8yVqroYurpAkEiz0P2BEV0= +github.com/prometheus/client_model v0.0.0-20180712105110-5c3871d89910/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190115171406-56726106282f/go.mod h1:MbSGuTsp3dbXC40dX6PRTWyKYBIrTGTE9sqQNg2J8bo= +github.com/prometheus/client_model v0.0.0-20190129233127-fd36f4220a90/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= github.com/prometheus/client_model v0.0.0-20190812154241-14fe0d1b01d4/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/client_model v0.1.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= github.com/prometheus/client_model v0.2.0/go.mod h1:xMI15A0UPsDsEKsMN9yxemIoYk6Tm2C1GtYGdfGttqA= +github.com/prometheus/common v0.0.0-20181113130724-41aa239b4cce/go.mod h1:daVV7qP5qjZbuso7PdcryaAu0sAZbrN9i7WWcTMWvro= +github.com/prometheus/common v0.2.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.4.0/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.4.1/go.mod h1:TNfzLD0ON7rHzMJeJkieUDPYmFC7Snx/y86RQel1bk4= +github.com/prometheus/common v0.7.0/go.mod h1:DjGbpBbp5NYNiECxcL/VnbXCCaQpKd3tt26CguLLsqA= +github.com/prometheus/common v0.10.0/go.mod h1:Tlit/dnDKsSWFlCLTWaA1cyBgKHSMdTB80sz/V91rCo= +github.com/prometheus/common v0.15.0/go.mod h1:U+gB1OBLb1lF3O42bTCL+FK18tX9Oar16Clt/msog/s= +github.com/prometheus/common v0.26.0/go.mod h1:M7rCNAaPfAosfx8veZJCuw84e35h3Cfd9VFqTh1DIvc= +github.com/prometheus/procfs v0.0.0-20181005140218-185b4288413d/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.0-20190117184657-bf6a532e95b1/go.mod h1:c3At6R/oaqEKCNdg8wHV1ftS6bRYblBhIjjI8uT2IGk= +github.com/prometheus/procfs v0.0.0-20190507164030-5867b95ac084/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/prometheus/procfs v0.0.2/go.mod h1:TjEm7ze935MbeOT/UhFTIMYKhuLP4wbCsTZCD3I8kEA= +github.com/prometheus/procfs v0.0.8/go.mod h1:7Qr8sr6344vo1JqZ6HhLceV9o3AJ1Ff+GxbHq6oeK9A= +github.com/prometheus/procfs v0.1.3/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= +github.com/prometheus/procfs v0.3.0/go.mod h1:lV6e/gmhEcM9IjHGsFOCxxuZ+z1YqCvr4OA4YeYWdaU= +github.com/prometheus/procfs v0.6.0/go.mod h1:cz+aTbrPOrUb4q7XlbU9ygM+/jj0fzG6c1xBZuNvfVA= +github.com/prometheus/tsdb v0.7.1/go.mod h1:qhTCs0VvXwvX/y3TZrWD7rabWM+ijKTux40TwIPHuXU= +github.com/rcrowley/go-metrics v0.0.0-20181016184325-3113b8401b8a/go.mod h1:bCqnVzQkZxMG4s8nGwiZ5l3QUCyqpo9Y+/ZMZ9VjZe4= github.com/remyoudompheng/bigfft v0.0.0-20200410134404-eec4a21b6bb0/go.mod h1:qqbHyh8v60DhA7CoWK5oRCqLrMHRGoxYCSS9EjAz6Eo= +github.com/rogpeppe/fastuuid v0.0.0-20150106093220-6724a57986af/go.mod h1:XWv6SoW27p1b0cqNHllgS5HIMJraePCO15w5zCzIWYg= +github.com/rogpeppe/fastuuid v1.1.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/fastuuid v1.2.0/go.mod h1:jVj6XXZzXRy/MSR5jhDC/2q6DgLz+nrA6LYCDYWNEvQ= github.com/rogpeppe/go-internal v1.3.0/go.mod h1:M8bDsm7K2OlrFYOpmOWEs/qY81heoFRclV5y23lUDJ4= github.com/rogpeppe/go-internal v1.6.1/go.mod h1:xXDCJY+GAPziupqXw64V24skbSoqbTEfhy4qGm1nDQc= github.com/rogpeppe/go-internal v1.9.0 h1:73kH8U+JUqXU8lRuOHeVHaa/SZPifC7BkcraZVejAe8= github.com/rogpeppe/go-internal v1.9.0/go.mod h1:WtVeX8xhTBvf0smdhujwtBcq4Qrzq/fJaraNFVN+nFs= +github.com/rs/cors v1.7.0/go.mod h1:gFx+x8UowdsKA9AchylcLynDq+nNFfI8FkUZdN/jGCU= +github.com/rs/xid v1.3.0/go.mod h1:trrq9SKmegXys3aeAKXMUTdJsYXVwGY3RLcfgqegfbg= +github.com/rs/zerolog v1.25.0/go.mod h1:7KHcEGe0QZPOm2IE4Kpb5rTh6n1h2hIgS5OOnu1rUaI= +github.com/rs/zerolog v1.26.0 h1:ORM4ibhEZeTeQlCojCK2kPz1ogAY4bGs4tD+SaAdGaE= +github.com/rs/zerolog v1.26.0/go.mod h1:yBiM87lvSqX8h0Ww4sdzNSkVYZ8dL2xjZJG1lAuGZEo= +github.com/russross/blackfriday/v2 v2.0.1/go.mod h1:+Rmxgy9KzJVeS9/2gXHxylqXiyQDYRxCVz55jmeOWTM= github.com/ruudk/golang-pdf417 v0.0.0-20181029194003-1af4ab5afa58/go.mod h1:6lfFZQK844Gfx8o5WFuvpxWRwnSoipWe/p622j1v06w= github.com/ruudk/golang-pdf417 v0.0.0-20201230142125-a7e3863a1245/go.mod h1:pQAZKsJ8yyVxGRWYNEm9oFB8ieLgKFnamEyDmSA0BRk= +github.com/ryanuber/columnize v0.0.0-20160712163229-9b3edd62028f/go.mod h1:sm1tb6uqfes/u+d4ooFouqFdy9/2g9QGwK3SQygK0Ts= +github.com/sagikazarmark/crypt v0.1.0/go.mod h1:B/mN0msZuINBtQ1zZLEQcegFJJf9vnYIR88KRMEuODE= +github.com/samuel/go-zookeeper v0.0.0-20190923202752-2cc03de413da/go.mod h1:gi+0XIa01GRL2eRQVjQkKGqKF3SF9vZR/HnPullcV2E= +github.com/sassoftware/go-rpmutils v0.0.0-20190420191620-a8f1baeba37b/go.mod h1:am+Fp8Bt506lA3Rk3QCmSqmYmLMnPDhdDUcosQCAx+I= github.com/satori/go.uuid v1.2.1-0.20181028125025-b2ce2384e17b h1:gQZ0qzfKHQIybLANtM3mBXNUtOfsCFXeTsnBqCsx1KM= github.com/satori/go.uuid v1.2.1-0.20181028125025-b2ce2384e17b/go.mod h1:dA0hQrYB0VpLJoorglMZABFdXlWrHn1NEOzdhQKdks0= +github.com/sclevine/agouti v3.0.0+incompatible/go.mod h1:b4WX9W9L1sfQKXeJf1mUTLZKJ48R1S7H23Ji7oFO5Bw= +github.com/sean-/seed v0.0.0-20170313163322-e2103e2c3529/go.mod h1:DxrIzT+xaE7yg65j358z/aeFdxmN0P9QXhEzd20vsDc= +github.com/sergi/go-diff v1.0.0/go.mod h1:0CfEIISq7TuYL3j771MWULgwwjU+GofnZX9QAmXWZgo= +github.com/shurcooL/sanitized_anchor_name v1.0.0/go.mod h1:1NzhyTcUVG4SuEtjjoZeVRXNmyL/1OwPU0+IJeTBvfc= +github.com/sirupsen/logrus v1.2.0/go.mod h1:LxeOpSwHxABJmUn/MG1IvRgCAasNZTLOkJPxbbu5VWo= github.com/sirupsen/logrus v1.4.2/go.mod h1:tLMulIdttU9McNUspp0xgXVQah82FyeX6MwdIuYE2rE= +github.com/sirupsen/logrus v1.6.0/go.mod h1:7uNnSEd1DgxDLC74fIahvMZmmYsHGZGEOFrfsX/uA88= +github.com/sirupsen/logrus v1.7.0/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= +github.com/sirupsen/logrus v1.8.1/go.mod h1:yWOB1SBYBC5VeMP7gHvWumXLIWorT60ONWic61uBYv0= github.com/sirupsen/logrus v1.9.0 h1:trlNQbNUG3OdDrDil03MCb1H2o9nJ1x4/5LYw7byDE0= github.com/sirupsen/logrus v1.9.0/go.mod h1:naHLuLoDiP4jHNo9R0sCBMtWGeIprob74mVsIT4qYEQ= +github.com/smartystreets/assertions v0.0.0-20180927180507-b2de0cb4f26d/go.mod h1:OnSkiWE9lh6wB0YB77sQom3nweQdgAjqCqsofrRNTgc= +github.com/smartystreets/assertions v1.0.0/go.mod h1:kHHU4qYBaI3q23Pp3VPrmWhuIUrLW/7eUrw0BU5VaoM= +github.com/smartystreets/go-aws-auth v0.0.0-20180515143844-0c1422d1fdb9/go.mod h1:SnhjPscd9TpLiy1LpzGSKh3bXCfxxXuqd9xmQJy3slM= +github.com/smartystreets/goconvey v1.6.4/go.mod h1:syvi0/a8iFYH4r/RixwvyeAJjdLS9QV7WQ/tjFTllLA= +github.com/smartystreets/gunit v1.0.0/go.mod h1:qwPWnhz6pn0NnRBP++URONOVyNkPyr4SauJk4cUOwJs= +github.com/soheilhy/cmux v0.1.4/go.mod h1:IM3LyeVVIOuxMH7sFAkER9+bJ4dT7Ms6E4xg4kGIyLM= +github.com/soheilhy/cmux v0.1.5/go.mod h1:T7TcVDs9LWfQgPlPsdngu6I6QIoyIFZDDC6sNE1GqG0= +github.com/sony/gobreaker v0.4.1/go.mod h1:ZKptC7FHNvhBz7dN2LGjPVBz2sZJmc0/PkyDJOjmxWY= +github.com/sourcegraph/jsonrpc2 v0.1.0/go.mod h1:ZafdZgk/axhT1cvZAPOhw+95nz2I/Ra5qMlU4gTRwIo= github.com/spaolacci/murmur3 v0.0.0-20180118202830-f09979ecbc72/go.mod h1:JwIasOWyU6f++ZhiEuf87xNszmSA2myDM2Kzu9HwQUA= +github.com/spf13/afero v1.1.2/go.mod h1:j4pytiNVoe2o6bmDsKpLACNPDBIoEAkihy7loJ1B0CQ= github.com/spf13/afero v1.3.3/go.mod h1:5KUK8ByomD5Ti5Artl0RtHeI5pTF7MIDuXL3yY520V4= github.com/spf13/afero v1.6.0/go.mod h1:Ai8FlHk4v/PARR026UzYexafAt9roJ7LcLMAmO6Z93I= github.com/spf13/afero v1.9.2/go.mod h1:iUV7ddyEEZPO5gA3zD4fJt6iStLlL+Lg4m2cihcDf8Y= +github.com/spf13/cast v1.3.0/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= +github.com/spf13/cast v1.4.1/go.mod h1:Qx5cxh0v+4UWYiBimWS+eyWzqEqokIECu5etghLkUJE= +github.com/spf13/cobra v0.0.3/go.mod h1:1l0Ry5zgKvJasoi3XT1TypsSe7PqH0Sj9dhYf7v3XqQ= +github.com/spf13/cobra v1.0.0/go.mod h1:/6GTrnGXV9HjY+aR4k0oJ5tcvakLuG6EuKReYlHNrgE= +github.com/spf13/jwalterweatherman v1.0.0/go.mod h1:cQK4TGJAtQXfYWX+Ddv3mKDzgVb68N+wFjFa4jdeBTo= +github.com/spf13/jwalterweatherman v1.1.0/go.mod h1:aNWZUN0dPAAO/Ljvb5BEdw96iTZ0EXowPYD95IqWIGo= +github.com/spf13/pflag v1.0.1/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.3/go.mod h1:DYY7MBk1bdzusC3SYhjObp+wFpr4gzcvqqNjLnInEg4= +github.com/spf13/pflag v1.0.5/go.mod h1:McXfInJRrz4CZXVZOBLb0bTZqETkiAhM9Iw0y3An2Bg= +github.com/spf13/viper v1.4.0/go.mod h1:PTJ7Z/lr49W6bUbkmS1V3by4uWynFiR9p7+dSq/yZzE= +github.com/spf13/viper v1.9.0/go.mod h1:+i6ajR7OX2XaiBkrcZJFK21htRk7eDeLg7+O6bhUPP4= +github.com/streadway/amqp v0.0.0-20190404075320-75d898a42a94/go.mod h1:AZpEONHx3DKn8O/DFsRAY58/XVQiIPMTMB1SddzLXVw= +github.com/streadway/amqp v0.0.0-20190827072141-edfb9018d271/go.mod h1:AZpEONHx3DKn8O/DFsRAY58/XVQiIPMTMB1SddzLXVw= +github.com/streadway/handy v0.0.0-20190108123426-d5acb3125c2a/go.mod h1:qNTQ5P5JnDBl6z3cMAg/SywNDC5ABu5ApDIw6lUbRmI= github.com/stretchr/objx v0.1.0/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= github.com/stretchr/objx v0.1.1/go.mod h1:HFkY916IF+rwdDfMAkV7OtwuqBVzrE8GR6GFx+wExME= +github.com/stretchr/objx v0.2.0/go.mod h1:qt09Ya8vawLte6SNmTgCsAVtYtaKzEcn8ATUoHMkEqE= github.com/stretchr/objx v0.4.0/go.mod h1:YvHI0jy2hoMjB+UWwv71VJQ9isScKT/TqJzVSSt89Yw= github.com/stretchr/objx v0.5.0/go.mod h1:Yh+to48EsGEfYuaHDzXPcE3xhTkx73EhmCGUpEOglKo= github.com/stretchr/testify v1.2.2/go.mod h1:a8OnRcib4nhh0OaRAV+Yts87kKdq0PP7pXfy6kDkUVs= +github.com/stretchr/testify v1.3.0/go.mod h1:M5WIy9Dh21IEIfnGCwXGc5bZfKNJtfHm1UVUgZn+9EI= github.com/stretchr/testify v1.4.0/go.mod h1:j7eGeouHqKxXV5pUuKE4zz7dFj8WfuZ+81PSLYec5m4= github.com/stretchr/testify v1.5.1/go.mod h1:5W2xD1RspED5o8YsWQXVCued0rvSQ+mT+I5cxcmMvtA= github.com/stretchr/testify v1.6.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.0/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.7.1/go.mod h1:6Fq8oRcR53rry900zMqJjRRixrwX3KX962/h/Wwjteg= github.com/stretchr/testify v1.8.0/go.mod h1:yNjHg4UonilssWZ8iaSj1OCr/vHnekPRkoO+kdMU+MU= -github.com/stretchr/testify v1.8.1 h1:w7B6lhMri9wdJUVmEZPGGhZzrYTPvgJArz7wNPgYKsk= github.com/stretchr/testify v1.8.1/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/stretchr/testify v1.8.2 h1:+h33VjcLVPDHtOdpUCuF+7gSuG3yGIftsP1YvFihtJ8= +github.com/stretchr/testify v1.8.2/go.mod h1:w2LPCIKwWwSfY2zedu0+kehJoqGctiVI29o6fzry7u4= +github.com/subosito/gotenv v1.2.0/go.mod h1:N0PQaV/YGNqwC0u51sEeR/aUtSLEXKX9iv69rRypqCw= github.com/tevino/abool v0.0.0-20170917061928-9b9efcf221b5/go.mod h1:f1SCnEOt6sc3fOJfPQDRDzHOtSXuTtnz0ImG9kPRDV0= github.com/tevino/abool v1.2.0 h1:heAkClL8H6w+mK5md9dzsuohKeXHUpY7Vw0ZCKW+huA= github.com/tevino/abool v1.2.0/go.mod h1:qc66Pna1RiIsPa7O4Egxxs9OqkuxDX55zznh9K07Tzg= +github.com/tj/assert v0.0.0-20171129193455-018094318fb0/go.mod h1:mZ9/Rh9oLWpLLDRpvE+3b7gP/C2YyLFYxNmcLnPTMe0= +github.com/tj/assert v0.0.3/go.mod h1:Ne6X72Q+TB1AteidzQncjw9PabbMp4PBMZ1k+vd1Pvk= +github.com/tj/go-elastic v0.0.0-20171221160941-36157cbbebc2/go.mod h1:WjeM0Oo1eNAjXGDx2yma7uG2XoyRZTq1uv3M/o7imD0= +github.com/tj/go-kinesis v0.0.0-20171128231115-08b17f58cb1b/go.mod h1:/yhzCV0xPfx6jb1bBgRFjl5lytqVqZXEaeqWP8lTEao= +github.com/tj/go-spin v1.1.0/go.mod h1:Mg1mzmePZm4dva8Qz60H2lHwmJ2loum4VIrLgVnKwh4= +github.com/tmc/grpc-websocket-proxy v0.0.0-20170815181823-89b8d40f7ca8/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= +github.com/tmc/grpc-websocket-proxy v0.0.0-20190109142713-0ad062ec5ee5/go.mod h1:ncp9v5uamzpCO7NfCPTXjqaC+bZgJeR0sMTm6dMHP7U= +github.com/ugorji/go v1.1.4/go.mod h1:uQMGLiO92mf5W77hV/PUCpI3pbzQx3CRekS0kk+RGrc= +github.com/ugorji/go v1.1.7/go.mod h1:kZn38zHttfInRq0xu/PH0az30d+z6vm202qpg1oXVMw= +github.com/ugorji/go/codec v1.1.7/go.mod h1:Ax+UKWsSmolVDwsd+7N3ZtXu+yMGCf907BLYF3GoBXY= +github.com/ulikunitz/xz v0.5.6/go.mod h1:2bypXElzHzzJZwzH67Y6wb67pO62Rzfn7BSiF4ABRW8= +github.com/ulikunitz/xz v0.5.7/go.mod h1:nbz6k7qbPmH4IRqmfOplQw/tblSgqTqBwxkY0oWt/14= +github.com/urfave/cli v1.20.0/go.mod h1:70zkFmudgCuE/ngEzBv17Jvp/497gISqfk5gWijbERA= +github.com/urfave/cli v1.22.1/go.mod h1:Gos4lmkARVdJ6EkW0WaNv/tZAAMe9V7XWyB60NtXRu0= github.com/x-cray/logrus-prefixed-formatter v0.5.2 h1:00txxvfBM9muc0jiLIEAkAcIMJzfthRT6usrui8uGmg= github.com/x-cray/logrus-prefixed-formatter v0.5.2/go.mod h1:2duySbKsL6M18s5GU7VPsoEPHyzalCE06qoARUCeBBE= +github.com/xanzy/go-gitlab v0.31.0/go.mod h1:sPLojNBn68fMUWSxIJtdVVIP8uSBYqesTfDUseX11Ug= +github.com/xi2/xz v0.0.0-20171230120015-48954b6210f8/go.mod h1:HUYIGzjTL3rfEspMxjDjgmT5uz5wzYJKVo23qUhYTos= +github.com/xiang90/probing v0.0.0-20190116061207-43a291ad63a2/go.mod h1:UETIi67q53MR2AWcXfiuqkDkRtnGDLqkBTpCHuJHxtU= +github.com/xordataexchange/crypt v0.0.3-0.20170626215501-b2862e3d0a77/go.mod h1:aYKd//L2LvnjZzWKhF00oedf4jCCReLcmhLdhm1A27Q= github.com/yuin/goldmark v1.1.25/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.27/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.1.32/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.2.1/go.mod h1:3hX8gzYuyVAZsxl0MRgGTJEmQBFcNTphYh9decYSb74= github.com/yuin/goldmark v1.3.5/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= +github.com/yuin/goldmark v1.4.0/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.4.1/go.mod h1:mwnBkeHKe2W/ZEtQ+71ViKU8L12m81fl3OWwC1Zlc8k= github.com/yuin/goldmark v1.4.13/go.mod h1:6yULJ656Px+3vBD8DxQVa3kxgyrAnzto9xy5taEt/CY= github.com/zeebo/assert v1.3.0/go.mod h1:Pq9JiuJQpG8JLJdtkwrJESF0Foym2/D9XMU5ciN/wJ0= github.com/zeebo/xxh3 v1.0.2/go.mod h1:5NWz9Sef7zIDm2JHfFlcQvNekmcEl9ekUZQQKCYaDcA= +go.etcd.io/bbolt v1.3.2/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= +go.etcd.io/bbolt v1.3.3/go.mod h1:IbVyRI1SCnLcuJnV2u8VeU0CEYM7e686BmAb1XKL+uU= +go.etcd.io/etcd v0.0.0-20191023171146-3cf2f69b5738/go.mod h1:dnLIgRNXwCJa5e+c6mIZCrds/GIG4ncV9HhK5PX7jPg= +go.etcd.io/etcd/api/v3 v3.5.0/go.mod h1:cbVKeC6lCfl7j/8jBhAK6aIYO9XOjdptoxU/nLQcPvs= +go.etcd.io/etcd/client/pkg/v3 v3.5.0/go.mod h1:IJHfcCEKxYu1Os13ZdwCwIUTUVGYTSAM3YSwc9/Ac1g= +go.etcd.io/etcd/client/v2 v2.305.0/go.mod h1:h9puh54ZTgAKtEbut2oe9P4L/oqKCVB6xsXlzd7alYQ= +go.opencensus.io v0.15.0/go.mod h1:UffZAU+4sDEINUGP/B7UfBBkq4fqLu9zXAX7ke6CHW0= +go.opencensus.io v0.20.1/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= +go.opencensus.io v0.20.2/go.mod h1:6WKK9ahsWS3RSO+PY9ZHZUfv2irvY6gN279GOPZjmmk= go.opencensus.io v0.21.0/go.mod h1:mSImk1erAIZhrmZN+AvHh14ztQfjbGwt4TtuofqLduU= go.opencensus.io v0.22.0/go.mod h1:+kGneAE2xo2IficOXnaByMWTGM9T73dGwxeWcUqIpI8= go.opencensus.io v0.22.2/go.mod h1:yxeiOL68Rb0Xd1ddK5vPZ/oVn4vY4Ynel7k9FzqtOIw= @@ -849,15 +1345,40 @@ go.opencensus.io v0.24.0 h1:y73uSU6J157QMP2kn2r30vwW1A2W2WFwSCGnAVxeaD0= go.opencensus.io v0.24.0/go.mod h1:vNK8G9p7aAivkbmorf4v+7Hgx+Zs0yY+0fOtgBfjQKo= go.opentelemetry.io/proto/otlp v0.7.0/go.mod h1:PqfVotwruBrMGOCsRd/89rSnXhoiJIqeYNgFYFoEGnI= go.opentelemetry.io/proto/otlp v0.15.0/go.mod h1:H7XAot3MsfNsj7EXtrA2q5xSNQ10UqI405h3+duxN4U= +go.uber.org/atomic v1.3.2/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= +go.uber.org/atomic v1.4.0/go.mod h1:gD2HeocX3+yG+ygLZcrzQJaqmWj9AIm7n08wl/qW/PE= +go.uber.org/atomic v1.5.0/go.mod h1:sABNBOSYdrvTF6hTgEIbc7YasKWGhgEQZyfxyTvoXHQ= +go.uber.org/atomic v1.7.0/go.mod h1:fEN4uk6kAWBTFdckzkM89CLk9XfWZrxpCo0nPH17wJc= +go.uber.org/multierr v1.1.0/go.mod h1:wR5kodmAFQ0UK8QlbwjlSNy0Z68gJhDJUG5sjR94q/0= +go.uber.org/multierr v1.3.0/go.mod h1:VgVr7evmIr6uPjLBxg28wmKNXyqE9akIJ5XnfpiKl+4= +go.uber.org/multierr v1.6.0/go.mod h1:cdWPpRnG4AhwMwsgIHip0KRBQjJy5kYEpYjJxpXp9iU= +go.uber.org/tools v0.0.0-20190618225709-2cfd321de3ee/go.mod h1:vJERXedbb3MVM5f9Ejo0C68/HhF8uaILCdgjnY+goOA= +go.uber.org/zap v1.10.0/go.mod h1:vwi/ZaCAaUcBkycHslxD9B2zi4UTXhF60s6SWpuDF0Q= +go.uber.org/zap v1.13.0/go.mod h1:zwrFLgMcdUuIBviXEYEH1YKNaOBnKXsx2IPda5bBwHM= +go.uber.org/zap v1.17.0/go.mod h1:MXVU+bhUf/A7Xi2HNOnopQOrmycQ5Ih87HtOu4q5SSo= +gocloud.dev v0.19.0/go.mod h1:SmKwiR8YwIMMJvQBKLsC3fHNyMwXLw3PMDO+VVteJMI= +golang.org/x/crypto v0.0.0-20180904163835-0709b304e793/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= +golang.org/x/crypto v0.0.0-20181029021203-45a5f77698d3/go.mod h1:6SG95UA2DQfeDnfUPMdvaQW0Q7yPrPDi9nlGo2tz2b4= golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w= +golang.org/x/crypto v0.0.0-20190426145343-a29dc8fdc734/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190510104115-cbcb75029529/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190605123033-f99c8df09eb5/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190701094942-4def268fd1a4/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20190820162420-60c769a6c586/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20190923035154-9ee001bba392/go.mod h1:/lpIB1dKB+9EgE3H3cr1v9wB50oz8l4C4h62xy7jSTY= +golang.org/x/crypto v0.0.0-20191002192127-34f69633bfdc/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= golang.org/x/crypto v0.0.0-20191011191535-87dc89f01550/go.mod h1:yigFU9vqHzYiE8UmvKecakEJjdnWj3jj499lnFckfCI= +golang.org/x/crypto v0.0.0-20200323165209-0ec3e9974c59/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto= +golang.org/x/crypto v0.0.0-20210314154223-e6e6c4f2bb5b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= +golang.org/x/crypto v0.0.0-20210322153248-0c34fe9e7dc2/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= golang.org/x/crypto v0.0.0-20210421170649-83a5a9bb288b/go.mod h1:T9bdIzuCu7OtxOm1hfPfRQxPLYneinmdGuTeoZ9dtd4= +golang.org/x/crypto v0.0.0-20210503195802-e9a32991a82e/go.mod h1:P+XmwS30IXTQdn5tA2iutPOUgjI07+tq3H3K9MVA1s8= +golang.org/x/crypto v0.0.0-20210817164053-32db794688a5/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.0.0-20210921155107-089bfa567519/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= golang.org/x/crypto v0.0.0-20211108221036-ceb1ce70b4fa/go.mod h1:GvvjBRRGRdwPK5ydBHafDWAxML/pGHZbMvKqRZ5+Abc= +golang.org/x/crypto v0.5.0/go.mod h1:NK/OQwhpMQP3MwtdjgLlYHnH9ebylxKWv3e0fK+mkQU= +golang.org/x/crypto v0.6.0/go.mod h1:OFC/31mSvZgRz0V1QTNCzfAI1aIRzbiufJtkMIlEp58= golang.org/x/crypto v0.7.0 h1:AvwMYaRytfdeVt3u6mLaxYtErKYjxA2OXjJ1HHq6t3A= golang.org/x/crypto v0.7.0/go.mod h1:pYwdfH91IfpZVANVyUOhSIPZaFoJGxTFbZhFTx+dXZU= golang.org/x/exp v0.0.0-20180321215751-8460e604b9de/go.mod h1:CJ0aWSM057203Lf6IL+f9T1iT9GByDxfZKAQTCR3kQA= @@ -874,6 +1395,7 @@ golang.org/x/exp v0.0.0-20191227195350-da58074b4299/go.mod h1:2RIsYlXP63K8oxa1u0 golang.org/x/exp v0.0.0-20200119233911-0405dc783f0a/go.mod h1:2RIsYlXP63K8oxa1u096TMicItID8zy7Y6sNkU49FU4= golang.org/x/exp v0.0.0-20200207192155-f17229e696bd/go.mod h1:J/WKrq2StrnmMY6+EHIKF9dgMWnmCNThgcyBT1FY9mM= golang.org/x/exp v0.0.0-20200224162631-6cc2880d07d6/go.mod h1:3jZMyOhIsHpP37uCMkUooju7aAi5cS1Q23tOzKc+0MU= +golang.org/x/exp v0.0.0-20200331195152-e8c3332aa8e5/go.mod h1:4M0jN8W1tt0AVLNr8HDosyJCDCDuyL9N9+3m7wDWgKw= golang.org/x/exp v0.0.0-20220827204233-334a2380cb91/go.mod h1:cyybsKvd6eL0RnXn6p/Grxp8F5bW7iYuBgsNCOHpMYE= golang.org/x/image v0.0.0-20180708004352-c73c2afc3b81/go.mod h1:ux5Hcp/YLpHSI86hEcLt0YII63i6oz57MZXIpbrjZUs= golang.org/x/image v0.0.0-20190227222117-0694c2d4d067/go.mod h1:kZ7UVZpmo3dzQBMxlp+ypCbDeSB+sBbTgSJuh5dn5js= @@ -919,16 +1441,27 @@ golang.org/x/mod v0.8.0/go.mod h1:iBbtSCu2XBx23ZKBPSOrRkjjQPZFPuis4dIYUhu/chs= golang.org/x/net v0.0.0-20180724234803-3673e40ba225/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180826012351-8a410e7b638d/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20180906233101-161cd47e91fd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181023162649-9b4f9f5ad519/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181108082009-03003ca0c849/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181114220301-adae6a3d119a/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181201002055-351d144fa1fc/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20181220203305-927f97764cc3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190108225652-1e06a53dbb7e/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= +golang.org/x/net v0.0.0-20190125091013-d26f9f9a57f3/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190213061140-3a22650c66bd/go.mod h1:mL1N/T3taQHkDXs73rZJwtUhF3w3ftmwwsq0BUmARs4= golang.org/x/net v0.0.0-20190311183353-d8887717615a/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190404232315-eb5bcb51f2a3/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190501004415-9ce7a6920f09/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= golang.org/x/net v0.0.0-20190503192946-f4e77d36d62c/go.mod h1:t9HGtf8HONx5eT2rtn7q6eTqICYqUVnKs3thJo3Qplg= +golang.org/x/net v0.0.0-20190522155817-f3200d17e092/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= golang.org/x/net v0.0.0-20190603091049-60506f45cf65/go.mod h1:HSz+uSET+XFnRR8LxR5pz3Of3rY3CfYBVs4xY44aLks= +golang.org/x/net v0.0.0-20190613194153-d28f0bde5980/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190619014844-b5b0513f8c1b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190620200207-3b0461eec859/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190628185345-da137c7871d7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20190724013045-ca1201d0de80/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190813141303-74dc4d7220e7/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= +golang.org/x/net v0.0.0-20190923162816-aa69164e4478/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20191209160850-c0dbc17a3553/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200114155413-6afb5195e5aa/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200202094626-16171245cfb2/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= @@ -936,6 +1469,7 @@ golang.org/x/net v0.0.0-20200222125558-5a598a2470a0/go.mod h1:z5CRVTTTmAJ677TzLL golang.org/x/net v0.0.0-20200226121028-0de0cce0169b/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200301022130-244492dfa37a/go.mod h1:z5CRVTTTmAJ677TzLLGU+0bjPO0LkuOLi4/5GtJWs/s= golang.org/x/net v0.0.0-20200324143707-d3edc9973b7e/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= +golang.org/x/net v0.0.0-20200421231249-e086a090c8fd/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= golang.org/x/net v0.0.0-20200501053045-e0ff5e5a1de5/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= golang.org/x/net v0.0.0-20200506145744-7e3656a0809f/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= golang.org/x/net v0.0.0-20200513185701-a91f0712d120/go.mod h1:qpuaurCH72eLCgpAm/N6yyVIVM9cpaDIP3A8BGJEC5A= @@ -947,16 +1481,22 @@ golang.org/x/net v0.0.0-20200822124328-c89045814202/go.mod h1:/O7V0waA8r7cgGh81R golang.org/x/net v0.0.0-20201021035429-f5854403a974/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20201031054903-ff519b6c9102/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20201110031124-69a78807bb2b/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= +golang.org/x/net v0.0.0-20201201195509-5d6afe98e0b7/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20201202161906-c7110b5ffcbb/go.mod h1:sp8m0HH+o8qH0wwXwYZr8TS3Oi6o0r6Gce1SSxlDquU= golang.org/x/net v0.0.0-20201209123823-ac852fbbde11/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20201224014010-6772e930b67b/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210119194325-5f4716e94777/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210226172049-e18ecbb05110/go.mod h1:m0MpNAwzfU5UDzcl9v0D8zg8gWTRqZa9RBIspLL5mdg= golang.org/x/net v0.0.0-20210316092652-d523dce5a7f4/go.mod h1:RBQZq4jEuRlivfhVLdyRGr576XBO4/greRjx4P4O3yc= +golang.org/x/net v0.0.0-20210331212208-0fccb6fa2b5c/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= golang.org/x/net v0.0.0-20210405180319-a5a99cb37ef4/go.mod h1:p54w0d4576C0XHj96bSt6lcn1PtDYWL6XObtHCRCNQM= +golang.org/x/net v0.0.0-20210428140749-89ef3d95e781/go.mod h1:OJAsFXCWl8Ukc7SiCT/9KSuxbyM7479/AVlXFRxuMCk= golang.org/x/net v0.0.0-20210503060351-7fd8e65b6420/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20210805182204-aaa1db679c0d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20210813160813-60bc85c4be6d/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20211005001312-d4b1ae081e3b/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20211015210444-4f30a5c0130f/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= +golang.org/x/net v0.0.0-20211020060615-d418f374d309/go.mod h1:9nx3DQGgdP8bBQD5qxJ1jj9UTztislL4KSBs9R2vV5Y= golang.org/x/net v0.0.0-20220127200216-cd36cc0744dd/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= golang.org/x/net v0.0.0-20220225172249-27dd8689420f/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= golang.org/x/net v0.0.0-20220325170049-de3da57026de/go.mod h1:CfG3xpIq0wQ8r1q4Su4UZFWDARRcnwPjda9FqA0JpMk= @@ -969,6 +1509,7 @@ golang.org/x/net v0.0.0-20220722155237-a158d28d115b/go.mod h1:XRhObCWvk6IyKnWLug golang.org/x/net v0.0.0-20220909164309-bea034e7d591/go.mod h1:YDH+HFinaLZZlnHAfSS6ZXJJ9M9t4Dl22yv3iI2vPwk= golang.org/x/net v0.0.0-20221012135044-0b7e1fb9d458/go.mod h1:YDH+HFinaLZZlnHAfSS6ZXJJ9M9t4Dl22yv3iI2vPwk= golang.org/x/net v0.0.0-20221014081412-f15817d10f9b/go.mod h1:YDH+HFinaLZZlnHAfSS6ZXJJ9M9t4Dl22yv3iI2vPwk= +golang.org/x/net v0.1.0/go.mod h1:Cx3nUiGt4eDBEyega/BKRp+/AlGL8hYe7U9odMt2Cco= golang.org/x/net v0.2.0/go.mod h1:KqCZLdyyvdV855qA2rE3GC2aiw5xGR5TEjj8smXukLY= golang.org/x/net v0.5.0/go.mod h1:DivGGAXEgPSlEBzxGzZI+ZLohi+xUj054jfeKui00ws= golang.org/x/net v0.6.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= @@ -976,7 +1517,9 @@ golang.org/x/net v0.7.0/go.mod h1:2Tu9+aMcznHK/AK1HMvgo6xiTLG5rD5rZLDS+rp2Bjs= golang.org/x/net v0.8.0 h1:Zrh2ngAOFYneWTAIAPethzeaQLuHwhuBkuV6ZiRnUaQ= golang.org/x/net v0.8.0/go.mod h1:QVkue5JL9kW//ek3r6jTKnTFis1tRmNAW2P1shuFdJc= golang.org/x/oauth2 v0.0.0-20180821212333-d2e6202438be/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= +golang.org/x/oauth2 v0.0.0-20181106182150-f42d05182288/go.mod h1:N/0e6XlmueqKjAGxoOufVs8QHGRruUQn6yWY3a++T0U= golang.org/x/oauth2 v0.0.0-20190226205417-e64efc72b421/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= +golang.org/x/oauth2 v0.0.0-20190402181905-9f3314589c9a/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20190604053449-0f29369cfe45/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20191202225959-858c2ad4c8b6/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= golang.org/x/oauth2 v0.0.0-20200107190931-bf48bf16ab8d/go.mod h1:gOpvHmFTYa4IltrdGE7lF6nIHvwfUNPOp7c8zoXwtLw= @@ -1021,8 +1564,14 @@ golang.org/x/sync v0.0.0-20220819030929-7fc1605a5dde/go.mod h1:RxMgew5VJxzue5/jJ golang.org/x/sync v0.0.0-20220929204114-8fcdb60fdcc0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= golang.org/x/sync v0.1.0 h1:wsuoTGHzEhffawBOhz5CYhcrV4IdKZbEyZjBMuTp12o= golang.org/x/sync v0.1.0/go.mod h1:RxMgew5VJxzue5/jJTE5uejpjVlOe/izrB70Jof72aM= +golang.org/x/sys v0.0.0-20180823144017-11551d06cbcc/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180830151530-49385e6e1522/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20180905080454-ebe1bf3edb33/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20180909124046-d0be0721c37e/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181026203630-95b1ffbd15a5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181107165924-66b7b1311ac8/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181116152217-5ac8a444bdc5/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= +golang.org/x/sys v0.0.0-20181122145206-62eef0e2fa9b/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190215142949-d0b11bdaac8a/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190222072716-a9d3bda3a223/go.mod h1:STP8DvDyc/dI5b8T5hshtkjS+E42TnysNCUPdjciGhY= golang.org/x/sys v0.0.0-20190312061237-fead79001313/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -1031,29 +1580,41 @@ golang.org/x/sys v0.0.0-20190422165155-953cdadca894/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20190502145724-3ef323f4f1fd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190507160741-ecd444e8653b/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190606165138-5da285871e9c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190620070143-6f217b454f45/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190624142023-c5567b49c5d0/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190726091711-fc99dfbffb4e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190826190057-c7b8b68b1456/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20190904154756-749cb33beabd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190922100055-0a153f010e69/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20190924154521-2837fb4f24fe/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191001151750-bb3f8db39f24/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191005200804-aed5e4c7ecf9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191008105621-543471e840be/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191010194322-b09406accb47/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191026070338-33540a1f6037/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191120155948-bd437916bb0e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191204072324-ce4227a45e2e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20191220142924-d4481acd189f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20191228213918-04cbcbbfeed8/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200106162015-b016eb3dc98e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200113162924-86b910548bc1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200116001909-b77594299b42/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200122134326-e047566fdf82/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200124204421-9fbb57f87de9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200202164722-d101bd2416d5/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200212091648-12a6c2dcc1e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200223170610-d5e6a3e2c0ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200302150141-5c8b2ff67527/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200323222414-85ca7c5b95cd/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200331124033-c3d80250170d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200420163511-1957bb5e6d1f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200501052902-10377860bb8e/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200511232937-7e40ca221e25/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200515095857-1151b9dac4a9/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200519105757-fe76b779f299/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200523222454-059865788121/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200615200032-f1bc736245b1/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20200625212154-ddb9806d33ae/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200803210538-64077c9b5642/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200905004654-be1d3432aa8f/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20200909081042-eff7692f9009/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -1063,7 +1624,9 @@ golang.org/x/sys v0.0.0-20201119102817-f84b799fce68/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20201201145000-ef89a241ccb3/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20201207223542-d4d67f95c62d/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210104204734-6f8348627aad/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210112080510-489259a85091/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210119212857-b64e53b001e4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210124154548-22da62e12c0c/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210220050731-9a76102bfb43/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210225134936-a50acf3fe073/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210304124612-50617c2ba197/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= @@ -1071,20 +1634,24 @@ golang.org/x/sys v0.0.0-20210305230114-8fe3ee5dd75b/go.mod h1:h1NjWce9XRLGQEsW7w golang.org/x/sys v0.0.0-20210315160823-c6e025ad8005/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210320140829-1e4c9ba3b0c4/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210330210617-4fbd30eecc44/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= +golang.org/x/sys v0.0.0-20210403161142-5e06dd20ab57/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423082822-04245dca01da/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210423185535-09eb48e85fd7/go.mod h1:h1NjWce9XRLGQEsW7wpKNCjG9DtNlClVuFLEZdDNbEs= golang.org/x/sys v0.0.0-20210510120138-977fb7262007/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210514084401-e8d321eab015/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210603081109-ebe580a85c40/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210603125802-9665404d3644/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210615035016-665e8c7367d1/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210616094352-59db8d763f22/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210630005230-0f9fa26af87c/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210806184541-e5e7981a1069/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20210809222454-d867a43fc93e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210816183151-1e6c022a8912/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210823070655-63515b42dcdf/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20210908233432-aa78b53d3365/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211007075335-d3039528d8ac/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211019181941-9d821ace8654/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= +golang.org/x/sys v0.0.0-20211025201205-69cdffdb9359/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211124211545-fe61309f8881/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211210111614-af8b64212486/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/sys v0.0.0-20211216021012-1d35b9e2eb4e/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= @@ -1112,6 +1679,7 @@ golang.org/x/sys v0.6.0 h1:MVltZSvRTcU2ljQOhs94SXPftV6DCNnZViHeQps87pQ= golang.org/x/sys v0.6.0/go.mod h1:oPkhp1MJrh7nUepCBck5+mAzfO9JrbApNNgaTdGDITg= golang.org/x/term v0.0.0-20201126162022-7de9c90e9dd1/go.mod h1:bj7SfCRtBDWHUb9snDiAeCFNEtKQo2Wmx5Cou7ajbmo= golang.org/x/term v0.0.0-20210927222741-03fcf44c2211/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= +golang.org/x/term v0.1.0/go.mod h1:jbD1KX2456YbFQfuXm/mYQcufACuNUgVhRMnK/tPxf8= golang.org/x/term v0.2.0/go.mod h1:TVmDHMZPmdnySmBfhjOoOdhjzdE1h4u1VwSiw2l1Nuc= golang.org/x/term v0.4.0/go.mod h1:9P2UbLfCdcvo3p/nzKvsmas4TnlujnuoV9hGgYzW1lQ= golang.org/x/term v0.5.0/go.mod h1:jMB1sMXY+tzblOD4FWmEbocvup2/aLOaQEp7JmGp78k= @@ -1133,13 +1701,16 @@ golang.org/x/text v0.6.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.7.0/go.mod h1:mrYo+phRRbMaCq/xk9113O4dZlRixOauAjOtrjsXDZ8= golang.org/x/text v0.8.0 h1:57P1ETyNKtuIjB4SRd15iJxuhj8Gc416Y78H3qgMh68= golang.org/x/text v0.8.0/go.mod h1:e1OnstbJyHTd6l/uOt8jFFHp6TRDWZR/bV3emEE/zU8= +golang.org/x/time v0.0.0-20180412165947-fbb02b2291d2/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20181108054448-85acf8d2951c/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20190308202827-9d24e82272b4/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20191024005414-555d28b269f0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.0.0-20220922220347-f3bd1da661af/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= golang.org/x/time v0.1.0 h1:xYY+Bajn2a7VBmTM5GikTmnK8ZuX8YgnQCqZpbBNtmA= golang.org/x/time v0.1.0/go.mod h1:tRJNPiyCQ0inRvYxbN9jk5I+vvW/OXSQhTDSoE431IQ= +golang.org/x/tools v0.0.0-20180221164845-07fd8470d635/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180525024113-a5b4c53f6e8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= +golang.org/x/tools v0.0.0-20180828015842-6cd1fcedba52/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20180917221912-90fa682c2a6e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190114222345-bf090417da8b/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= golang.org/x/tools v0.0.0-20190206041539-40960b6deb8e/go.mod h1:n7NCudcB/nEzxVGmLbDWY5pfWTLqBcC2KZ6jyYvM4mQ= @@ -1147,6 +1718,8 @@ golang.org/x/tools v0.0.0-20190226205152-f727befe758c/go.mod h1:9Yl7xja0Znq3iFh3 golang.org/x/tools v0.0.0-20190311212946-11955173bddd/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190312151545-0bb0c0a6e846/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190312170243-e65039ee4138/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190328211700-ab21143f2384/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= +golang.org/x/tools v0.0.0-20190422233926-fe54fb35175b/go.mod h1:LCzVGOaR6xXOjkQ3onu1FJEFr0SW1gC7cKk1uF8kGRs= golang.org/x/tools v0.0.0-20190425150028-36563e24a262/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= golang.org/x/tools v0.0.0-20190506145303-2d16b83fe98c/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= golang.org/x/tools v0.0.0-20190524140312-2c0ae7006135/go.mod h1:RgjU9mgBXZiqYHBnxXauZ1Gv1EHHAz9KjViQ78xBX0Q= @@ -1154,9 +1727,12 @@ golang.org/x/tools v0.0.0-20190606124116-d0a3d012864b/go.mod h1:/rFqwRUd4F7ZHNgw golang.org/x/tools v0.0.0-20190621195816-6e04913cbbac/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190628153133-6cdbf07be9d0/go.mod h1:/rFqwRUd4F7ZHNgwSSTFct+R/Kf4OFW1sUzUTQQTgfc= golang.org/x/tools v0.0.0-20190816200558-6889da9d5479/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20190907020128-2ca718005c18/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20190911174233-4f2ddba30aff/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20190927191325-030b2cf1153e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191012152004-8de300cfc20a/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191029041327-9cc4af7d6b2c/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= +golang.org/x/tools v0.0.0-20191029190741-b9c20aec41a5/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191113191852-77e3bb0ad9e7/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191115202509-3a792d9c32b2/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191119224855-298f0cb1881e/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= @@ -1164,6 +1740,7 @@ golang.org/x/tools v0.0.0-20191125144606-a911d9008d1f/go.mod h1:b+2E5dAYhXwXZwtn golang.org/x/tools v0.0.0-20191130070609-6e064ea0cf2d/go.mod h1:b+2E5dAYhXwXZwtnZ6UAqBI28+e2cm9otk0dWdXHAEo= golang.org/x/tools v0.0.0-20191216173652-a0e659d51361/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20191227053925-7b8e75db28f4/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= +golang.org/x/tools v0.0.0-20200103221440-774c71fcf114/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200117161641-43d50277825c/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200122220014-bf1340f18c4a/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= golang.org/x/tools v0.0.0-20200130002326-2f3ba24bd6e7/go.mod h1:TB2adYChydJhpapKDTa4BR/hXlZSLoq2Wpct/0txZ28= @@ -1175,10 +1752,14 @@ golang.org/x/tools v0.0.0-20200227222343-706bc42d1f0d/go.mod h1:TB2adYChydJhpapK golang.org/x/tools v0.0.0-20200304193943-95d2e580d8eb/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= golang.org/x/tools v0.0.0-20200312045724-11d5b4c81c7d/go.mod h1:o4KQGtdN14AW+yjsvvwRTJJuXz8XRtIHtEnmAXLyFUw= golang.org/x/tools v0.0.0-20200331025713-a30bf2db82d4/go.mod h1:Sl4aGygMT6LrqrWclx+PTx3U+LnKx/seiNR+3G19Ar8= +golang.org/x/tools v0.0.0-20200426102838-f3a5411a4c3b/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200501065659-ab2804fb9c9d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200512131952-2bc93b1c0c88/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200515010526-7d3b6ebf133d/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200522201501-cb1345f3a375/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= golang.org/x/tools v0.0.0-20200618134242-20370b0cb4b2/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200619180055-7c47624df98f/go.mod h1:EkVYQZoAsY45+roYkvgYkIh4xh/qjgUK9TdY2XT94GE= +golang.org/x/tools v0.0.0-20200717024301-6ddee64345a6/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200729194436-6467de6f59a7/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200804011535-6c149bb5ef0d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= golang.org/x/tools v0.0.0-20200825202427-b303f430e36d/go.mod h1:njjCfa9FT2d7l9Bc6FUM5FLjQPp3cFF28FI3qnDFljA= @@ -1187,7 +1768,9 @@ golang.org/x/tools v0.0.0-20201110124207-079ba7bd75cd/go.mod h1:emZCQorbCU4vsT4f golang.org/x/tools v0.0.0-20201124115921-2c860bdd6e78/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20201201161351-ac6f37ff4c2a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20201208233053-a543418bbed2/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20201224043029-2b0845dc783e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210105154028-b0ab187a4818/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= +golang.org/x/tools v0.0.0-20210106214847-113979e3529a/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.0.0-20210108195828-e2f9c7f1fc8e/go.mod h1:emZCQorbCU4vsT4fOWvOPXz4eW1wZW4PmDk9uLelYpA= golang.org/x/tools v0.1.0/go.mod h1:xkSsbof2nBLbhDlRMhhhyNLN/zl3eTqcnHD5viDpcZ0= golang.org/x/tools v0.1.1/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= @@ -1195,6 +1778,7 @@ golang.org/x/tools v0.1.2/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.3/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.4/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= golang.org/x/tools v0.1.5/go.mod h1:o0xws9oXOQQZyjljx8fwUC0k7L1pTE6eaCbjGeHmOkk= +golang.org/x/tools v0.1.7/go.mod h1:LGqMHiF4EqQNHR1JncWGqT5BVaXmza+X+BDGol+dOxo= golang.org/x/tools v0.1.9/go.mod h1:nABZi5QlRsZVlzPpHl034qft6wpY4eDcsTt5AaioBiU= golang.org/x/tools v0.1.12/go.mod h1:hNGJHUnrk76NpqgfD5Aqm5Crs+Hm0VOH/i9J2+nxYbc= golang.org/x/tools v0.3.0/go.mod h1:/rWhSS2+zyEVwoJf8YAX6L2f0ntZ7Kn/mGgAWcipA5k= @@ -1216,7 +1800,10 @@ gonum.org/v1/netlib v0.0.0-20190313105609-8cb42192e0e0/go.mod h1:wa6Ws7BG/ESfp6d gonum.org/v1/plot v0.0.0-20190515093506-e2840ee46a6b/go.mod h1:Wt8AAjI+ypCyYX3nZBvf6cAIx93T+c/OS2HFAYskSZc= gonum.org/v1/plot v0.9.0/go.mod h1:3Pcqqmp6RHvJI72kgb8fThyUnav364FOsdDo2aGW5lY= gonum.org/v1/plot v0.10.1/go.mod h1:VZW5OlhkL1mysU9vaqNHnsy86inf6Ot+jB3r+BczCEo= +google.golang.org/api v0.3.1/go.mod h1:6wY9I6uQWHQ8EM57III9mq/AjF+i8G65rmVagqKMtkk= google.golang.org/api v0.4.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= +google.golang.org/api v0.5.0/go.mod h1:8k5glujaEP+g9n7WNsDg8QP6cUVNI86fCNMcbazEtwE= +google.golang.org/api v0.6.0/go.mod h1:btoxGiFvQNVUZQ8W08zLtrVS08CNpINPEfxXxgJL1Q4= google.golang.org/api v0.7.0/go.mod h1:WtwebWUNSVBH/HAw79HIFXZNqEvBhG+Ra+ax0hx3E3M= google.golang.org/api v0.8.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= google.golang.org/api v0.9.0/go.mod h1:o4eAsZoiT+ibD93RtjEohWalFOjRDx6CVaqeizhEnKg= @@ -1275,6 +1862,8 @@ google.golang.org/api v0.111.0/go.mod h1:qtFHvU9mhgTJegR31csQ+rwxyUTHOKFqCKWp1J0 google.golang.org/api v0.114.0 h1:1xQPji6cO2E2vLiI+C/XiFAnsn1WV3mjaEwGLhi3grE= google.golang.org/api v0.114.0/go.mod h1:ifYI2ZsFK6/uGddGfAD5BMxlnkBqCmqHSDUVi45N5Yg= google.golang.org/appengine v1.1.0/go.mod h1:EbEs0AVv82hx2wNQdGPgUI5lhzA/G0D9YwlJXL52JkM= +google.golang.org/appengine v1.2.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= +google.golang.org/appengine v1.3.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.4.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.5.0/go.mod h1:xpcJRLb0r/rnEns0DIKYYv+WjYCduHsrkT7/EB5XEv4= google.golang.org/appengine v1.6.1/go.mod h1:i06prIuMbXzDqacNJfV5OdTW448YApPu5ww/cMBSeb0= @@ -1282,11 +1871,15 @@ google.golang.org/appengine v1.6.5/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCID google.golang.org/appengine v1.6.6/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= google.golang.org/appengine v1.6.7 h1:FZR1q0exgwxzPzp/aF+VccGrSfxfPpkBqjIIEq3ru6c= google.golang.org/appengine v1.6.7/go.mod h1:8WjMMxjGQR8xUklV/ARdw2HLXBOI7O7uCIDZVag1xfc= +google.golang.org/genproto v0.0.0-20170818010345-ee236bd376b0/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= google.golang.org/genproto v0.0.0-20180817151627-c66870c02cf8/go.mod h1:JiN7NxoALGmiZfu7CAH4rXhgtRTLTxftemlI0sWmxmc= google.golang.org/genproto v0.0.0-20190307195333-5fe7a883aa19/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190418145605-e7d98fc518a7/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190425155659-357c62f0e4bb/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= google.golang.org/genproto v0.0.0-20190502173448-54afdca5d873/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190508193815-b515fa19cec8/go.mod h1:VzzqZJRnGkLBvHegQrXjBqPurQTc5/KpmUdxsrq26oE= +google.golang.org/genproto v0.0.0-20190530194941-fb225487d101/go.mod h1:z3L6/3dTEVtUr6QSP8miRzeRqwQOioJ9I66odjN4I7s= +google.golang.org/genproto v0.0.0-20190620144150-6af8c5fc6601/go.mod h1:z3L6/3dTEVtUr6QSP8miRzeRqwQOioJ9I66odjN4I7s= google.golang.org/genproto v0.0.0-20190801165951-fa694d86fc64/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= google.golang.org/genproto v0.0.0-20190819201941-24fa4b261c55/go.mod h1:DMBHOl98Agz4BDEuKkezgsaosCRResVns1a3J2ZsMNc= google.golang.org/genproto v0.0.0-20190911173649-1774047e7e51/go.mod h1:IbNlFCBrqXvoKpeg0TB2l7cyZUmoaFKYIwrEpbDKLA8= @@ -1303,6 +1896,7 @@ google.golang.org/genproto v0.0.0-20200228133532-8c2c7df3a383/go.mod h1:55QSHmfG google.golang.org/genproto v0.0.0-20200305110556-506484158171/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200312145019-da6875a35672/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200331122359-1ee6d9798940/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= +google.golang.org/genproto v0.0.0-20200423170343-7949de9c1215/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200430143042-b979b6f78d84/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200511104702-f5ebc3bea380/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= google.golang.org/genproto v0.0.0-20200513103714-09dca8ec2884/go.mod h1:55QSHmfGQM9UVYDPBsyGGes0y52j32PQ3BqQfXhyH3c= @@ -1318,6 +1912,7 @@ google.golang.org/genproto v0.0.0-20201201144952-b05cb90ed32e/go.mod h1:FWY/as6D google.golang.org/genproto v0.0.0-20201210142538-e3217bee35cc/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20201214200347-8c77b98c765d/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210108203827-ffc7fda8c3d7/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= +google.golang.org/genproto v0.0.0-20210126160654-44e461bb6506/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210222152913-aa3ee6e6a81c/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210226172003-ab064af71705/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210303154014-9728d6b83eeb/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= @@ -1325,6 +1920,7 @@ google.golang.org/genproto v0.0.0-20210310155132-4ce2db91004e/go.mod h1:FWY/as6D google.golang.org/genproto v0.0.0-20210319143718-93e7006c17a6/go.mod h1:FWY/as6DDZQgahTzZj3fqbO1CbirC29ZNUFHwi0/+no= google.golang.org/genproto v0.0.0-20210329143202-679c6ae281ee/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= google.golang.org/genproto v0.0.0-20210402141018-6c239bbf2bb1/go.mod h1:9lPAdzaEmUacj36I+k7YKbEc5CXzPIeORRgDAUOu28A= +google.golang.org/genproto v0.0.0-20210505142820-a42aa055cf76/go.mod h1:P3QM42oQyzQSnHPnZ/vqoCdDmzH28fzWByN9asMeM8A= google.golang.org/genproto v0.0.0-20210513213006-bf773b8c8384/go.mod h1:P3QM42oQyzQSnHPnZ/vqoCdDmzH28fzWByN9asMeM8A= google.golang.org/genproto v0.0.0-20210602131652-f16073e35f0c/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= google.golang.org/genproto v0.0.0-20210604141403-392c879c8b08/go.mod h1:UODoCrxHCcBojKKwX1terBiRUaqAsFqJiF615XL43r0= @@ -1410,10 +2006,16 @@ google.golang.org/genproto v0.0.0-20230223222841-637eb2293923/go.mod h1:3Dl5ZL0q google.golang.org/genproto v0.0.0-20230303212802-e74f57abe488/go.mod h1:TvhZT5f700eVlTNwND1xoEZQeWTB2RY/65kplwl/bFA= google.golang.org/genproto v0.0.0-20230306155012-7f2fa6fef1f4 h1:DdoeryqhaXp1LtT/emMP1BRJPHHKFi5akj/nbx/zNTA= google.golang.org/genproto v0.0.0-20230306155012-7f2fa6fef1f4/go.mod h1:NWraEVixdDnqcqQ30jipen1STv2r/n24Wb7twVTGR4s= +google.golang.org/grpc v1.8.0/go.mod h1:yo6s7OP7yaDglbqo1J04qKzAhqBH6lvTonzMVmEdcZw= +google.golang.org/grpc v1.17.0/go.mod h1:6QZJwpn2B+Zp71q/5VxRsJ6NXXVCE5NRUHRo+f3cWCs= google.golang.org/grpc v1.19.0/go.mod h1:mqu4LbDTu4XGKhr4mRzUsmM4RtVoemTSY81AxZiDr8c= +google.golang.org/grpc v1.20.0/go.mod h1:chYK+tFQF0nDUGJgXMSgLCQk3phJEuONr2DCgLDdAQM= google.golang.org/grpc v1.20.1/go.mod h1:10oTOabMzJvdu6/UiuZezV6QK5dSlG84ov/aaiqXj38= +google.golang.org/grpc v1.21.0/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= google.golang.org/grpc v1.21.1/go.mod h1:oYelfM1adQP15Ek0mdvEgi9Df8B9CZIaU1084ijfRaM= +google.golang.org/grpc v1.22.1/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= google.golang.org/grpc v1.23.0/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= +google.golang.org/grpc v1.23.1/go.mod h1:Y5yQAOtifL1yxbo5wqy6BxZv8vAUGQwXBOALyacEbxg= google.golang.org/grpc v1.25.1/go.mod h1:c3i+UQWmh7LiEpx4sFZnkU36qjEYZ0imhYfXVyQciAY= google.golang.org/grpc v1.26.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= google.golang.org/grpc v1.27.0/go.mod h1:qbnxyOmOxrQa7FizSgH+ReBfzJrCY1pSN7KXBS8abTk= @@ -1423,6 +2025,7 @@ google.golang.org/grpc v1.29.1/go.mod h1:itym6AZVZYACWQqET3MqgPpjcuV5QH3BxFS3Iji google.golang.org/grpc v1.30.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= google.golang.org/grpc v1.31.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= google.golang.org/grpc v1.31.1/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= +google.golang.org/grpc v1.32.0/go.mod h1:N36X2cJ7JwdamYAgDz+s+rVMFjt3numwzf/HckM8pak= google.golang.org/grpc v1.33.1/go.mod h1:fr5YgcSWrqhRRxogOsw7RzIpsmvOZ6IcH4kBYTpR3n0= google.golang.org/grpc v1.33.2/go.mod h1:JMHMWHQWaTccqQQlmk3MJZS+GWXOdAesneDmEnv2fbc= google.golang.org/grpc v1.34.0/go.mod h1:WotjhfgOW/POjDeRt8vscBtXq+2VjORFy659qA51WJ8= @@ -1436,6 +2039,7 @@ google.golang.org/grpc v1.39.0/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnD google.golang.org/grpc v1.39.1/go.mod h1:PImNr+rS9TWYb2O4/emRugxiyHZ5JyHW5F+RPnDzfrE= google.golang.org/grpc v1.40.0/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= google.golang.org/grpc v1.40.1/go.mod h1:ogyxbiOoUXAkP+4+xa6PZSE9DZgIHtSpzjDTB9KAK34= +google.golang.org/grpc v1.41.0/go.mod h1:U3l9uK9J0sini8mHphKoXyaqDA/8VyGnDee1zzIUK6k= google.golang.org/grpc v1.42.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= google.golang.org/grpc v1.44.0/go.mod h1:k+4IHHFw41K8+bbowsex27ge2rCb65oeWqe4jJ590SU= google.golang.org/grpc v1.45.0/go.mod h1:lN7owxKUQEqMfSyQikvvk5tf/6zMPsrK+ONuO11+0rQ= @@ -1460,6 +2064,7 @@ google.golang.org/protobuf v1.23.0/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2 google.golang.org/protobuf v1.23.1-0.20200526195155-81db48ad09cc/go.mod h1:EGpADcykh3NcUnDUJcl1+ZksZNG86OlYog2l/sGQquU= google.golang.org/protobuf v1.24.0/go.mod h1:r/3tXBNzIEhYS9I1OUVjXDlt8tc493IdKGjtUeSXeh4= google.golang.org/protobuf v1.25.0/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= +google.golang.org/protobuf v1.25.1-0.20200805231151-a709e31e5d12/go.mod h1:9JNX74DMeImyA3h4bdi1ymwjUzf21/xIlbajtzgsN7c= google.golang.org/protobuf v1.26.0-rc.1/go.mod h1:jlhhOSvTdKEhbULTjvd4ARK9grFBp09yW+WbY/TyQbw= google.golang.org/protobuf v1.26.0/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= google.golang.org/protobuf v1.27.1/go.mod h1:9q0QmTI4eRPtz6boOQmLYwt+qCgq0jsYwAQnmE0givc= @@ -1467,22 +2072,37 @@ google.golang.org/protobuf v1.28.0/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqw google.golang.org/protobuf v1.28.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= google.golang.org/protobuf v1.29.1 h1:7QBf+IK2gx70Ap/hDsOmam3GE0v9HicjfEdAxE62UoM= google.golang.org/protobuf v1.29.1/go.mod h1:HV8QOd/L58Z+nl8r43ehVNZIU/HEI6OcFqwMG9pJV4I= +gopkg.in/alecthomas/kingpin.v2 v2.2.6/go.mod h1:FMv+mEhP44yOT+4EoQTLFTRgOQ1FBLkstjWtayDeSgw= gopkg.in/check.v1 v0.0.0-20161208181325-20d25e280405/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20180628173108-788fd7840127/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= +gopkg.in/check.v1 v1.0.0-20190902080502-41f04d3bba15/go.mod h1:Co6ibVJAznAaIkqp8huTwlJQCZ016jof/cbN4VW5Yz0= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c h1:Hei/4ADfdWqJk1ZMxUNpqntNwaWcugrBjAiHlqqRiVk= gopkg.in/check.v1 v1.0.0-20201130134442-10cb98267c6c/go.mod h1:JHkPIbrfpd72SG/EVd6muEfDQjcINNoR0C8j2r3qZ4Q= +gopkg.in/cheggaaa/pb.v1 v1.0.25/go.mod h1:V/YB90LKu/1FcN3WVnfiiE5oMCibMjukxqG/qStrOgw= gopkg.in/errgo.v2 v2.1.0/go.mod h1:hNsd1EY+bozCKY1Ytp96fpM3vjJbqLJn88ws8XvfDNI= gopkg.in/fsnotify.v1 v1.4.7/go.mod h1:Tz8NjZHkW78fSQdbUxIjBTcgA1z1m8ZHf0WmKUhAMys= +gopkg.in/gcfg.v1 v1.2.3/go.mod h1:yesOnuUOFQAhST5vPY4nbZsb/huCgGGXlipJsBn0b3o= +gopkg.in/ini.v1 v1.63.2/go.mod h1:pNLf8WUiyNEtQjuu5G5vTm06TEv9tsIgeAvK8hOrP4k= +gopkg.in/resty.v1 v1.12.0/go.mod h1:mDo4pnntr5jdWRML875a/NmxYqAlA73dVijT2AXvQQo= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7 h1:uRGJdciOHaEIrze2W8Q3AKkepLTh2hOroT7a+7czfdQ= gopkg.in/tomb.v1 v1.0.0-20141024135613-dd632973f1e7/go.mod h1:dt/ZhP58zS4L8KSrWDmTeBkI65Dw0HsyUHuEVlX15mw= +gopkg.in/warnings.v0 v0.1.2/go.mod h1:jksf8JmL6Qr/oQM2OXTHunEvvTAsrWBLb6OOjuVWRNI= +gopkg.in/yaml.v2 v2.0.0-20170812160011-eb3733d160e7/go.mod h1:JAlM8MvJe8wmxCU4Bli9HhUf9+ttbYbLASfIpnQbh74= +gopkg.in/yaml.v2 v2.2.1/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.2/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.3/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.2.4/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= -gopkg.in/yaml.v2 v2.3.0 h1:clyUAQHOM3G0M3f5vQj7LuJrETvjVot3Z5el9nffUtU= +gopkg.in/yaml.v2 v2.2.5/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.2.8/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= gopkg.in/yaml.v2 v2.3.0/go.mod h1:hI93XBmqTisBFMUTm0b8Fm+jr3Dg1NNxqwp+5A1VGuI= +gopkg.in/yaml.v2 v2.4.0 h1:D8xgwECY7CYvx+Y2n4sBz93Jn9JRvxdiyyo8CTfuKaY= +gopkg.in/yaml.v2 v2.4.0/go.mod h1:RDklbk79AGWmwhnvt/jBztapEOGDOx6ZbXqjP6csGnQ= gopkg.in/yaml.v3 v3.0.0-20200313102051-9f266ea9e77c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20200605160147-a5ece683394c/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +gopkg.in/yaml.v3 v3.0.0-20210107192922-496545a6307b/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= gopkg.in/yaml.v3 v3.0.1 h1:fxVm/GzAzEWqLHuvctI91KS9hhNmmWOoWu0XTYJS7CA= gopkg.in/yaml.v3 v3.0.1/go.mod h1:K4uyk7z7BCEPqu6E+C64Yfv1cQ7kz7rIZviUmN+EgEM= +honnef.co/go/tools v0.0.0-20180728063816-88497007e858/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190102054323-c2f93a96b099/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190106161140-3f1c8253044a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= honnef.co/go/tools v0.0.0-20190418001031-e561f6794a2a/go.mod h1:rf3lG4BRIbNafJWhAfAdb/ePZxsR/4RtNHQocxwk9r4= @@ -1525,7 +2145,11 @@ modernc.org/strutil v1.1.3/go.mod h1:MEHNA7PdEnEwLvspRMtWTNnp2nnyvMfkimT1NKNAGbw modernc.org/tcl v1.13.1/go.mod h1:XOLfOwzhkljL4itZkK6T72ckMgvj0BDsnKNdZVUOecw= modernc.org/token v1.0.0/go.mod h1:UGzOrNV1mAFSEB63lOFHIpNRUVMvYTc6yu1SMY/XTDM= modernc.org/z v1.5.1/go.mod h1:eWFB510QWW5Th9YGZT81s+LwvaAs3Q2yr4sP0rmLkv8= +nhooyr.io/websocket v1.8.6/go.mod h1:B70DZP8IakI65RVQ51MsWP/8jndNma26DVA/nFSCgW0= +pack.ag/amqp v0.11.2/go.mod h1:4/cbmt4EJXSKlG6LCfWHoqmN0uFdy5i/+YFz+fTfhV4= rsc.io/binaryregexp v0.2.0/go.mod h1:qTv7/COck+e2FymRvadv62gMdZztPaShugOCi3I+8D8= rsc.io/pdf v0.1.1/go.mod h1:n8OzWcQ6Sp37PL01nO98y4iUCRdTGarVfzxY20ICaU4= rsc.io/quote/v3 v3.1.0/go.mod h1:yEA65RcK8LyAZtP9Kv3t0HmxON59tX3rD+tICJqUlj0= rsc.io/sampler v1.3.0/go.mod h1:T1hPZKmBbMNahiBKFy5HrXp6adAjACjK9JXDnKaTXpA= +sigs.k8s.io/yaml v1.1.0/go.mod h1:UJmg0vDUVViEyp3mgSv9WPwZCDxu4rQW1olrI1uml+o= +sourcegraph.com/sourcegraph/appdash v0.0.0-20190731080439-ebfcffb1b5c0/go.mod h1:hI742Nqp5OhwiqlzhgfbWU4mW4yO10fP+LoT9WOswdU= diff --git a/pkg/account/profile.go b/pkg/account/profile.go index 89cbdb2..df3b95c 100644 --- a/pkg/account/profile.go +++ b/pkg/account/profile.go @@ -16,10 +16,10 @@ var ( ) func init() { - logger = utils.NewLogrusLogger(log.DebugLevel, "UserAgent", nil) + logger = utils.NewLogrusLogger(utils.DefaultLogLevel, "UserAgent", nil) } -//AuthInfo . +// AuthInfo . type AuthInfo struct { AuthUser string Realm string @@ -66,7 +66,7 @@ func (p *Profile) Contact() *sip.Address { return contact } -//NewProfile . +// NewProfile . func NewProfile( uri sip.Uri, displayName string, @@ -104,7 +104,7 @@ func NewProfile( return p } -//RegisterState . +// RegisterState . type RegisterState struct { Account *Profile StatusCode sip.StatusCode diff --git a/pkg/auth/server.go b/pkg/auth/server.go index 1b9c79e..e6198f4 100644 --- a/pkg/auth/server.go +++ b/pkg/auth/server.go @@ -50,7 +50,7 @@ func NewServerAuthorizer(callback RequestCredentialCallback, realm string, authI useAuthInt: authInt, realm: realm, } - auth.log = utils.NewLogrusLogger(log.DebugLevel, "ServerAuthorizer", nil) + auth.log = utils.NewLogrusLogger(utils.DefaultLogLevel, "ServerAuthorizer", nil) go func() { for now := range time.Tick(NonceExpire) { auth.mx.Lock() diff --git a/pkg/media/rtp/udp.go b/pkg/media/rtp/udp.go index ed7c118..cee7257 100644 --- a/pkg/media/rtp/udp.go +++ b/pkg/media/rtp/udp.go @@ -23,7 +23,7 @@ type RtpUDPStream struct { func NewRtpUDPStream(bind string, portMin, portMax int, callback func(pkt []byte, raddr net.Addr)) *RtpUDPStream { - logger := utils.NewLogrusLogger(log.DebugLevel, "Media", nil) + logger := utils.NewLogrusLogger(utils.DefaultLogLevel, "Media", nil) lAddr := &net.UDPAddr{IP: net.ParseIP(bind), Port: 0} var err error diff --git a/pkg/session/session.go b/pkg/session/session.go index a1b2c64..e510c93 100644 --- a/pkg/session/session.go +++ b/pkg/session/session.go @@ -46,7 +46,7 @@ func NewInviteSession(reqcb RequestCallback, uaType string, contact: contact, } - s.logger = utils.NewLogrusLogger(log.DebugLevel, "Session", nil) + s.logger = utils.NewLogrusLogger(utils.DefaultLogLevel, "Session", nil) to, _ := req.To() from, _ := req.From() @@ -104,11 +104,11 @@ func (s *Session) RemoteSdp() string { } func (s *Session) Contact() string { - return s.contact.String() + return s.contact.Address.String() } -func (s *Session) CallID() *sip.CallID { - return &s.callID +func (s *Session) CallID() sip.CallID { + return s.callID } func (s *Session) Request() sip.Request { diff --git a/pkg/stack/stack.go b/pkg/stack/stack.go index 15939dc..3655bd8 100644 --- a/pkg/stack/stack.go +++ b/pkg/stack/stack.go @@ -78,7 +78,7 @@ func NewSipStack(config *SipStackConfig) *SipStack { config = &SipStackConfig{} } - logger := utils.NewLogrusLogger(log.DebugLevel, "SipStack", nil) + logger := utils.NewLogrusLogger(utils.DefaultLogLevel, "SipStack", nil) var host string var ip net.IP @@ -134,12 +134,12 @@ func NewSipStack(config *SipStackConfig) *SipStack { } s.log = logger - s.tp = transport.NewLayer(ip, dnsResolver, config.MsgMapper, utils.NewLogrusLogger(log.DebugLevel, "transport.Layer", nil)) + s.tp = transport.NewLayer(ip, dnsResolver, config.MsgMapper, utils.NewLogrusLogger(utils.DefaultLogLevel, "transport.Layer", nil)) sipTp := &sipTransport{ tpl: s.tp, s: s, } - s.tx = transaction.NewLayer(sipTp, utils.NewLogrusLogger(log.DebugLevel, "transaction.Layer", nil)) + s.tx = transaction.NewLayer(sipTp, utils.NewLogrusLogger(utils.DefaultLogLevel, "transaction.Layer", nil)) s.running.Set() go s.serve() diff --git a/pkg/ua/ua.go b/pkg/ua/ua.go index b6b57ab..6892f0d 100644 --- a/pkg/ua/ua.go +++ b/pkg/ua/ua.go @@ -38,13 +38,13 @@ type UserAgentConfig struct { SipStack *stack.SipStack } -//InviteSessionHandler . +// InviteSessionHandler . type InviteSessionHandler func(s *session.Session, req *sip.Request, resp *sip.Response, status session.Status) -//RegisterHandler . +// RegisterHandler . type RegisterHandler func(regState account.RegisterState) -//UserAgent . +// UserAgent . type UserAgent struct { InviteStateHandler InviteSessionHandler RegisterStateHandler RegisterHandler @@ -53,14 +53,14 @@ type UserAgent struct { log log.Logger } -//NewUserAgent . +// NewUserAgent . func NewUserAgent(config *UserAgentConfig) *UserAgent { ua := &UserAgent{ config: config, iss: sync.Map{}, InviteStateHandler: nil, RegisterStateHandler: nil, - log: utils.NewLogrusLogger(log.DebugLevel, "UserAgent", nil), + log: utils.NewLogrusLogger(utils.DefaultLogLevel, "UserAgent", nil), } stack := config.SipStack stack.OnRequest(sip.INVITE, ua.handleInvite) @@ -234,18 +234,29 @@ func (ua *UserAgent) handleBye(request sip.Request, tx sip.ServerTransaction) { tx.Respond(response) callID, ok := request.CallID() - fromHeader, ok2 := request.From() - if ok && ok2 { - fromTag, _ := fromHeader.Params.Get("tag") - if v, found := ua.iss.Load(NewSessionKey(*callID, fromTag)); found { - is := v.(*session.Session) - ua.iss.Delete(NewSessionKey(*callID, fromTag)) + if ok { + if sess, sessKey := ua.findSessionByCallID(*callID); sess != nil { + ua.iss.Delete(sessKey) var transaction sip.Transaction = tx.(sip.Transaction) - ua.handleInviteState(is, &request, &response, session.Terminated, &transaction) + ua.handleInviteState(sess, &request, &response, session.Terminated, &transaction) } } } +func (ua *UserAgent) findSessionByCallID(callID sip.CallID) (*session.Session, SessionKey) { + var ret *session.Session = nil + var sessKey SessionKey + ua.iss.Range(func(key, value interface{}) bool { + if key.(SessionKey).CallID == callID { + ret = value.(*session.Session) + sessKey = key.(SessionKey) + return false + } + return true + }) + return ret, sessKey +} + func (ua *UserAgent) handleCancel(request sip.Request, tx sip.ServerTransaction) { ua.Log().Debugf("handleCancel: Request => %s, body => %s", request.Short(), request.Body()) diff --git a/pkg/utils/atomic.go b/pkg/utils/atomic.go new file mode 100644 index 0000000..de63651 --- /dev/null +++ b/pkg/utils/atomic.go @@ -0,0 +1,34 @@ +package utils + +import "sync/atomic" + +type AtomicBool int32 + +func (a *AtomicBool) Set(value bool) (swapped bool) { + if value { + return atomic.SwapInt32((*int32)(a), 1) == 0 + } + return atomic.SwapInt32((*int32)(a), 0) == 1 +} + +func (a *AtomicBool) Get() bool { + return atomic.LoadInt32((*int32)(a)) != 0 +} + +type AtomicUInt32 uint32 + +func (ai *AtomicUInt32) Set(value uint32) (result uint32) { + return atomic.SwapUint32((*uint32)(ai), value) +} + +func (ai *AtomicUInt32) Get() uint32 { + return atomic.LoadUint32((*uint32)(ai)) +} + +func (ai *AtomicUInt32) Incr() { + ai.Set(ai.Get() + 1) +} + +func (ai *AtomicUInt32) Decr() { + ai.Set(ai.Get() - 1) +} diff --git a/pkg/utils/log.go b/pkg/utils/log.go index d3f6cdb..3fd1eeb 100644 --- a/pkg/utils/log.go +++ b/pkg/utils/log.go @@ -34,7 +34,8 @@ func (ml *MyLogger) Level() string { } var ( - loggers map[string]*MyLogger + loggers map[string]*MyLogger + DefaultLogLevel = log.DebugLevel ) func init() { @@ -59,14 +60,14 @@ func NewLogrusLogger(level log.Level, prefix string, fields log.Fields) log.Logg Logger: logger, level: level, } - logger.SetLevel(level) + logger.SetLevel(uint32(level)) return logger.WithPrefix(prefix) } func SetLogLevel(prefix string, level log.Level) error { if logger, found := loggers[prefix]; found { logger.level = level - logger.Logger.SetLevel(level) + logger.Logger.SetLevel(uint32(level)) return nil } return fmt.Errorf("logger [%v] not found", prefix)