Verify email reminders (#4510)
* Clarify intent * Increase email reminder period to once per day * Fallback * Snooze immediately after account creation, prevent showing right after signup * Fix e2e test exports * Remove redundant check * Better simple date generation * Replace in DateField * Use non-string comparison * Revert change to unrelated code * Also parse * Remove side effectzio/stable
parent
853c32b4d8
commit
32b4063185
|
@ -54,8 +54,8 @@ import {useModalControls} from './state/modals'
|
||||||
import {useUnreadNotifications} from './state/queries/notifications/unread'
|
import {useUnreadNotifications} from './state/queries/notifications/unread'
|
||||||
import {useSession} from './state/session'
|
import {useSession} from './state/session'
|
||||||
import {
|
import {
|
||||||
setEmailConfirmationRequested,
|
|
||||||
shouldRequestEmailConfirmation,
|
shouldRequestEmailConfirmation,
|
||||||
|
snoozeEmailConfirmationPrompt,
|
||||||
} from './state/shell/reminders'
|
} from './state/shell/reminders'
|
||||||
import {AccessibilitySettingsScreen} from './view/screens/AccessibilitySettings'
|
import {AccessibilitySettingsScreen} from './view/screens/AccessibilitySettings'
|
||||||
import {CommunityGuidelinesScreen} from './view/screens/CommunityGuidelines'
|
import {CommunityGuidelinesScreen} from './view/screens/CommunityGuidelines'
|
||||||
|
@ -585,7 +585,7 @@ function RoutesContainer({children}: React.PropsWithChildren<{}>) {
|
||||||
|
|
||||||
if (currentAccount && shouldRequestEmailConfirmation(currentAccount)) {
|
if (currentAccount && shouldRequestEmailConfirmation(currentAccount)) {
|
||||||
openModal({name: 'verify-email', showReminder: true})
|
openModal({name: 'verify-email', showReminder: true})
|
||||||
setEmailConfirmationRequested()
|
snoozeEmailConfirmationPrompt()
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -19,3 +19,14 @@ export function getAge(birthDate: Date): number {
|
||||||
}
|
}
|
||||||
return age
|
return age
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Compares two dates by year, month, and day only
|
||||||
|
*/
|
||||||
|
export function simpleAreDatesEqual(a: Date, b: Date): boolean {
|
||||||
|
return (
|
||||||
|
a.getFullYear() === b.getFullYear() &&
|
||||||
|
a.getMonth() === b.getMonth() &&
|
||||||
|
a.getDate() === b.getDate()
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
|
@ -11,6 +11,7 @@ import {
|
||||||
import {tryFetchGates} from '#/lib/statsig/statsig'
|
import {tryFetchGates} from '#/lib/statsig/statsig'
|
||||||
import {getAge} from '#/lib/strings/time'
|
import {getAge} from '#/lib/strings/time'
|
||||||
import {logger} from '#/logger'
|
import {logger} from '#/logger'
|
||||||
|
import {snoozeEmailConfirmationPrompt} from '#/state/shell/reminders'
|
||||||
import {
|
import {
|
||||||
configureModerationForAccount,
|
configureModerationForAccount,
|
||||||
configureModerationForGuest,
|
configureModerationForGuest,
|
||||||
|
@ -191,6 +192,13 @@ export async function createAgentAndCreateAccount(
|
||||||
agent.setPersonalDetails({birthDate: birthDate.toISOString()})
|
agent.setPersonalDetails({birthDate: birthDate.toISOString()})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
// snooze first prompt after signup, defer to next prompt
|
||||||
|
snoozeEmailConfirmationPrompt()
|
||||||
|
} catch (e: any) {
|
||||||
|
logger.error(e, {context: `session: failed snoozeEmailConfirmationPrompt`})
|
||||||
|
}
|
||||||
|
|
||||||
return prepareAgent(agent, gates, moderation, onSessionChange)
|
return prepareAgent(agent, gates, moderation, onSessionChange)
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -1,7 +1,5 @@
|
||||||
export function init() {}
|
|
||||||
|
|
||||||
export function shouldRequestEmailConfirmation() {
|
export function shouldRequestEmailConfirmation() {
|
||||||
return false
|
return false
|
||||||
}
|
}
|
||||||
|
|
||||||
export function setEmailConfirmationRequested() {}
|
export function snoozeEmailConfirmationPrompt() {}
|
||||||
|
|
|
@ -1,36 +1,45 @@
|
||||||
|
import {simpleAreDatesEqual} from '#/lib/strings/time'
|
||||||
|
import {logger} from '#/logger'
|
||||||
import * as persisted from '#/state/persisted'
|
import * as persisted from '#/state/persisted'
|
||||||
import {toHashCode} from 'lib/strings/helpers'
|
|
||||||
import {isOnboardingActive} from './onboarding'
|
|
||||||
import {SessionAccount} from '../session'
|
import {SessionAccount} from '../session'
|
||||||
|
import {isOnboardingActive} from './onboarding'
|
||||||
|
|
||||||
export function shouldRequestEmailConfirmation(account: SessionAccount) {
|
export function shouldRequestEmailConfirmation(account: SessionAccount) {
|
||||||
if (!account) {
|
// ignore logged out
|
||||||
return false
|
if (!account) return false
|
||||||
}
|
// ignore confirmed accounts, this is the success state of this reminder
|
||||||
if (account.emailConfirmed) {
|
if (account.emailConfirmed) return false
|
||||||
return false
|
// wait for onboarding to complete
|
||||||
}
|
if (isOnboardingActive()) return false
|
||||||
if (isOnboardingActive()) {
|
|
||||||
return false
|
const snoozedAt = persisted.get('reminders').lastEmailConfirm
|
||||||
}
|
|
||||||
// only prompt once
|
|
||||||
if (persisted.get('reminders').lastEmailConfirm) {
|
|
||||||
return false
|
|
||||||
}
|
|
||||||
const today = new Date()
|
const today = new Date()
|
||||||
// shard the users into 2 day of the week buckets
|
|
||||||
// (this is to avoid a sudden influx of email updates when
|
logger.debug('Checking email confirmation reminder', {
|
||||||
// this feature rolls out)
|
today,
|
||||||
const code = toHashCode(account.did) % 7
|
snoozedAt,
|
||||||
if (code !== today.getDay() && code !== (today.getDay() + 1) % 7) {
|
})
|
||||||
return false
|
|
||||||
}
|
// never been snoozed, new account
|
||||||
|
if (!snoozedAt) {
|
||||||
return true
|
return true
|
||||||
}
|
}
|
||||||
|
|
||||||
export function setEmailConfirmationRequested() {
|
// already snoozed today
|
||||||
|
if (simpleAreDatesEqual(new Date(Date.parse(snoozedAt)), new Date())) {
|
||||||
|
return false
|
||||||
|
}
|
||||||
|
|
||||||
|
return true
|
||||||
|
}
|
||||||
|
|
||||||
|
export function snoozeEmailConfirmationPrompt() {
|
||||||
|
const lastEmailConfirm = new Date().toISOString()
|
||||||
|
logger.debug('Snoozing email confirmation reminder', {
|
||||||
|
snoozedAt: lastEmailConfirm,
|
||||||
|
})
|
||||||
persisted.write('reminders', {
|
persisted.write('reminders', {
|
||||||
...persisted.get('reminders'),
|
...persisted.get('reminders'),
|
||||||
lastEmailConfirm: new Date().toISOString(),
|
lastEmailConfirm,
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in New Issue