18 Commits

Author SHA1 Message Date
Smile Rex
8ae941df3b fix
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m15s
2026-01-22 17:45:11 +03:00
Smile Rex
73cb832728 add logs
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m11s
2026-01-22 17:31:55 +03:00
Smile Rex
4e04de5581 new arch
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m11s
2026-01-22 16:34:47 +03:00
68ba9bb2e2 Merge pull request 'fix' (#5) from feauture into main
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m11s
Reviewed-on: #5
2026-01-22 10:01:36 +03:00
Smile Rex
b7d33889fd fix 2026-01-22 10:01:15 +03:00
2fd959255d Merge pull request 'add log' (#4) from feauture into main
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m17s
Reviewed-on: #4
2026-01-22 09:32:09 +03:00
Smile Rex
12e789b22e add log 2026-01-22 09:31:16 +03:00
49d6fb9815 Merge pull request 'feauture' (#3) from feauture into main
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m9s
Reviewed-on: #3
2026-01-21 19:15:19 +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
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
Smile Rex
280d9801d6 new server 2026-01-21 18:49:12 +03:00
Smile Rex
cb586cd5ce add hsh responce
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m4s
2026-01-21 14:50:28 +03:00
Smile Rex
1bb1e3bfd4 fix
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 3m25s
2026-01-21 13:10:15 +03:00
Smile Rex
9cd779fbea cors fix
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m9s
2026-01-21 12:36:25 +03:00
Smile Rex
3c989c33f8 new server
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m19s
2026-01-17 16:38:45 +03:00
Smile Rex
1c60264d1d add readme
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m8s
2026-01-15 11:55:05 +03:00
61b7754c5c Merge pull request 'hub_feauture' (#1) from hub_feauture into main
All checks were successful
Create and publish a Docker image 🚀 / build-and-push-image (push) Successful in 1m8s
Reviewed-on: #1
2026-01-14 23:11:26 +03:00
19 changed files with 240 additions and 379 deletions

View File

@@ -1,55 +0,0 @@
package controllers
import (
"log"
"server/models"
"github.com/gorilla/websocket"
)
func (h *Hub) readLoop(conn *websocket.Conn) {
handMessage := models.Message{Type: MSG_WELCOME}
_ = conn.WriteMessage(websocket.BinaryMessage, handMessage.Encode())
var player *models.Player
defer func() {
if player != nil {
h.removePlayer(player.ID)
log.Println("Player left:", player.ID)
}
conn.Close()
}()
for {
_, bytes, err := conn.ReadMessage()
if err != nil {
return
}
msg, err := models.Decode(bytes)
if err != nil {
log.Println(err)
continue
}
switch msg.Type {
case MSG_WELCOME:
if player != nil {
continue
}
player = h.handShake(msg.Payload, conn)
log.Println("Player joined:", player.ID)
case MSG_INPUT:
if player == nil {
continue
}
reader := models.NewReader(msg.Payload)
x := reader.ReadF32()
y := reader.ReadF32()
h.updatePosition(x, y, player)
}
}
}

View File

@@ -1,26 +0,0 @@
package controllers
import (
"log"
"net/http"
"server/models"
"sync"
)
type Hub struct {
Players map[uint32]*models.Player
Mu sync.RWMutex
}
func NewHub() *Hub {
return &Hub{
Players: make(map[uint32]*models.Player),
}
}
func (h *Hub) Start() {
go h.updateWorld()
http.HandleFunc("/ws", h.ws)
log.Println("Server listen port 8080")
log.Fatal(http.ListenAndServe(":8080", nil))
}

View File

@@ -1,43 +0,0 @@
package controllers
import (
"server/models"
"github.com/gorilla/websocket"
)
func (h *Hub) handShake(msg []byte, conn *websocket.Conn) *models.Player {
reader := models.NewReader(msg)
newID := reader.ReadU32()
name := reader.ReadString()
player := &models.Player{
Entity: models.Entity{
Type: models.EntityPlayer,
ID: newID,
},
Name: name,
Conn: conn,
}
h.Mu.Lock()
h.Players[player.ID] = player
h.Mu.Unlock()
return player
}
func (h *Hub) updatePosition(x, y float32, player *models.Player) {
h.Mu.Lock()
defer h.Mu.Unlock()
player.X = x
player.Y = y
}
func (h *Hub) removePlayer(id uint32) {
h.Mu.Lock()
defer h.Mu.Unlock()
delete(h.Players, id)
}

View File

@@ -1,7 +0,0 @@
package controllers
const (
MSG_WELCOME = 0
MSG_INPUT = 1
MSG_SNAPSHOT = 2
)

View File

@@ -1,46 +0,0 @@
package controllers
import (
"server/models"
"time"
"github.com/gorilla/websocket"
)
func (h *Hub) broadcastSnapshot() {
h.Mu.RLock()
defer h.Mu.RUnlock()
w := models.Writer{}
w.WriteU16(uint16(len(h.Players)))
for _, p := range h.Players {
w.WriteU32(p.ID)
w.WriteU8(uint8(models.EntityPlayer))
w.WriteF32(p.X)
w.WriteF32(p.Y)
w.WriteF32(p.Z)
w.WriteF32(p.Yaw)
}
msg := models.Message{
Type: MSG_SNAPSHOT,
Version: 1,
Payload: w.Bytes(),
}
for _, p := range h.Players {
if p.Conn != nil {
_ = p.Conn.WriteMessage(websocket.BinaryMessage, msg.Encode())
}
}
}
func (h *Hub) updateWorld() {
ticker := time.NewTicker(50 * time.Millisecond)
defer ticker.Stop()
for range ticker.C {
h.broadcastSnapshot()
}
}

View File

@@ -1,22 +0,0 @@
package controllers
import (
"log"
"net/http"
"github.com/gorilla/websocket"
)
func (h *Hub) ws(w http.ResponseWriter, r *http.Request) {
upgrader := websocket.Upgrader{
CheckOrigin: func(r *http.Request) bool { return true },
}
conn, err := upgrader.Upgrade(w, r, nil)
if err != nil {
log.Println(err)
return
}
go h.readLoop(conn)
}

5
go.mod
View File

@@ -2,4 +2,7 @@ module server
go 1.25.0 go 1.25.0
require github.com/gorilla/websocket v1.5.3 require (
github.com/gorilla/websocket v1.5.3
github.com/telegram-mini-apps/init-data-golang v1.5.0
)

2
go.sum
View File

@@ -1,2 +1,4 @@
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=

100
main.go
View File

@@ -1,10 +1,102 @@
package main package main
import ( import (
"server/controllers" "fmt"
"log"
"net/http"
"server/src/controllers"
"server/src/models"
"github.com/gorilla/websocket"
) )
func main() { var upgrader = websocket.Upgrader{
hub := controllers.NewHub() CheckOrigin: func(r *http.Request) bool { return true },
hub.Start() }
func ServeWS(room *controllers.Room, w http.ResponseWriter, r *http.Request) {
conn, err := upgrader.Upgrade(w, r, nil)
if err != nil {
log.Println("Failed to upgrade connection:", err)
return
}
var auth models.AuthMessage
if err := conn.ReadJSON(&auth); err != nil || auth.Type != "auth" {
log.Println("Invalid auth message. Connection closed!")
conn.Close()
return
}
data, err := controllers.VerifyTelegramInitData(auth.InitData, "7697757472:AAESD9HfkWwbIZe-HXR7IazUShr69hZTLmE")
if err != nil {
log.Println("initData is empty. Connection closed!")
conn.Close()
return
}
userID := data.User.ID
username := data.User.Username
if username == "" {
username = data.User.FirstName
}
log.Printf("Player %d connected!", userID)
player := &models.Player{
ID: userID,
Username: username,
Conn: conn,
X: 180,
Y: 320,
}
room.Players[player.ID] = player
player.Conn.WriteJSON(map[string]any{
"type": "init",
"payload": map[string]string{
"id": fmt.Sprintf("%d", player.ID),
},
})
go readLoop(room, player)
}
func readLoop(room *controllers.Room, player *models.Player) {
defer func() {
log.Printf("Player %d disconnected!", player.ID)
delete(room.Players, player.ID)
player.Conn.Close()
}()
for {
var msg models.StateMessage
err := player.Conn.ReadJSON(&msg)
if err != nil {
return
}
switch msg.Type {
case models.InputMsgType:
var input models.InputMessage = msg.Payload.(models.InputMessage)
player.X = input.DX
player.Y = input.DY
room.Input <- input
}
}
}
func main() {
room := controllers.NewRoom()
go room.Run()
http.HandleFunc("/ws", func(w http.ResponseWriter, r *http.Request) {
ServeWS(room, w, r)
})
log.Println("Server started on :8080")
log.Fatal(http.ListenAndServe(":8080", nil))
} }

View File

@@ -1,19 +0,0 @@
package models
type EntityType uint8
const (
EntityPlayer EntityType = 1
EntityNPC EntityType = 2
EntityBullet EntityType = 3
)
type Entity struct {
ID uint32
Type EntityType
X float32
Y float32
Z float32
Yaw float32
}

View File

@@ -1,48 +0,0 @@
package models
import (
"encoding/binary"
"errors"
)
type Message struct {
Type uint16
Version uint16
Payload []byte
}
func (m *Message) Encode() []byte {
payloadLen := len(m.Payload)
if payloadLen > 0xFFFF {
return nil
}
buf := make([]byte, 8+payloadLen)
binary.LittleEndian.PutUint16(buf[0:2], m.Type)
binary.LittleEndian.PutUint16(buf[2:4], m.Version)
binary.LittleEndian.PutUint16(buf[4:6], uint16(payloadLen))
copy(buf[8:], m.Payload)
return buf
}
func Decode(data []byte) (*Message, error) {
if len(data) < 8 {
return nil, errors.New("message too short")
}
payloadLen := binary.LittleEndian.Uint16(data[4:6])
if len(data) != int(8+payloadLen) {
return nil, errors.New("invalid payload length")
}
msg := &Message{
Type: binary.LittleEndian.Uint16(data[0:2]),
Version: binary.LittleEndian.Uint16(data[2:4]),
Payload: make([]byte, payloadLen),
}
copy(msg.Payload, data[8:])
return msg, nil
}

View File

@@ -1,9 +0,0 @@
package models
import "github.com/gorilla/websocket"
type Player struct {
Entity
Name string
Conn *websocket.Conn
}

View File

@@ -1,50 +0,0 @@
package models
import (
"encoding/binary"
"math"
)
type Reader struct {
buf []byte
off int
}
func NewReader(b []byte) *Reader {
return &Reader{buf: b}
}
func (r *Reader) ReadU8() uint8 {
v := r.buf[r.off]
r.off++
return v
}
func (r *Reader) ReadU16() uint16 {
v := binary.LittleEndian.Uint16(r.buf[r.off:])
r.off += 2
return v
}
func (r *Reader) ReadU32() uint32 {
v := binary.LittleEndian.Uint32(r.buf[r.off:])
r.off += 4
return v
}
func (r *Reader) ReadF32() float32 {
v := math.Float32frombits(binary.LittleEndian.Uint32(r.buf[r.off:]))
r.off += 4
return v
}
func (r *Reader) ReadBool() bool {
return r.ReadU8() == 1
}
func (r *Reader) ReadString() string {
l := r.ReadU16()
s := string(r.buf[r.off : r.off+int(l)])
r.off += int(l)
return s
}

View File

@@ -1,49 +0,0 @@
package models
import (
"encoding/binary"
"math"
)
type Writer struct {
buf []byte
}
func (w *Writer) Bytes() []byte {
return w.buf
}
func (w *Writer) WriteU8(v uint8) {
w.buf = append(w.buf, v)
}
func (w *Writer) WriteU16(v uint16) {
tmp := make([]byte, 2)
binary.LittleEndian.PutUint16(tmp, v)
w.buf = append(w.buf, tmp...)
}
func (w *Writer) WriteU32(v uint32) {
tmp := make([]byte, 4)
binary.LittleEndian.PutUint32(tmp, v)
w.buf = append(w.buf, tmp...)
}
func (w *Writer) WriteF32(v float32) {
tmp := make([]byte, 4)
binary.LittleEndian.PutUint32(tmp, math.Float32bits(v))
w.buf = append(w.buf, tmp...)
}
func (w *Writer) WriteBool(v bool) {
if v {
w.WriteU8(1)
} else {
w.WriteU8(0)
}
}
func (w *Writer) WriteString(s string) {
w.WriteU16(uint16(len(s)))
w.buf = append(w.buf, []byte(s)...)
}

83
src/controllers/room.go Normal file
View File

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

View File

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

6
src/models/auth.go Normal file
View File

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

14
src/models/player.go Normal file
View File

@@ -0,0 +1,14 @@
package models
import (
"github.com/gorilla/websocket"
)
type Player struct {
ID int64
Username string
Conn *websocket.Conn
X, Y float64
DX, DY float64
}

19
src/models/protocol.go Normal file
View File

@@ -0,0 +1,19 @@
package models
type MsgType string
const (
InputMsgType MsgType = "input"
ChatMsgType MsgType = "chat"
)
type InputMessage struct {
PlayerID int64
DX float64 `json:"dx"`
DY float64 `json:"dy"`
}
type StateMessage struct {
Type MsgType `json:"type"`
Payload any `json:"payload"`
}