Teensy TCP echo server

This commit is contained in:
Dan Buch 2025-07-07 21:23:01 -04:00
parent 0b2d033cd8
commit 8b39d6cc82
Signed by: meatballhat
GPG Key ID: A12F782281063434
2 changed files with 92 additions and 0 deletions

View File

@ -0,0 +1,89 @@
package main
import (
"context"
"io"
"log"
"net"
"os"
"os/signal"
"strconv"
"syscall"
)
func main() {
ctx, stop := signal.NotifyContext(context.Background(), syscall.SIGINT, syscall.SIGTERM)
defer stop()
addr := ":7765"
if len(os.Args) > 1 {
addr = os.Args[1]
}
poolSize := 10
if len(os.Args) > 2 {
if v, err := strconv.ParseInt(os.Args[2], 10, 64); err != nil {
log.Fatal(err)
} else {
poolSize = int(v)
}
}
log.Printf("listening at %v (workers=%v)", addr, poolSize)
listener, err := net.Listen("tcp", addr)
if err != nil {
log.Fatal(err)
}
workers := []*worker{}
requests := make(chan net.Conn)
for i := 0; i < poolSize; i++ {
w := &worker{id: i, r: requests}
go w.Start(ctx)
workers = append(workers, w)
}
for {
select {
case <-ctx.Done():
return
default:
log.Println("accepting connection")
conn, err := listener.Accept()
if err != nil {
log.Printf("failed to accept connection: %v", err)
continue
}
log.Println("accepted connection")
requests <- conn
}
}
}
type worker struct {
id int
r chan net.Conn
}
func (w *worker) Start(ctx context.Context) {
log.Printf("[worker %v] starting loop", w.id)
for {
select {
case <-ctx.Done():
log.Printf("[worker %v] exiting", w.id)
return
case conn := <-w.r:
log.Printf("[worker %v] handling connection", w.id)
io.Copy(conn, conn)
if err := conn.Close(); err != nil {
log.Printf("[worker %v] failed to echo: %v", w.id, err)
}
}
}
}

View File

@ -0,0 +1,3 @@
module git.meatballhat.com/x/box-o-sand/tcp-challenge/server
go 1.21.4