4 Commits

Author SHA1 Message Date
Smile Rex
b7d33889fd fix 2026-01-22 10:01:15 +03:00
Smile Rex
12e789b22e add log 2026-01-22 09:31:16 +03:00
Smile Rex
ca06471d9e dsd 2026-01-21 19:14:25 +03:00
Smile Rex
9033264a15 fix 2026-01-21 19:14:05 +03:00
8 changed files with 90 additions and 25 deletions

6
auth.go Normal file
View File

@@ -0,0 +1,6 @@
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=

52
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,22 +13,46 @@ 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, _ := upgrader.Upgrade(w, r, nil) conn, err := upgrader.Upgrade(w, r, nil)
if err != nil {
id := uuid.New().String() return
player := &Player{
ID: id,
Conn: conn,
X: 180,
Y: 320,
} }
room.Players[id] = player 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)
player := &Player{
ID: userID,
Username: username,
Conn: conn,
X: 180,
Y: 320,
}
room.Players[player.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": id, "id": fmt.Sprintf("%d", player.ID),
}, },
}) })
@@ -36,11 +60,15 @@ 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,9 @@ import (
) )
type Player struct { type Player struct {
ID string ID int64
Conn *websocket.Conn Username string
Conn *websocket.Conn
X, Y float64 X, Y float64
DX, DY float64 DX, DY float64

View File

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

28
room.go
View File

@@ -1,52 +1,64 @@
package main package main
import ( import (
"sync"
"time" "time"
) )
type Room struct { type Room struct {
Players map[string]*Player Players map[int64]*Player
Input chan InputMessage Input chan InputMessage
mu sync.Mutex
} }
func NewRoom() *Room { func NewRoom() *Room {
return &Room{ return &Room{
Players: make(map[string]*Player), Players: make(map[int64]*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() {
state := map[string]map[string]float64{} r.mu.Lock()
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]float64{ state[id] = map[string]any{
"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{
@@ -54,13 +66,15 @@ 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()

16
telegram.go Normal file
View File

@@ -0,0 +1,16 @@
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)
}