diff --git a/application.yaml b/application.yaml index 8649637..9d7f922 100644 --- a/application.yaml +++ b/application.yaml @@ -90,16 +90,16 @@ notifications: ¬ifications wintr/multimedia/picture: urlDownload: https://ice-staging.b-cdn.net socials: - - notificationType: follow_ion_on_x - link: https://x.com/ice_blockchain/status/1743221098349535361 - notificationType: follow_us_on_x - link: https://x.com/ice_blockchain/status/1793988190614823165 - - notificationType: follow_zeus_on_x - link: https://x.com/ice_blockchain/status/1793256299519418521 - - notificationType: join_ion_on_telegram - link: https://x.com/ice_blockchain/status/1793256299519418521 + link: https://x.com/sunwaves_token - notificationType: join_our_telegram - link: https://x.com/ice_blockchain/status/1793256299519418521 + link: 'https://t.me/sunwavestoken' + - notificationType: follow_ion_on_x + link: https://x.com/ice_blockchain + - notificationType: join_ion_on_telegram + link: 'https://t.me/iceblockchain' + - notificationType: follow_zeus_on_x + link: https://x.com/ice_z3us weeklyStats: weekday: 1 hour: 10 @@ -118,6 +118,8 @@ notifications: ¬ifications password: pass replicaURLs: - postgresql://root:pass@localhost:5434/husky + wintr/connectors/storage/v3: + url: redis://default:@localhost:6379 tenantName: BogusName tokenName: BN wintr/notifications/telegram: diff --git a/go.mod b/go.mod index c3b645d..c910eed 100644 --- a/go.mod +++ b/go.mod @@ -6,8 +6,8 @@ require ( github.com/goccy/go-json v0.10.3 github.com/google/uuid v1.6.0 github.com/hashicorp/go-multierror v1.1.1 - github.com/ice-blockchain/eskimo v1.368.0 - github.com/ice-blockchain/freezer v1.485.0 + github.com/ice-blockchain/eskimo v1.369.0 + github.com/ice-blockchain/freezer v1.488.0 github.com/ice-blockchain/go-tarantool-client v0.0.0-20230327200757-4fc71fa3f7bb github.com/ice-blockchain/wintr v1.144.0 github.com/imroc/req/v3 v3.43.7 @@ -95,7 +95,7 @@ require ( github.com/golang-jwt/jwt/v5 v5.2.1 // indirect github.com/golang/groupcache v0.0.0-20210331224755-41bb18bfe9da // indirect github.com/golang/protobuf v1.5.4 // indirect - github.com/google/pprof v0.0.0-20240625030939-27f56978b8b0 // indirect + github.com/google/pprof v0.0.0-20240711041743-f6c9dda6c6da // indirect github.com/google/s2a-go v0.1.7 // indirect github.com/googleapis/enterprise-certificate-proxy v0.3.2 // indirect github.com/googleapis/gax-go/v2 v2.12.5 // indirect diff --git a/go.sum b/go.sum index 17217ff..2af738c 100644 --- a/go.sum +++ b/go.sum @@ -247,8 +247,8 @@ github.com/google/gofuzz v1.2.0 h1:xRy4A+RhZaiKjJ1bPfwQ8sedCA+YS2YcCHW6ec7JMi0= github.com/google/gofuzz v1.2.0/go.mod h1:dBl0BpW6vV/+mYPU4Po3pmUjxk6FQPldtuIdl/M65Eg= github.com/google/martian/v3 v3.3.3 h1:DIhPTQrbPkgs2yJYdXU/eNACCG5DVQjySNRNlflZ9Fc= github.com/google/martian/v3 v3.3.3/go.mod h1:iEPrYcgCF7jA9OtScMFQyAlZZ4YXTKEtJ1E6RWzmBA0= -github.com/google/pprof v0.0.0-20240625030939-27f56978b8b0 h1:e+8XbKB6IMn8A4OAyZccO4pYfB3s7bt6azNIPE7AnPg= -github.com/google/pprof v0.0.0-20240625030939-27f56978b8b0/go.mod h1:K1liHPHnj73Fdn/EKuT8nrFqBihUSKXoLYU0BuatOYo= +github.com/google/pprof v0.0.0-20240711041743-f6c9dda6c6da h1:xRmpO92tb8y+Z85iUOMOicpCfaYcv7o3Cg3wKrIpg8g= +github.com/google/pprof v0.0.0-20240711041743-f6c9dda6c6da/go.mod h1:K1liHPHnj73Fdn/EKuT8nrFqBihUSKXoLYU0BuatOYo= github.com/google/s2a-go v0.1.7 h1:60BLSyTrOV4/haCDW4zb1guZItoSq8foHCXrAnjBo/o= github.com/google/s2a-go v0.1.7/go.mod h1:50CgR4k1jNlWBu4UfS4AcfhVe1r6pdZPygJ3R8F0Qdw= github.com/google/subcommands v1.2.0/go.mod h1:ZjhPrFU+Olkh9WazFPsl27BQ4UPiG37m3yTrtFlrHVk= @@ -282,10 +282,10 @@ github.com/holiman/uint256 v1.3.0 h1:4wdcm/tnd0xXdu7iS3ruNvxkWwrb4aeBQv19ayYn8F4 github.com/holiman/uint256 v1.3.0/go.mod h1:EOMSn4q6Nyt9P6efbI3bueV4e1b3dGlUCXeiRV4ng7E= github.com/huin/goupnp v1.3.0 h1:UvLUlWDNpoUdYzb2TCn+MuTWtcjXKSza2n6CBdQ0xXc= github.com/huin/goupnp v1.3.0/go.mod h1:gnGPsThkYa7bFi/KWmEysQRf48l2dvR5bxr2OFckNX8= -github.com/ice-blockchain/eskimo v1.368.0 h1:4xougCqbB62CEvBRzfoqEzFa/6ZYf1OeKoa2EoV53f8= -github.com/ice-blockchain/eskimo v1.368.0/go.mod h1:OpR88RU/bYLmjugzyumwo/sx0fvG8B9pYsLZIy/DFgI= -github.com/ice-blockchain/freezer v1.485.0 h1:znX6LMcYFbOArD/zHvwm/bYvgAzyX6Z/aYFmrJet90Q= -github.com/ice-blockchain/freezer v1.485.0/go.mod h1:NmqNJ34CIRUHzBZ7Tgd42Ei30qKeSqflQh4PM/iW4M4= +github.com/ice-blockchain/eskimo v1.369.0 h1:DFguku6xR4rb8G1Ay4LAh0s6qzYPnoKsO28uFJGvAko= +github.com/ice-blockchain/eskimo v1.369.0/go.mod h1:l4MZKGo/Lpq+LFr65HUAGc/SvN4IclA0kpFqXjjQsZ8= +github.com/ice-blockchain/freezer v1.488.0 h1:Q3XGS16eDQ4zeSTqfYLLwAt+H3TbDda9hI+VhEfhTx0= +github.com/ice-blockchain/freezer v1.488.0/go.mod h1:bSwFbBC95BBF+/WrPJp+l9qyrGruim0pnTpAmkkWoLQ= github.com/ice-blockchain/go-tarantool-client v0.0.0-20230327200757-4fc71fa3f7bb h1:8TnFP3mc7O+tc44kv2e0/TpZKnEVUaKH+UstwfBwRkk= github.com/ice-blockchain/go-tarantool-client v0.0.0-20230327200757-4fc71fa3f7bb/go.mod h1:ZsQU7i3mxhgBBu43Oev7WPFbIjP4TniN/b1UPNGbrq8= github.com/ice-blockchain/wintr v1.144.0 h1:YQE0olkPdSI6AOlw7r/j5jGI6uLciZQrvXFIkN4C4l4= diff --git a/notifications/contract.go b/notifications/contract.go index 4f67aeb..ab7ca9a 100644 --- a/notifications/contract.go +++ b/notifications/contract.go @@ -72,6 +72,7 @@ const ( SocialsFollowOurTelegramNotificationType NotificationType = "join_our_telegram" WeeklyStatsNotificationType NotificationType = "weekly_stats" ReplyNotificationType NotificationType = "reply" + SocialsNotificationType NotificationType = "socials" ) var ( @@ -125,11 +126,8 @@ var ( MiningExpiredNotificationType, MiningNotActiveNotificationType, InviteFriendNotificationType, - SocialsFollowIceOnXNotificationType, - SocialsFollowUsOnXNotificationType, - SocialsFollowZeusOnXNotificationType, - SocialsFollowIONOnTelegramNotificationType, - SocialsFollowOurTelegramNotificationType, + NewReferralNotificationType, + SocialsNotificationType, ReplyNotificationType, } //nolint:gochecknoglobals // It's just for more descriptive validation messages. @@ -218,8 +216,9 @@ const ( requestingUserIDCtxValueKey = "requestingUserIDCtxValueKey" requestDeadline = 30 * stdlibtime.Second - schedulerWorkersCount int64 = 10 - schedulerPushBatchSize int64 = 250 + schedulerWorkersCount int64 = 10 + schedulerPushBatchSize int64 = 250 + telegramLongPollingLimit int64 = 100 defaultLanguage = "en" // Protection from getting ErrDuplicate on session creation due to ReferralsCountChangeGuardUpdatedAt on freezer. diff --git a/notifications/notification_invite_friend.go b/notifications/notification_invite_friend.go index ccdce37..b7aa134 100644 --- a/notifications/notification_invite_friend.go +++ b/notifications/notification_invite_friend.go @@ -27,8 +27,8 @@ func (r *repository) addScheduledInviteFriendNotifications(ctx context.Context, dayDuration = 24 * stdlibtime.Hour firstNotificationDuration = 1 * stdlibtime.Hour } - availableChannels := []NotificationChannel{PushNotificationChannel, TelegramNotificationChannel} - scheduled := make([]*scheduledNotification, 0, 2*len(availableChannels)) //nolint:gomnd,mnd // . + availableChannels := []NotificationChannel{PushNotificationChannel} + scheduled := make([]*scheduledNotification, 0, len(availableChannels)) for _, channel := range availableChannels { scheduled = append(scheduled, &scheduledNotification{ ScheduledAt: now, diff --git a/notifications/notification_type_badge_unlocked.go b/notifications/notification_type_badge_unlocked.go index eae3518..83aed39 100644 --- a/notifications/notification_type_badge_unlocked.go +++ b/notifications/notification_type_badge_unlocked.go @@ -137,8 +137,8 @@ func (s *achievedBadgesSource) Process(ctx context.Context, msg *messagebroker.M return errors.Wrapf(s.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", notifType, in) }) } - if tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:nestif // . - tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID { + if false && (tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:revive,nestif // . + tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID) { tmplTelegram, found := allTelegramNotificationTemplates[notifType][tokens.Language] if !found { log.Warn(fmt.Sprintf("language `%v` was not found in the `%v` telegram config", tokens.Language, notifType)) @@ -162,8 +162,8 @@ func (s *achievedBadgesSource) Process(ctx context.Context, msg *messagebroker.M }, }, } - buttonText := tmplTelegram.getButtonText(nil) - buttonLink := getTelegramDeeplink(notifType, s.cfg, "", "") + buttonText := tmplTelegram.getButtonText(nil, 0) + buttonLink := getTelegramDeeplink(notifType, s.cfg, "", "", "") if buttonText != "" && buttonLink != "" { tn.tn.Buttons = append(tn.tn.Buttons, struct { Text string `json:"text,omitempty"` diff --git a/notifications/notification_type_level_changed.go b/notifications/notification_type_level_changed.go index af8d732..cd5d097 100644 --- a/notifications/notification_type_level_changed.go +++ b/notifications/notification_type_level_changed.go @@ -115,8 +115,8 @@ func (s *completedLevelsSource) Process(ctx context.Context, msg *messagebroker. return errors.Wrapf(s.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", LevelChangedNotificationType, in) }) } - if tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:nestif // . - tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID { + if false && (tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:revive,nestif // . + tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID) { if tmplTelegram, found := allTelegramNotificationTemplates[LevelChangedNotificationType][tokens.Language]; !found { log.Warn(fmt.Sprintf("language `%v` was not found in the `%v` telegram config", tokens.Language, LevelChangedNotificationType)) } else { //nolint:dupl // . @@ -139,8 +139,8 @@ func (s *completedLevelsSource) Process(ctx context.Context, msg *messagebroker. }, }, } - buttonText := tmplTelegram.getButtonText(nil) - buttonLink := getTelegramDeeplink(LevelChangedNotificationType, s.cfg, "", "") + buttonText := tmplTelegram.getButtonText(nil, 0) + buttonLink := getTelegramDeeplink(LevelChangedNotificationType, s.cfg, "", "", "") if buttonText != "" && buttonLink != "" { tn.tn.Buttons = append(tn.tn.Buttons, struct { Text string `json:"text,omitempty"` diff --git a/notifications/notification_type_mining.go b/notifications/notification_type_mining.go index 8a5b5b4..3526014 100644 --- a/notifications/notification_type_mining.go +++ b/notifications/notification_type_mining.go @@ -106,8 +106,8 @@ func (m *miningSessionSource) insertMiningScheduledNotifications(ctx context.Con dayDuration = 24 * stdlibtime.Hour uniquenessTime = fmt.Sprintf("%v:%02d:%02d %02d:%02d:%02d", now.Year(), int(now.Month()), now.Day(), now.Hour(), 0, 0) } - scheduled := make([]*scheduledNotification, 0, 2*notificationsNum) //nolint:gomnd,mnd // . - for _, channel := range []NotificationChannel{PushNotificationChannel, TelegramNotificationChannel} { + scheduled := make([]*scheduledNotification, 0, notificationsNum) + for _, channel := range []NotificationChannel{PushNotificationChannel} { scheduled = append(scheduled, &scheduledNotification{ ScheduledAt: now, ScheduledFor: message.ResettableStartingAt, diff --git a/notifications/notification_type_new_referral.go b/notifications/notification_type_new_referral.go index b037a2e..da5ec20 100644 --- a/notifications/notification_type_new_referral.go +++ b/notifications/notification_type_new_referral.go @@ -5,6 +5,7 @@ package notifications import ( "context" "fmt" + "strings" "github.com/hashicorp/go-multierror" "github.com/pkg/errors" @@ -16,10 +17,11 @@ import ( "github.com/ice-blockchain/wintr/log" "github.com/ice-blockchain/wintr/notifications/inapp" "github.com/ice-blockchain/wintr/notifications/push" + "github.com/ice-blockchain/wintr/notifications/telegram" "github.com/ice-blockchain/wintr/time" ) -func (r *repository) sendNewReferralNotification(ctx context.Context, us *users.UserSnapshot) error { //nolint:funlen,gocyclo,revive,cyclop // . +func (r *repository) sendNewReferralNotification(ctx context.Context, us *users.UserSnapshot) error { //nolint:funlen,gocyclo,revive,cyclop,gocognit // . if ctx.Err() != nil { return errors.Wrap(ctx.Err(), "unexpected deadline") } @@ -66,19 +68,13 @@ func (r *repository) sendNewReferralNotification(ctx context.Context, us *users. }, } tokens, err := r.getPushNotificationTokens(ctx, MicroCommunityNotificationDomain, us.User.ReferredBy) - if err != nil || tokens == nil || tokens.PushNotificationTokens == nil || len(*tokens.PushNotificationTokens) == 0 { + if err != nil || tokens == nil { return multierror.Append( //nolint:wrapcheck // . err, errors.Wrapf(r.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", NewReferralNotificationType, in), ).ErrorOrNil() } - tmpl, found := allPushNotificationTemplates[NewReferralNotificationType][tokens.Language] - if !found { - log.Warn(fmt.Sprintf("language `%v` was not found in the `%v` push config", tokens.Language, NewReferralNotificationType)) - - return errors.Wrapf(r.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", NewReferralNotificationType, in) - } - pn := make([]*pushNotification, 0, len(*tokens.PushNotificationTokens)) + var exConcurrently []func() error data := struct { Username, Coin string Amount uint64 @@ -87,40 +83,90 @@ func (r *repository) sendNewReferralNotification(ctx context.Context, us *users. Coin: r.cfg.TokenName, Amount: r.getNewReferralCoinAmount(ctx, us.User.ReferredBy), } - for _, token := range *tokens.PushNotificationTokens { - pn = append(pn, &pushNotification{ - pn: &push.Notification[push.DeviceToken]{ - Data: map[string]string{"deeplink": deeplink}, - Target: token, - Title: tmpl.getTitle(data), - Body: tmpl.getBody(data), - ImageURL: us.User.ProfilePictureURL, - }, - sn: &sentNotification{ - SentAt: now, - Language: tokens.Language, - sentNotificationPK: sentNotificationPK{ - UserID: us.User.ReferredBy, - Uniqueness: us.User.ID, - NotificationType: NewReferralNotificationType, - NotificationChannel: PushNotificationChannel, - NotificationChannelValue: string(token), + if tokens.PushNotificationTokens != nil && len(*tokens.PushNotificationTokens) != 0 { + tmpl, found := allPushNotificationTemplates[NewReferralNotificationType][tokens.Language] + if !found { + log.Warn(fmt.Sprintf("language `%v` was not found in the `%v` push config", tokens.Language, NewReferralNotificationType)) + + return errors.Wrapf(r.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", NewReferralNotificationType, in) + } + pn := make([]*pushNotification, 0, len(*tokens.PushNotificationTokens)) + for _, token := range *tokens.PushNotificationTokens { + pn = append(pn, &pushNotification{ + pn: &push.Notification[push.DeviceToken]{ + Data: map[string]string{"deeplink": deeplink}, + Target: token, + Title: tmpl.getTitle(data), + Body: tmpl.getBody(data), + ImageURL: us.User.ProfilePictureURL, }, - }, + sn: &sentNotification{ + SentAt: now, + Language: tokens.Language, + sentNotificationPK: sentNotificationPK{ + UserID: us.User.ReferredBy, + Uniqueness: us.User.ID, + NotificationType: NewReferralNotificationType, + NotificationChannel: PushNotificationChannel, + NotificationChannelValue: string(token), + }, + }, + }) + } + exConcurrently = append(exConcurrently, func() error { + return errors.Wrapf(runConcurrently(ctx, r.sendPushNotification, pn), "failed to sendPushNotifications atleast to some devices for %v, args:%#v", NewReferralNotificationType, pn) //nolint:lll // . + }, func() error { + return errors.Wrapf(r.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", NewReferralNotificationType, in) }) } + if tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:nestif // . + tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID { + tmplTelegram, found := allTelegramNotificationTemplates[NewReferralNotificationType][tokens.Language] + if !found { + log.Warn(fmt.Sprintf("language `%v` was not found in the `%v` telegram config", tokens.Language, NewReferralNotificationType)) + } else { + if botInfo, botFound := r.cfg.TelegramBots[strings.ToLower(tokens.TelegramBotID)]; botFound { + tn := &telegramNotification{ + tn: &telegram.Notification{ + ChatID: tokens.TelegramUserID, + Text: tmplTelegram.getBody(data), + BotToken: botInfo.BotToken, + }, + sn: &sentNotification{ + SentAt: now, + Language: tokens.Language, + sentNotificationPK: sentNotificationPK{ + UserID: us.User.ReferredBy, + Uniqueness: us.User.ID, + NotificationType: NewReferralNotificationType, + NotificationChannel: TelegramNotificationChannel, + NotificationChannelValue: us.User.ID, + }, + }, + } + buttonText := tmplTelegram.getButtonText(nil, 0) + buttonLink := getTelegramDeeplink(NewReferralNotificationType, r.cfg, "", "", us.User.ID) + if buttonText != "" && buttonLink != "" { + tn.tn.Buttons = append(tn.tn.Buttons, struct { + Text string `json:"text,omitempty"` + URL string `json:"url,omitempty"` + }{ + Text: buttonText, + URL: buttonLink, + }) + } + exConcurrently = append(exConcurrently, func() error { + return errors.Wrapf(r.sendTelegramNotification(ctx, tn), "failed to send telegram notification for %v, notif:%#v", NewReferralNotificationType, in) + }) + } + } + } - return errors.Wrap(executeConcurrently(func() error { - return errors.Wrapf(runConcurrently(ctx, r.sendPushNotification, pn), "failed to sendPushNotifications atleast to some devices for %v, args:%#v", NewReferralNotificationType, pn) //nolint:lll // . - }, func() error { - return errors.Wrapf(r.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", NewReferralNotificationType, in) - }), "failed to executeConcurrently") + return errors.Wrap(executeConcurrently(exConcurrently...), "failed to executeConcurrently") } func (r *repository) getNewReferralCoinAmount(ctx context.Context, referredBy string) uint64 { - const defaultNewReferralCoinAmount = 500.0 - //nolint:gocritic,godot // TODO: Uncomment this asap! - // const defaultNewReferralCoinAmount = tokenomics.WelcomeBonusV2Amount + const defaultNewReferralCoinAmount = tokenomics.WelcomeBonusV2Amount freezerInternalID, err := tokenomics.GetInternalID(ctx, r.freezerDB, referredBy) if err != nil { log.Error(errors.Wrapf(err, "failed to tokenomics.GetInternalID for referredBy: %v", referredBy)) diff --git a/notifications/notification_type_role_changed.go b/notifications/notification_type_role_changed.go index 27a797b..4bd7c16 100644 --- a/notifications/notification_type_role_changed.go +++ b/notifications/notification_type_role_changed.go @@ -115,8 +115,8 @@ func (s *enabledRolesSource) Process(ctx context.Context, msg *messagebroker.Mes return errors.Wrapf(s.sendInAppNotification(ctx, in), "failed to sendInAppNotification for %v, notif:%#v", RoleChangedNotificationType, in) }) } - if tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:nestif // . - tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID { + if false && (tokens.TelegramBotID != "" && tokens.TelegramBotID != tokens.UserID && //nolint:revive,nestif // . + tokens.TelegramUserID != "" && tokens.TelegramUserID != tokens.UserID) { tmplTelegram, found := allTelegramNotificationTemplates[RoleChangedNotificationType][tokens.Language] if !found { log.Warn(fmt.Sprintf("language `%v` was not found in the `%v` telegram config", tokens.Language, RoleChangedNotificationType)) @@ -140,8 +140,8 @@ func (s *enabledRolesSource) Process(ctx context.Context, msg *messagebroker.Mes }, }, } - buttonText := tmplTelegram.getButtonText(nil) - buttonLink := getTelegramDeeplink(LevelChangedNotificationType, s.cfg, "", "") + buttonText := tmplTelegram.getButtonText(nil, 0) + buttonLink := getTelegramDeeplink(LevelChangedNotificationType, s.cfg, "", "", "") if buttonText != "" && buttonLink != "" { tn.tn.Buttons = append(tn.tn.Buttons, struct { Text string `json:"text,omitempty"` diff --git a/notifications/notification_type_socials.go b/notifications/notification_type_socials.go index 445f3b7..ba567dc 100644 --- a/notifications/notification_type_socials.go +++ b/notifications/notification_type_socials.go @@ -19,14 +19,14 @@ func (r *repository) addScheduledSocialsNotifications(ctx context.Context, us *u return errors.Wrap(ctx.Err(), "unexpected deadline") } now := time.Now() - scheduled := make([]*scheduledNotification, 0, 2*len(r.cfg.Socials)) //nolint:gomnd,mnd // . + scheduled := make([]*scheduledNotification, 0, len(r.cfg.Socials)+1) var dayDuration stdlibtime.Duration if r.cfg.Development { dayDuration = 1 * stdlibtime.Minute } else { dayDuration = 24 * stdlibtime.Hour } - for _, channel := range []NotificationChannel{PushNotificationChannel, TelegramNotificationChannel} { + for _, channel := range []NotificationChannel{PushNotificationChannel} { for ix := range r.cfg.Socials { scheduled = append(scheduled, &scheduledNotification{ ScheduledAt: now, @@ -44,6 +44,19 @@ func (r *repository) addScheduledSocialsNotifications(ctx context.Context, us *u }) } } + scheduled = append(scheduled, &scheduledNotification{ + ScheduledAt: now, + ScheduledFor: time.New(us.CreatedAt.Add(dayDuration)), + Language: us.Language, + UserID: us.ID, + NotificationType: string(SocialsNotificationType), + Uniqueness: fmt.Sprintf("%v_%v", us.ID, SocialsNotificationType), + NotificationChannel: string(TelegramNotificationChannel), + NotificationChannelValue: us.ID, + Data: &users.JSON{ + "TenantName": r.cfg.TenantName, + }, + }) return errors.Wrapf(insertScheduledNotifications(ctx, r.db, scheduled), "can't execute insertScheduledNotifications:%#v", scheduled) } diff --git a/notifications/scheduler.go b/notifications/scheduler.go index e19044f..f8e34e6 100644 --- a/notifications/scheduler.go +++ b/notifications/scheduler.go @@ -43,7 +43,9 @@ func MustStartScheduler(ctx context.Context, cancel context.CancelFunc) *Schedul schedulerTelegramNotificationsMX: &sync.Mutex{}, } go sh.startWeeklyStatsUpdater(ctx) - go sh.startGetUpdatesTelegramLongPolling(ctx) + if false { + go sh.startGetUpdatesTelegramLongPolling(ctx) + } sh.wg = new(sync.WaitGroup) sh.wg.Add(3 * int(schedulerWorkersCount)) //nolint:gomnd,mnd // . sh.cancel = cancel diff --git a/notifications/scheduler_telegram_notifications.go b/notifications/scheduler_telegram_notifications.go index b40d7fa..1a22bd9 100644 --- a/notifications/scheduler_telegram_notifications.go +++ b/notifications/scheduler_telegram_notifications.go @@ -109,18 +109,10 @@ func (s *Scheduler) runTelegramNotificationsProcessor(ctx context.Context, worke }, scheduled: notification.scheduledNotification, } - buttonText := tmpl.getButtonText(notification.Data) - buttonLink := getTelegramDeeplink(NotificationType(notification.NotificationType), s.cfg, notification.Username, tmpl.getInviteText(notification.Data)) - if buttonText != "" && buttonLink != "" { - tn.tn.Buttons = append(tn.tn.Buttons, struct { - Text string `json:"text,omitempty"` - URL string `json:"url,omitempty"` - }{ - Text: buttonText, - URL: buttonLink, - }) - } - if notification.NotificationType == string(ReplyNotificationType) { + switch notification.NotificationType { + case string(SocialsNotificationType): + tn.tn.Buttons = append(tn.tn.Buttons, prepareTelegramButtonsForSocialNotificationType(s.cfg, tmpl.ButtonText)...) + case string(ReplyNotificationType): replyMessageID, pErr := strconv.ParseInt(notification.NotificationChannelValue, 10, 64) if pErr != nil { log.Warn("can't convert notification chanel value to integer for reply notification type", pErr) @@ -129,6 +121,18 @@ func (s *Scheduler) runTelegramNotificationsProcessor(ctx context.Context, worke continue } tn.tn.ReplyMessageID = replyMessageID + default: + buttonText := tmpl.getButtonText(notification.Data, 0) + buttonLink := getTelegramDeeplink(NotificationType(notification.NotificationType), s.cfg, notification.Username, tmpl.getInviteText(notification.Data), notification.UserID) //nolint:lll // . + if buttonText != "" && buttonLink != "" { + tn.tn.Buttons = append(tn.tn.Buttons, struct { + Text string `json:"text,omitempty"` + URL string `json:"url,omitempty"` + }{ + Text: buttonText, + URL: buttonLink, + }) + } } toSendTelegramNotifications = append(toSendTelegramNotifications, tn) } diff --git a/notifications/telegram_notifications.go b/notifications/telegram_notifications.go index 6fbea82..73b2177 100644 --- a/notifications/telegram_notifications.go +++ b/notifications/telegram_notifications.go @@ -24,19 +24,23 @@ import ( type ( telegramNotificationTemplate struct { - body, buttonText, inviteText *template.Template - Body string `json:"body"` //nolint:revive // That's intended. - ButtonText string `json:"buttonText"` //nolint:revive // That's intended. - InviteText string `json:"inviteText"` //nolint:revive // That's intended. + body, inviteText *template.Template + Body string `json:"body"` //nolint:revive // That's intended. + InviteText string `json:"inviteText"` //nolint:revive // That's intended. + ButtonText []string `json:"buttonText"` + buttonText []*template.Template //nolint:revive // That's intended. } ) -func (t *telegramNotificationTemplate) getButtonText(data any) string { +func (t *telegramNotificationTemplate) getButtonText(data any, ix int) string { //nolint:unparam // . + if ix > len(t.buttonText)-1 { + return "" + } if data == nil { - return t.ButtonText + return t.ButtonText[ix] } bf := new(bytes.Buffer) - log.Panic(errors.Wrapf(t.buttonText.Execute(bf, data), "failed to execute button text template for data:%#v", data)) + log.Panic(errors.Wrapf(t.buttonText[ix].Execute(bf, data), "failed to execute button text 1 template for data:%#v", data)) return bf.String() } @@ -66,7 +70,7 @@ func (t *telegramNotificationTemplate) getInviteText(data any) string { return bf.String() } -func loadTelegramNotificationTranslationTemplates() { +func loadTelegramNotificationTranslationTemplates() { //nolint:funlen // . const totalLanguages = 50 allTelegramNotificationTemplates = make(map[NotificationType]map[languageCode]*telegramNotificationTemplate, len(AllTelegramNotificationTypes)) for _, notificationType := range AllTelegramNotificationTypes { @@ -76,9 +80,9 @@ func loadTelegramNotificationTranslationTemplates() { } allTelegramNotificationTemplates[notificationType] = make(map[languageCode]*telegramNotificationTemplate, totalLanguages) var translations map[string]*struct { - Body string `json:"body"` - ButtonText string `json:"buttonText"` - InviteText string `json:"inviteText"` + Body string `json:"body"` + InviteText string `json:"inviteText"` + ButtonText []string `json:"buttonText"` } err := json.Unmarshal(content, &translations) if err != nil { @@ -87,12 +91,14 @@ func loadTelegramNotificationTranslationTemplates() { for language, data := range translations { var tmpl telegramNotificationTemplate tmpl.Body = data.Body - tmpl.ButtonText = data.ButtonText if notificationType == InviteFriendNotificationType { tmpl.InviteText = data.InviteText tmpl.inviteText = template.Must(template.New(fmt.Sprintf("telegram_%v_%v_invite_text", notificationType, language)).Parse(data.InviteText)) } - tmpl.buttonText = template.Must(template.New(fmt.Sprintf("telegram_%v_%v_button_text", notificationType, language)).Parse(data.ButtonText)) + tmpl.ButtonText = data.ButtonText + for ix := range data.ButtonText { + tmpl.buttonText = append(tmpl.buttonText, template.Must(template.New(fmt.Sprintf("telegram_%v_%v_button_text_%v", notificationType, language, ix)).Parse(data.ButtonText[ix]))) //nolint:lll // . + } tmpl.body = template.Must(template.New(fmt.Sprintf("telegram_%v_%v_body", notificationType, language)).Parse(data.Body)) allTelegramNotificationTemplates[notificationType][language] = &tmpl } @@ -100,29 +106,20 @@ func loadTelegramNotificationTranslationTemplates() { } //nolint:exhaustive // We know what cases need to be handled only. -func getTelegramDeeplink(nt NotificationType, cfg *config, username, inviteText string) string { - urls := getSocialsMapURL(cfg) +func getTelegramDeeplink(nt NotificationType, cfg *config, username, inviteText, userID string) string { switch nt { case MiningExtendNotificationType, MiningEndingSoonNotificationType, MiningExpiredNotificationType, MiningNotActiveNotificationType: return cfg.WebAppLink case InviteFriendNotificationType: return fmt.Sprintf("%[1]v?url=%[2]v/@%[3]v&text=%[4]v", cfg.InviteURL, url.QueryEscape(cfg.WebSiteURL), url.QueryEscape(username), url.QueryEscape(inviteText)) //nolint:lll // . - case SocialsFollowIceOnXNotificationType: - return urls[string(SocialsFollowIceOnXNotificationType)] - case SocialsFollowUsOnXNotificationType: - return urls[string(SocialsFollowUsOnXNotificationType)] - case SocialsFollowZeusOnXNotificationType: - return urls[string(SocialsFollowZeusOnXNotificationType)] - case SocialsFollowIONOnTelegramNotificationType: - return urls[string(SocialsFollowIONOnTelegramNotificationType)] - case SocialsFollowOurTelegramNotificationType: - return urls[string(SocialsFollowOurTelegramNotificationType)] case CoinBadgeUnlockedNotificationType, LevelBadgeUnlockedNotificationType, SocialBadgeUnlockedNotificationType: return fmt.Sprintf("%v?startapp=goto_profile_badges", cfg.WebAppLink) case LevelChangedNotificationType: return fmt.Sprintf("%v?startapp=goto_profile", cfg.WebAppLink) case ReplyNotificationType: return cfg.WebAppLink + case NewReferralNotificationType: + return fmt.Sprintf("%v://profile?userId=%v", cfg.WebAppLink, userID) default: log.Panic(fmt.Sprintf("wrong notification type:%v", nt)) } @@ -130,13 +127,40 @@ func getTelegramDeeplink(nt NotificationType, cfg *config, username, inviteText return "" } -func getSocialsMapURL(cfg *config) map[string]string { +func prepareTelegramButtonsForSocialNotificationType(cfg *config, buttonTexts []string) []struct { + Text string `json:"text,omitempty"` + URL string `json:"url,omitempty"` +} { + urls := getSocialsMapURL(cfg) + if len(urls) != len(buttonTexts) { + log.Error(errors.New("socials cfg/translation misconfiguration")) + + return nil + } + res := make([]struct { + Text string `json:"text,omitempty"` + URL string `json:"url,omitempty"` + }, 0, len(buttonTexts)) + for ix, text := range buttonTexts { + res = append(res, struct { + Text string `json:"text,omitempty"` + URL string `json:"url,omitempty"` + }{ + Text: text, + URL: urls[ix], + }) + } + + return res +} + +func getSocialsMapURL(cfg *config) map[int]string { if len(cfg.Socials) == 0 { log.Panic("no urls for socials") } - urls := make(map[string]string, len(cfg.Socials)) + urls := make(map[int]string, len(cfg.Socials)) for ix := range cfg.Socials { - urls[cfg.Socials[ix].NotificationType] = cfg.Socials[ix].Link + urls[ix] = cfg.Socials[ix].Link } return urls @@ -172,7 +196,7 @@ func (s *Scheduler) getTelegramLongPollingUpdates(ctx context.Context) (err erro upd, gErr := s.telegramNotificationsClient.GetUpdates(ctx, &telegram.GetUpdatesArg{ BotToken: bot.BotToken, AllowedUpdates: []string{"message", "callback_query"}, - Limit: int64(1), + Limit: telegramLongPollingLimit, Offset: nextOffset, }) if gErr != nil { diff --git a/notifications/translations/telegram/coin_badge_unlocked.txt b/notifications/translations/telegram/coin_badge_unlocked.txt index e3124d1..f1cf524 100644 --- a/notifications/translations/telegram/coin_badge_unlocked.txt +++ b/notifications/translations/telegram/coin_badge_unlocked.txt @@ -1,6 +1,6 @@ { "en":{ "body":"🎉 New badge earned!\n\nYou've unlocked the {{.BadgeName}} coin badge! Check it out in your profile.", - "buttonText": "🤩 View Badge" + "buttonText": ["🤩 View Badge"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/follow_ion_on_x.txt b/notifications/translations/telegram/follow_ion_on_x.txt deleted file mode 100644 index 7b75bca..0000000 --- a/notifications/translations/telegram/follow_ion_on_x.txt +++ /dev/null @@ -1,6 +0,0 @@ -{ - "en": { - "body": "❄️ Follow Ice Open Network on X\n\nStay updated with the latest news and insights from Ice Open Network", - "buttonText": "👉 Follow ION on X" - } -} diff --git a/notifications/translations/telegram/follow_us_on_x.txt b/notifications/translations/telegram/follow_us_on_x.txt deleted file mode 100644 index e9cc3a5..0000000 --- a/notifications/translations/telegram/follow_us_on_x.txt +++ /dev/null @@ -1,6 +0,0 @@ -{ - "en": { - "body": "👋 Follow us on X\n\nConnect with us on X and join our fast-growing community.", - "buttonText": "🎉 Follow {{.TenantName}} on X" - } -} \ No newline at end of file diff --git a/notifications/translations/telegram/follow_zeus_on_x.txt b/notifications/translations/telegram/follow_zeus_on_x.txt deleted file mode 100644 index 4c32452..0000000 --- a/notifications/translations/telegram/follow_zeus_on_x.txt +++ /dev/null @@ -1,6 +0,0 @@ -{ - "en": { - "body": "🙌 Follow Zeus on X\n\nStay connected with the founder of Ice Open Network on X", - "buttonText": "🌟 Follow Zeus on X" - } -} \ No newline at end of file diff --git a/notifications/translations/telegram/invite_friend.txt b/notifications/translations/telegram/invite_friend.txt index 88aeead..6f6d018 100644 --- a/notifications/translations/telegram/invite_friend.txt +++ b/notifications/translations/telegram/invite_friend.txt @@ -1,7 +1,7 @@ { "en": { "body": "👋  Invite friends, earn more {{.TenantName}}!\n\nBring your friends to the network and watch your {{.TenantName}} balance grow!", - "buttonText": "⭐️ Invite Now", + "buttonText": ["⭐️ Invite Now"], "inviteText": "🌟 One of the biggest and oldest electronic music festivals in the WORLD is launching their token on #IceOpenNetwork.\n\n Join {{.TenantName}} and receive 10 SW coins when you sign-up using my referral code: {{.Username}}\n\n 🎶 Music is the anSWer" } } \ No newline at end of file diff --git a/notifications/translations/telegram/join_ion_on_telegram.txt b/notifications/translations/telegram/join_ion_on_telegram.txt deleted file mode 100644 index 1245e32..0000000 --- a/notifications/translations/telegram/join_ion_on_telegram.txt +++ /dev/null @@ -1,6 +0,0 @@ -{ - "en": { - "body": "🤩 Join Ice Open Network on Telegram\n\nStay connected and follow the latest updates from Ice Open Network on Telegram", - "buttonText": "💙 Join Now" - } -} diff --git a/notifications/translations/telegram/join_our_telegram.txt b/notifications/translations/telegram/join_our_telegram.txt deleted file mode 100644 index 8236bba..0000000 --- a/notifications/translations/telegram/join_our_telegram.txt +++ /dev/null @@ -1,6 +0,0 @@ -{ - "en": { - "body": "📣 Join our Telegram community!\n\nJoin our community channel for exclusive updates.", - "buttonText": "✅ Join Now" - } -} \ No newline at end of file diff --git a/notifications/translations/telegram/level_badge_unlocked.txt b/notifications/translations/telegram/level_badge_unlocked.txt index ea07b8f..bdc7368 100644 --- a/notifications/translations/telegram/level_badge_unlocked.txt +++ b/notifications/translations/telegram/level_badge_unlocked.txt @@ -1,6 +1,6 @@ { "en":{ "body":"🎉 New badge earned!\n\nYou've unlocked the {{.BadgeName}} level badge! Check it out in your profile.", - "buttonText": "🤩 View Badge" + "buttonText": ["🤩 View Badge"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/level_changed.txt b/notifications/translations/telegram/level_changed.txt index 0dce712..86da149 100644 --- a/notifications/translations/telegram/level_changed.txt +++ b/notifications/translations/telegram/level_changed.txt @@ -1,6 +1,6 @@ { "en":{ "body":"🚀 A new level achieved!\n\nCongratulations! Your hard work is paying off, keep leveling up! 🏆", - "buttonText": "🙌 Check Your Level" + "buttonText": ["🙌 Check Your Level"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/mining_ending_soon.txt b/notifications/translations/telegram/mining_ending_soon.txt index 2819d66..990508f 100644 --- a/notifications/translations/telegram/mining_ending_soon.txt +++ b/notifications/translations/telegram/mining_ending_soon.txt @@ -1,6 +1,6 @@ { "en": { "body": "🚨 Time's Running Out 🚨\n\nYou've got 1 hour left to extend your check-in session.", - "buttonText": "👉 Extend Session" + "buttonText": ["👉 Extend Session"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/mining_expired.txt b/notifications/translations/telegram/mining_expired.txt index 90f84c7..0197d7b 100644 --- a/notifications/translations/telegram/mining_expired.txt +++ b/notifications/translations/telegram/mining_expired.txt @@ -1,6 +1,6 @@ { "en": { "body": "💔 Don't lose more coins!\n\nYou're losing coins for inactivity, come back and start your check-in session again.", - "buttonText": "👉 Start Session" + "buttonText": ["👉 Start Session"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/mining_extend.txt b/notifications/translations/telegram/mining_extend.txt index 9d78a7c..6c21c61 100644 --- a/notifications/translations/telegram/mining_extend.txt +++ b/notifications/translations/telegram/mining_extend.txt @@ -1,6 +1,6 @@ { "en": { "body": "🚀 Extend your check-in session!🔓\n\n Press and hold the {{.TenantName}} logo button to keep earning {{.TokenName}} tokens. 💰", - "buttonText": "👉 Extend Session" + "buttonText": ["👉 Extend Session"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/mining_not_active.txt b/notifications/translations/telegram/mining_not_active.txt index 4b76fc4..87eaced 100644 --- a/notifications/translations/telegram/mining_not_active.txt +++ b/notifications/translations/telegram/mining_not_active.txt @@ -1,6 +1,6 @@ { "en": { "body": "⏰ Tick-tock, tick-tock!\n\nKeep your coins safe by starting a new check-in session now.", - "buttonText": "👉 Start Session" + "buttonText": ["👉 Start Session"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/new_referral.txt b/notifications/translations/telegram/new_referral.txt new file mode 100644 index 0000000..e17a350 --- /dev/null +++ b/notifications/translations/telegram/new_referral.txt @@ -0,0 +1,6 @@ +{ + "en":{ + "body":"🔥 {{.Username}} has joined your team.\n\nYou earned {{.Amount}} {{.Coin}} tokens! 🌟", + "buttonText": ["👉 View Profile"] + } +} diff --git a/notifications/translations/telegram/reply.txt b/notifications/translations/telegram/reply.txt index 736de87..785dae4 100644 --- a/notifications/translations/telegram/reply.txt +++ b/notifications/translations/telegram/reply.txt @@ -1,6 +1,6 @@ { "en":{ "body":"🎉 Hi, {{.Username}}!\n\nJoin to our app.", - "buttonText": "👉 Join the app" + "buttonText": ["👉 Join the app"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/role_changed.txt b/notifications/translations/telegram/role_changed.txt index 8dd4b0b..31959af 100644 --- a/notifications/translations/telegram/role_changed.txt +++ b/notifications/translations/telegram/role_changed.txt @@ -1,6 +1,6 @@ { "en":{ "body":"🌟You’re a community hero!\n\nYou made it! Welcome to the AMBASSADOR team. 🚀", - "buttonText": "💼 View Role" + "buttonText": ["💼 View Role"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/social_badge_unlocked.txt b/notifications/translations/telegram/social_badge_unlocked.txt index 9996c55..e8d6d58 100644 --- a/notifications/translations/telegram/social_badge_unlocked.txt +++ b/notifications/translations/telegram/social_badge_unlocked.txt @@ -1,6 +1,6 @@ { "en":{ "body":"🎉 New badge earned!\n\nYou've unlocked the {{.BadgeName}} social badge! Check it out in your profile.", - "buttonText": "🤩 View Badge" + "buttonText": ["🤩 View Badge"] } } \ No newline at end of file diff --git a/notifications/translations/telegram/socials.txt b/notifications/translations/telegram/socials.txt new file mode 100644 index 0000000..7099482 --- /dev/null +++ b/notifications/translations/telegram/socials.txt @@ -0,0 +1,12 @@ +{ + "en": { + "body": "🚀 Join our socials to stay updated on all the latest news and important updates!", + "buttonText": [ + "Follow us on X", + "Join us on Telegram", + "Follow Ice Open Network on X", + "Join Ice Open Network on Telegram", + "Follow Zeus on X" + ] + } +}