Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Support
Keyboard shortcuts
?
Submit feedback
Contribute to GitLab
Sign in / Register
Toggle navigation
N
neo
Project overview
Project overview
Details
Activity
Releases
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Labels
Merge Requests
2
Merge Requests
2
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Analytics
Analytics
CI / CD
Repository
Value Stream
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Jobs
Commits
Open sidebar
Kirill Smelkov
neo
Commits
454b6576
Commit
454b6576
authored
Dec 22, 2020
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
6e312d66
Changes
3
Hide whitespace changes
Inline
Side-by-side
Showing
3 changed files
with
98 additions
and
56 deletions
+98
-56
go/internal/xtracing/tracetest/tracetest.go
go/internal/xtracing/tracetest/tracetest.go
+82
-40
go/neo/neo_test.go
go/neo/neo_test.go
+1
-1
go/neo/t_events_test.go
go/neo/t_events_test.go
+15
-15
No files found.
go/internal/xtracing/tracetest/tracetest.go
View file @
454b6576
...
...
@@ -89,7 +89,10 @@ import (
"github.com/kylelemons/godebug/pretty"
)
var
chatty
=
flag
.
Bool
(
"tracetest.v"
,
false
,
"verbose: print events as they are sent on trace channels"
)
var
(
chatty
=
flag
.
Bool
(
"tracetest.v"
,
false
,
"verbose: print events as they are sent on trace channels"
)
deadTime
=
flag
.
Duration
(
"tracetest.deadtime"
,
3
*
time
.
Second
,
"time after which no events activity is considered to be a deadlock"
)
)
// Chan provides synchronous channel with additional property that send
// blocks until receiving side explicitly acknowledges message was received and
...
...
@@ -98,9 +101,31 @@ var chatty = flag.Bool("tracetest.v", false, "verbose: print events as they are
// New channels must be created via NewChan.
//
// It is safe to use Chan from multiple goroutines simultaneously.
type
Chan
struct
{
msgq
chan
*
Msg
name
string
type
Chan
interface
{
ChanTx
ChanRx
name
()
string
// name of the channel
}
// ChanTx represents "send-only" half of Chan.
// It is similar to chan<- .
type
ChanTx
interface
{
// Send sends event to a consumer and waits for ack.
// if main testing goroutine detects any problem Send panics. XXX
Send
(
event
interface
{})
}
// ChanRx represents "receive-only" half of Chan.
// It is similar to <-chan .
type
ChanRx
interface
{
// Recv receives message from a producer.
//
// The consumer, after dealing with the message, must send back an ack.
Recv
()
*
Msg
// _rxq returns raw channel that is serving ChanRx.
// it is used internally to use ChanRx in select.
_rxq
()
<-
chan
*
Msg
}
// Msg represents message with 1 event sent over Chan.
...
...
@@ -112,58 +137,74 @@ type Msg struct {
}
// Send sends event to a consumer and waits for ack.
//
// if main testing goroutine detects any problem Send panics. XXX
func
(
ch
*
Chan
)
Send
(
event
interface
{})
{
// _chan implements Chan.
type
_chan
struct
{
msgq
chan
*
Msg
_name
string
}
// name implements Chan.
func
(
ch
*
_chan
)
name
()
string
{
return
ch
.
_name
}
// Send implements ChanTx.
func
(
ch
*
_chan
)
Send
(
event
interface
{})
{
if
*
chatty
{
fmt
.
Printf
(
"%s <- %T %v
\n
"
,
ch
.
name
,
event
,
event
)
fmt
.
Printf
(
"%s <- %T %v
\n
"
,
ch
.
name
()
,
event
,
event
)
}
ack
:=
make
(
chan
bool
)
ch
.
msgq
<-
&
Msg
{
event
,
ack
}
ok
:=
<-
ack
if
!
ok
{
panicf
(
"%s: send: deadlock"
,
ch
.
name
)
panicf
(
"%s: send: deadlock"
,
ch
.
name
()
)
}
}
// Recv receives message from a producer.
//
// The consumer, after dealing with the message, must send back an ack.
func
(
ch
*
Chan
)
Recv
()
*
Msg
{
// Recv implements ChanRx.
func
(
ch
*
_chan
)
Recv
()
*
Msg
{
msg
:=
<-
ch
.
msgq
return
msg
}
// _rxq implements ChanRx.
func
(
ch
*
_chan
)
_rxq
()
<-
chan
*
Msg
{
return
ch
.
msgq
}
// XXX -> Unpause?
// Ack acknowledges the event was processed and unblocks producer goroutine.
func
(
m
*
Msg
)
Ack
()
{
m
.
ack
<-
true
}
// NewChan creates new Chan channel.
func
NewChan
(
name
string
)
*
Chan
{
func
NewChan
(
name
string
)
Chan
{
// XXX somehow avoid channels with duplicate names
// (only allow to create named channels from under dispatcher?)
return
&
Chan
{
msgq
:
make
(
chan
*
Msg
),
name
:
name
}
return
&
_chan
{
msgq
:
make
(
chan
*
Msg
),
_
name
:
name
}
}
// ----------------------------------------
// XXX -> tracetest.TSerial ? tracetest.TSeq?
// EventChecker is testing utility to verify that sequence of events coming
// from a single Chan is as expected.
type
EventChecker
struct
{
t
testing
.
TB
in
*
Chan
dispatch
*
EventDispatcher
in
Chan
// XXX -> ChanRx ?
dispatch
*
EventDispatcher
// XXX why here?
}
// NewEventChecker constructs new EventChecker that will retrieve events from
// `in` and use `t` for tests reporting.
//
// XXX -> dispatch.NewChecker() ?
func
NewEventChecker
(
t
testing
.
TB
,
dispatch
*
EventDispatcher
,
in
*
Chan
)
*
EventChecker
{
// XXX kill dispatch arg in NewEventChecker (and set it only from dispatch.NewChecker)
func
NewEventChecker
(
t
testing
.
TB
,
dispatch
*
EventDispatcher
,
in
Chan
)
*
EventChecker
{
return
&
EventChecker
{
t
:
t
,
in
:
in
,
dispatch
:
dispatch
}
}
...
...
@@ -175,17 +216,17 @@ func (evc *EventChecker) xget1(eventp interface{}) *Msg {
var
msg
*
Msg
select
{
case
msg
=
<-
evc
.
in
.
msgq
:
// unwrapped Recv
case
msg
=
<-
evc
.
in
.
_rxq
()
:
// unwrapped Recv
// ok
case
<-
time
.
After
(
2
*
time
.
Second
)
:
// XXX timeout hardcoded
case
<-
time
.
After
(
*
deadTime
)
:
evc
.
deadlock
(
eventp
)
// XXX -> Dispatcher + allow to tune?
}
reventp
:=
reflect
.
ValueOf
(
eventp
)
if
reventp
.
Type
()
.
Elem
()
!=
reflect
.
TypeOf
(
msg
.
Event
)
{
// msg.ack <- error
evc
.
t
.
Fatalf
(
"%s: expect: %s: got %#v"
,
evc
.
in
.
name
,
reventp
.
Elem
()
.
Type
(),
msg
.
Event
)
evc
.
t
.
Fatalf
(
"%s: expect: %s: got %#v"
,
evc
.
in
.
name
()
,
reventp
.
Elem
()
.
Type
(),
msg
.
Event
)
}
// *eventp = msg.Event
...
...
@@ -209,7 +250,7 @@ func (evc *EventChecker) expect1(eventExpect interface{}) *Msg {
if
!
reflect
.
DeepEqual
(
revent
.
Interface
(),
reventExpect
.
Interface
())
{
// msg.ack <- error
evc
.
t
.
Fatalf
(
"%s: expect: %s:
\n
want: %v
\n
have: %v
\n
diff: %s"
,
evc
.
in
.
name
,
evc
.
in
.
name
()
,
reventExpect
.
Type
(),
reventExpect
,
revent
,
pretty
.
Compare
(
reventExpect
.
Interface
(),
revent
.
Interface
()))
}
...
...
@@ -252,14 +293,14 @@ func (evc *EventChecker) ExpectNoACK(expected interface{}) *Msg {
func
(
evc
*
EventChecker
)
deadlock
(
eventp
interface
{})
{
evc
.
t
.
Helper
()
bad
:=
fmt
.
Sprintf
(
"%s: deadlock waiting for %T
\n
"
,
evc
.
in
.
name
,
eventp
)
type
sendInfo
struct
{
dst
*
Chan
;
event
interface
{}}
bad
:=
fmt
.
Sprintf
(
"%s: deadlock waiting for %T
\n
"
,
evc
.
in
.
name
()
,
eventp
)
type
sendInfo
struct
{
dst
Chan
;
event
interface
{}}
var
sendv
[]
sendInfo
for
_
,
dst
:=
range
evc
.
dispatch
.
streams
{
// check whether someone is sending on a dst without blocking.
// if yes - report to sender there is a problem - so it can cancel its task.
select
{
case
msg
:=
<-
dst
.
msgq
:
case
msg
:=
<-
dst
.
_rxq
()
:
sendv
=
append
(
sendv
,
sendInfo
{
dst
,
msg
.
Event
})
//msg.ack <- false
...
...
@@ -269,13 +310,13 @@ func (evc *EventChecker) deadlock(eventp interface{}) {
// XXX panic triggering disabled because if sender panics we have no chance to continue
// TODO retest this
// in any case close channel where fut
er
Sends may arrive so that will panic too.
// in any case close channel where fut
ure
Sends may arrive so that will panic too.
//close(dst.msgq)
}
// order channels by name
sort
.
Slice
(
sendv
,
func
(
i
,
j
int
)
bool
{
return
strings
.
Compare
(
sendv
[
i
]
.
dst
.
name
,
sendv
[
j
]
.
dst
.
name
)
<
0
return
strings
.
Compare
(
sendv
[
i
]
.
dst
.
name
(),
sendv
[
j
]
.
dst
.
name
()
)
<
0
})
if
len
(
sendv
)
==
0
{
...
...
@@ -283,7 +324,7 @@ func (evc *EventChecker) deadlock(eventp interface{}) {
}
else
{
bad
+=
fmt
.
Sprintf
(
"there are %d sender(s) on other channel(s):
\n
"
,
len
(
sendv
))
for
_
,
__
:=
range
sendv
{
bad
+=
fmt
.
Sprintf
(
"%s:
\t
%T %v
\n
"
,
__
.
dst
.
name
,
__
.
event
,
__
.
event
)
bad
+=
fmt
.
Sprintf
(
"%s:
\t
%T %v
\n
"
,
__
.
dst
.
name
()
,
__
.
event
,
__
.
event
)
}
}
...
...
@@ -312,20 +353,21 @@ type T = EventDispatcher
// EventDispatcher dispatches events to appropriate Chan for checking
// according to provided streams.
type
EventDispatcher
struct
{
streams
map
[
/*name*/
string
]
*
Chan
routeEvent
func
(
event
interface
{})
*
Chan
streams
map
[
/*name*/
string
]
Chan
// XXX -> ChanTx? just str
routeEvent
func
(
event
interface
{})
Chan
// XXX ----//----
}
// NewEventDispatcher creates new dispatcher and provides router to it.
func
NewEventDispatcher
(
streams
Streams
)
*
EventDispatcher
{
// build streams as dict and make sure all streams are named uniquely
streamDict
:=
make
(
map
[
string
]
*
Chan
)
streamDict
:=
make
(
map
[
string
]
Chan
)
for
_
,
ch
:=
range
streams
.
AllStreams
()
{
_
,
already
:=
streamDict
[
ch
.
name
]
name
:=
ch
.
name
()
_
,
already
:=
streamDict
[
name
]
if
already
{
panicf
(
"duplicate stream name %q"
,
ch
.
name
)
panicf
(
"duplicate stream name %q"
,
name
)
}
streamDict
[
ch
.
name
]
=
ch
streamDict
[
name
]
=
ch
}
return
&
EventDispatcher
{
streams
:
streamDict
,
...
...
@@ -361,10 +403,10 @@ func (d *EventDispatcher) Dispatch(event interface{}) {
}
*/
// TODO it is possible to emp
e
rically detect here if a test incorrectly
// TODO it is possible to emp
i
rically detect here if a test incorrectly
// decomposed its system into serial streams: consider unrelated to each
// other events A and B are incorrectly routed to the same channel. It
// could be so happen
n
ing that the order of checks on the test side is
// could be so happening that the order of checks on the test side is
// almost always correct and so the error is not visible. However
//
// if we add delays to delivery of either A or B
...
...
@@ -388,7 +430,7 @@ func (d *EventDispatcher) Dispatch(event interface{}) {
// http://www.1024cores.net/home/relacy-race-detector/rrd-introduction
// TODO timeout: deadlock? (print all-in-flight events on timout)
// TODO timeout: deadlock? (print all-in-flight events on tim
e
out)
// XXX or better ^^^ to do on receiver side?
//
// XXX -> if deadlock detection is done on receiver side (so in
...
...
@@ -401,11 +443,11 @@ func (d *EventDispatcher) Dispatch(event interface{}) {
type
Streams
interface
{
// AllStreams should return all streams of the system.
// Streams must have unique names.
AllStreams
()
[]
*
Chan
AllStreams
()
[]
Chan
// RouteEvent should return particular stream into which event should be routed.
// Returned stream must be one of reported by AllStreams.
RouteEvent
(
event
interface
{})
*
Chan
RouteEvent
(
event
interface
{})
Chan
}
...
...
go/neo/neo_test.go
View file @
454b6576
...
...
@@ -428,7 +428,7 @@ func TestMasterStorage(t0 *testing.T) {
// XXX hack - better we don't need it.
// XXX -> with testenv.MkCluster() we won't need it
type
tdispatch1
struct
{
outch
*
tracetest
.
Chan
outch
tracetest
.
Chan
}
func
(
d
tdispatch1
)
Dispatch
(
event
interface
{})
{
...
...
go/neo/t_events_test.go
View file @
454b6576
...
...
@@ -158,11 +158,11 @@ type EventRouter struct {
mu
sync
.
Mutex
// XXX all unbranched events go to here?
defaultq
*
tracetest
.
Chan
defaultq
tracetest
.
Chan
// events specific to particular node - e.g. node starts listening,
// state on that node changes, etc...
byNode
map
[
string
/*host*/
]
*
tracetest
.
Chan
byNode
map
[
string
/*host*/
]
tracetest
.
Chan
// state on host changes. Takes precendece over byNode.
//
...
...
@@ -172,7 +172,7 @@ type EventRouter struct {
// byNode("C") and simply verify events on tMC and then tC in that order.
// keeping events local to C on tC, not tMC helps TestCluster to
// organize trace channels in uniform way )
byState
map
[
string
/*host*/
]
*
tracetest
.
Chan
byState
map
[
string
/*host*/
]
tracetest
.
Chan
// event on a-b link
byLink
map
[
string
/*host-host*/
]
*
linkDst
...
...
@@ -187,23 +187,23 @@ type EventRouter struct {
// Events go to either a or b depending on which side initiated particular
// connection on top of the link.
type
linkDst
struct
{
a
*
tracetest
.
Chan
// net cause was on dialer
b
*
tracetest
.
Chan
// net cause was on listener
a
tracetest
.
Chan
// net cause was on dialer
b
tracetest
.
Chan
// net cause was on listener
}
func
NewEventRouter
()
*
EventRouter
{
return
&
EventRouter
{
defaultq
:
tracetest
.
NewChan
(
"default"
),
byNode
:
make
(
map
[
string
]
*
tracetest
.
Chan
),
byState
:
make
(
map
[
string
]
*
tracetest
.
Chan
),
byNode
:
make
(
map
[
string
]
tracetest
.
Chan
),
byState
:
make
(
map
[
string
]
tracetest
.
Chan
),
byLink
:
make
(
map
[
string
]
*
linkDst
),
connected
:
make
(
map
[
string
]
bool
),
}
}
func
(
r
*
EventRouter
)
AllStreams
()
[]
*
tracetest
.
Chan
{
rtset
:=
map
[
*
tracetest
.
Chan
]
int
{}
func
(
r
*
EventRouter
)
AllStreams
()
[]
tracetest
.
Chan
{
rtset
:=
map
[
tracetest
.
Chan
]
int
{}
rtset
[
r
.
defaultq
]
=
1
for
_
,
dst
:=
range
r
.
byNode
{
rtset
[
dst
]
=
1
...
...
@@ -216,7 +216,7 @@ func (r *EventRouter) AllStreams() []*tracetest.Chan {
rtset
[
ldst
.
b
]
=
1
}
var
rtv
[]
*
tracetest
.
Chan
var
rtv
[]
tracetest
.
Chan
for
dst
:=
range
rtset
{
rtv
=
append
(
rtv
,
dst
)
}
...
...
@@ -245,7 +245,7 @@ func host(addr string) string {
}
// Route routes events according to rules specified via Branch*().
func
(
r
*
EventRouter
)
RouteEvent
(
event
interface
{})
(
dst
*
tracetest
.
Chan
)
{
func
(
r
*
EventRouter
)
RouteEvent
(
event
interface
{})
(
dst
tracetest
.
Chan
)
{
r
.
mu
.
Lock
()
defer
r
.
mu
.
Unlock
()
...
...
@@ -329,7 +329,7 @@ func (r *EventRouter) RouteEvent(event interface{}) (dst *tracetest.Chan) {
}
// routeState routes event corresponding to state change on host
func
(
r
*
EventRouter
)
routeState
(
host
string
)
(
dst
*
tracetest
.
Chan
)
{
func
(
r
*
EventRouter
)
routeState
(
host
string
)
(
dst
tracetest
.
Chan
)
{
// lookup dst by state rules
dst
=
r
.
byState
[
host
]
if
dst
!=
nil
{
...
...
@@ -341,7 +341,7 @@ func (r *EventRouter) routeState(host string) (dst *tracetest.Chan) {
}
// BranchNode branches events corresponding to host.
func
(
r
*
EventRouter
)
BranchNode
(
host
string
,
dst
*
tracetest
.
Chan
)
{
func
(
r
*
EventRouter
)
BranchNode
(
host
string
,
dst
tracetest
.
Chan
)
{
r
.
mu
.
Lock
()
defer
r
.
mu
.
Unlock
()
...
...
@@ -355,7 +355,7 @@ func (r *EventRouter) BranchNode(host string, dst *tracetest.Chan) {
// BranchState branches events corresponding to state changes on host.
//
// XXX not needed?
func
(
r
*
EventRouter
)
BranchState
(
host
string
,
dst
*
tracetest
.
Chan
)
{
func
(
r
*
EventRouter
)
BranchState
(
host
string
,
dst
tracetest
.
Chan
)
{
r
.
mu
.
Lock
()
defer
r
.
mu
.
Unlock
()
...
...
@@ -375,7 +375,7 @@ func (r *EventRouter) BranchState(host string, dst *tracetest.Chan) {
// networking cause root coming from b - go to dstb.
//
// XXX extend to support multiple ongoing streams (e.g. prefetch) ?
func
(
r
*
EventRouter
)
BranchLink
(
link
string
,
dsta
,
dstb
*
tracetest
.
Chan
)
{
func
(
r
*
EventRouter
)
BranchLink
(
link
string
,
dsta
,
dstb
tracetest
.
Chan
)
{
r
.
mu
.
Lock
()
defer
r
.
mu
.
Unlock
()
...
...
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