1 Commits

Author SHA1 Message Date
dde276f22f Merge pull request 'new server' (#2) from feauture into main
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m4s
Reviewed-on: #2
2026-01-21 18:49:43 +03:00
8 changed files with 23 additions and 88 deletions

View File

@@ -1,6 +0,0 @@
package main
type AuthMessage struct {
Type string `json:"type"`
InitData string `json:"initData"`
}

2
go.mod
View File

@@ -3,6 +3,6 @@ module server
go 1.25.0 go 1.25.0
require ( require (
github.com/google/uuid v1.6.0
github.com/gorilla/websocket v1.5.3 github.com/gorilla/websocket v1.5.3
github.com/telegram-mini-apps/init-data-golang v1.5.0
) )

4
go.sum
View File

@@ -1,4 +1,4 @@
github.com/google/uuid v1.6.0 h1:NIvaJDMOsjHA8n1jAhLSgzrAzy1Hgr+hNrb57e+94F0=
github.com/google/uuid v1.6.0/go.mod h1:TIyPZe4MgqvfeYDBFedMoGGpEw/LqOeaOT+nhxU+yHo=
github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg= github.com/gorilla/websocket v1.5.3 h1:saDtZ6Pbx/0u+bgYQ3q96pZgCzfhKXGPqt7kZ72aNNg=
github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE= github.com/gorilla/websocket v1.5.3/go.mod h1:YR8l580nyteQvAITg2hZ9XVh4b55+EU/adAjf1fMHhE=
github.com/telegram-mini-apps/init-data-golang v1.5.0 h1:rtpsmQ/nihkicPvnrdRXmHHtTnPvG1FmxMRZJwMKPz0=
github.com/telegram-mini-apps/init-data-golang v1.5.0/go.mod h1:GG4HnRx9ocjD4MjjzOw7gf9Ptm0NvFbDr5xqnfFOYuY=

42
main.go
View File

@@ -1,10 +1,10 @@
package main package main
import ( import (
"fmt"
"log" "log"
"net/http" "net/http"
"github.com/google/uuid"
"github.com/gorilla/websocket" "github.com/gorilla/websocket"
) )
@@ -13,46 +13,22 @@ var upgrader = websocket.Upgrader{
} }
func ServeWS(room *Room, w http.ResponseWriter, r *http.Request) { func ServeWS(room *Room, w http.ResponseWriter, r *http.Request) {
conn, err := upgrader.Upgrade(w, r, nil) conn, _ := upgrader.Upgrade(w, r, nil)
if err != nil {
return
}
var auth AuthMessage
if err := conn.ReadJSON(&auth); err != nil || auth.Type != "auth" {
conn.Close()
return
}
data, err := VerifyTelegramInitData(auth.InitData, "7697757472:AAESD9HfkWwbIZe-HXR7IazUShr69hZTLmE")
if err != nil {
conn.Close()
return
}
userID := data.User.ID
username := data.User.Username
if username == "" {
username = data.User.FirstName
}
log.Printf("Player %d connected!", userID)
id := uuid.New().String()
player := &Player{ player := &Player{
ID: userID, ID: id,
Username: username,
Conn: conn, Conn: conn,
X: 180, X: 180,
Y: 320, Y: 320,
} }
room.Players[player.ID] = player room.Players[id] = player
player.Conn.WriteJSON(map[string]any{ player.Conn.WriteJSON(map[string]any{
"type": "init", "type": "init",
"payload": map[string]string{ "payload": map[string]string{
"id": fmt.Sprintf("%d", player.ID), "id": id,
}, },
}) })
@@ -60,15 +36,11 @@ func ServeWS(room *Room, w http.ResponseWriter, r *http.Request) {
} }
func readLoop(room *Room, player *Player) { func readLoop(room *Room, player *Player) {
defer func() {
delete(room.Players, player.ID)
player.Conn.Close()
}()
for { for {
var msg InputMessage var msg InputMessage
err := player.Conn.ReadJSON(&msg) err := player.Conn.ReadJSON(&msg)
if err != nil { if err != nil {
delete(room.Players, player.ID)
return return
} }

View File

@@ -5,8 +5,7 @@ import (
) )
type Player struct { type Player struct {
ID int64 ID string
Username string
Conn *websocket.Conn Conn *websocket.Conn
X, Y float64 X, Y float64

View File

@@ -1,7 +1,7 @@
package main package main
type InputMessage struct { type InputMessage struct {
PlayerID int64 PlayerID string
DX float64 `json:"dx"` DX float64 `json:"dx"`
DY float64 `json:"dy"` DY float64 `json:"dy"`
} }

24
room.go
View File

@@ -1,64 +1,52 @@
package main package main
import ( import (
"sync"
"time" "time"
) )
type Room struct { type Room struct {
Players map[int64]*Player Players map[string]*Player
Input chan InputMessage Input chan InputMessage
mu sync.Mutex
} }
func NewRoom() *Room { func NewRoom() *Room {
return &Room{ return &Room{
Players: make(map[int64]*Player), Players: make(map[string]*Player),
Input: make(chan InputMessage, 128), Input: make(chan InputMessage, 128),
} }
} }
func (r *Room) update() { func (r *Room) update() {
// 1⃣ обрабатываем input
for { for {
select { select {
case input := <-r.Input: case input := <-r.Input:
r.mu.Lock()
p := r.Players[input.PlayerID] p := r.Players[input.PlayerID]
if p != nil { if p != nil {
p.DX = input.DX p.DX = input.DX
p.DY = input.DY p.DY = input.DY
} }
r.mu.Unlock()
default: default:
goto DONE goto DONE
} }
} }
DONE: DONE:
// 2⃣ двигаем игроков
r.mu.Lock()
for _, p := range r.Players { for _, p := range r.Players {
p.X += p.DX * 4 p.X += p.DX * 4
p.Y += p.DY * 4 p.Y += p.DY * 4
} }
r.mu.Unlock()
} }
func (r *Room) broadcast() { func (r *Room) broadcast() {
r.mu.Lock() state := map[string]map[string]float64{}
state := make(map[int64]map[string]any, len(r.Players))
for id, p := range r.Players { for id, p := range r.Players {
state[id] = map[string]any{ state[id] = map[string]float64{
"x": p.X, "x": p.X,
"y": p.Y, "y": p.Y,
"name": p.Username,
} }
} }
r.mu.Unlock()
msg := StateMessage{ msg := StateMessage{
Type: "state", Type: "state",
Payload: map[string]any{ Payload: map[string]any{
@@ -66,15 +54,13 @@ func (r *Room) broadcast() {
}, },
} }
// отправляем БЕЗ mutex — важно
for _, p := range r.Players { for _, p := range r.Players {
_ = p.Conn.WriteJSON(msg) p.Conn.WriteJSON(msg)
} }
} }
func (r *Room) Run() { func (r *Room) Run() {
ticker := time.NewTicker(time.Second / 30) ticker := time.NewTicker(time.Second / 30)
defer ticker.Stop()
for range ticker.C { for range ticker.C {
r.update() r.update()

View File

@@ -1,16 +0,0 @@
package main
import (
"time"
initdata "github.com/telegram-mini-apps/init-data-golang"
)
func VerifyTelegramInitData(initData, botToken string) (initdata.InitData, error) {
expIn := 24 * time.Hour
err := initdata.Validate(initData, botToken, expIn)
if err != nil {
return initdata.InitData{}, err
}
return initdata.Parse(initData)
}