whisper.go 2.7 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121
  1. // QWhisper package. This package is temporarily on hold until QML DApp dev will reemerge.
  2. package qwhisper
  3. import (
  4. "time"
  5. "github.com/ethereum/go-ethereum/crypto"
  6. "github.com/ethereum/go-ethereum/ethutil"
  7. "github.com/ethereum/go-ethereum/logger"
  8. "github.com/ethereum/go-ethereum/whisper"
  9. "github.com/obscuren/qml"
  10. )
  11. var qlogger = logger.NewLogger("QSHH")
  12. func fromHex(s string) []byte {
  13. if len(s) > 1 {
  14. return ethutil.Hex2Bytes(s[2:])
  15. }
  16. return nil
  17. }
  18. func toHex(b []byte) string { return "0x" + ethutil.Bytes2Hex(b) }
  19. type Whisper struct {
  20. *whisper.Whisper
  21. view qml.Object
  22. watches map[int]*Watch
  23. }
  24. func New(w *whisper.Whisper) *Whisper {
  25. return &Whisper{w, nil, make(map[int]*Watch)}
  26. }
  27. func (self *Whisper) SetView(view qml.Object) {
  28. self.view = view
  29. }
  30. func (self *Whisper) Post(payload []string, to, from string, topics []string, priority, ttl uint32) {
  31. var data []byte
  32. for _, d := range payload {
  33. data = append(data, fromHex(d)...)
  34. }
  35. pk := crypto.ToECDSAPub(fromHex(from))
  36. if key := self.Whisper.GetIdentity(pk); key != nil {
  37. msg := whisper.NewMessage(data)
  38. envelope, err := msg.Seal(time.Duration(priority*100000), whisper.Opts{
  39. Ttl: time.Duration(ttl) * time.Second,
  40. To: crypto.ToECDSAPub(fromHex(to)),
  41. From: key,
  42. Topics: whisper.TopicsFromString(topics...),
  43. })
  44. if err != nil {
  45. qlogger.Infoln(err)
  46. // handle error
  47. return
  48. }
  49. if err := self.Whisper.Send(envelope); err != nil {
  50. qlogger.Infoln(err)
  51. // handle error
  52. return
  53. }
  54. } else {
  55. qlogger.Infoln("unmatched pub / priv for seal")
  56. }
  57. }
  58. func (self *Whisper) NewIdentity() string {
  59. key := self.Whisper.NewIdentity()
  60. return toHex(crypto.FromECDSAPub(&key.PublicKey))
  61. }
  62. func (self *Whisper) HasIdentity(key string) bool {
  63. return self.Whisper.HasIdentity(crypto.ToECDSAPub(fromHex(key)))
  64. }
  65. func (self *Whisper) Watch(opts map[string]interface{}, view *qml.Common) int {
  66. filter := filterFromMap(opts)
  67. var i int
  68. filter.Fn = func(msg *whisper.Message) {
  69. if view != nil {
  70. view.Call("onShhMessage", ToQMessage(msg), i)
  71. }
  72. }
  73. i = self.Whisper.Watch(filter)
  74. self.watches[i] = &Watch{}
  75. return i
  76. }
  77. func (self *Whisper) Messages(id int) (messages *ethutil.List) {
  78. msgs := self.Whisper.Messages(id)
  79. messages = ethutil.EmptyList()
  80. for _, message := range msgs {
  81. messages.Append(ToQMessage(message))
  82. }
  83. return
  84. }
  85. func filterFromMap(opts map[string]interface{}) (f whisper.Filter) {
  86. if to, ok := opts["to"].(string); ok {
  87. f.To = crypto.ToECDSA(fromHex(to))
  88. }
  89. if from, ok := opts["from"].(string); ok {
  90. f.From = crypto.ToECDSAPub(fromHex(from))
  91. }
  92. if topicList, ok := opts["topics"].(*qml.List); ok {
  93. var topics []string
  94. topicList.Convert(&topics)
  95. f.Topics = whisper.TopicsFromString(topics...)
  96. }
  97. return
  98. }