Commit 599d17bc authored by Kirill Smelkov's avatar Kirill Smelkov

.

parent dd3bb8b4
......@@ -18,6 +18,7 @@ import (
"errors"
"io"
"net"
"sync"
"unsafe"
)
......@@ -45,16 +46,21 @@ import (
type NodeLink struct {
peerLink net.Conn // raw conn to peer
// TODO locking
connTab map[uint32]*Conn // msgid -> connection associated with msgid
connMu sync.Mutex // TODO -> RW ?
connTab map[uint32]*Conn // connid -> connection associated with connid
nextConnId uint32 // next connId to use for Conn initiated by us
handleNewConn func(conn *Conn) // handler for new connections XXX -> ConnHandler (a-la Handler in net/http) ?
// TODO peerLink .LocalAddr() vs .RemoteAddr() -> msgid even/odd ? (XXX vs NAT ?)
txreq chan txReq // tx requests from Conns go here
// (received pkt go dispatched to connTab[connid].rxq)
closed chan struct{}
}
// Conn is a connection established over NodeLink
//
// Data can be sent and received over it.
......@@ -63,12 +69,14 @@ type NodeLink struct {
// It is safe to use Conn from multiple goroutines simultaneously.
type Conn struct {
nodeLink *NodeLink
connId uint32
rxq chan *PktBuf
txerr chan error // transmit errors go back here
closed chan struct{}
}
// Buffer with packet data
// XXX move me out of here
type PktBuf struct {
Data []byte // whole packet data including all headers XXX -> Buf ?
}
......@@ -85,11 +93,34 @@ func (pkt *PktBuf) Payload() []byte {
}
type ConnRole int
const (
ConnServer ConnRole = iota // connection created as server
ConnClient // connection created as client
)
// Make a new NodeLink from already established net.Conn
func NewNodeLink(c net.Conn) *NodeLink {
//
// role specifies how to treat conn - either as server or client one.
// The differrence in between client and server roles are in connid % 2 XXX text
//
// Usually server role should be used for connections created via
// net.Listen/net.Accept and client role for connections created via net.Dial.
func NewNodeLink(conn net.Conn, role ConnRole) *NodeLink {
var nextConnId uint32
switch role {
case ConnServer:
nextConnId = 0
case ConnClient:
nextConnId = 1
default:
panic("invalid conn role")
}
nl := NodeLink{
peerLink: c,
peerLink: conn,
connTab: map[uint32]*Conn{},
nextConnId: nextConnId,
txreq: make(chan txReq),
closed: make(chan struct{}),
}
......@@ -103,10 +134,18 @@ func NewNodeLink(c net.Conn) *NodeLink {
func (nl *NodeLink) Close() error {
close(nl.closed)
err := nl.peerLink.Close()
// TODO close active Conns
// close active Conns
nl.connMu.Lock()
defer nl.connMu.Unlock()
for _, conn := range nl.connTab {
// FIXME it also wants to lock conntab -> conn.close() ?
println("conn", conn.connId, " -> closing ...")
conn.Close() // XXX err
}
nl.connTab = nil // XXX ok? vs panic on NewConn after close ?
// XXX wait for serve{Send,Recv} to complete
nl.wg.Wait()
//nl.wg.Wait()
return err
}
// send raw packet to peer
......@@ -171,8 +210,11 @@ func (nl *NodeLink) NewConn() *Conn {
txerr: make(chan error),
closed: make(chan struct{}),
}
// XXX locking
nl.connTab[0] = c // FIXME 0 -> msgid; XXX also check not a duplicate
nl.connMu.Lock()
defer nl.connMu.Unlock()
c.connId = nl.nextConnId
nl.nextConnId += 2
nl.connTab[c.connId] = c
return c
}
......@@ -193,10 +235,10 @@ func (nl *NodeLink) serveRecv() {
panic(err) // XXX err
}
// if we don't yet have connection established for pkt.MsgId -
// if we don't yet have connection established for pkt.ConnId -
// spawn connection-serving goroutine
// XXX connTab locking
conn := nl.connTab[ntoh32(pkt.Header().MsgId)]
conn := nl.connTab[ntoh32(pkt.Header().ConnId)]
if conn == nil {
if nl.handleNewConn == nil {
// we are not accepting incoming connections - ignore packet
......@@ -231,7 +273,6 @@ func (nl *NodeLink) serveSend() {
break
case txreq := <-nl.txreq:
pkt.Header().MsgId = hton32(0) // TODO next msgid, or using same msgid as received
err := nl.sendPkt(txreq.pkt)
if err != nil {
// XXX also close whole nodeLink since tx framing now can be broken?
......@@ -254,6 +295,10 @@ var ErrClosedConn = errors.New("read/write on closed connection")
// Send packet via connection
func (c *Conn) Send(pkt *PktBuf) error {
// set pkt connid associated with this connection
// TODO for new Conn - it should be set by serveSend ?
pkt.Header().ConnId = hton32(c.connId)
select {
case <-c.closed:
return ErrClosedConn
......@@ -291,6 +336,11 @@ func (c *Conn) Close() error { // XXX do we need error here?
// TODO
//func Dial(ctx context.Context, network, address string) (*NodeLink, error) // + tls.Config
//func Listen(network, laddr string) (net.Listener, error) // + tls.Config
// ln.Accept -> will return net.Conn wrapped in NodeLink
// ----------------------------------------
......
......@@ -88,10 +88,10 @@ func xwait(w interface { Wait() error }) {
}
// Prepare PktBuf with content
func mkpkt(msgid uint32, msgcode uint16, payload []byte) *PktBuf {
func mkpkt(connid uint32, msgcode uint16, payload []byte) *PktBuf {
pkt := &PktBuf{make([]byte, PktHeadLen + len(payload))}
h := pkt.Header()
h.MsgId = hton32(msgid)
h.ConnId = hton32(connid)
h.MsgCode = hton16(msgcode)
h.Len = hton32(PktHeadLen + 4)
copy(pkt.Payload(), payload)
......@@ -99,12 +99,12 @@ func mkpkt(msgid uint32, msgcode uint16, payload []byte) *PktBuf {
}
// Verify PktBuf is as expected
func xverifyPkt(pkt *PktBuf, msgid uint32, msgcode uint16, payload []byte) {
func xverifyPkt(pkt *PktBuf, connid uint32, msgcode uint16, payload []byte) {
errv := xerr.Errorv{}
h := pkt.Header()
// TODO include caller location
if ntoh32(h.MsgId) != msgid {
errv.Appendf("header: unexpected msgid %v (want %v)", ntoh32(h.MsgId), msgid)
if ntoh32(h.ConnId) != connid {
errv.Appendf("header: unexpected connid %v (want %v)", ntoh32(h.ConnId), connid)
}
if ntoh16(h.MsgCode) != msgcode {
errv.Appendf("header: unexpected msgcode %v (want %v)", ntoh16(h.MsgCode), msgcode)
......@@ -129,8 +129,8 @@ func tdelay() {
// create NodeLinks connected via net.Pipe
func nodeLinkPipe() (nl1, nl2 *NodeLink) {
node1, node2 := net.Pipe()
nl1 = NewNodeLink(node1)
nl2 = NewNodeLink(node2)
nl1 = NewNodeLink(node1, ConnClient)
nl2 = NewNodeLink(node2, ConnServer)
return nl1, nl2
}
......@@ -138,6 +138,7 @@ func nodeLinkPipe() (nl1, nl2 *NodeLink) {
func TestNodeLink(t *testing.T) {
// TODO catch exception -> add proper location from it -> t.Fatal (see git-backup)
println("111")
nl1, nl2 := nodeLinkPipe()
// Close vs recvPkt
......@@ -197,6 +198,7 @@ func TestNodeLink(t *testing.T) {
// test channels on top of nodelink
println("222")
nl1, nl2 = nodeLinkPipe()
// Close vs Recv
......@@ -227,6 +229,7 @@ func TestNodeLink(t *testing.T) {
xwait(wg)
// NodeLink.Close vs Conn.Send/Recv
println("333")
c11 := nl1.NewConn()
c12 := nl1.NewConn()
wg = WorkGroup()
......@@ -248,6 +251,7 @@ func TestNodeLink(t *testing.T) {
xwait(wg)
xclose(nl2) // for completeness
println("444")
/*
......
......@@ -127,7 +127,7 @@ type RowInfo struct {
// XXX link request <-> answer ?
// TODO ensure len(encoded packet header) == 10
type PktHead struct {
MsgId be32
ConnId be32 // NOTE is .msgid in py
MsgCode be16
Len be32 // whole packet length (including header)
}
......
Markdown is supported
0%
or
You are about to add 0 people to the discussion. Proceed with caution.
Finish editing this message first!
Please register or to comment