Skip to content

Commit

Permalink
Merge pull request #4325 from tloncorp/lb/more-invite-logging
Browse files Browse the repository at this point in the history
invites: enhanced error and debug reporting during invite creation
  • Loading branch information
latter-bolden authored Jan 10, 2025
2 parents ef0b289 + 4e7462e commit 0ca965b
Show file tree
Hide file tree
Showing 4 changed files with 130 additions and 60 deletions.
14 changes: 10 additions & 4 deletions packages/app/lib/bootHelpers.ts
Original file line number Diff line number Diff line change
@@ -1,11 +1,13 @@
import { AppInvite } from '@tloncorp/shared';
import { AnalyticsEvent, AppInvite, createDevLogger } from '@tloncorp/shared';
import { getLandscapeAuthCookie } from '@tloncorp/shared/api';
import * as db from '@tloncorp/shared/db';

import * as hostingApi from '../lib/hostingApi';
import { trackOnboardingAction } from '../utils/posthog';
import { getShipFromCookie, getShipUrl } from '../utils/ship';

const logger = createDevLogger('bootHelpers', true);

export enum NodeBootPhase {
IDLE = 1,
RESERVING = 2,
Expand Down Expand Up @@ -133,9 +135,13 @@ async function getInvitedGroupAndDm(lureMeta: AppInvite | null): Promise<{
const tlonTeam = `~wittyr-witbes`;
const isPersonalInvite = inviteType === 'user';
if (!inviterUserId || (!isPersonalInvite && !invitedGroupId)) {
throw new Error(
`invalid invite metadata: group[${invitedGroupId}] inviter[${inviterUserId}]`
);
logger.trackEvent(AnalyticsEvent.InviteError, {
message: 'invite is missing metadata',
context:
'this will prevent the group from being auto-joined, but an invite should still be delivered',
invite: lureMeta,
});
throw new Error('invite is missing metadata');
}
// use api client to see if you have pending DM and group invite
const invitedDm = await db.getChannel({ id: inviterUserId });
Expand Down
3 changes: 3 additions & 0 deletions packages/shared/src/logic/analytics.ts
Original file line number Diff line number Diff line change
Expand Up @@ -14,4 +14,7 @@ export enum AnalyticsEvent {
ChannelTemplateSetup = 'Channel Created from Template',
ChannelLoadComplete = 'Channel Load Complete',
SessionInitialized = 'Session Initialized',
InviteError = 'Invite Error',
InviteDebug = 'Invite Debug',
InviteButtonShown = 'Invite Button Shown',
}
161 changes: 105 additions & 56 deletions packages/shared/src/store/lure.ts
Original file line number Diff line number Diff line change
@@ -1,18 +1,17 @@
import { useQuery } from '@tanstack/react-query';
import produce from 'immer';
import { useCallback, useEffect, useMemo, useRef } from 'react';
import { useCallback, useEffect, useMemo, useRef, useState } from 'react';
import create from 'zustand';

import { getCurrentUserId, poke, scry, subscribeOnce } from '../api/urbit';
import * as db from '../db';
import { createDevLogger } from '../debug';
import { AnalyticsEvent } from '../logic';
import { DeepLinkMetadata, createDeepLink } from '../logic/branch';
import { asyncWithDefault, getFlagParts, withRetry } from '../logic/utils';
import { stringToTa } from '../urbit';
import { GroupMeta } from '../urbit/groups';

const logger = createDevLogger('lure', true);

interface LureMetadata {
tag: string;
fields: Record<string, string | undefined>;
Expand Down Expand Up @@ -73,29 +72,37 @@ export const useLureState = create<LureState>((set, get) => ({
});
}

await poke({
app: 'reel',
mark: 'reel-describe',
json: {
token: flag,
metadata: groupsDescribe({
// legacy keys
title: group?.title ?? '',
description: group?.description ?? '',
cover: group?.coverImage ?? '',
image: group?.iconImage ?? '',

// new-style metadata keys
inviterUserId: currentUserId,
inviterNickname: user?.nickname ?? '',
inviterAvatarImage: user?.avatarImage ?? '',
invitedGroupId: flag,
invitedGroupTitle: group?.title ?? '',
invitedGroupDescription: group?.description ?? '',
invitedGroupIconImageUrl: group?.iconImage ?? '',
}),
},
});
try {
await poke({
app: 'reel',
mark: 'reel-describe',
json: {
token: flag,
metadata: groupsDescribe({
// legacy keys
title: group?.title ?? '',
description: group?.description ?? '',
cover: group?.coverImage ?? '',
image: group?.iconImage ?? '',

// new-style metadata keys
inviterUserId: currentUserId,
inviterNickname: user?.nickname ?? '',
inviterAvatarImage: user?.avatarImage ?? '',
invitedGroupId: flag,
invitedGroupTitle: group?.title ?? '',
invitedGroupDescription: group?.description ?? '',
invitedGroupIconImageUrl: group?.iconImage ?? '',
}),
},
});
} catch (e) {
lureLogger.trackError(AnalyticsEvent.InviteError, {
context: 'reel describe failed',
errorMessage: e.message,
errorStack: e.stack,
});
}
},
toggle: async (flag) => {
const { name } = getFlagParts(flag);
Expand Down Expand Up @@ -130,16 +137,24 @@ export const useLureState = create<LureState>((set, get) => ({
});
},
start: async () => {
const bait = await scry<Bait>({
app: 'reel',
path: '/bait',
});
try {
const bait = await scry<Bait>({
app: 'reel',
path: '/bait',
});

set(
produce((draft: LureState) => {
draft.bait = bait;
})
);
set(
produce((draft: LureState) => {
draft.bait = bait;
})
);
} catch (e) {
lureLogger.trackEvent(AnalyticsEvent.InviteError, {
context: 'failed to get bait on start',
errorMessage: e.message,
errorStack: e.stack,
});
}
},
fetchLure: async (flag, inviteServiceEndpoint, inviteServiceIsDev) => {
const { name } = getFlagParts(flag);
Expand All @@ -162,7 +177,11 @@ export const useLureState = create<LureState>((set, get) => ({
return en;
})
.catch((e) => {
lureLogger.error(`group-enabled failed`, e);
lureLogger.trackEvent(AnalyticsEvent.InviteError, {
context: `group-enabled failed`,
errorMessage: e.message,
errorStack: e.stack,
});
return prevLure?.enabled;
});
}, prevLure?.enabled),
Expand All @@ -178,7 +197,11 @@ export const useLureState = create<LureState>((set, get) => ({
return u;
})
.catch((e) => {
lureLogger.error(`id-link failed`, e);
lureLogger.trackError(AnalyticsEvent.InviteDebug, {
context: `id-link failed`,
errorMessage: e.message,
errorStack: e.stack,
});
return prevLure?.url;
});
}, prevLure?.url),
Expand Down Expand Up @@ -227,6 +250,14 @@ export const useLureState = create<LureState>((set, get) => ({
lureLogger.crumb('deepLinkUrl created', deepLinkUrl);
}

lureLogger.trackEvent(AnalyticsEvent.InviteDebug, {
context: 'fetchLure result',
flag,
enabled,
url,
deepLinkUrl,
});

set(
produce((draft: LureState) => {
draft.lures[flag] = {
Expand Down Expand Up @@ -343,6 +374,7 @@ export function useLureLinkStatus({
inviteServiceEndpoint: string;
inviteServiceIsDev: boolean;
}) {
const [lastLoggedStatus, setLastLoggedStatus] = useState('');
const { supported, fetched, enabled, url, deepLinkUrl, toggle, describe } =
useLure({
flag,
Expand All @@ -351,16 +383,21 @@ export function useLureLinkStatus({
});
const { good, checked } = useLureLinkChecked(url, !!enabled);

lureLogger.crumb('useLureLinkStatus', {
flag,
supported,
fetched,
enabled,
checked,
good,
url,
deepLinkUrl,
});
const inviteInfo = useMemo(
() => ({
flag,
supported,
fetched,
enabled,
checked,
good,
url,
deepLinkUrl,
}),
[flag, supported, fetched, enabled, checked, good, url, deepLinkUrl]
);

lureLogger.crumb('useLureLinkStatus', inviteInfo);

const status = useMemo(() => {
if (!supported) {
Expand All @@ -381,21 +418,33 @@ export function useLureLinkStatus({
}

if (checked && !good) {
lureLogger.trackError('useLureLinkStatus has error status', {
flag,
enabled,
checked,
good,
url,
deepLinkUrl,
});
return 'error';
}

return 'ready';
}, [supported, fetched, enabled, url, checked, deepLinkUrl, good, flag]);

lureLogger.crumb('url', url, 'deepLinkUrl', deepLinkUrl, 'status', status);
// prevent over zealous logging
const statusKey = useMemo(() => {
return `${status}-${fetched}-${checked}`;
}, [status, fetched, checked]);

if (statusKey !== lastLoggedStatus) {
if (status === 'error') {
lureLogger.trackEvent(AnalyticsEvent.InviteError, {
context: 'useLureLinkStatus has error status',
inviteStatus: status,
inviteInfo,
});
} else {
lureLogger.trackEvent(AnalyticsEvent.InviteDebug, {
context: 'useLureLinkStatus log',
inviteStatus: status,
inviteInfo,
});
}
setLastLoggedStatus(statusKey);
}

return { status, shareUrl: deepLinkUrl, toggle, describe };
}
Expand Down
12 changes: 12 additions & 0 deletions packages/ui/src/components/InviteFriendsToTlonButton.tsx
Original file line number Diff line number Diff line change
Expand Up @@ -30,6 +30,10 @@ export function InviteFriendsToTlonButton({
const title = useGroupTitle(group);
const { doCopy } = useCopy(shareUrl || '');

useEffect(() => {
logger.trackEvent('Invite Button Shown', { group: group?.id });
}, []);

const handleInviteButtonPress = useCallback(async () => {
if (shareUrl && status === 'ready' && group) {
if (isWeb) {
Expand Down Expand Up @@ -74,9 +78,17 @@ export function InviteFriendsToTlonButton({
await toggle();
};
if (status === 'disabled' && isGroupAdmin) {
logger.trackEvent(AnalyticsEvent.InviteDebug, {
group: group?.id,
context: 'invite button: disabled and isAdmin, toggling',
});
toggleLink();
}
if (status === 'stale') {
logger.trackEvent(AnalyticsEvent.InviteDebug, {
group: group?.id,
context: 'invite button: stale, describing',
});
describe();
}
}, [group, toggle, status, isGroupAdmin, describe]);
Expand Down

0 comments on commit 0ca965b

Please sign in to comment.