| package main |
| |
| import ( |
| "context" |
| "net" |
| "strings" |
| |
| "github.com/golang/glog" |
| ) |
| |
| type packetFrom struct { |
| addr net.Addr |
| data []byte |
| } |
| |
| func (s *service) runProxy(ctx context.Context, updates chan *lockUpdate, laserAddr net.Addr, laserNet, clientNet net.PacketConn) { |
| glog.Infof("Proxy starting... (laser: %v, laser network: %v, client network: %v)", laserAddr, laserNet, clientNet) |
| |
| |
| pipe := func(conn net.PacketConn, C chan *packetFrom) { |
| for { |
| buf := make([]byte, 1500) |
| n, addr, err := conn.ReadFrom(buf) |
| if err != nil { |
| glog.Errorf("pipe failed: %v", err) |
| } |
| C <- &packetFrom{ addr, buf[:n] } |
| } |
| } |
| |
| laserC := make(chan *packetFrom) |
| go pipe(laserNet, laserC) |
| clientC := make(chan *packetFrom) |
| go pipe(clientNet, clientC) |
| |
| var allowedClient string |
| var curClient *net.Addr |
| for { |
| select { |
| case <-ctx.Done(): |
| err := ctx.Err() |
| glog.Errorf("Proxy stopped: %v", err) |
| return |
| case u := <-updates: |
| allowedClient = u.addr |
| glog.Infof("New allowed client: %q", allowedClient) |
| case p := <-laserC: |
| s.lockCtrl <- &lockCtrl{ |
| bump: &lockCtrlBump{}, |
| } |
| if curClient == nil { |
| glog.Warningf("Packet from laser without client connected, dropping.") |
| break |
| } |
| clientNet.WriteTo(p.data, *curClient) |
| case p := <-clientC: |
| s.lockCtrl <- &lockCtrl{ |
| bump: &lockCtrlBump{}, |
| } |
| if strings.Split(p.addr.String(), ":")[0] == allowedClient { |
| curClient = &p.addr |
| laserNet.WriteTo(p.data, laserAddr) |
| } else { |
| glog.Infof("Rejecting packet from %s", p.addr.String()) |
| } |
| } |
| } |
| } |
| |