Refactor ChangeHandle modal (#1929)

* Refactor ChangeHandle to use new methods

* Better telemetry

* Remove unused logic

* Remove caching

* Add error message

* Persist service changes, don't fall back on change handle
This commit is contained in:
Eric Bailey 2023-11-16 11:16:16 -06:00 committed by GitHub
parent e6efeea7c0
commit a652b52b88
No known key found for this signature in database
GPG key ID: 4AEE18F83AFDEB23
4 changed files with 154 additions and 101 deletions

View file

@ -1,9 +1,10 @@
import React from 'react'
import {useQueryClient} from '@tanstack/react-query'
import {useQueryClient, useMutation} from '@tanstack/react-query'
import {useSession} from '#/state/session'
const fetchHandleQueryKey = (handleOrDid: string) => ['handle', handleOrDid]
const fetchDidQueryKey = (handleOrDid: string) => ['did', handleOrDid]
export function useFetchHandle() {
const {agent} = useSession()
@ -23,3 +24,35 @@ export function useFetchHandle() {
[agent, queryClient],
)
}
export function useUpdateHandleMutation() {
const {agent} = useSession()
return useMutation({
mutationFn: async ({handle}: {handle: string}) => {
await agent.updateHandle({handle})
},
})
}
export function useFetchDid() {
const {agent} = useSession()
const queryClient = useQueryClient()
return React.useCallback(
async (handleOrDid: string) => {
return queryClient.fetchQuery({
queryKey: fetchDidQueryKey(handleOrDid),
queryFn: async () => {
let identifier = handleOrDid
if (!identifier.startsWith('did:')) {
const res = await agent.resolveHandle({handle: identifier})
identifier = res.data.did
}
return identifier
},
})
},
[agent, queryClient],
)
}

View file

@ -0,0 +1,16 @@
import {useQuery} from '@tanstack/react-query'
import {useSession} from '#/state/session'
export const RQKEY = (serviceUrl: string) => ['service', serviceUrl]
export function useServiceQuery() {
const {agent} = useSession()
return useQuery({
queryKey: RQKEY(agent.service.toString()),
queryFn: async () => {
const res = await agent.com.atproto.server.describeServer()
return res.data
},
})
}

View file

@ -14,8 +14,8 @@ export type SessionState = {
agent: BskyAgent
isInitialLoad: boolean
isSwitchingAccounts: boolean
accounts: persisted.PersistedAccount[]
currentAccount: persisted.PersistedAccount | undefined
accounts: SessionAccount[]
currentAccount: SessionAccount | undefined
}
export type StateContext = SessionState & {
hasSession: boolean
@ -70,15 +70,15 @@ const ApiContext = React.createContext<ApiContext>({
})
function createPersistSessionHandler(
account: persisted.PersistedAccount,
account: SessionAccount,
persistSessionCallback: (props: {
expired: boolean
refreshedAccount: persisted.PersistedAccount
refreshedAccount: SessionAccount
}) => void,
): AtpPersistSessionHandler {
return function persistSession(event, session) {
const expired = !(event === 'create' || event === 'update')
const refreshedAccount = {
const refreshedAccount: SessionAccount = {
service: account.service,
did: session?.did || account.did,
handle: session?.handle || account.handle,
@ -128,7 +128,7 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
)
const upsertAccount = React.useCallback(
(account: persisted.PersistedAccount, expired = false) => {
(account: SessionAccount, expired = false) => {
setStateAndPersist(s => {
return {
...s,
@ -164,8 +164,8 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
throw new Error(`session: createAccount failed to establish a session`)
}
const account: persisted.PersistedAccount = {
service,
const account: SessionAccount = {
service: agent.service.toString(),
did: agent.session.did,
handle: agent.session.handle,
email: agent.session.email!, // TODO this is always defined?
@ -215,8 +215,8 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
throw new Error(`session: login failed to establish a session`)
}
const account: persisted.PersistedAccount = {
service,
const account: SessionAccount = {
service: agent.service.toString(),
did: agent.session.did,
handle: agent.session.handle,
email: agent.session.email!, // TODO this is always defined?
@ -293,9 +293,24 @@ export function Provider({children}: React.PropsWithChildren<{}>) {
}),
)
if (!agent.session) {
throw new Error(`session: initSession failed to establish a session`)
}
// ensure changes in handle/email etc are captured on reload
const freshAccount: SessionAccount = {
service: agent.service.toString(),
did: agent.session.did,
handle: agent.session.handle,
email: agent.session.email!, // TODO this is always defined?
emailConfirmed: agent.session.emailConfirmed || false,
refreshJwt: agent.session.refreshJwt,
accessJwt: agent.session.accessJwt,
}
setState(s => ({...s, agent}))
upsertAccount(account)
emitSessionLoaded(account, agent)
upsertAccount(freshAccount)
emitSessionLoaded(freshAccount, agent)
},
[upsertAccount],
)