aboutsummaryrefslogtreecommitdiffstats
path: root/ui/qt/qwhisper/whisper.go
blob: 644c147b7bbd24452c64c5bde78b993582cdadcd (plain) (blame)
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
package qwhisper

import (
    "time"

    "github.com/ethereum/go-ethereum/crypto"
    "github.com/ethereum/go-ethereum/ethutil"
    "github.com/ethereum/go-ethereum/logger"
    "github.com/ethereum/go-ethereum/whisper"
    "gopkg.in/qml.v1"
)

var qlogger = logger.NewLogger("QSHH")

func fromHex(s string) []byte {
    if len(s) > 1 {
        return ethutil.Hex2Bytes(s[2:])
    }
    return nil
}
func toHex(b []byte) string { return "0x" + ethutil.Bytes2Hex(b) }

type Whisper struct {
    *whisper.Whisper
    view qml.Object

    watches map[int]*Watch
}

func New(w *whisper.Whisper) *Whisper {
    return &Whisper{w, nil, make(map[int]*Watch)}
}

func (self *Whisper) SetView(view qml.Object) {
    self.view = view
}

func (self *Whisper) Post(payload []string, to, from string, topics []string, priority, ttl uint32) {
    var data []byte
    for _, d := range payload {
        data = append(data, fromHex(d)...)
    }

    pk := crypto.ToECDSAPub(fromHex(from))
    if key := self.Whisper.GetIdentity(pk); key != nil {
        msg := whisper.NewMessage(data)
        envelope, err := msg.Seal(time.Duration(priority*100000), whisper.Opts{
            Ttl:    time.Duration(ttl) * time.Second,
            To:     crypto.ToECDSAPub(fromHex(to)),
            From:   key,
            Topics: whisper.TopicsFromString(topics...),
        })

        if err != nil {
            qlogger.Infoln(err)
            // handle error
            return
        }

        if err := self.Whisper.Send(envelope); err != nil {
            qlogger.Infoln(err)
            // handle error
            return
        }
    } else {
        qlogger.Infoln("unmatched pub / priv for seal")
    }

}

func (self *Whisper) NewIdentity() string {
    key := self.Whisper.NewIdentity()

    return toHex(crypto.FromECDSAPub(&key.PublicKey))
}

func (self *Whisper) HasIdentity(key string) bool {
    return self.Whisper.HasIdentity(crypto.ToECDSAPub(fromHex(key)))
}

func (self *Whisper) Watch(opts map[string]interface{}, view *qml.Common) int {
    filter := filterFromMap(opts)
    var i int
    filter.Fn = func(msg *whisper.Message) {
        if view != nil {
            view.Call("onShhMessage", ToQMessage(msg), i)
        }
    }

    i = self.Whisper.Watch(filter)
    self.watches[i] = &Watch{}

    return i
}

func (self *Whisper) Messages(id int) (messages *ethutil.List) {
    msgs := self.Whisper.Messages(id)
    messages = ethutil.EmptyList()
    for _, message := range msgs {
        messages.Append(ToQMessage(message))
    }

    return
}

func filterFromMap(opts map[string]interface{}) (f whisper.Filter) {
    if to, ok := opts["to"].(string); ok {
        f.To = crypto.ToECDSA(fromHex(to))
    }
    if from, ok := opts["from"].(string); ok {
        f.From = crypto.ToECDSAPub(fromHex(from))
    }
    if topicList, ok := opts["topics"].(*qml.List); ok {
        var topics []string
        topicList.Convert(&topics)
        f.Topics = whisper.TopicsFromString(topics...)
    }

    return
}