Skip to content
Projects
Groups
Snippets
Help
Loading...
Help
Submit feedback
Contribute to GitLab
Sign in
Toggle navigation
G
Geth-Modification
Project
Project
Details
Activity
Releases
Cycle Analytics
Repository
Repository
Files
Commits
Branches
Tags
Contributors
Graph
Compare
Charts
Issues
0
Issues
0
List
Board
Labels
Milestones
Merge Requests
0
Merge Requests
0
CI / CD
CI / CD
Pipelines
Jobs
Schedules
Charts
Wiki
Wiki
Snippets
Snippets
Members
Members
Collapse sidebar
Close sidebar
Activity
Graph
Charts
Create a new issue
Jobs
Commits
Issue Boards
Open sidebar
张蕾
Geth-Modification
Commits
6fec5bd3
Commit
6fec5bd3
authored
Oct 10, 2014
by
Jeffrey Wilcke
Browse files
Options
Browse Files
Download
Plain Diff
Merge pull request #56 from fjl/feature/raceless-eventer
Fix Eventer race
parents
a38dafcc
44674cb9
Changes
2
Hide whitespace changes
Inline
Side-by-side
Showing
2 changed files
with
60 additions
and
9 deletions
+60
-9
eventer.go
eventer/eventer.go
+8
-4
eventer_test.go
eventer/eventer_test.go
+52
-5
No files found.
eventer/eventer.go
View file @
6fec5bd3
package
eventer
package
eventer
import
"sync"
// Basic receiver interface.
// Basic receiver interface.
type
Receiver
interface
{
type
Receiver
interface
{
Send
(
Event
)
Send
(
Event
)
...
@@ -27,17 +29,18 @@ type Event struct {
...
@@ -27,17 +29,18 @@ type Event struct {
type
Channels
map
[
string
][]
Receiver
type
Channels
map
[
string
][]
Receiver
type
EventMachine
struct
{
type
EventMachine
struct
{
mu
sync
.
RWMutex
channels
Channels
channels
Channels
}
}
func
New
()
*
EventMachine
{
func
New
()
*
EventMachine
{
return
&
EventMachine
{
return
&
EventMachine
{
channels
:
make
(
Channels
)}
channels
:
make
(
Channels
),
}
}
}
func
(
self
*
EventMachine
)
add
(
typ
string
,
r
Receiver
)
{
func
(
self
*
EventMachine
)
add
(
typ
string
,
r
Receiver
)
{
self
.
mu
.
Lock
()
self
.
channels
[
typ
]
=
append
(
self
.
channels
[
typ
],
r
)
self
.
channels
[
typ
]
=
append
(
self
.
channels
[
typ
],
r
)
self
.
mu
.
Unlock
()
}
}
// Generalised methods for the known receiver types
// Generalised methods for the known receiver types
...
@@ -64,11 +67,11 @@ func (self *EventMachine) RegisterFunc(typ string, f Function) {
...
@@ -64,11 +67,11 @@ func (self *EventMachine) RegisterFunc(typ string, f Function) {
func
(
self
*
EventMachine
)
Register
(
typ
string
)
Channel
{
func
(
self
*
EventMachine
)
Register
(
typ
string
)
Channel
{
c
:=
make
(
Channel
,
1
)
c
:=
make
(
Channel
,
1
)
self
.
add
(
typ
,
c
)
self
.
add
(
typ
,
c
)
return
c
return
c
}
}
func
(
self
*
EventMachine
)
Post
(
typ
string
,
data
interface
{})
{
func
(
self
*
EventMachine
)
Post
(
typ
string
,
data
interface
{})
{
self
.
mu
.
RLock
()
if
self
.
channels
[
typ
]
!=
nil
{
if
self
.
channels
[
typ
]
!=
nil
{
ev
:=
Event
{
typ
,
data
}
ev
:=
Event
{
typ
,
data
}
for
_
,
receiver
:=
range
self
.
channels
[
typ
]
{
for
_
,
receiver
:=
range
self
.
channels
[
typ
]
{
...
@@ -76,4 +79,5 @@ func (self *EventMachine) Post(typ string, data interface{}) {
...
@@ -76,4 +79,5 @@ func (self *EventMachine) Post(typ string, data interface{}) {
receiver
.
Send
(
ev
)
receiver
.
Send
(
ev
)
}
}
}
}
self
.
mu
.
RUnlock
()
}
}
eventer/eventer_test.go
View file @
6fec5bd3
package
eventer
package
eventer
import
"testing"
import
(
"math/rand"
"testing"
"time"
)
func
TestChannel
(
t
*
testing
.
T
)
{
func
TestChannel
(
t
*
testing
.
T
)
{
eventer
:=
New
(
nil
)
eventer
:=
New
()
c
:=
make
(
Channel
,
1
)
c
:=
make
(
Channel
,
1
)
eventer
.
RegisterChannel
(
"test"
,
c
)
eventer
.
RegisterChannel
(
"test"
,
c
)
...
@@ -17,7 +21,7 @@ func TestChannel(t *testing.T) {
...
@@ -17,7 +21,7 @@ func TestChannel(t *testing.T) {
}
}
func
TestFunction
(
t
*
testing
.
T
)
{
func
TestFunction
(
t
*
testing
.
T
)
{
eventer
:=
New
(
nil
)
eventer
:=
New
()
var
data
string
var
data
string
eventer
.
RegisterFunc
(
"test"
,
func
(
ev
Event
)
{
eventer
.
RegisterFunc
(
"test"
,
func
(
ev
Event
)
{
...
@@ -31,7 +35,7 @@ func TestFunction(t *testing.T) {
...
@@ -31,7 +35,7 @@ func TestFunction(t *testing.T) {
}
}
func
TestRegister
(
t
*
testing
.
T
)
{
func
TestRegister
(
t
*
testing
.
T
)
{
eventer
:=
New
(
nil
)
eventer
:=
New
()
c
:=
eventer
.
Register
(
"test"
)
c
:=
eventer
.
Register
(
"test"
)
eventer
.
Post
(
"test"
,
"hello world"
)
eventer
.
Post
(
"test"
,
"hello world"
)
...
@@ -44,7 +48,7 @@ func TestRegister(t *testing.T) {
...
@@ -44,7 +48,7 @@ func TestRegister(t *testing.T) {
}
}
func
TestOn
(
t
*
testing
.
T
)
{
func
TestOn
(
t
*
testing
.
T
)
{
eventer
:=
New
(
nil
)
eventer
:=
New
()
c
:=
make
(
Channel
,
1
)
c
:=
make
(
Channel
,
1
)
eventer
.
On
(
"test"
,
c
)
eventer
.
On
(
"test"
,
c
)
...
@@ -64,3 +68,46 @@ func TestOn(t *testing.T) {
...
@@ -64,3 +68,46 @@ func TestOn(t *testing.T) {
t
.
Error
(
"Expected function event with data 'hello world'. Got"
,
data
)
t
.
Error
(
"Expected function event with data 'hello world'. Got"
,
data
)
}
}
}
}
func
TestConcurrentUsage
(
t
*
testing
.
T
)
{
rand
.
Seed
(
time
.
Now
()
.
Unix
())
eventer
:=
New
()
stop
:=
make
(
chan
struct
{})
recv
:=
make
(
chan
int
)
poster
:=
func
()
{
for
{
select
{
case
<-
stop
:
return
default
:
eventer
.
Post
(
"test"
,
"hi"
)
}
}
}
listener
:=
func
(
i
int
)
{
time
.
Sleep
(
time
.
Duration
(
rand
.
Intn
(
99
))
*
time
.
Millisecond
)
c
:=
eventer
.
Register
(
"test"
)
// wait for the first event
<-
c
recv
<-
i
// keep receiving to prevent deadlock
for
{
select
{
case
<-
stop
:
return
case
<-
c
:
}
}
}
nlisteners
:=
200
go
poster
()
for
i
:=
0
;
i
<
nlisteners
;
i
++
{
go
listener
(
i
)
}
// wait until everyone has been served
for
i
:=
0
;
i
<
nlisteners
;
i
++
{
<-
recv
}
close
(
stop
)
}
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