2018-05-10 16:47:54 +02:00
|
|
|
package shhext
|
|
|
|
|
|
|
|
import (
|
2018-12-14 12:21:34 +01:00
|
|
|
"context"
|
2018-12-11 11:23:47 +01:00
|
|
|
"encoding/hex"
|
2018-05-10 16:47:54 +02:00
|
|
|
"fmt"
|
2019-07-18 00:25:42 +02:00
|
|
|
"github.com/ethereum/go-ethereum/crypto"
|
|
|
|
"github.com/status-im/status-go/params"
|
|
|
|
"io/ioutil"
|
2018-05-10 16:47:54 +02:00
|
|
|
"testing"
|
|
|
|
"time"
|
|
|
|
|
2018-12-11 11:23:47 +01:00
|
|
|
"github.com/ethereum/go-ethereum/common"
|
|
|
|
"github.com/status-im/status-go/mailserver"
|
|
|
|
|
2019-07-18 00:25:42 +02:00
|
|
|
protocol "github.com/status-im/status-protocol-go"
|
2018-10-15 23:15:04 +02:00
|
|
|
whisper "github.com/status-im/whisper/whisperv6"
|
|
|
|
"github.com/stretchr/testify/assert"
|
2018-05-10 16:47:54 +02:00
|
|
|
"github.com/stretchr/testify/require"
|
|
|
|
)
|
|
|
|
|
|
|
|
func TestMessagesRequest_setDefaults(t *testing.T) {
|
|
|
|
daysAgo := func(now time.Time, days int) uint32 {
|
|
|
|
return uint32(now.UTC().Add(-24 * time.Hour * time.Duration(days)).Unix())
|
|
|
|
}
|
|
|
|
|
|
|
|
tnow := time.Now()
|
|
|
|
now := uint32(tnow.UTC().Unix())
|
|
|
|
yesterday := daysAgo(tnow, 1)
|
|
|
|
|
|
|
|
scenarios := []struct {
|
|
|
|
given *MessagesRequest
|
|
|
|
expected *MessagesRequest
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
&MessagesRequest{From: 0, To: 0},
|
2018-06-15 17:12:31 +02:00
|
|
|
&MessagesRequest{From: yesterday, To: now, Timeout: defaultRequestTimeout},
|
2018-05-10 16:47:54 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
&MessagesRequest{From: 1, To: 0},
|
2018-06-15 17:12:31 +02:00
|
|
|
&MessagesRequest{From: uint32(1), To: now, Timeout: defaultRequestTimeout},
|
2018-05-10 16:47:54 +02:00
|
|
|
},
|
|
|
|
{
|
|
|
|
&MessagesRequest{From: 0, To: yesterday},
|
2018-06-15 17:12:31 +02:00
|
|
|
&MessagesRequest{From: daysAgo(tnow, 2), To: yesterday, Timeout: defaultRequestTimeout},
|
2018-05-10 16:47:54 +02:00
|
|
|
},
|
|
|
|
// 100 - 1 day would be invalid, so we set From to 0
|
|
|
|
{
|
|
|
|
&MessagesRequest{From: 0, To: 100},
|
2018-06-15 17:12:31 +02:00
|
|
|
&MessagesRequest{From: 0, To: 100, Timeout: defaultRequestTimeout},
|
|
|
|
},
|
|
|
|
// set Timeout
|
|
|
|
{
|
|
|
|
&MessagesRequest{From: 0, To: 0, Timeout: 100},
|
|
|
|
&MessagesRequest{From: yesterday, To: now, Timeout: 100},
|
2018-05-10 16:47:54 +02:00
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for i, s := range scenarios {
|
|
|
|
t.Run(fmt.Sprintf("Scenario %d", i), func(t *testing.T) {
|
|
|
|
s.given.setDefaults(tnow)
|
|
|
|
require.Equal(t, s.expected, s.given)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2018-10-15 23:15:04 +02:00
|
|
|
|
2018-12-11 11:23:47 +01:00
|
|
|
func TestMakeMessagesRequestPayload(t *testing.T) {
|
2019-05-09 12:58:02 +02:00
|
|
|
var emptyTopic whisper.TopicType
|
2018-12-11 11:23:47 +01:00
|
|
|
testCases := []struct {
|
|
|
|
Name string
|
|
|
|
Req MessagesRequest
|
|
|
|
Err string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Name: "empty cursor",
|
|
|
|
Req: MessagesRequest{Cursor: ""},
|
|
|
|
Err: "",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "invalid cursor size",
|
|
|
|
Req: MessagesRequest{Cursor: hex.EncodeToString([]byte{0x01, 0x02, 0x03})},
|
2019-05-09 12:58:02 +02:00
|
|
|
Err: fmt.Sprintf("invalid cursor size: expected %d but got 3", mailserver.CursorLength),
|
2018-12-11 11:23:47 +01:00
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "valid cursor",
|
|
|
|
Req: MessagesRequest{
|
2019-05-09 12:58:02 +02:00
|
|
|
Cursor: hex.EncodeToString(mailserver.NewDBKey(123, emptyTopic, common.Hash{}).Cursor()),
|
2018-12-11 11:23:47 +01:00
|
|
|
},
|
|
|
|
Err: "",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.Name, func(t *testing.T) {
|
|
|
|
_, err := makeMessagesRequestPayload(tc.Req)
|
|
|
|
if tc.Err == "" {
|
|
|
|
require.NoError(t, err)
|
|
|
|
} else {
|
|
|
|
require.EqualError(t, err, tc.Err)
|
|
|
|
}
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2018-10-15 23:15:04 +02:00
|
|
|
func TestTopicsToBloom(t *testing.T) {
|
|
|
|
t1 := stringToTopic("t1")
|
|
|
|
b1 := whisper.TopicToBloom(t1)
|
|
|
|
t2 := stringToTopic("t2")
|
|
|
|
b2 := whisper.TopicToBloom(t2)
|
|
|
|
t3 := stringToTopic("t3")
|
|
|
|
b3 := whisper.TopicToBloom(t3)
|
|
|
|
|
|
|
|
reqBloom := topicsToBloom(t1)
|
|
|
|
assert.True(t, whisper.BloomFilterMatch(reqBloom, b1))
|
|
|
|
assert.False(t, whisper.BloomFilterMatch(reqBloom, b2))
|
|
|
|
assert.False(t, whisper.BloomFilterMatch(reqBloom, b3))
|
|
|
|
|
|
|
|
reqBloom = topicsToBloom(t1, t2)
|
|
|
|
assert.True(t, whisper.BloomFilterMatch(reqBloom, b1))
|
|
|
|
assert.True(t, whisper.BloomFilterMatch(reqBloom, b2))
|
|
|
|
assert.False(t, whisper.BloomFilterMatch(reqBloom, b3))
|
|
|
|
|
|
|
|
reqBloom = topicsToBloom(t1, t2, t3)
|
|
|
|
assert.True(t, whisper.BloomFilterMatch(reqBloom, b1))
|
|
|
|
assert.True(t, whisper.BloomFilterMatch(reqBloom, b2))
|
|
|
|
assert.True(t, whisper.BloomFilterMatch(reqBloom, b3))
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestCreateBloomFilter(t *testing.T) {
|
|
|
|
t1 := stringToTopic("t1")
|
|
|
|
t2 := stringToTopic("t2")
|
|
|
|
|
|
|
|
req := MessagesRequest{Topic: t1}
|
|
|
|
bloom := createBloomFilter(req)
|
|
|
|
assert.Equal(t, topicsToBloom(t1), bloom)
|
|
|
|
|
|
|
|
req = MessagesRequest{Topics: []whisper.TopicType{t1, t2}}
|
|
|
|
bloom = createBloomFilter(req)
|
|
|
|
assert.Equal(t, topicsToBloom(t1, t2), bloom)
|
|
|
|
}
|
|
|
|
|
|
|
|
func stringToTopic(s string) whisper.TopicType {
|
|
|
|
return whisper.BytesToTopic([]byte(s))
|
|
|
|
}
|
2018-12-14 12:21:34 +01:00
|
|
|
|
|
|
|
func TestCreateSyncMailRequest(t *testing.T) {
|
|
|
|
testCases := []struct {
|
|
|
|
Name string
|
|
|
|
Req SyncMessagesRequest
|
|
|
|
Verify func(*testing.T, whisper.SyncMailRequest)
|
|
|
|
Error string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Name: "no topics",
|
|
|
|
Req: SyncMessagesRequest{},
|
|
|
|
Verify: func(t *testing.T, r whisper.SyncMailRequest) {
|
|
|
|
require.Equal(t, whisper.MakeFullNodeBloom(), r.Bloom)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "some topics",
|
|
|
|
Req: SyncMessagesRequest{
|
|
|
|
Topics: []whisper.TopicType{{0x01, 0xff, 0xff, 0xff}},
|
|
|
|
},
|
|
|
|
Verify: func(t *testing.T, r whisper.SyncMailRequest) {
|
|
|
|
expectedBloom := whisper.TopicToBloom(whisper.TopicType{0x01, 0xff, 0xff, 0xff})
|
|
|
|
require.Equal(t, expectedBloom, r.Bloom)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "decode cursor",
|
|
|
|
Req: SyncMessagesRequest{
|
|
|
|
Cursor: hex.EncodeToString([]byte{0x01, 0x02, 0x03}),
|
|
|
|
},
|
|
|
|
Verify: func(t *testing.T, r whisper.SyncMailRequest) {
|
|
|
|
require.Equal(t, []byte{0x01, 0x02, 0x03}, r.Cursor)
|
|
|
|
},
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.Name, func(t *testing.T) {
|
|
|
|
r, err := createSyncMailRequest(tc.Req)
|
|
|
|
if tc.Error != "" {
|
|
|
|
require.EqualError(t, err, tc.Error)
|
|
|
|
}
|
|
|
|
tc.Verify(t, r)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
func TestSyncMessagesErrors(t *testing.T) {
|
|
|
|
validEnode := "enode://e8a7c03b58911e98bbd66accb2a55d57683f35b23bf9dfca89e5e244eb5cc3f25018b4112db507faca34fb69ffb44b362f79eda97a669a8df29c72e654416784@127.0.0.1:30404"
|
|
|
|
|
|
|
|
testCases := []struct {
|
|
|
|
Name string
|
|
|
|
Req SyncMessagesRequest
|
|
|
|
Resp SyncMessagesResponse
|
|
|
|
Error string
|
|
|
|
}{
|
|
|
|
{
|
|
|
|
Name: "invalid MailServerPeer",
|
|
|
|
Req: SyncMessagesRequest{MailServerPeer: "invalid-scheme://"},
|
|
|
|
Error: `invalid MailServerPeer: invalid URL scheme, want "enode"`,
|
|
|
|
},
|
|
|
|
{
|
|
|
|
Name: "failed to create SyncMailRequest",
|
|
|
|
Req: SyncMessagesRequest{
|
|
|
|
MailServerPeer: validEnode,
|
|
|
|
Cursor: "a", // odd number of characters is an invalid hex representation
|
|
|
|
},
|
|
|
|
Error: "failed to create a sync mail request: encoding/hex: odd length hex string",
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
for _, tc := range testCases {
|
|
|
|
t.Run(tc.Name, func(t *testing.T) {
|
|
|
|
api := PublicAPI{}
|
|
|
|
resp, err := api.SyncMessages(context.TODO(), tc.Req)
|
|
|
|
if tc.Error != "" {
|
|
|
|
require.EqualError(t, err, tc.Error)
|
|
|
|
}
|
|
|
|
require.EqualValues(t, tc.Resp, resp)
|
|
|
|
})
|
|
|
|
}
|
|
|
|
}
|
2019-05-20 11:10:26 +03:00
|
|
|
|
|
|
|
func TestExpiredOrCompleted(t *testing.T) {
|
|
|
|
timeout := time.Millisecond
|
|
|
|
events := make(chan whisper.EnvelopeEvent)
|
|
|
|
errors := make(chan error, 1)
|
|
|
|
hash := common.Hash{1}
|
|
|
|
go func() {
|
|
|
|
_, err := waitForExpiredOrCompleted(hash, events, timeout)
|
|
|
|
errors <- err
|
|
|
|
}()
|
|
|
|
select {
|
|
|
|
case <-time.After(time.Second):
|
|
|
|
require.FailNow(t, "timed out waiting for waitForExpiredOrCompleted to complete")
|
|
|
|
case err := <-errors:
|
|
|
|
require.EqualError(t, err, fmt.Sprintf("request %x expired", hash))
|
|
|
|
}
|
|
|
|
}
|
2019-07-18 00:25:42 +02:00
|
|
|
|
|
|
|
func TestAfterPostIsCalled(t *testing.T) {
|
|
|
|
shhConfig := whisper.DefaultConfig
|
|
|
|
shhConfig.MinimumAcceptedPOW = 0
|
|
|
|
w := whisper.New(&shhConfig)
|
|
|
|
symKeyID, err := w.AddSymKeyFromPassword("abc")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
identity, err := crypto.GenerateKey()
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
tmpDir, err := ioutil.TempDir("", "test-after-post")
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
service := New(w, nil, nil, params.ShhextConfig{})
|
|
|
|
service.messenger, err = protocol.NewMessenger(
|
|
|
|
identity,
|
|
|
|
nil,
|
|
|
|
w,
|
|
|
|
tmpDir,
|
|
|
|
"encKey",
|
|
|
|
"installation1",
|
|
|
|
)
|
|
|
|
require.NoError(t, err)
|
|
|
|
|
|
|
|
api := NewPublicAPI(service)
|
|
|
|
|
|
|
|
// Using api.Post()
|
|
|
|
hash1, err := api.Post(context.Background(), whisper.NewMessage{
|
|
|
|
SymKeyID: symKeyID,
|
|
|
|
Topic: stringToTopic("topic"),
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, EnvelopePosted, service.envelopesMonitor.GetMessageState(common.BytesToHash(hash1)))
|
|
|
|
|
|
|
|
// Using api.SendPublicMessage()
|
|
|
|
hash2, err := api.SendPublicMessage(context.Background(), SendPublicMessageRPC{
|
|
|
|
Chat: "test-channel",
|
|
|
|
Payload: []byte("abc"),
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, EnvelopePosted, service.envelopesMonitor.GetMessageState(common.BytesToHash(hash2)))
|
|
|
|
|
|
|
|
// Using api.SendDirectMessage()
|
|
|
|
recipient, err := crypto.GenerateKey()
|
|
|
|
require.NoError(t, err)
|
|
|
|
hash3, err := api.SendDirectMessage(context.Background(), SendDirectMessageRPC{
|
|
|
|
PubKey: crypto.FromECDSAPub(&recipient.PublicKey),
|
|
|
|
Payload: []byte("abc"),
|
|
|
|
})
|
|
|
|
require.NoError(t, err)
|
|
|
|
require.Equal(t, EnvelopePosted, service.envelopesMonitor.GetMessageState(common.BytesToHash(hash3)))
|
|
|
|
}
|