Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neoppod
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Issues
1
Issues
1
List
Boards
Labels
Milestones
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
nexedi
neoppod
Commits
7f7e2a02
Commit
7f7e2a02
authored
Jun 13, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
X Move net + net_trace to xcommon/xnet/
parent
ad8bdfdd
Changes
13
Show whitespace changes
Inline
Side-by-side
Showing
13 changed files
with
142 additions
and
102 deletions
+142
-102
go/neo/client/client.go
go/neo/client/client.go
+3
-2
go/neo/cluster.go
go/neo/cluster.go
+35
-0
go/neo/connection.go
go/neo/connection.go
+5
-5
go/neo/neotools/master.go
go/neo/neotools/master.go
+2
-2
go/neo/neotools/storage.go
go/neo/neotools/storage.go
+2
-2
go/neo/proto-str.go
go/neo/proto-str.go
+47
-0
go/neo/server/cluster_test.go
go/neo/server/cluster_test.go
+4
-4
go/neo/server/master.go
go/neo/server/master.go
+7
-6
go/neo/server/storage.go
go/neo/server/storage.go
+3
-2
go/xcommon/xnet/net.go
go/xcommon/xnet/net.go
+8
-61
go/xcommon/xnet/pipenet/pipenet.go
go/xcommon/xnet/pipenet/pipenet.go
+0
-0
go/xcommon/xnet/pipenet/pipenet_test.go
go/xcommon/xnet/pipenet/pipenet_test.go
+0
-0
go/xcommon/xnet/trace.go
go/xcommon/xnet/trace.go
+26
-18
No files found.
go/neo/client/client.go
View file @
7f7e2a02
...
@@ -24,6 +24,7 @@ import (
...
@@ -24,6 +24,7 @@ import (
"../../neo"
"../../neo"
"../../zodb"
"../../zodb"
"../../xcommon/xnet"
)
)
// Client talks to NEO cluster and exposes access it via ZODB interfaces
// Client talks to NEO cluster and exposes access it via ZODB interfaces
...
@@ -33,7 +34,7 @@ type Client struct {
...
@@ -33,7 +34,7 @@ type Client struct {
myInfo
neo
.
NodeInfo
// XXX -> only NodeUUID
myInfo
neo
.
NodeInfo
// XXX -> only NodeUUID
clusterName
string
clusterName
string
net
neo
.
Network
// network we are sending/receiving on
net
xnet
.
Network
// network we are sending/receiving on
masterAddr
string
// address of master XXX -> Address ?
masterAddr
string
// address of master XXX -> Address ?
// ---- 8< ----
// ---- 8< ----
...
@@ -134,7 +135,7 @@ func NewClient(storLink *neo.NodeLink) (*Client, error) {
...
@@ -134,7 +135,7 @@ func NewClient(storLink *neo.NodeLink) (*Client, error) {
func
openClientByURL
(
ctx
context
.
Context
,
u
*
url
.
URL
)
(
zodb
.
IStorage
,
error
)
{
func
openClientByURL
(
ctx
context
.
Context
,
u
*
url
.
URL
)
(
zodb
.
IStorage
,
error
)
{
// XXX for now url is treated as storage node URL
// XXX for now url is treated as storage node URL
// XXX check/use other url fields
// XXX check/use other url fields
net
:=
neo
.
NetPlain
(
"tcp"
)
// TODO + TLS; not only "tcp" ?
net
:=
xnet
.
NetPlain
(
"tcp"
)
// TODO + TLS; not only "tcp" ?
storLink
,
err
:=
neo
.
Dial
(
ctx
,
net
,
u
.
Host
)
storLink
,
err
:=
neo
.
Dial
(
ctx
,
net
,
u
.
Host
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
...
...
go/neo/cluster.go
0 → 100644
View file @
7f7e2a02
// Copyright (C) 2017 Nexedi SA and Contributors.
// Kirill Smelkov <kirr@nexedi.com>
//
// This program is free software: you can Use, Study, Modify and Redistribute
// it under the terms of the GNU General Public License version 3, or (at your
// option) any later version, as published by the Free Software Foundation.
//
// You can also Link and Combine this program with other software covered by
// the terms of any of the Open Source Initiative approved licenses and Convey
// the resulting work. Corresponding source of such a combination shall include
// the source code for all other software used.
//
// This program is distributed WITHOUT ANY WARRANTY; without even the implied
// warranty of MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.
//
// See COPYING file for full licensing terms.
// XXX goes away - we don't need it.
package
neo
// cluster state XXX
// ClusterInfo represents information about state and participants of a NEO cluster
//
// Usually ClusterInfo is Master's idea about the cluster which Master shares
// with other nodes. XXX text ok?
//
// XXX naming -> ClusterState ? (but conflict with proto.ClusterState)
type
ClusterInfo
struct
{
State
ClusterState
// what is cluster currently doing: recovering/verification/service/...
NodeTab
NodeTable
// nodes participating in the cluster
PartTab
PartitionTable
// data space partitioning
// XXX do we want to put data movement scheduling plans here ?
}
go/neo/connection.go
View file @
7f7e2a02
...
@@ -20,16 +20,16 @@ package neo
...
@@ -20,16 +20,16 @@ package neo
import
(
import
(
"context"
"context"
"encoding/binary"
"errors"
"errors"
"fmt"
"io"
"io"
"net"
"net"
"reflect"
"sync"
"sync"
"sync/atomic"
"sync/atomic"
"encoding/binary"
"../xcommon/xnet"
"fmt"
"reflect"
)
)
// NodeLink is a node-node link in NEO
// NodeLink is a node-node link in NEO
...
@@ -683,7 +683,7 @@ func handshake(ctx context.Context, conn net.Conn, version uint32) (err error) {
...
@@ -683,7 +683,7 @@ func handshake(ctx context.Context, conn net.Conn, version uint32) (err error) {
// ---- for convenience: Dial ----
// ---- for convenience: Dial ----
// Dial connects to address on given network, handshakes and wraps the connection as NodeLink
// Dial connects to address on given network, handshakes and wraps the connection as NodeLink
func
Dial
(
ctx
context
.
Context
,
net
Network
,
addr
string
)
(
nl
*
NodeLink
,
err
error
)
{
func
Dial
(
ctx
context
.
Context
,
net
xnet
.
Network
,
addr
string
)
(
nl
*
NodeLink
,
err
error
)
{
peerConn
,
err
:=
net
.
Dial
(
ctx
,
addr
)
peerConn
,
err
:=
net
.
Dial
(
ctx
,
addr
)
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
...
...
go/neo/neotools/master.go
View file @
7f7e2a02
...
@@ -26,8 +26,8 @@ import (
...
@@ -26,8 +26,8 @@ import (
"io"
"io"
"os"
"os"
"../../neo"
"../../neo/server"
"../../neo/server"
"../../xcommon/xnet"
)
)
const
masterSummary
=
"run master node"
const
masterSummary
=
"run master node"
...
@@ -64,7 +64,7 @@ func masterMain(argv []string) {
...
@@ -64,7 +64,7 @@ func masterMain(argv []string) {
os
.
Exit
(
2
)
os
.
Exit
(
2
)
}
}
net
:=
neo
.
NetPlain
(
"tcp"
)
// TODO + TLS; not only "tcp" ?
net
:=
xnet
.
NetPlain
(
"tcp"
)
// TODO + TLS; not only "tcp" ?
masterSrv
:=
server
.
NewMaster
(
*
cluster
,
*
bind
,
net
)
masterSrv
:=
server
.
NewMaster
(
*
cluster
,
*
bind
,
net
)
...
...
go/neo/neotools/storage.go
View file @
7f7e2a02
...
@@ -27,9 +27,9 @@ import (
...
@@ -27,9 +27,9 @@ import (
"os"
"os"
"strings"
"strings"
"../../neo"
"../../neo/server"
"../../neo/server"
"../../zodb/storage/fs1"
"../../zodb/storage/fs1"
"../../xcommon/xnet"
)
)
const
storageSummary
=
"run storage node"
const
storageSummary
=
"run storage node"
...
@@ -81,7 +81,7 @@ func storageMain(argv []string) {
...
@@ -81,7 +81,7 @@ func storageMain(argv []string) {
log
.
Fatal
(
err
)
log
.
Fatal
(
err
)
}
}
net
:=
neo
.
NetPlain
(
"tcp"
)
// TODO + TLS; not only "tcp" ?
net
:=
xnet
.
NetPlain
(
"tcp"
)
// TODO + TLS; not only "tcp" ?
storSrv
:=
server
.
NewStorage
(
*
cluster
,
master
,
*
bind
,
net
,
zstor
)
storSrv
:=
server
.
NewStorage
(
*
cluster
,
master
,
*
bind
,
net
,
zstor
)
...
...
go/neo/proto-str.go
View file @
7f7e2a02
...
@@ -4,6 +4,8 @@ package neo
...
@@ -4,6 +4,8 @@ package neo
import
(
import
(
"fmt"
"fmt"
"net"
"strconv"
"strings"
"strings"
)
)
...
@@ -42,3 +44,48 @@ func (nodeUUID NodeUUID) String() string {
...
@@ -42,3 +44,48 @@ func (nodeUUID NodeUUID) String() string {
return
s
return
s
}
}
// ----------------------------------------
// Addr converts network address string into NEO Address
// TODO make neo.Address just string without host:port split
func
AddrString
(
network
,
addr
string
)
(
Address
,
error
)
{
// e.g. on unix, pipenet, etc networks there is no host/port split - the address there
// is single string -> we put it into .Host and set .Port=0 to indicate such cases
if
strings
.
HasPrefix
(
network
,
"tcp"
)
||
strings
.
HasPrefix
(
network
,
"udp"
)
{
// networks that have host:port split
host
,
portstr
,
err
:=
net
.
SplitHostPort
(
addr
)
if
err
!=
nil
{
return
Address
{},
err
}
// XXX also lookup portstr in /etc/services (net.LookupPort) ?
port
,
err
:=
strconv
.
ParseUint
(
portstr
,
10
,
16
)
if
err
!=
nil
{
return
Address
{},
&
net
.
AddrError
{
Err
:
"invalid port"
,
Addr
:
addr
}
}
return
Address
{
Host
:
host
,
Port
:
uint16
(
port
)},
nil
}
return
Address
{
Host
:
addr
,
Port
:
0
},
nil
}
// Addr converts net.Addr into NEO Address
func
Addr
(
addr
net
.
Addr
)
(
Address
,
error
)
{
return
AddrString
(
addr
.
Network
(),
addr
.
String
())
}
// String formats Address to networked address string
func
(
addr
Address
)
String
()
string
{
// XXX in py if .Host == "" -> whole Address is assumed to be empty
// see Addr ^^^ about .Port=0 meaning no host:port split was applied
switch
addr
.
Port
{
case
0
:
return
addr
.
Host
default
:
return
net
.
JoinHostPort
(
addr
.
Host
,
fmt
.
Sprintf
(
"%d"
,
addr
.
Port
))
}
}
go/neo/server/cluster_test.go
View file @
7f7e2a02
...
@@ -25,12 +25,12 @@ import (
...
@@ -25,12 +25,12 @@ import (
//"reflect"
//"reflect"
"testing"
"testing"
"../../neo"
//"../../neo/client"
//"../../neo/client"
//"../../zodb"
//"../../zodb"
"../../zodb/storage/fs1"
"../../zodb/storage/fs1"
"../../xcommon/xnet/pipenet"
"../../xcommon/xsync"
"../../xcommon/xsync"
"lab.nexedi.com/kirr/go123/exc"
"lab.nexedi.com/kirr/go123/exc"
...
@@ -50,7 +50,7 @@ func xfs1stor(path string) *fs1.FileStorage {
...
@@ -50,7 +50,7 @@ func xfs1stor(path string) *fs1.FileStorage {
// M drives cluster with 1 S through recovery -> verification -> service -> shutdown
// M drives cluster with 1 S through recovery -> verification -> service -> shutdown
func
TestMasterStorage
(
t
*
testing
.
T
)
{
func
TestMasterStorage
(
t
*
testing
.
T
)
{
net
:=
neo
.
NetPipe
(
""
)
// test network
net
:=
pipenet
.
New
(
""
)
// test network
Maddr
:=
"0"
Maddr
:=
"0"
Saddr
:=
"1"
Saddr
:=
"1"
...
@@ -91,11 +91,11 @@ func TestClientStorage(t *testing.T) {
...
@@ -91,11 +91,11 @@ func TestClientStorage(t *testing.T) {
/*
/*
Cnl, Snl := NodeLinkPipe()
Cnl, Snl := NodeLinkPipe()
wg :=
neo.WorkGroup()
wg :=
&xsync.WorkGroup{}
Sctx, Scancel := context.WithCancel(context.Background())
Sctx, Scancel := context.WithCancel(context.Background())
net :=
neo.NetPipe
("") // XXX here? (or a bit above?)
net :=
pipenet.New
("") // XXX here? (or a bit above?)
zstor := xfs1stor("../../zodb/storage/fs1/testdata/1.fs") // XXX +readonly
zstor := xfs1stor("../../zodb/storage/fs1/testdata/1.fs") // XXX +readonly
S := NewStorage("cluster", "Maddr", "Saddr", net, zstor)
S := NewStorage("cluster", "Maddr", "Saddr", net, zstor)
wg.Gox(func() {
wg.Gox(func() {
...
...
go/neo/server/master.go
View file @
7f7e2a02
...
@@ -28,6 +28,7 @@ import (
...
@@ -28,6 +28,7 @@ import (
"../../neo"
"../../neo"
"../../zodb"
"../../zodb"
"../../xcommon/xnet"
"lab.nexedi.com/kirr/go123/xerr"
"lab.nexedi.com/kirr/go123/xerr"
)
)
...
@@ -44,10 +45,13 @@ type Master struct {
...
@@ -44,10 +45,13 @@ type Master struct {
// master manages node and partition tables and broadcast their updates
// master manages node and partition tables and broadcast their updates
// to all nodes in cluster
// to all nodes in cluster
///*
stateMu
sync
.
RWMutex
// XXX recheck: needed ?
stateMu
sync
.
RWMutex
// XXX recheck: needed ?
nodeTab
neo
.
NodeTable
nodeTab
neo
.
NodeTable
partTab
neo
.
PartitionTable
partTab
neo
.
PartitionTable
clusterState
neo
.
ClusterState
clusterState
neo
.
ClusterState
//*/
clusterInfo
neo
.
ClusterInfo
// channels controlling main driver
// channels controlling main driver
ctlStart
chan
chan
error
// request to start cluster
ctlStart
chan
chan
error
// request to start cluster
...
@@ -73,7 +77,7 @@ type nodeLeave struct {
...
@@ -73,7 +77,7 @@ type nodeLeave struct {
}
}
// NewMaster TODO ...
// NewMaster TODO ...
func
NewMaster
(
clusterName
,
serveAddr
string
,
net
neo
.
Network
)
*
Master
{
func
NewMaster
(
clusterName
,
serveAddr
string
,
net
xnet
.
Network
)
*
Master
{
// XXX serveAddr + net
// XXX serveAddr + net
m
:=
&
Master
{
clusterName
:
clusterName
}
m
:=
&
Master
{
clusterName
:
clusterName
}
...
@@ -115,11 +119,8 @@ func (m *Master) Shutdown() error {
...
@@ -115,11 +119,8 @@ func (m *Master) Shutdown() error {
// setClusterState sets .clusterState and notifies subscribers
// setClusterState sets .clusterState and notifies subscribers
func
(
m
*
Master
)
setClusterState
(
state
neo
.
ClusterState
)
{
func
(
m
*
Master
)
setClusterState
(
state
neo
.
ClusterState
)
{
if
state
==
m
.
clusterState
{
// <- XXX do we really need this ?
return
}
m
.
clusterState
=
state
m
.
clusterState
=
state
// TODO notify subscribers
// TODO notify subscribers
}
}
...
@@ -316,7 +317,7 @@ func storCtlRecovery(ctx context.Context, link *neo.NodeLink, res chan storRecov
...
@@ -316,7 +317,7 @@ func storCtlRecovery(ctx context.Context, link *neo.NodeLink, res chan storRecov
}
}
resp
:=
neo
.
AnswerPartitionTable
{}
resp
:=
neo
.
AnswerPartitionTable
{}
err
=
conn
.
Ask
(
&
neo
.
X_
PartitionTable
{},
&
resp
)
err
=
conn
.
Ask
(
&
neo
.
Ask
PartitionTable
{},
&
resp
)
if
err
!=
nil
{
if
err
!=
nil
{
return
return
}
}
...
...
go/neo/server/storage.go
View file @
7f7e2a02
...
@@ -25,6 +25,7 @@ import (
...
@@ -25,6 +25,7 @@ import (
"../../neo"
"../../neo"
"../../zodb"
"../../zodb"
"../../xcommon/xnet"
)
)
// XXX fmt -> log
// XXX fmt -> log
...
@@ -36,7 +37,7 @@ type Storage struct {
...
@@ -36,7 +37,7 @@ type Storage struct {
myInfo
neo
.
NodeInfo
// XXX -> only Address + NodeUUID ?
myInfo
neo
.
NodeInfo
// XXX -> only Address + NodeUUID ?
clusterName
string
clusterName
string
net
neo
.
Network
// network we are sending/receiving on
net
xnet
.
Network
// network we are sending/receiving on
masterAddr
string
// address of master
masterAddr
string
// address of master
// ---- 8< ----
// ---- 8< ----
...
@@ -46,7 +47,7 @@ type Storage struct {
...
@@ -46,7 +47,7 @@ type Storage struct {
// NewStorage creates new storage node that will listen on serveAddr and talk to master on masterAddr
// NewStorage creates new storage node that will listen on serveAddr and talk to master on masterAddr
// The storage uses zstor as underlying backend for storing data.
// The storage uses zstor as underlying backend for storing data.
// To actually start running the node - call Run. XXX text
// To actually start running the node - call Run. XXX text
func
NewStorage
(
cluster
,
masterAddr
,
serveAddr
string
,
net
neo
.
Network
,
zstor
zodb
.
IStorage
)
*
Storage
{
func
NewStorage
(
cluster
,
masterAddr
,
serveAddr
string
,
net
xnet
.
Network
,
zstor
zodb
.
IStorage
)
*
Storage
{
// convert serveAddr into neo format
// convert serveAddr into neo format
addr
,
err
:=
neo
.
AddrString
(
net
.
Network
(),
serveAddr
)
addr
,
err
:=
neo
.
AddrString
(
net
.
Network
(),
serveAddr
)
if
err
!=
nil
{
if
err
!=
nil
{
...
...
go/
neo
/net.go
→
go/
xcommon/xnet
/net.go
View file @
7f7e2a02
...
@@ -15,22 +15,17 @@
...
@@ -15,22 +15,17 @@
//
//
// See COPYING file for full licensing terms.
// See COPYING file for full licensing terms.
package
neo
// Package xnet provides addons to std package net
// unified interface for accessing various kinds of networks
package
xnet
import
(
import
(
"context"
"context"
"fmt"
"net"
"net"
"strconv"
"strings"
"crypto/tls"
"crypto/tls"
"../xcommon/pipenet"
)
)
// Network is the interface to work with various kinds of streaming networks
// Network represents interface to work with some kind of streaming network
//
//
// NOTE in NEO a node usually needs to both 1) listen and serve incoming
// NOTE in NEO a node usually needs to both 1) listen and serve incoming
// connections, and 2) dial peers. For this reason the interface is not split
// connections, and 2) dial peers. For this reason the interface is not split
...
@@ -45,6 +40,8 @@ type Network interface {
...
@@ -45,6 +40,8 @@ type Network interface {
// Listen starts listening on local address laddr on underlying network
// Listen starts listening on local address laddr on underlying network
// see net.Listen for semantic details
// see net.Listen for semantic details
//
// XXX also introduce xnet.Listener in which Accept() accepts also ctx?
Listen
(
laddr
string
)
(
net
.
Listener
,
error
)
Listen
(
laddr
string
)
(
net
.
Listener
,
error
)
}
}
...
@@ -70,15 +67,10 @@ func (n netPlain) Listen(laddr string) (net.Listener, error) {
...
@@ -70,15 +67,10 @@ func (n netPlain) Listen(laddr string) (net.Listener, error) {
return
net
.
Listen
(
string
(
n
),
laddr
)
return
net
.
Listen
(
string
(
n
),
laddr
)
}
}
// NetPipe creates Network corresponding to in-memory pipenet
// name is passed directly to pipenet.New
func
NetPipe
(
name
string
)
Network
{
return
pipenet
.
New
(
name
)
}
// NetTLS wraps underlying network with TLS layer according to config
// NetTLS wraps underlying network with TLS layer according to config
// The config must be valid for both tls.Client and tls.Server for Dial and Listen to work
// The config must be valid:
// - for tls.Client -- for Dial to work,
// - for tls.Server -- for Listen to work.
func
NetTLS
(
inner
Network
,
config
*
tls
.
Config
)
Network
{
func
NetTLS
(
inner
Network
,
config
*
tls
.
Config
)
Network
{
return
&
netTLS
{
inner
,
config
}
return
&
netTLS
{
inner
,
config
}
}
}
...
@@ -107,48 +99,3 @@ func (n *netTLS) Listen(laddr string) (net.Listener, error) {
...
@@ -107,48 +99,3 @@ func (n *netTLS) Listen(laddr string) (net.Listener, error) {
}
}
return
tls
.
NewListener
(
l
,
n
.
config
),
nil
return
tls
.
NewListener
(
l
,
n
.
config
),
nil
}
}
// ----------------------------------------
// Addr converts network address string into NEO Address
// TODO make neo.Address just string without host:port split
func
AddrString
(
network
,
addr
string
)
(
Address
,
error
)
{
// e.g. on unix, pipenet, etc networks there is no host/port split - the address there
// is single string -> we put it into .Host and set .Port=0 to indicate such cases
if
strings
.
HasPrefix
(
network
,
"tcp"
)
||
strings
.
HasPrefix
(
network
,
"udp"
)
{
// networks that have host:port split
host
,
portstr
,
err
:=
net
.
SplitHostPort
(
addr
)
if
err
!=
nil
{
return
Address
{},
err
}
// XXX also lookup portstr in /etc/services (net.LookupPort) ?
port
,
err
:=
strconv
.
ParseUint
(
portstr
,
10
,
16
)
if
err
!=
nil
{
return
Address
{},
&
net
.
AddrError
{
Err
:
"invalid port"
,
Addr
:
addr
}
}
return
Address
{
Host
:
host
,
Port
:
uint16
(
port
)},
nil
}
return
Address
{
Host
:
addr
,
Port
:
0
},
nil
}
// Addr converts net.Addre into NEO Address
func
Addr
(
addr
net
.
Addr
)
(
Address
,
error
)
{
return
AddrString
(
addr
.
Network
(),
addr
.
String
())
}
// String formats Address to networked address string
func
(
addr
Address
)
String
()
string
{
// XXX in py if .Host == "" -> whole Address is assumed to be empty
// see Addr ^^^ about .Port=0 meaning no host:port split was applied
switch
addr
.
Port
{
case
0
:
return
addr
.
Host
default
:
return
net
.
JoinHostPort
(
addr
.
Host
,
fmt
.
Sprintf
(
"%d"
,
addr
.
Port
))
}
}
go/xcommon/pipenet/pipenet.go
→
go/xcommon/
xnet/
pipenet/pipenet.go
View file @
7f7e2a02
File moved
go/xcommon/pipenet/pipenet_test.go
→
go/xcommon/
xnet/
pipenet/pipenet_test.go
View file @
7f7e2a02
File moved
go/
neo/net_
trace.go
→
go/
xcommon/xnet/
trace.go
View file @
7f7e2a02
...
@@ -15,52 +15,60 @@
...
@@ -15,52 +15,60 @@
//
//
// See COPYING file for full licensing terms.
// See COPYING file for full licensing terms.
// +build test
package
xnet
package
neo
// network tracing
// network tracing
// XXX move to xnet/trace ?
import
(
"context"
"net"
)
// NetTrace wraps underlying network with IO tracing layer
// NetTrace wraps underlying network with IO tracing layer
//
//
//
the t
racing is done via calling trace func right before corresponding packet
//
T
racing is done via calling trace func right before corresponding packet
// is sent for Tx to underlying network. No synchronization for notification is
// is sent for Tx to underlying network. No synchronization for notification is
// performed - if one is required tracing func must implement such
// performed - if one is required tracing func must implement such
// synchronization itself.
// synchronization itself.
//
//
// only Tx events are traced:
// only Tx events are traced:
// - because Write, contrary to Read, never writes partial data on non-error
// - because Write, contrary to Read, never writes partial data on non-error
// - because in case of
NetPipe
tracing writes only is enough to get whole network exchange picture
// - because in case of
pipenet
tracing writes only is enough to get whole network exchange picture
func
NetTrace
(
inner
Network
,
trace
func
(
t
*
t
raceTx
))
Network
{
func
NetTrace
(
inner
Network
,
trace
func
(
t
*
T
raceTx
))
Network
{
&
netTrace
{
inner
,
trace
}
return
&
netTrace
{
inner
,
trace
}
}
}
//
t
raceTx is event corresponding to network transmission
//
T
raceTx is event corresponding to network transmission
type
t
raceTx
struct
{
type
T
raceTx
struct
{
src
,
d
st
net
.
Addr
Src
,
D
st
net
.
Addr
p
kt
[]
byte
P
kt
[]
byte
}
}
// netTrace wraps underlying Network such that whenever a connection is created
// netTrace wraps underlying Network such that whenever a connection is created
// it is wrapped with traceConn
// it is wrapped with traceConn
type
netTrace
struct
{
type
netTrace
struct
{
inner
Network
inner
Network
trace
func
(
t
*
traceTx
)
trace
func
(
t
*
TraceTx
)
}
func
(
nt
*
netTrace
)
Network
()
string
{
return
nt
.
inner
.
Network
()
// XXX + "+trace" ?
}
}
func
(
nt
*
netTrace
)
Dial
(
ctx
context
.
Context
,
addr
string
)
(
net
.
Conn
,
error
)
{
func
(
nt
*
netTrace
)
Dial
(
ctx
context
.
Context
,
addr
string
)
(
net
.
Conn
,
error
)
{
c
,
err
:=
nt
.
inner
.
Dial
(
ctx
,
addr
)
c
,
err
:=
nt
.
inner
.
Dial
(
ctx
,
addr
)
if
err
!=
nil
{
if
err
!=
nil
{
return
err
return
nil
,
err
}
}
return
&
traceConn
{
nt
,
c
}
return
&
traceConn
{
nt
,
c
}
,
nil
}
}
func
(
nt
*
netTrace
)
Listen
(
laddr
string
)
(
net
.
Listener
,
error
)
{
func
(
nt
*
netTrace
)
Listen
(
laddr
string
)
(
net
.
Listener
,
error
)
{
l
,
err
:=
nt
.
inner
.
Listen
(
laddr
)
l
,
err
:=
nt
.
inner
.
Listen
(
laddr
)
if
err
!=
nil
{
if
err
!=
nil
{
return
err
return
nil
,
err
}
}
return
&
netTraceListener
{
nt
,
l
}
return
&
netTraceListener
{
nt
,
l
}
,
nil
}
}
// netTraceListener wraps net.Listener to wrap accepted connections with traceConn
// netTraceListener wraps net.Listener to wrap accepted connections with traceConn
...
@@ -74,7 +82,7 @@ func (ntl *netTraceListener) Accept() (net.Conn, error) {
...
@@ -74,7 +82,7 @@ func (ntl *netTraceListener) Accept() (net.Conn, error) {
if
err
!=
nil
{
if
err
!=
nil
{
return
nil
,
err
return
nil
,
err
}
}
return
&
traceConn
{
ntl
.
nt
,
c
}
return
&
traceConn
{
ntl
.
nt
,
c
}
,
nil
}
}
// traceConn wraps net.Conn and notifies tracer on Writes
// traceConn wraps net.Conn and notifies tracer on Writes
...
@@ -84,7 +92,7 @@ type traceConn struct {
...
@@ -84,7 +92,7 @@ type traceConn struct {
}
}
func
(
tc
*
traceConn
)
Write
(
b
[]
byte
)
(
int
,
error
)
{
func
(
tc
*
traceConn
)
Write
(
b
[]
byte
)
(
int
,
error
)
{
t
:=
&
traceTx
{
src
:
tc
.
LocalAddr
(),
dst
:
tc
.
RemoteAddr
(),
p
kt
:
b
}
t
:=
&
TraceTx
{
Src
:
tc
.
LocalAddr
(),
Dst
:
tc
.
RemoteAddr
(),
P
kt
:
b
}
tc
.
nt
.
trace
(
t
)
tc
.
nt
.
trace
(
t
)
return
tc
.
Conn
.
Write
(
b
)
return
tc
.
Conn
.
Write
(
b
)
}
}
Write
Preview
Markdown
is supported
0%
Try again
or
attach a new file
Attach a file
Cancel
You are about to add
0
people
to the discussion. Proceed with caution.
Finish editing this message first!
Cancel
Please
register
or
sign in
to comment