Created
April 18, 2021 21:38
-
-
Save cablehead/7a7c5a2e573624444b109d5a108bc18d to your computer and use it in GitHub Desktop.
cli to merge / split stdio over a network
This file contains hidden or bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
package main | |
import ( | |
"bufio" | |
"fmt" | |
"net" | |
"os" | |
"sync" | |
"github.com/alexflint/go-arg" | |
) | |
// port range: 8163-8180 | |
type commandSplit struct { | |
Port int `default:"8163"` | |
} | |
type commandMerge struct { | |
Port int `default:"8164"` | |
} | |
func main() { | |
var args struct { | |
Split *commandSplit `arg:"subcommand:split"` | |
Merge *commandMerge `arg:"subcommand:merge"` | |
} | |
p := arg.MustParse(&args) | |
if p.Subcommand() == nil { | |
p.Fail("missing subcommand") | |
} | |
if args.Split != nil { | |
doSplit(args.Split) | |
} | |
if args.Merge != nil { | |
doMerge(args.Merge) | |
} | |
} | |
func doMerge(args *commandMerge) { | |
s, err := net.Listen("tcp", fmt.Sprintf(":%d", args.Port)) | |
if err != nil { | |
panic(err) | |
} | |
handle := func(conn net.Conn) { | |
defer func() { _ = conn.Close() }() | |
scanner := bufio.NewScanner(conn) | |
for scanner.Scan() { | |
fmt.Println(scanner.Text()) | |
} | |
if err := scanner.Err(); err != nil { | |
panic(err) | |
} | |
} | |
for { | |
conn, err := s.Accept() | |
if err != nil { | |
panic(err) | |
} | |
go handle(conn) | |
} | |
} | |
func NewRoom() *Room { | |
return &Room{members: make(map[net.Conn]chan string)} | |
} | |
type Room struct { | |
members map[net.Conn]chan string | |
sync.Mutex | |
closed bool | |
} | |
func (r *Room) Join(conn net.Conn) <-chan string { | |
r.Lock() | |
defer r.Unlock() | |
if r.closed { | |
panic("join on closed room") | |
} | |
c := make(chan string) | |
r.members[conn] = c | |
return c | |
} | |
func (r *Room) Leave(conn net.Conn) { | |
r.Lock() | |
defer r.Unlock() | |
// TODO: return err if conn isn't a member | |
c := r.members[conn] | |
close(c) | |
delete(r.members, conn) | |
} | |
func (r *Room) Send(msg string) { | |
r.Lock() | |
defer r.Unlock() | |
if r.closed { | |
panic("send on closed room") | |
} | |
for _, c := range r.members { | |
c <- msg | |
} | |
} | |
func (r *Room) Close() { | |
r.Lock() | |
defer r.Unlock() | |
if r.closed { | |
panic("close on closed room") | |
} | |
r.closed = true | |
for conn, c := range r.members { | |
close(c) | |
delete(r.members, conn) | |
} | |
} | |
func doSplit(args *commandSplit) { | |
r := NewRoom() | |
s, err := net.Listen("tcp", fmt.Sprintf(":%d", args.Port)) | |
if err != nil { | |
panic(err) | |
} | |
// main loop to read from stdin | |
go func() { | |
scanner := bufio.NewScanner(os.Stdin) | |
for scanner.Scan() { | |
r.Send(scanner.Text()) | |
} | |
if err := scanner.Err(); err != nil { | |
panic(err) | |
} | |
_ = s.Close() | |
r.Close() | |
}() | |
handle := func(conn net.Conn) { | |
c := r.Join(conn) | |
defer func() { | |
r.Leave(conn) | |
_ = conn.Close() | |
}() | |
for { | |
msg := <-c | |
_, err := fmt.Fprintf(conn, "%s\n", msg) | |
if err != nil { | |
break | |
} | |
} | |
} | |
for { | |
conn, err := s.Accept() | |
if err != nil { | |
break | |
} | |
go handle(conn) | |
} | |
} |
Sign up for free
to join this conversation on GitHub.
Already have an account?
Sign in to comment