2019-11-21 16:19:22 +00:00
|
|
|
package protocol
|
2019-07-30 18:39:16 +00:00
|
|
|
|
2019-08-29 06:33:46 +00:00
|
|
|
import (
|
|
|
|
"crypto/ecdsa"
|
|
|
|
|
2019-11-23 17:57:05 +00:00
|
|
|
"github.com/status-im/status-go/eth-node/crypto"
|
|
|
|
"github.com/status-im/status-go/eth-node/types"
|
2020-12-15 15:28:05 +00:00
|
|
|
"github.com/status-im/status-go/images"
|
2019-11-21 16:19:22 +00:00
|
|
|
"github.com/status-im/status-go/protocol/identity/alias"
|
|
|
|
"github.com/status-im/status-go/protocol/identity/identicon"
|
2019-08-29 06:33:46 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
const (
|
Move to protobuf for Message type (#1706)
* Use a single Message type `v1/message.go` and `message.go` are the same now, and they embed `protobuf.ChatMessage`
* Use `SendChatMessage` for sending chat messages, this is basically the old `Send` but a bit more flexible so we can send different message types (stickers,commands), and not just text.
* Remove dedup from services/shhext. Because now we process in status-protocol, dedup makes less sense, as those messages are going to be processed anyway, so removing for now, we can re-evaluate if bringing it to status-go or not.
* Change the various retrieveX method to a single one:
`RetrieveAll` will be processing those messages that it can process (Currently only `Message`), and return the rest in `RawMessages` (still transit). The format for the response is:
`Chats`: -> The chats updated by receiving the message
`Messages`: -> The messages retrieved (already matched to a chat)
`Contacts`: -> The contacts updated by the messages
`RawMessages` -> Anything else that can't be parsed, eventually as we move everything to status-protocol-go this will go away.
2019-12-05 16:25:34 +00:00
|
|
|
contactBlocked = ":contact/blocked"
|
|
|
|
contactAdded = ":contact/added"
|
|
|
|
contactRequestReceived = ":contact/request-received"
|
2019-08-29 06:33:46 +00:00
|
|
|
)
|
|
|
|
|
2019-07-30 18:39:16 +00:00
|
|
|
// ContactDeviceInfo is a struct containing information about a particular device owned by a contact
|
|
|
|
type ContactDeviceInfo struct {
|
|
|
|
// The installation id of the device
|
|
|
|
InstallationID string `json:"id"`
|
|
|
|
// Timestamp represents the last time we received this info
|
|
|
|
Timestamp int64 `json:"timestamp"`
|
|
|
|
// FCMToken is to be used for push notifications
|
|
|
|
FCMToken string `json:"fcmToken"`
|
|
|
|
}
|
|
|
|
|
2021-03-31 16:23:45 +00:00
|
|
|
func (c *Contact) CanonicalName() string {
|
|
|
|
if c.LocalNickname != "" {
|
|
|
|
return c.LocalNickname
|
|
|
|
}
|
|
|
|
|
|
|
|
if c.ENSVerified {
|
|
|
|
return c.Name
|
|
|
|
}
|
|
|
|
|
|
|
|
return c.Alias
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c *Contact) CanonicalImage() string {
|
|
|
|
return c.Identicon
|
|
|
|
}
|
|
|
|
|
2019-07-30 18:39:16 +00:00
|
|
|
// Contact has information about a "Contact". A contact is not necessarily one
|
|
|
|
// that we added or added us, that's based on SystemTags.
|
|
|
|
type Contact struct {
|
2019-08-29 06:33:46 +00:00
|
|
|
// ID of the contact. It's a hex-encoded public key (prefixed with 0x).
|
2019-07-30 18:39:16 +00:00
|
|
|
ID string `json:"id"`
|
|
|
|
// Ethereum address of the contact
|
2020-01-20 16:44:32 +00:00
|
|
|
Address string `json:"address,omitempty"`
|
2019-11-04 10:08:22 +00:00
|
|
|
// ENS name of contact
|
2019-09-26 07:01:17 +00:00
|
|
|
Name string `json:"name,omitempty"`
|
2019-11-04 10:08:22 +00:00
|
|
|
// EnsVerified whether we verified the name of the contact
|
|
|
|
ENSVerified bool `json:"ensVerified"`
|
2019-09-26 07:01:17 +00:00
|
|
|
// Generated username name of the contact
|
|
|
|
Alias string `json:"alias,omitempty"`
|
|
|
|
// Identicon generated from public key
|
|
|
|
Identicon string `json:"identicon"`
|
2019-07-30 18:39:16 +00:00
|
|
|
// LastUpdated is the last time we received an update from the contact
|
|
|
|
// updates should be discarded if last updated is less than the one stored
|
2020-01-10 18:59:01 +00:00
|
|
|
LastUpdated uint64 `json:"lastUpdated"`
|
2019-07-30 18:39:16 +00:00
|
|
|
// SystemTags contains information about whether we blocked/added/have been
|
|
|
|
// added.
|
|
|
|
SystemTags []string `json:"systemTags"`
|
|
|
|
|
|
|
|
DeviceInfo []ContactDeviceInfo `json:"deviceInfo"`
|
2020-08-20 14:06:38 +00:00
|
|
|
LocalNickname string `json:"localNickname,omitempty"`
|
2020-12-15 15:28:05 +00:00
|
|
|
|
|
|
|
Images map[string]images.IdentityImage `json:"images"`
|
2019-07-30 18:39:16 +00:00
|
|
|
}
|
2019-08-29 06:33:46 +00:00
|
|
|
|
|
|
|
func (c Contact) PublicKey() (*ecdsa.PublicKey, error) {
|
2019-11-23 17:57:05 +00:00
|
|
|
b, err := types.DecodeHex(c.ID)
|
2019-08-29 06:33:46 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
return crypto.UnmarshalPubkey(b)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c Contact) IsAdded() bool {
|
|
|
|
return existsInStringSlice(c.SystemTags, contactAdded)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c Contact) HasBeenAdded() bool {
|
|
|
|
return existsInStringSlice(c.SystemTags, contactRequestReceived)
|
|
|
|
}
|
|
|
|
|
|
|
|
func (c Contact) IsBlocked() bool {
|
|
|
|
return existsInStringSlice(c.SystemTags, contactBlocked)
|
|
|
|
}
|
|
|
|
|
2020-12-22 10:49:25 +00:00
|
|
|
func (c *Contact) Remove() {
|
|
|
|
var newSystemTags []string
|
|
|
|
// Remove the newSystemTags system-tag, so that the contact is
|
|
|
|
// not considered "added" anymore
|
|
|
|
for _, tag := range newSystemTags {
|
|
|
|
if tag != contactAdded {
|
|
|
|
newSystemTags = append(newSystemTags, tag)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
c.SystemTags = newSystemTags
|
|
|
|
}
|
|
|
|
|
2019-08-29 06:33:46 +00:00
|
|
|
// existsInStringSlice checks if a string is in a set.
|
|
|
|
func existsInStringSlice(set []string, find string) bool {
|
|
|
|
for _, s := range set {
|
|
|
|
if s == find {
|
|
|
|
return true
|
|
|
|
}
|
|
|
|
}
|
|
|
|
return false
|
|
|
|
}
|
2019-11-04 10:08:22 +00:00
|
|
|
|
2020-12-22 10:49:25 +00:00
|
|
|
func buildContactFromPkString(pkString string) (*Contact, error) {
|
|
|
|
publicKeyBytes, err := types.DecodeHex(pkString)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
publicKey, err := crypto.UnmarshalPubkey(publicKeyBytes)
|
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
return buildContact(pkString, publicKey)
|
|
|
|
}
|
|
|
|
|
|
|
|
func buildContactFromPublicKey(publicKey *ecdsa.PublicKey) (*Contact, error) {
|
|
|
|
id := types.EncodeHex(crypto.FromECDSAPub(publicKey))
|
|
|
|
return buildContact(id, publicKey)
|
|
|
|
}
|
2019-11-04 10:08:22 +00:00
|
|
|
|
2020-12-22 10:49:25 +00:00
|
|
|
func buildContact(publicKeyString string, publicKey *ecdsa.PublicKey) (*Contact, error) {
|
|
|
|
identicon, err := identicon.GenerateBase64(publicKeyString)
|
2019-11-04 10:08:22 +00:00
|
|
|
if err != nil {
|
|
|
|
return nil, err
|
|
|
|
}
|
|
|
|
|
|
|
|
contact := &Contact{
|
2020-12-22 10:49:25 +00:00
|
|
|
ID: publicKeyString,
|
2019-11-04 10:08:22 +00:00
|
|
|
Alias: alias.GenerateFromPublicKey(publicKey),
|
|
|
|
Identicon: identicon,
|
|
|
|
}
|
|
|
|
|
|
|
|
return contact, nil
|
|
|
|
}
|
2020-01-15 07:25:09 +00:00
|
|
|
|
2020-04-17 11:22:38 +00:00
|
|
|
// HasCustomFields returns whether the the contact has any field that is valuable
|
|
|
|
// to the client other than the computed name/image
|
|
|
|
func (c Contact) HasCustomFields() bool {
|
2020-12-15 16:23:48 +00:00
|
|
|
return c.IsAdded() || c.HasBeenAdded() || c.IsBlocked() || c.ENSVerified || c.LocalNickname != "" || len(c.Images) != 0
|
2020-04-17 11:22:38 +00:00
|
|
|
}
|
|
|
|
|
2020-01-15 07:25:09 +00:00
|
|
|
func contactIDFromPublicKey(key *ecdsa.PublicKey) string {
|
|
|
|
return types.EncodeHex(crypto.FromECDSAPub(key))
|
|
|
|
|
|
|
|
}
|