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
082ba218
Commit
082ba218
authored
Sep 05, 2017
by
Kirill Smelkov
Browse files
Options
Browse Files
Download
Email Patches
Plain Diff
.
parent
42fd27c5
Changes
9
Hide whitespace changes
Inline
Side-by-side
Showing
9 changed files
with
175 additions
and
49 deletions
+175
-49
go/neo/neo.go
go/neo/neo.go
+5
-6
go/neo/nodetab.go
go/neo/nodetab.go
+5
-1
go/neo/proto-misc.go
go/neo/proto-misc.go
+56
-10
go/neo/proto.go
go/neo/proto.go
+38
-9
go/neo/proto_test.go
go/neo/proto_test.go
+55
-7
go/neo/protogen.go
go/neo/protogen.go
+1
-1
go/neo/server/cluster_test.go
go/neo/server/cluster_test.go
+11
-11
go/neo/t/t.sh
go/neo/t/t.sh
+2
-2
go/neo/t/zsha1.go
go/neo/t/zsha1.go
+2
-2
No files found.
go/neo/neo.go
View file @
082ba218
...
...
@@ -30,7 +30,6 @@ package neo
import
(
"context"
"fmt"
"math"
"net"
"sync"
...
...
@@ -80,7 +79,7 @@ func NewNodeApp(net xnet.Networker, typ NodeType, clusterName, masterAddr, serve
}
app
:=
&
NodeApp
{
MyInfo
:
NodeInfo
{
Type
:
typ
,
Addr
:
addr
,
IdTime
stamp
:
math
.
NaN
()
},
MyInfo
:
NodeInfo
{
Type
:
typ
,
Addr
:
addr
,
IdTime
:
0
},
ClusterName
:
clusterName
,
Net
:
net
,
MasterAddr
:
masterAddr
,
...
...
@@ -129,7 +128,7 @@ func (app *NodeApp) Dial(ctx context.Context, peerType NodeType, addr string) (_
UUID
:
app
.
MyInfo
.
UUID
,
Address
:
app
.
MyInfo
.
Addr
,
ClusterName
:
app
.
ClusterName
,
IdTime
stamp
:
app
.
MyInfo
.
IdTimestamp
,
// XXX ok?
IdTime
:
app
.
MyInfo
.
IdTime
,
// XXX ok?
}
accept
:=
&
AcceptIdentification
{}
// FIXME error if peer sends us something with another connID
...
...
@@ -332,17 +331,17 @@ func (l *listener) Addr() net.Addr {
// UpdateNodeTab applies updates to .NodeTab from message and logs changes appropriately.
func
(
app
*
NodeApp
)
UpdateNodeTab
(
ctx
context
.
Context
,
msg
*
NotifyNodeInformation
)
{
// XXX msg.IdTime
stamp
?
// XXX msg.IdTime ?
for
_
,
nodeInfo
:=
range
msg
.
NodeList
{
log
.
Infof
(
ctx
,
"rx node update: %v"
,
nodeInfo
)
app
.
NodeTab
.
Update
(
nodeInfo
)
// XXX we have to provide IdTime
stamp
when requesting identification to other peers
// XXX we have to provide IdTime when requesting identification to other peers
// (e.g. Spy checks this is what master broadcast them and if not replis "unknown by master")
if
nodeInfo
.
UUID
==
app
.
MyInfo
.
UUID
{
// XXX recheck locking
// XXX do .MyInfo = nodeInfo ?
app
.
MyInfo
.
IdTime
stamp
=
nodeInfo
.
IdTimestamp
app
.
MyInfo
.
IdTime
=
nodeInfo
.
IdTime
}
}
...
...
go/neo/nodetab.go
View file @
082ba218
...
...
@@ -178,7 +178,11 @@ func (nt *NodeTable) String() string {
// XXX also for .storv
for
_
,
n
:=
range
nt
.
nodev
{
// XXX recheck output
fmt
.
Fprintf
(
&
buf
,
"%s (%s)
\t
%s
\t
%s
\n
"
,
n
.
UUID
,
n
.
Type
,
n
.
State
,
n
.
Addr
)
fmt
.
Fprintf
(
&
buf
,
"%s (%s)
\t
%s
\t
%s"
,
n
.
UUID
,
n
.
Type
,
n
.
State
,
n
.
Addr
)
if
n
.
IdTime
.
Valid
()
{
fmt
.
Fprintf
(
&
buf
,
"
\t
@ %v"
,
n
.
IdTime
)
}
fmt
.
Fprintf
(
&
buf
,
"
\n
"
)
}
return
buf
.
String
()
...
...
go/neo/proto-misc.go
View file @
082ba218
...
...
@@ -27,6 +27,7 @@ import (
"net"
"strconv"
"strings"
"time"
)
// XXX or better translate to some other errors ?
...
...
@@ -47,51 +48,96 @@ func (cs *ClusterState) Set(v ClusterState) {
traceClusterStateChanged
(
cs
)
}
const
nodeTypeChar
=
"MSCA4567"
// keep in sync with NodeType constants
//const nodeTypeChar = "MSCA????" // keep in sync with NodeType constants
const
nodeTypeChar
=
"SMCA"
// XXX neo/py does this out of sync with NodeType constants
// String returns string representation of a node uuid.
// It returns ex 'S1', 'M2', ...
func
(
nodeUUID
NodeUUID
)
String
()
string
{
if
nodeUUID
==
0
{
return
"?0"
return
"?
(0)
0"
}
typ
:=
nodeUUID
>>
24
num
:=
nodeUUID
&
(
1
<<
24
-
1
)
temp
:=
typ
&
(
1
<<
7
)
!=
0
typ
&=
1
<<
7
-
1
// XXX UUID_NAMESPACES description does not match neo/py code
//typ := nodeUUID >> 24
//temp := typ&(1 << 7) != 0
//typ &= 1<<7 - 1
//nodeType := typ >> 4
typ
:=
uint8
(
-
int8
(
nodeUUID
>>
24
))
>>
4
nodeType
:=
NodeType
(
typ
>>
4
)
s
:=
fmt
.
Sprintf
(
"%c%d"
,
nodeTypeChar
[
nodeType
],
num
)
if
typ
<
4
{
return
fmt
.
Sprintf
(
"%c%d"
,
nodeTypeChar
[
typ
],
num
)
}
return
fmt
.
Sprintf
(
"?(%d)%d"
,
typ
,
num
)
/*
// 's1', 'm2', for temporary nids
if temp {
s = strings.ToLower(s)
}
return s
*/
}
// XXX goes out of sync wrt NodeType constants
var
nodeTypeNum
=
[
...
]
int8
{
STORAGE
:
0x00
,
MASTER
:
-
0x10
,
CLIENT
:
-
0x20
,
ADMIN
:
-
0x30
,
}
// UUID creates node uuid from node type and number.
// XXX test
func
UUID
(
typ
NodeType
,
num
int32
)
NodeUUID
{
// XXX neo/py does not what UUID_NAMESPACES describes
/*
temp := uint32(0)
if num < 0 {
temp = 1
num = -num
}
*/
if
int
(
typ
)
>=
len
(
nodeTypeNum
)
{
panic
(
"typ invalid"
)
}
typn
:=
nodeTypeNum
[
typ
]
if
num
>>
24
!=
0
{
if
(
num
<
0
)
||
num
>>
24
!=
0
{
panic
(
"node number out of range"
)
}
uuid
:=
temp
<<
(
7
+
3
*
8
)
|
uint32
(
typ
)
<<
(
4
+
3
*
8
)
|
uint32
(
num
)
//uuid := temp << (7 + 3*8) | uint32(typ) << (4 + 3*8) | uint32(num)
uuid
:=
uint32
(
uint8
(
typn
))
<<
(
3
*
8
)
|
uint32
(
num
)
return
NodeUUID
(
uuid
)
}
// ----------------------------------------
// Valid returns whether t was initialized
func
(
t
IdTime
)
Valid
()
bool
{
return
t
!=
0
}
func
(
t
IdTime
)
String
()
string
{
if
!
t
.
Valid
()
{
return
"ø"
}
sec
:=
int64
(
t
)
nsec
:=
int64
((
float64
(
t
)
-
float64
(
sec
))
*
1E9
)
return
time
.
Unix
(
sec
,
nsec
)
.
String
()
}
// ----------------------------------------
// 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
)
{
...
...
go/neo/proto.go
View file @
082ba218
...
...
@@ -269,15 +269,44 @@ type Checksum [20]byte
// Zero value means "invalid id" (<-> None in py.PPTID)
type
PTid
uint64
// IdTime represents time of identification
type
IdTime
float64
func
(
t
IdTime
)
neoEncodedLen
()
int
{
return
8
}
func
(
t
IdTime
)
neoEncode
(
b
[]
byte
)
int
{
// use 0 as value for no data (NaN != NaN -> hard to use NaN in tests)
// NOTE neo/py uses None for "no data"; we use 0 for "no data" to avoid pointer
tt
:=
float64
(
t
)
if
tt
==
0
{
tt
=
math
.
NaN
()
}
float64_neoEncode
(
b
,
tt
)
return
8
}
func
(
t
*
IdTime
)
neoDecode
(
data
[]
byte
)
(
uint32
,
bool
)
{
if
len
(
data
)
<
8
{
return
0
,
false
}
tt
:=
float64_neoDecode
(
data
)
if
math
.
IsNaN
(
tt
)
{
tt
=
0
}
*
t
=
IdTime
(
tt
)
return
8
,
true
}
// NodeInfo is information about a node
//neo:proto typeonly
type
NodeInfo
struct
{
Type
NodeType
Addr
Address
// serving address
UUID
NodeUUID
State
NodeState
IdTime
stamp
float64
// FIXME clarify semantic where it is used
Type
NodeType
Addr
Address
// serving address
UUID
NodeUUID
State
NodeState
IdTime
IdTime
// FIXME clarify semantic where it is used
}
//neo:proto typeonly
...
...
@@ -311,7 +340,7 @@ type RequestIdentification struct {
UUID
NodeUUID
Address
Address
// where requesting node is also accepting connections
ClusterName
string
IdTime
stamp
float64
IdTime
IdTime
}
//neo:proto answer
...
...
@@ -352,7 +381,7 @@ type NotPrimaryMaster struct {
// Notify information about one or more nodes. PM -> Any.
type
NotifyNodeInformation
struct
{
// XXX in py this is monotonic_time() of call to broadcastNodesInformation() & friends
IdTime
stamp
float64
IdTime
IdTime
NodeList
[]
NodeInfo
}
...
...
@@ -1042,7 +1071,7 @@ func bool2byte(b bool) byte {
// NOTE py.None encodes as '\xff' * 8 (-> we use NaN for None)
// NOTE '\xff' * 8 represents FP NaN but many other NaN bits representations exist
func
float64_
NEO
Encode
(
b
[]
byte
,
f
float64
)
{
func
float64_
neo
Encode
(
b
[]
byte
,
f
float64
)
{
var
fu
uint64
if
!
math
.
IsNaN
(
f
)
{
fu
=
math
.
Float64bits
(
f
)
...
...
@@ -1054,7 +1083,7 @@ func float64_NEOEncode(b []byte, f float64) {
binary
.
BigEndian
.
PutUint64
(
b
,
fu
)
}
func
float64_
NEO
Decode
(
b
[]
byte
)
float64
{
func
float64_
neo
Decode
(
b
[]
byte
)
float64
{
fu
:=
binary
.
BigEndian
.
Uint64
(
b
)
return
math
.
Float64frombits
(
fu
)
}
...
...
go/neo/proto_test.go
View file @
082ba218
...
...
@@ -244,7 +244,7 @@ func TestMsgMarshal(t *testing.T) {
u32
(
7
)
+
u32
(
4
)
+
u32
(
1
)
+
u32
(
3
)
+
u32
(
9
)
+
u32
(
4
),
},
// uint32, Address, string,
float64
// uint32, Address, string,
IdTime
{
&
RequestIdentification
{
CLIENT
,
17
,
Address
{
"localhost"
,
7777
},
"myname"
,
0.12345678
},
u32
(
2
)
+
u32
(
17
)
+
u32
(
9
)
+
...
...
@@ -253,22 +253,21 @@ func TestMsgMarshal(t *testing.T) {
hex
(
"3fbf9add1091c895"
),
},
//
float64
, empty Address, int32
//
IdTime
, empty Address, int32
{
&
NotifyNodeInformation
{
1504466245.926185
,
[]
NodeInfo
{
{
CLIENT
,
Address
{},
UUID
(
CLIENT
,
1
),
RUNNING
,
1504466245.925599
}}},
hex
(
"41d66b15517b469d"
)
+
u32
(
1
)
+
u32
(
2
)
+
u32
(
0
)
/* <- ø Address */
+
hex
(
"
2
0000001"
)
+
u32
(
2
)
+
u32
(
2
)
+
u32
(
0
)
/* <- ø Address */
+
hex
(
"
e
0000001"
)
+
u32
(
2
)
+
hex
(
"41d66b15517b3d04"
),
},
// empty IdTime
{
&
NotifyNodeInformation
{
0
,
[]
NodeInfo
{}},
hex
(
"ffffffffffffffff"
)
+
hex
(
"00000000"
)},
// TODO we need tests for:
// []varsize + trailing
// map[]varsize + trailing
// TODO special cases for:
// - float64 (+ nan !nan ...)
}
for
_
,
tt
:=
range
testv
{
...
...
@@ -294,3 +293,52 @@ func TestMsgMarshalAllOverflowLightly(t *testing.T) {
testMsgMarshal
(
t
,
msg
,
string
(
zerol
))
}
}
func
TestUUID
(
t
*
testing
.
T
)
{
var
testv
=
[]
struct
{
typ
NodeType
;
num
int32
;
uuid
uint32
;
str
string
}{
{
STORAGE
,
1
,
0x00000001
,
"S1"
},
{
MASTER
,
2
,
0xf0000002
,
"M2"
},
{
CLIENT
,
3
,
0xe0000003
,
"C3"
},
{
ADMIN
,
4
,
0xd0000004
,
"A4"
},
}
for
_
,
tt
:=
range
testv
{
uuid
:=
UUID
(
tt
.
typ
,
tt
.
num
)
if
uint32
(
uuid
)
!=
tt
.
uuid
{
t
.
Errorf
(
"%v: uuid=%08x ; want %08x"
,
tt
,
uuid
,
tt
.
uuid
)
}
if
uuids
:=
uuid
.
String
();
uuids
!=
tt
.
str
{
t
.
Errorf
(
"%v: str(uuid): %q ; want %q"
,
tt
,
uuids
,
tt
.
str
)
}
}
}
func
TestUUIDDecode
(
t
*
testing
.
T
)
{
var
testv
=
[]
struct
{
uuid
uint32
;
str
string
}{
{
0
,
"?(0)0"
},
{
0x00000001
,
"S1"
},
{
0xf0000002
,
"M2"
},
{
0xe0000003
,
"C3"
},
{
0xd0000004
,
"A4"
},
{
0xc0000005
,
"?(4)5"
},
{
0xb0000006
,
"?(5)6"
},
{
0xa0000007
,
"?(6)7"
},
{
0x90000008
,
"?(7)8"
},
{
0x80000009
,
"?(8)9"
},
{
0x7000000a
,
"?(9)10"
},
{
0x6000000b
,
"?(10)11"
},
{
0x5000000c
,
"?(11)12"
},
{
0x4000000d
,
"?(12)13"
},
{
0x3000000e
,
"?(13)14"
},
{
0x2000000f
,
"?(14)15"
},
{
0x10000010
,
"?(15)16"
},
{
0x00000011
,
"S17"
},
}
for
_
,
tt
:=
range
testv
{
str
:=
NodeUUID
(
tt
.
uuid
)
.
String
()
if
str
!=
tt
.
str
{
t
.
Errorf
(
"%08x -> %q ; want %q"
,
tt
.
uuid
,
str
,
tt
.
str
)
}
}
}
go/neo/protogen.go
View file @
082ba218
...
...
@@ -390,7 +390,7 @@ var basicTypes = map[types.BasicKind]basicCodec{
types
.
Uint32
:
{
4
,
"binary.BigEndian.PutUint32(%v, %v)"
,
"binary.BigEndian.Uint32(%v)"
},
types
.
Uint64
:
{
8
,
"binary.BigEndian.PutUint64(%v, %v)"
,
"binary.BigEndian.Uint64(%v)"
},
types
.
Float64
:
{
8
,
"float64_
NEOEncode(%v, %v)"
,
"float64_NEO
Decode(%v)"
},
types
.
Float64
:
{
8
,
"float64_
neoEncode(%v, %v)"
,
"float64_neo
Decode(%v)"
},
}
// does a type have fixed wire size and, if yes, what it is?
...
...
go/neo/server/cluster_test.go
View file @
082ba218
...
...
@@ -229,7 +229,7 @@ func TestMasterStorage(t *testing.T) {
gwg
:=
&
xsync
.
WorkGroup
{}
// start master
Mclock
:=
&
vclock
{}
Mclock
:=
&
vclock
{
100
}
M
:=
NewMaster
(
"abc1"
,
":1"
,
Mhost
)
M
.
monotime
=
Mclock
.
monotime
Mctx
,
Mcancel
:=
context
.
WithCancel
(
bg
)
...
...
@@ -241,7 +241,7 @@ func TestMasterStorage(t *testing.T) {
// M starts listening
tc
.
Expect
(
netlisten
(
"m:1"
))
tc
.
Expect
(
node
(
M
.
node
,
"m:1"
,
neo
.
MASTER
,
1
,
neo
.
RUNNING
,
0
.0
))
tc
.
Expect
(
node
(
M
.
node
,
"m:1"
,
neo
.
MASTER
,
1
,
neo
.
RUNNING
,
0
))
tc
.
Expect
(
clusterState
(
&
M
.
node
.
ClusterState
,
neo
.
ClusterRecovering
))
// TODO create C; C tries connect to master - rejected ("not yet operational")
...
...
@@ -269,7 +269,7 @@ func TestMasterStorage(t *testing.T) {
IdTimestamp
:
0
,
}))
tc
.
Expect
(
node
(
M
.
node
,
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
PENDING
,
0.01
))
tc
.
Expect
(
node
(
M
.
node
,
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
PENDING
,
10
0.01
))
tc
.
Expect
(
conntx
(
"m:2"
,
"s:2"
,
1
,
&
neo
.
AcceptIdentification
{
NodeType
:
neo
.
MASTER
,
...
...
@@ -306,7 +306,7 @@ func TestMasterStorage(t *testing.T) {
exc
.
Raiseif
(
err
)
})
tc
.
Expect
(
node
(
M
.
node
,
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
RUNNING
,
0.01
))
tc
.
Expect
(
node
(
M
.
node
,
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
RUNNING
,
10
0.01
))
xwait
(
wg
)
// XXX M.partTab <- S1
...
...
@@ -367,7 +367,7 @@ func TestMasterStorage(t *testing.T) {
IdTimestamp
:
0
,
}))
tc
.
Expect
(
node
(
M
.
node
,
""
,
neo
.
CLIENT
,
1
,
neo
.
RUNNING
,
0.02
))
tc
.
Expect
(
node
(
M
.
node
,
""
,
neo
.
CLIENT
,
1
,
neo
.
RUNNING
,
10
0.02
))
tc
.
Expect
(
conntx
(
"m:3"
,
"c:1"
,
1
,
&
neo
.
AcceptIdentification
{
NodeType
:
neo
.
MASTER
,
...
...
@@ -392,16 +392,16 @@ func TestMasterStorage(t *testing.T) {
tc
.
Expect
(
conntx
(
"m:3"
,
"c:1"
,
0
,
&
neo
.
NotifyNodeInformation
{
IdTimestamp
:
0
,
// XXX ?
NodeList
:
[]
neo
.
NodeInfo
{
nodei
(
"m:1"
,
neo
.
MASTER
,
1
,
neo
.
RUNNING
,
0
.00
),
nodei
(
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
RUNNING
,
0.01
),
nodei
(
""
,
neo
.
CLIENT
,
1
,
neo
.
RUNNING
,
0.02
),
nodei
(
"m:1"
,
neo
.
MASTER
,
1
,
neo
.
RUNNING
,
0
),
nodei
(
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
RUNNING
,
10
0.01
),
nodei
(
""
,
neo
.
CLIENT
,
1
,
neo
.
RUNNING
,
10
0.02
),
},
}))
Cnode
:=
*
(
**
neo
.
NodeApp
)(
unsafe
.
Pointer
(
C
))
// XXX hack, fragile
tc
.
Expect
(
node
(
Cnode
,
"m:1"
,
neo
.
MASTER
,
1
,
neo
.
RUNNING
,
0
.00
))
tc
.
Expect
(
node
(
Cnode
,
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
RUNNING
,
0.01
))
tc
.
Expect
(
node
(
Cnode
,
""
,
neo
.
CLIENT
,
1
,
neo
.
RUNNING
,
0.02
))
tc
.
Expect
(
node
(
Cnode
,
"m:1"
,
neo
.
MASTER
,
1
,
neo
.
RUNNING
,
0
))
tc
.
Expect
(
node
(
Cnode
,
"s:1"
,
neo
.
STORAGE
,
1
,
neo
.
RUNNING
,
10
0.01
))
tc
.
Expect
(
node
(
Cnode
,
""
,
neo
.
CLIENT
,
1
,
neo
.
RUNNING
,
10
0.02
))
// C asks M about last tid XXX better master sends it itself on new client connected
...
...
go/neo/t/t.sh
View file @
082ba218
...
...
@@ -208,8 +208,8 @@ gensqlite() {
#neopylite
neopysql
#time demo-zbigarray read neo://$cluster@$Mbind
./zsha1.py neo://
$cluster
@
$Mbind
go run zsha1.go neo://
$cluster
@
$Mbind
#
./zsha1.py neo://$cluster@$Mbind
go run zsha1.go
-logtostderr
neo://
$cluster
@
$Mbind
xneoctl
set
cluster stopping
xmysql
-e
"SHUTDOWN"
...
...
go/neo/t/zsha1.go
View file @
082ba218
...
...
@@ -9,7 +9,7 @@ import (
"log"
"flag"
"fmt"
"os"
//
"os"
"time"
"lab.nexedi.com/kirr/neo/go/zodb"
...
...
@@ -20,7 +20,7 @@ import (
func
main
()
{
flag
.
Parse
()
url
:=
os
.
Args
[
1
]
// XXX dirty
url
:=
flag
.
Args
()[
0
]
// XXX dirty
bg
:=
context
.
Background
()
stor
,
err
:=
zodb
.
OpenStorageURL
(
bg
,
url
)
...
...
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