2021-01-24 05:00:26 -08:00
|
|
|
package main
|
2021-01-05 11:34:35 -08:00
|
|
|
|
|
|
|
import (
|
2021-01-09 03:26:30 -08:00
|
|
|
"errors"
|
|
|
|
"net"
|
|
|
|
"sync"
|
2021-01-18 13:08:22 -08:00
|
|
|
|
2021-02-23 11:09:00 -08:00
|
|
|
"github.com/anon55555/mt"
|
2021-01-18 13:08:22 -08:00
|
|
|
"github.com/anon55555/mt/rudp"
|
2021-01-05 11:34:35 -08:00
|
|
|
)
|
|
|
|
|
2021-01-06 05:42:55 -08:00
|
|
|
var ErrPlayerLimitReached = errors.New("player limit reached")
|
|
|
|
|
2021-01-05 11:34:35 -08:00
|
|
|
type Listener struct {
|
2021-01-18 13:08:22 -08:00
|
|
|
*rudp.Listener
|
2021-01-05 11:34:35 -08:00
|
|
|
}
|
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
var peerMu sync.RWMutex
|
|
|
|
var peers map[*Peer]struct{}
|
2021-01-06 14:39:54 -08:00
|
|
|
|
2021-01-05 11:34:35 -08:00
|
|
|
func Listen(conn net.PacketConn) *Listener {
|
2021-01-19 08:31:00 -08:00
|
|
|
return &Listener{
|
|
|
|
Listener: rudp.Listen(conn),
|
|
|
|
}
|
2021-01-05 11:34:35 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
// Accept waits for and returns a connecting Peer
|
|
|
|
// You should keep calling this until it returns ErrClosed
|
|
|
|
// so it doesn't leak a goroutine
|
|
|
|
func (l *Listener) Accept() (*Peer, error) {
|
2021-01-19 08:15:28 -08:00
|
|
|
rp, err := l.Listener.Accept()
|
2021-01-18 13:08:22 -08:00
|
|
|
if err != nil {
|
2021-01-05 11:34:35 -08:00
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
2021-01-19 08:15:28 -08:00
|
|
|
clt := &Peer{Peer: rp}
|
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
peerMu.Lock()
|
|
|
|
peers[clt] = struct{}{}
|
|
|
|
peerMu.Unlock()
|
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
go func() {
|
|
|
|
<-clt.Disco()
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
peerMu.Lock()
|
|
|
|
delete(peers, clt)
|
|
|
|
peerMu.Unlock()
|
2021-01-18 13:08:22 -08:00
|
|
|
}()
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
clt.aoIDs = make(map[uint16]bool)
|
2021-01-24 05:18:29 -08:00
|
|
|
clt.modChs = make(map[string]bool)
|
2021-02-13 10:52:41 -08:00
|
|
|
clt.huds = make(map[uint32]bool)
|
2021-02-21 03:04:54 -08:00
|
|
|
clt.sounds = make(map[int32]bool)
|
2021-02-23 11:09:00 -08:00
|
|
|
clt.inv = &mt.Inv{}
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-03-09 13:23:41 -08:00
|
|
|
maxPeers, ok := ConfKey("player_limit").(int)
|
2021-01-18 13:08:22 -08:00
|
|
|
if !ok {
|
2021-01-19 08:56:39 -08:00
|
|
|
maxPeers = int(^uint(0) >> 1)
|
2021-01-18 13:08:22 -08:00
|
|
|
}
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
if PeerCount() >= maxPeers {
|
2021-01-18 13:08:22 -08:00
|
|
|
data := []byte{
|
2021-01-19 11:37:35 -08:00
|
|
|
0, ToClientAccessDenied,
|
|
|
|
AccessDeniedTooManyUsers, 0, 0, 0, 0,
|
2021-01-06 05:42:55 -08:00
|
|
|
}
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
_, err := clt.Send(rudp.Pkt{Data: data})
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
2021-01-06 05:42:55 -08:00
|
|
|
}
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
return nil, ErrPlayerLimitReached
|
2021-01-05 11:34:35 -08:00
|
|
|
}
|
2021-03-09 09:06:05 -08:00
|
|
|
|
|
|
|
connectedPeersMu.Lock()
|
2021-01-19 08:56:39 -08:00
|
|
|
connectedPeers++
|
2021-03-09 09:06:05 -08:00
|
|
|
connectedPeersMu.Unlock()
|
2021-01-09 03:26:30 -08:00
|
|
|
|
2021-01-18 13:08:22 -08:00
|
|
|
return clt, nil
|
2021-01-05 11:34:35 -08:00
|
|
|
}
|
2021-01-06 14:39:54 -08:00
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
// PeerByUsername returns the Peer that is using the specified name
|
|
|
|
// for authentication
|
|
|
|
func PeerByUsername(name string) *Peer {
|
|
|
|
peerMu.RLock()
|
|
|
|
defer peerMu.RUnlock()
|
2021-01-18 13:08:22 -08:00
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
for p := range peers {
|
|
|
|
if p.Username() == name {
|
|
|
|
return p
|
2021-01-14 10:06:40 -08:00
|
|
|
}
|
|
|
|
}
|
2021-01-18 13:08:22 -08:00
|
|
|
|
2021-01-14 10:06:40 -08:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
// Peers returns an array containing all connected client Peers
|
|
|
|
func Peers() []*Peer {
|
|
|
|
peerMu.RLock()
|
|
|
|
defer peerMu.RUnlock()
|
2021-01-14 10:06:40 -08:00
|
|
|
|
|
|
|
var r []*Peer
|
2021-03-09 09:06:05 -08:00
|
|
|
for p := range peers {
|
2021-01-18 13:08:22 -08:00
|
|
|
r = append(r, p)
|
2021-01-14 10:06:40 -08:00
|
|
|
}
|
|
|
|
return r
|
2021-01-06 14:39:54 -08:00
|
|
|
}
|
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
func init() {
|
|
|
|
peerMu.Lock()
|
|
|
|
defer peerMu.Unlock()
|
2021-01-06 14:39:54 -08:00
|
|
|
|
2021-03-09 09:06:05 -08:00
|
|
|
peers = make(map[*Peer]struct{})
|
2021-01-06 14:39:54 -08:00
|
|
|
}
|