working lobby-ish
This commit is contained in:
parent
e90b9c26b1
commit
e5df746e7f
|
@ -25,7 +25,7 @@ func main() {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Panic("Bro your input is no good...")
|
log.Panic("Bro your input is no good...")
|
||||||
}
|
}
|
||||||
username := string(buf[:n])
|
username := string(buf[:n-1])
|
||||||
|
|
||||||
conn, err := netwrk.ConnectToLobby(username)
|
conn, err := netwrk.ConnectToLobby(username)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -43,6 +43,9 @@ func main() {
|
||||||
}
|
}
|
||||||
|
|
||||||
userMessage, err := client.HandleUserInput(buf[:n])
|
userMessage, err := client.HandleUserInput(buf[:n])
|
||||||
|
if err == io.EOF {
|
||||||
|
exit <- true
|
||||||
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
fmt.Println(err)
|
fmt.Println(err)
|
||||||
continue
|
continue
|
||||||
|
@ -88,7 +91,7 @@ func main() {
|
||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
log.Panic("Server disconnected sorry...")
|
log.Panic("Server disconnected sorry...")
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
log.Panic("Error reading from server connection...")
|
log.Panic("Error reading from server connection...", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
message := &netwrk.LobbyMessage{}
|
message := &netwrk.LobbyMessage{}
|
||||||
|
|
|
@ -10,7 +10,7 @@ var exit chan bool
|
||||||
func main() {
|
func main() {
|
||||||
fmt.Println("Starting sshpong server!")
|
fmt.Println("Starting sshpong server!")
|
||||||
|
|
||||||
netwrk.Listen()
|
netwrk.LobbyListen()
|
||||||
|
|
||||||
_ = <-exit
|
_ = <-exit
|
||||||
}
|
}
|
||||||
|
|
|
@ -8,9 +8,14 @@ import (
|
||||||
"strings"
|
"strings"
|
||||||
)
|
)
|
||||||
|
|
||||||
|
var help = fmt.Errorf("use invite <player name> to invite a player\nchat or / to send a message to the lobby\nq or quit to leave the game")
|
||||||
|
|
||||||
func HandleUserInput(buf []byte) (*netwrk.LobbyMessage, error) {
|
func HandleUserInput(buf []byte) (*netwrk.LobbyMessage, error) {
|
||||||
input := string(buf)
|
input := string(buf)
|
||||||
args := strings.Fields(input)
|
args := strings.Fields(input)
|
||||||
|
if len(args) == 0 {
|
||||||
|
return nil, help
|
||||||
|
}
|
||||||
switch args[0] {
|
switch args[0] {
|
||||||
case "invite":
|
case "invite":
|
||||||
if args[1] != "" {
|
if args[1] != "" {
|
||||||
|
@ -40,11 +45,11 @@ func HandleUserInput(buf []byte) (*netwrk.LobbyMessage, error) {
|
||||||
case "q":
|
case "q":
|
||||||
return nil, io.EOF
|
return nil, io.EOF
|
||||||
case "help":
|
case "help":
|
||||||
return nil, fmt.Errorf("use invite <player name> to invite a player\nchat or / to send a message to the lobby\nq or quit to leave the game")
|
return nil, help
|
||||||
case "h":
|
case "h":
|
||||||
return nil, fmt.Errorf("use invite <player name> to invite a player\nchat or / to send a message to the lobby\nq or quit to leave the game")
|
return nil, help
|
||||||
default:
|
default:
|
||||||
return nil, fmt.Errorf("use invite <player name> to invite a player\nchat or / to send a message to the lobby\nq or quit to leave the game")
|
return nil, help
|
||||||
}
|
}
|
||||||
return nil, nil
|
return nil, nil
|
||||||
}
|
}
|
||||||
|
|
|
@ -15,12 +15,12 @@ func handleGameConnection(conn net.Conn) {
|
||||||
log.Printf("Error reading game ID on connection", err)
|
log.Printf("Error reading game ID on connection", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
gameID := string(messageBytes[:n])
|
_ = string(messageBytes[:n])
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Printf("Game id was not a string?", err)
|
log.Printf("Game id was not a string?", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
clientChan := make(chan GameMessage)
|
_ = make(chan GameMessage)
|
||||||
|
|
||||||
n, err = conn.Read(messageBytes)
|
n, err = conn.Read(messageBytes)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
@ -28,16 +28,6 @@ func handleGameConnection(conn net.Conn) {
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
gameClients, ok := gameChans.games[gameID]
|
|
||||||
if !ok {
|
|
||||||
newGameClients := GameClients{
|
|
||||||
client1: clientChan,
|
|
||||||
client2: nil,
|
|
||||||
}
|
|
||||||
gameChans.games[gameID] = newGameClients
|
|
||||||
} else {
|
|
||||||
gameClients.client2 = clientChan
|
|
||||||
}
|
|
||||||
}
|
}
|
||||||
|
|
||||||
func handleGameMessage(conn net.Conn, message *GameMessage) error {
|
func handleGameMessage(conn net.Conn, message *GameMessage) error {
|
||||||
|
|
|
@ -9,8 +9,6 @@ import (
|
||||||
)
|
)
|
||||||
|
|
||||||
func handleLobbyConnection(conn net.Conn) {
|
func handleLobbyConnection(conn net.Conn) {
|
||||||
defer conn.Close()
|
|
||||||
|
|
||||||
messageBytes := make([]byte, 4096)
|
messageBytes := make([]byte, 4096)
|
||||||
|
|
||||||
ingress := make(chan *LobbyMessage)
|
ingress := make(chan *LobbyMessage)
|
||||||
|
@ -21,9 +19,11 @@ func handleLobbyConnection(conn net.Conn) {
|
||||||
for {
|
for {
|
||||||
n, err := conn.Read(messageBytes)
|
n, err := conn.Read(messageBytes)
|
||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
|
conn.Close()
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
|
conn.Close()
|
||||||
log.Printf("Error reading message %v", err)
|
log.Printf("Error reading message %v", err)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
@ -46,10 +46,15 @@ func handleLobbyConnection(conn net.Conn) {
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error marshalling message to send to user...", err)
|
log.Println("Error marshalling message to send to user...", err)
|
||||||
}
|
}
|
||||||
|
log.Println("sending message to user!")
|
||||||
_, err = conn.Write(bytes)
|
_, err = conn.Write(bytes)
|
||||||
if err == io.EOF {
|
if err == io.EOF {
|
||||||
|
conn.Close()
|
||||||
log.Println("User has disconnected", err)
|
log.Println("User has disconnected", err)
|
||||||
ingress <- &LobbyMessage{Type: "disconnect"}
|
ingress <- &LobbyMessage{
|
||||||
|
Type: "disconnect",
|
||||||
|
Content: msg.PlayerId,
|
||||||
|
}
|
||||||
}
|
}
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error writing to user...", err)
|
log.Println("Error writing to user...", err)
|
||||||
|
@ -61,7 +66,7 @@ func handleLobbyConnection(conn net.Conn) {
|
||||||
go func() {
|
go func() {
|
||||||
for {
|
for {
|
||||||
msg := <-ingress
|
msg := <-ingress
|
||||||
serverMsg, err := handleClientLobbyMessage(msg)
|
serverMsg, err := handleClientLobbyMessage(msg, conn)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error handling client lobby message...", err)
|
log.Println("Error handling client lobby message...", err)
|
||||||
}
|
}
|
||||||
|
@ -73,18 +78,29 @@ func handleLobbyConnection(conn net.Conn) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// Returns a bool of whether the player has disconnected from the lobby and an error
|
// Returns a bool of whether the player has disconnected from the lobby and an error
|
||||||
func handleClientLobbyMessage(message *LobbyMessage) (*LobbyMessage, error) {
|
func handleClientLobbyMessage(message *LobbyMessage, conn net.Conn) (*LobbyMessage, error) {
|
||||||
switch message.Type {
|
switch message.Type {
|
||||||
|
|
||||||
|
// Handle an name/login message from a player
|
||||||
|
// Store the new player in the lobbyMembers
|
||||||
|
// Send a connection message for each of the lobbyMembers to the new player
|
||||||
|
// Send a connection message to all members in the lobby
|
||||||
case "name":
|
case "name":
|
||||||
|
log.Println("Got a name message!", message)
|
||||||
_, ok := lobbyMembers.Load(message.Content)
|
_, ok := lobbyMembers.Load(message.Content)
|
||||||
if ok {
|
if ok {
|
||||||
return &LobbyMessage{Type: "name_error", Content: "Sorry, that name is already taken, please try a different name"}, nil
|
return &LobbyMessage{Type: "name_error", Content: "Sorry, that name is already taken, please try a different name"}, nil
|
||||||
}
|
}
|
||||||
username := message.Content
|
username := message.Content
|
||||||
|
|
||||||
|
lobbyMembers.Store(username, Client{Username: username, Conn: conn})
|
||||||
|
|
||||||
|
log.Println("Storing new user!")
|
||||||
// Send all client messages
|
// Send all client messages
|
||||||
lobbyMembers.Range(func(lobbyUsername string, client Client) bool {
|
lobbyMembers.Range(func(lobbyUsername any, client any) bool {
|
||||||
externalMessageChan <- ExternalMessage{Target: username, Message: &LobbyMessage{Type: "connect", Content: lobbyUsername}}
|
log.Println("ranging over users!")
|
||||||
|
usernameString, _ := lobbyUsername.(string)
|
||||||
|
externalMessageChan <- ExternalMessage{Target: username, Message: &LobbyMessage{Type: "connect", Content: usernameString}}
|
||||||
return true
|
return true
|
||||||
})
|
})
|
||||||
|
|
||||||
|
@ -93,44 +109,74 @@ func handleClientLobbyMessage(message *LobbyMessage) (*LobbyMessage, error) {
|
||||||
broadcastToLobby(&LobbyMessage{PlayerId: "", Type: "connect", Content: username})
|
broadcastToLobby(&LobbyMessage{PlayerId: "", Type: "connect", Content: username})
|
||||||
|
|
||||||
return &LobbyMessage{PlayerId: username, Type: "name", Content: username}, nil
|
return &LobbyMessage{PlayerId: username, Type: "name", Content: username}, nil
|
||||||
|
|
||||||
|
// Handle an invite message by sending a message to the target player
|
||||||
|
// Send an invite message to the invitee: message.Content
|
||||||
|
// Send an ack message to the inviter: message.PlayerId
|
||||||
case "invite":
|
case "invite":
|
||||||
log.Println("Got invite for player:", message.Content)
|
externalMessageChan <- ExternalMessage{
|
||||||
invitee, ok := lobbyMembers[message.Content]
|
Target: message.Content,
|
||||||
if !ok {
|
Message: message,
|
||||||
return &LobbyMessage{Type: "text", Content: "Sorry, that player is not available..."}, nil
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return &LobbyMessage{Type: "invite", Content: message.PlayerId}, nil
|
return &LobbyMessage{Type: "invite", Content: message.PlayerId}, nil
|
||||||
|
|
||||||
|
// Handle a accept message from a player that was invited
|
||||||
|
// Send an ack message back to the player: message.PlayerId
|
||||||
|
// Send an invite ack message back to the inviter: message.Content
|
||||||
case "accept_game":
|
case "accept_game":
|
||||||
player := lobbyMembers[message.Content]
|
externalMessageChan <- ExternalMessage{
|
||||||
|
Target: message.Content,
|
||||||
|
Message: &LobbyMessage{Type: "accept", Content: ""},
|
||||||
|
}
|
||||||
|
|
||||||
return &LobbyMessage{Type: "accept", Content: ""}, nil
|
return &LobbyMessage{Type: "accept", Content: ""}, nil
|
||||||
|
|
||||||
|
// Handle a chat message from a player with PlayerId
|
||||||
case "chat":
|
case "chat":
|
||||||
broadcastToLobby(&LobbyMessage{PlayerId: message.PlayerId, Type: "text", Content: message.Content})
|
broadcastToLobby(&LobbyMessage{PlayerId: message.PlayerId, Type: "text", Content: message.Content})
|
||||||
return nil, nil
|
return nil, nil
|
||||||
|
|
||||||
|
// Handle a decline_game message from a player that was invited
|
||||||
|
// Send an ack message back to the invitee: message.PlayerId
|
||||||
|
// Send an ack message to the inviter: message.Content
|
||||||
case "decline_game":
|
case "decline_game":
|
||||||
inviter := lobbyMembers[message.Content]
|
externalMessageChan <- ExternalMessage{
|
||||||
|
Target: message.Content,
|
||||||
|
Message: &LobbyMessage{Type: "decline", Content: ""},
|
||||||
|
}
|
||||||
|
|
||||||
return &LobbyMessage{Type: "decline_game", Content: message.PlayerId}, nil
|
return &LobbyMessage{Type: "decline_game", Content: message.PlayerId}, nil
|
||||||
|
|
||||||
|
// Handle a quit message from a player that was connected
|
||||||
|
// broadcast the player quit to the lobby
|
||||||
case "quit":
|
case "quit":
|
||||||
delete(lobbyMembers, message.PlayerId)
|
lobbyMembers.Delete(message.PlayerId)
|
||||||
broadcastToLobby(&LobbyMessage{Type: "disconnect", Content: message.PlayerId})
|
broadcastToLobby(&LobbyMessage{Type: "disconnect", Content: message.PlayerId})
|
||||||
return nil, nil
|
return nil, nil
|
||||||
|
|
||||||
|
// Ping and pong
|
||||||
case "ping":
|
case "ping":
|
||||||
return &LobbyMessage{Type: "pong", Content: "pong"}, nil
|
return &LobbyMessage{Type: "pong", Content: "pong"}, nil
|
||||||
|
|
||||||
|
// Ping and pong
|
||||||
default:
|
default:
|
||||||
return &LobbyMessage{Type: "pong", Content: "pong"}, nil
|
return &LobbyMessage{Type: "pong", Content: "pong"}, nil
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
func broadcastToLobby(message *LobbyMessage) {
|
func broadcastToLobby(message *LobbyMessage) {
|
||||||
for _, player := range lobbyMembers {
|
lobbyMembers.Range(func(playerId, player interface{}) bool {
|
||||||
bytes, err := proto.Marshal(message)
|
bytes, err := proto.Marshal(message)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error marshalling broadcast message", err)
|
log.Println("Error marshalling broadcast message", err)
|
||||||
}
|
}
|
||||||
_, err = player.Conn.Write(bytes)
|
|
||||||
|
client := player.(Client)
|
||||||
|
_, err = client.Conn.Write(bytes)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
log.Println("Error broadcasting to clients...", err)
|
log.Println("Error broadcasting to clients...", err)
|
||||||
}
|
}
|
||||||
}
|
return true
|
||||||
|
})
|
||||||
}
|
}
|
||||||
|
|
|
@ -4,6 +4,8 @@ import (
|
||||||
"log"
|
"log"
|
||||||
"net"
|
"net"
|
||||||
sync "sync"
|
sync "sync"
|
||||||
|
|
||||||
|
"google.golang.org/protobuf/proto"
|
||||||
)
|
)
|
||||||
|
|
||||||
type Client struct {
|
type Client struct {
|
||||||
|
@ -32,6 +34,19 @@ func init() {
|
||||||
externalMessageChan = make(chan ExternalMessage)
|
externalMessageChan = make(chan ExternalMessage)
|
||||||
|
|
||||||
lobbyMembers = sync.Map{}
|
lobbyMembers = sync.Map{}
|
||||||
|
|
||||||
|
go func() {
|
||||||
|
for {
|
||||||
|
msg := <-externalMessageChan
|
||||||
|
player, ok := lobbyMembers.Load(msg.Target)
|
||||||
|
if !ok {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
client, _ := player.(Client)
|
||||||
|
bytes, _ := proto.Marshal(msg.Message)
|
||||||
|
client.Conn.Write(bytes)
|
||||||
|
}
|
||||||
|
}()
|
||||||
}
|
}
|
||||||
|
|
||||||
// Starts listening on port 12345 for TCP connections
|
// Starts listening on port 12345 for TCP connections
|
||||||
|
|
Loading…
Reference in New Issue
Block a user