2022-02-17 15:13:10 +00:00
|
|
|
package protocol
|
|
|
|
|
|
|
|
import (
|
2023-04-19 22:59:09 +00:00
|
|
|
"context"
|
2022-02-17 15:13:10 +00:00
|
|
|
"errors"
|
2022-11-24 20:48:26 +00:00
|
|
|
"fmt"
|
2022-02-17 15:13:10 +00:00
|
|
|
"regexp"
|
2023-03-08 14:04:02 +00:00
|
|
|
"runtime"
|
2022-02-17 15:13:10 +00:00
|
|
|
"strings"
|
|
|
|
|
2022-03-23 18:47:00 +00:00
|
|
|
"github.com/status-im/status-go/multiaccounts/settings"
|
2023-06-05 11:10:26 +00:00
|
|
|
sociallinkssettings "github.com/status-im/status-go/multiaccounts/settings_social_links"
|
2022-11-24 20:48:26 +00:00
|
|
|
"github.com/status-im/status-go/protocol/encryption/multidevice"
|
2022-08-05 11:22:35 +00:00
|
|
|
"github.com/status-im/status-go/protocol/identity"
|
2022-02-17 15:13:10 +00:00
|
|
|
"github.com/status-im/status-go/protocol/identity/alias"
|
2022-11-24 20:48:26 +00:00
|
|
|
"github.com/status-im/status-go/server"
|
2022-02-17 15:13:10 +00:00
|
|
|
)
|
|
|
|
|
2022-08-10 13:09:15 +00:00
|
|
|
const (
|
|
|
|
maxBioLength = 240
|
|
|
|
maxSocialLinkTextLength = 24
|
|
|
|
)
|
|
|
|
|
2022-02-17 15:13:10 +00:00
|
|
|
var ErrInvalidDisplayNameRegExp = errors.New("only letters, numbers, underscores and hyphens allowed")
|
|
|
|
var ErrInvalidDisplayNameEthSuffix = errors.New(`usernames ending with "eth" are not allowed`)
|
|
|
|
var ErrInvalidDisplayNameNotAllowed = errors.New("name is not allowed")
|
2022-08-10 13:09:15 +00:00
|
|
|
var ErrInvalidBioLength = errors.New("invalid bio length")
|
|
|
|
var ErrInvalidSocialLinkTextLength = errors.New("invalid social link text length")
|
2024-03-01 12:46:48 +00:00
|
|
|
var ErrDisplayNameDupeOfCommunityMember = errors.New("display name duplicates on of community members")
|
2022-02-17 15:13:10 +00:00
|
|
|
|
|
|
|
func ValidateDisplayName(displayName *string) error {
|
|
|
|
name := strings.TrimSpace(*displayName)
|
|
|
|
*displayName = name
|
|
|
|
|
|
|
|
if name == "" {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
// ^[\\w-\\s]{5,24}$ to allow spaces
|
|
|
|
if match, _ := regexp.MatchString("^[\\w-\\s]{5,24}$", name); !match {
|
|
|
|
return ErrInvalidDisplayNameRegExp
|
|
|
|
}
|
|
|
|
|
|
|
|
// .eth should not happen due to the regexp above, but let's keep it here in case the regexp is changed in the future
|
|
|
|
if strings.HasSuffix(name, "_eth") || strings.HasSuffix(name, ".eth") || strings.HasSuffix(name, "-eth") {
|
|
|
|
return ErrInvalidDisplayNameEthSuffix
|
|
|
|
}
|
|
|
|
|
|
|
|
if alias.IsAlias(name) {
|
|
|
|
return ErrInvalidDisplayNameNotAllowed
|
|
|
|
}
|
|
|
|
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-30 10:18:13 +00:00
|
|
|
func (m *Messenger) SetDisplayName(displayName string) error {
|
2022-02-17 15:13:10 +00:00
|
|
|
currDisplayName, err := m.settings.DisplayName()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if currDisplayName == displayName {
|
|
|
|
return nil // Do nothing
|
|
|
|
}
|
|
|
|
|
|
|
|
if err = ValidateDisplayName(&displayName); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-03-01 12:46:48 +00:00
|
|
|
isDupe, err := m.IsDisplayNameDupeOfCommunityMember(displayName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if isDupe {
|
|
|
|
return ErrDisplayNameDupeOfCommunityMember
|
|
|
|
}
|
|
|
|
|
2023-03-30 10:18:13 +00:00
|
|
|
m.account.Name = displayName
|
2024-02-01 15:43:41 +00:00
|
|
|
err = m.multiAccounts.UpdateDisplayName(m.account.KeyUID, displayName)
|
2022-02-17 15:13:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-03-23 18:47:00 +00:00
|
|
|
err = m.settings.SaveSettingField(settings.DisplayName, displayName)
|
2022-02-17 15:13:10 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-05-24 14:42:31 +00:00
|
|
|
err = m.UpdateKeypairName(m.account.KeyUID, displayName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-03-30 10:18:13 +00:00
|
|
|
err = m.resetLastPublishedTimeForChatIdentity()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-11-30 09:41:35 +00:00
|
|
|
|
2023-03-30 10:18:13 +00:00
|
|
|
return m.publishContactCode()
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Messenger) SaveSyncDisplayName(displayName string, clock uint64) error {
|
|
|
|
err := m.settings.SaveSyncSetting(settings.DisplayName, displayName, clock)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2022-02-17 15:13:10 +00:00
|
|
|
}
|
2024-02-29 12:44:35 +00:00
|
|
|
|
2024-03-25 15:43:45 +00:00
|
|
|
preferredName, err := m.settings.GetPreferredUsername()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2024-03-05 13:03:10 +00:00
|
|
|
preferredNameClock, err := m.settings.GetSettingLastSynced(settings.PreferredName)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
// When either the display name or preferred name changes, m.account.Name should be updated.
|
|
|
|
// However, a race condition can occur during BackupData, where m.account.Name could be incorrectly updated.
|
2024-03-25 15:43:45 +00:00
|
|
|
// The final value of m.account.Name depends on which backup message(BackedUpProfile/BackedUpSettings) arrives later.
|
2024-03-05 13:03:10 +00:00
|
|
|
// So we should check the clock of the preferred name and only update m.account.Name if it's older than the display name.
|
2024-03-25 15:43:45 +00:00
|
|
|
// Yet even if the preferred name clock is older, but the preferred name was empty, we should still update m.account.Name.
|
|
|
|
|
|
|
|
if preferredNameClock < clock || preferredName == "" {
|
2024-03-05 13:03:10 +00:00
|
|
|
m.account.Name = displayName
|
|
|
|
return m.multiAccounts.SaveAccount(*m.account)
|
|
|
|
}
|
|
|
|
return nil
|
2022-02-17 15:13:10 +00:00
|
|
|
}
|
2022-08-05 11:22:35 +00:00
|
|
|
|
2022-08-10 13:09:15 +00:00
|
|
|
func ValidateBio(bio *string) error {
|
|
|
|
if len(*bio) > maxBioLength {
|
|
|
|
return ErrInvalidBioLength
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2022-08-05 11:22:35 +00:00
|
|
|
func (m *Messenger) SetBio(bio string) error {
|
|
|
|
currentBio, err := m.settings.Bio()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
if currentBio == bio {
|
|
|
|
return nil // Do nothing
|
|
|
|
}
|
|
|
|
|
2022-08-10 13:09:15 +00:00
|
|
|
if err = ValidateBio(&bio); err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2022-08-05 11:22:35 +00:00
|
|
|
|
2022-08-10 13:09:15 +00:00
|
|
|
if err = m.settings.SaveSettingField(settings.Bio, bio); err != nil {
|
2022-08-05 11:22:35 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-08-10 13:09:15 +00:00
|
|
|
if err = m.resetLastPublishedTimeForChatIdentity(); err != nil {
|
2022-08-05 11:22:35 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
return m.publishContactCode()
|
|
|
|
}
|
|
|
|
|
2023-06-05 11:10:26 +00:00
|
|
|
func ValidateSocialLinks(socialLinks identity.SocialLinks) error {
|
|
|
|
for _, link := range socialLinks {
|
2023-04-19 22:59:09 +00:00
|
|
|
l := link
|
2023-06-05 11:10:26 +00:00
|
|
|
if err := ValidateSocialLink(l); err != nil {
|
2023-04-19 22:59:09 +00:00
|
|
|
return err
|
2022-08-10 13:09:15 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-04-19 22:59:09 +00:00
|
|
|
func ValidateSocialLink(link *identity.SocialLink) error {
|
|
|
|
if len(link.Text) > maxSocialLinkTextLength {
|
|
|
|
return ErrInvalidSocialLinkTextLength
|
|
|
|
}
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-06-05 11:10:26 +00:00
|
|
|
func (m *Messenger) AddOrReplaceSocialLinks(socialLinks identity.SocialLinks) error {
|
|
|
|
if len(socialLinks) > sociallinkssettings.MaxNumOfSocialLinks {
|
|
|
|
return errors.New("exceeded maximum number of social links")
|
|
|
|
}
|
|
|
|
|
2022-08-05 11:22:35 +00:00
|
|
|
currentSocialLinks, err := m.settings.GetSocialLinks()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-06-05 11:10:26 +00:00
|
|
|
if currentSocialLinks.Equal(socialLinks) {
|
2022-08-05 11:22:35 +00:00
|
|
|
return nil // Do nothing
|
|
|
|
}
|
|
|
|
|
2023-06-05 11:10:26 +00:00
|
|
|
err = ValidateSocialLinks(socialLinks)
|
|
|
|
if err != nil {
|
2022-08-10 13:09:15 +00:00
|
|
|
return err
|
|
|
|
}
|
2022-08-05 11:22:35 +00:00
|
|
|
|
2023-04-19 22:59:09 +00:00
|
|
|
err = m.withChatClock(func(chatID string, clock uint64) error {
|
2023-06-05 11:10:26 +00:00
|
|
|
err = m.settings.AddOrReplaceSocialLinksIfNewer(socialLinks, clock)
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2023-04-19 22:59:09 +00:00
|
|
|
}
|
2023-10-24 10:15:32 +00:00
|
|
|
m.selfContact.SocialLinks = socialLinks
|
2023-11-13 20:07:35 +00:00
|
|
|
m.publishSelfContactSubscriptions(&SelfContactChangeEvent{
|
|
|
|
SocialLinksChanged: true,
|
|
|
|
})
|
2023-06-05 11:10:26 +00:00
|
|
|
|
|
|
|
err = m.syncSocialLinks(context.Background(), m.dispatchMessage)
|
|
|
|
return err
|
2023-04-19 22:59:09 +00:00
|
|
|
})
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2022-08-10 13:09:15 +00:00
|
|
|
if err = m.resetLastPublishedTimeForChatIdentity(); err != nil {
|
2022-08-05 11:22:35 +00:00
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
2023-06-05 11:10:26 +00:00
|
|
|
return m.publishContactCode()
|
2023-04-19 22:59:09 +00:00
|
|
|
}
|
|
|
|
|
2023-06-05 11:10:26 +00:00
|
|
|
func (m *Messenger) GetSocialLinks() (identity.SocialLinks, error) {
|
|
|
|
return m.settings.GetSocialLinks()
|
2022-08-05 11:22:35 +00:00
|
|
|
}
|
2022-11-24 20:48:26 +00:00
|
|
|
|
|
|
|
func (m *Messenger) setInstallationHostname() error {
|
2023-02-28 12:32:45 +00:00
|
|
|
imd, err := m.getOurInstallationMetadata()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
2022-11-24 20:48:26 +00:00
|
|
|
}
|
|
|
|
|
2023-03-08 14:04:02 +00:00
|
|
|
// If the name and device are already set, don't do anything
|
|
|
|
if len(imd.Name) != 0 && len(imd.DeviceType) != 0 {
|
2022-11-24 20:48:26 +00:00
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
2023-03-08 14:04:02 +00:00
|
|
|
if len(imd.Name) == 0 {
|
2023-06-27 07:54:49 +00:00
|
|
|
deviceName, err := m.settings.DeviceName()
|
2023-03-08 14:04:02 +00:00
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
2023-06-27 07:54:49 +00:00
|
|
|
if deviceName != "" {
|
|
|
|
imd.Name = deviceName
|
|
|
|
} else {
|
|
|
|
hn, err := server.GetDeviceName()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
imd.Name = fmt.Sprintf("%s %s", hn, imd.Name)
|
|
|
|
}
|
2023-03-08 14:04:02 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
if len(imd.DeviceType) == 0 {
|
|
|
|
imd.DeviceType = runtime.GOOS
|
2022-11-24 20:48:26 +00:00
|
|
|
}
|
2023-03-08 14:04:02 +00:00
|
|
|
|
2022-11-24 20:48:26 +00:00
|
|
|
return m.setInstallationMetadata(m.installationID, imd)
|
2023-03-08 14:04:02 +00:00
|
|
|
|
2022-11-24 20:48:26 +00:00
|
|
|
}
|
2023-02-28 12:32:45 +00:00
|
|
|
|
|
|
|
func (m *Messenger) getOurInstallationMetadata() (*multidevice.InstallationMetadata, error) {
|
|
|
|
ourInstallation, ok := m.allInstallations.Load(m.installationID)
|
|
|
|
if !ok {
|
|
|
|
return nil, fmt.Errorf("messenger's installationID is not set or not loadable")
|
|
|
|
}
|
|
|
|
|
|
|
|
if ourInstallation.InstallationMetadata == nil {
|
|
|
|
return new(multidevice.InstallationMetadata), nil
|
|
|
|
}
|
|
|
|
|
|
|
|
return ourInstallation.InstallationMetadata, nil
|
|
|
|
}
|
|
|
|
|
|
|
|
func (m *Messenger) SetInstallationDeviceType(deviceType string) error {
|
|
|
|
if strings.TrimSpace(deviceType) == "" {
|
|
|
|
return errors.New("device type is empty")
|
|
|
|
}
|
|
|
|
|
|
|
|
imd, err := m.getOurInstallationMetadata()
|
|
|
|
if err != nil {
|
|
|
|
return err
|
|
|
|
}
|
|
|
|
|
|
|
|
// If the name is already set, don't do anything
|
|
|
|
if len(imd.DeviceType) != 0 {
|
|
|
|
return nil
|
|
|
|
}
|
|
|
|
|
|
|
|
imd.DeviceType = deviceType
|
|
|
|
return m.setInstallationMetadata(m.installationID, imd)
|
|
|
|
}
|