-
Notifications
You must be signed in to change notification settings - Fork 1
/
Copy pathmain.go
121 lines (104 loc) · 2.58 KB
/
main.go
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
// main.go
package main
import (
"context"
"fmt"
"net"
"os"
"os/signal"
"strconv"
"strings"
"sync"
"syscall"
"github.com/rs/zerolog"
"github.com/rs/zerolog/log"
)
var (
clients = make([]*AuthenticatedClient, 0)
clientsMu sync.RWMutex
)
func main() {
output := zerolog.NewConsoleWriter(func(w *zerolog.ConsoleWriter) {
w.TimeFormat = "15:04:05.000"
})
log.Logger = zerolog.New(output).With().Timestamp().Logger()
host := getEnv("HOST", "0.0.0.0")
portStr := getEnv("PORT", "27200")
port, err := strconv.Atoi(portStr)
if err != nil {
log.Fatal().Err(err).Msg("Invalid PORT environment variable")
}
// Parse WHITELIST environment (e.g. "127.0.0.1:27115,62.226.205.78:27015")
wlEnv := getEnv("WHITELIST", "")
whitelist, err := parseWhitelistEnv(wlEnv)
if err != nil {
log.Fatal().Err(err).Msg("Failed to parse WHITELIST")
}
log.Info().Msgf("Loaded %d whitelist entries from environment", len(whitelist))
addr := fmt.Sprintf("%s:%d", host, port)
listener, err := net.Listen("tcp", addr)
if err != nil {
log.Fatal().Err(err).Msg("Failed to start TCP server")
}
defer listener.Close()
log.Info().Msgf("Relay server ready at tcp://%s", addr)
// Listen for Ctrl+C / SIGTERM to shut down gracefully
ctx, stop := signal.NotifyContext(context.Background(), os.Interrupt, syscall.SIGTERM)
defer stop()
go func() {
for {
conn, err := listener.Accept()
if err != nil {
select {
case <-ctx.Done():
return
default:
log.Error().Err(err).Msg("Failed to accept connection")
continue
}
}
go handleConnection(conn, whitelist)
}
}()
<-ctx.Done()
log.Info().Msg("Shutting down gracefully...")
_ = listener.Close()
clientsMu.Lock()
for _, c := range clients {
_ = c.conn.Close()
}
clients = nil
clientsMu.Unlock()
log.Info().Msg("All connections closed. Bye!")
}
func getEnv(key, fallback string) string {
if val := os.Getenv(key); val != "" {
return val
}
return fallback
}
func parseWhitelistEnv(envVal string) ([]WhitelistEntry, error) {
if envVal == "" {
return nil, nil
}
var result []WhitelistEntry
parts := strings.Split(envVal, ",")
for _, part := range parts {
part = strings.TrimSpace(part)
if part == "" {
continue
}
sub := strings.Split(part, ":")
if len(sub) != 2 {
return nil, fmt.Errorf("invalid format (need ip:port): %s", part)
}
ip := strings.TrimSpace(sub[0])
portStr := strings.TrimSpace(sub[1])
p, err := strconv.Atoi(portStr)
if err != nil {
return nil, fmt.Errorf("invalid port: %s", portStr)
}
result = append(result, WhitelistEntry{IP: ip, Port: p})
}
return result, nil
}