Various search fixes (#2145)

* Add posts-search query to shadow cache search

* Update user search to use correct endpoint

* Fix: include cursor in post search
zio/stable
Paul Frazee 2023-12-07 17:20:17 -08:00 committed by GitHub
parent 448a403c81
commit 7b50331188
No known key found for this signature in database
GPG Key ID: 4AEE18F83AFDEB23
5 changed files with 88 additions and 34 deletions

View File

@ -6,6 +6,7 @@ import {Shadow, castAsShadow} from './types'
import {findAllPostsInQueryData as findAllPostsInNotifsQueryData} from '../queries/notifications/feed' import {findAllPostsInQueryData as findAllPostsInNotifsQueryData} from '../queries/notifications/feed'
import {findAllPostsInQueryData as findAllPostsInFeedQueryData} from '../queries/post-feed' import {findAllPostsInQueryData as findAllPostsInFeedQueryData} from '../queries/post-feed'
import {findAllPostsInQueryData as findAllPostsInThreadQueryData} from '../queries/post-thread' import {findAllPostsInQueryData as findAllPostsInThreadQueryData} from '../queries/post-thread'
import {findAllPostsInQueryData as findAllPostsInSearchQueryData} from '../queries/search-posts'
import {queryClient} from 'lib/react-query' import {queryClient} from 'lib/react-query'
export type {Shadow} from './types' export type {Shadow} from './types'
@ -98,4 +99,7 @@ function* findPostsInCache(
yield node.post yield node.post
} }
} }
for (let post of findAllPostsInSearchQueryData(queryClient, uri)) {
yield post
}
} }

View File

@ -11,6 +11,7 @@ import {findAllProfilesInQueryData as findAllProfilesInProfileQueryData} from '.
import {findAllProfilesInQueryData as findAllProfilesInProfileFollowersQueryData} from '../queries/profile-followers' import {findAllProfilesInQueryData as findAllProfilesInProfileFollowersQueryData} from '../queries/profile-followers'
import {findAllProfilesInQueryData as findAllProfilesInProfileFollowsQueryData} from '../queries/profile-follows' import {findAllProfilesInQueryData as findAllProfilesInProfileFollowsQueryData} from '../queries/profile-follows'
import {findAllProfilesInQueryData as findAllProfilesInSuggestedFollowsQueryData} from '../queries/suggested-follows' import {findAllProfilesInQueryData as findAllProfilesInSuggestedFollowsQueryData} from '../queries/suggested-follows'
import {findAllProfilesInQueryData as findAllProfilesInActorSearchQueryData} from '../queries/actor-search'
import {Shadow, castAsShadow} from './types' import {Shadow, castAsShadow} from './types'
import {queryClient} from 'lib/react-query' import {queryClient} from 'lib/react-query'
export type {Shadow} from './types' export type {Shadow} from './types'
@ -98,4 +99,5 @@ function* findProfilesInCache(did: string): Generator<ProfileView, void> {
yield* findAllProfilesInProfileFollowersQueryData(queryClient, did) yield* findAllProfilesInProfileFollowersQueryData(queryClient, did)
yield* findAllProfilesInProfileFollowsQueryData(queryClient, did) yield* findAllProfilesInProfileFollowsQueryData(queryClient, did)
yield* findAllProfilesInSuggestedFollowsQueryData(queryClient, did) yield* findAllProfilesInSuggestedFollowsQueryData(queryClient, did)
yield* findAllProfilesInActorSearchQueryData(queryClient, did)
} }

View File

@ -0,0 +1,42 @@
import {AppBskyActorDefs} from '@atproto/api'
import {QueryClient, useQuery} from '@tanstack/react-query'
import {getAgent} from '#/state/session'
import {STALE} from '#/state/queries'
export const RQKEY = (prefix: string) => ['actor-search', prefix]
export function useActorSearch(prefix: string) {
return useQuery<AppBskyActorDefs.ProfileView[]>({
staleTime: STALE.MINUTES.ONE,
queryKey: RQKEY(prefix || ''),
async queryFn() {
const res = await getAgent().searchActors({
term: prefix,
})
return res.data.actors
},
enabled: !!prefix,
})
}
export function* findAllProfilesInQueryData(
queryClient: QueryClient,
did: string,
) {
const queryDatas = queryClient.getQueriesData<AppBskyActorDefs.ProfileView[]>(
{
queryKey: ['actor-search'],
},
)
for (const [_queryKey, queryData] of queryDatas) {
if (!queryData) {
continue
}
for (const actor of queryData) {
if (actor.did === did) {
yield actor
}
}
}
}

View File

@ -1,7 +1,13 @@
import {AppBskyFeedSearchPosts} from '@atproto/api' import {AppBskyFeedDefs, AppBskyFeedSearchPosts} from '@atproto/api'
import {useInfiniteQuery, InfiniteData, QueryKey} from '@tanstack/react-query' import {
useInfiniteQuery,
InfiniteData,
QueryKey,
QueryClient,
} from '@tanstack/react-query'
import {getAgent} from '#/state/session' import {getAgent} from '#/state/session'
import {embedViewRecordToPostView, getEmbeddedPost} from './util'
const searchPostsQueryKey = ({query}: {query: string}) => [ const searchPostsQueryKey = ({query}: {query: string}) => [
'search-posts', 'search-posts',
@ -17,10 +23,11 @@ export function useSearchPostsQuery({query}: {query: string}) {
string | undefined string | undefined
>({ >({
queryKey: searchPostsQueryKey({query}), queryKey: searchPostsQueryKey({query}),
queryFn: async () => { queryFn: async ({pageParam}) => {
const res = await getAgent().app.bsky.feed.searchPosts({ const res = await getAgent().app.bsky.feed.searchPosts({
q: query, q: query,
limit: 25, limit: 25,
cursor: pageParam,
}) })
return res.data return res.data
}, },
@ -28,3 +35,30 @@ export function useSearchPostsQuery({query}: {query: string}) {
getNextPageParam: lastPage => lastPage.cursor, getNextPageParam: lastPage => lastPage.cursor,
}) })
} }
export function* findAllPostsInQueryData(
queryClient: QueryClient,
uri: string,
): Generator<AppBskyFeedDefs.PostView, undefined> {
const queryDatas = queryClient.getQueriesData<
InfiniteData<AppBskyFeedSearchPosts.OutputSchema>
>({
queryKey: ['search-posts'],
})
for (const [_queryKey, queryData] of queryDatas) {
if (!queryData?.pages) {
continue
}
for (const page of queryData?.pages) {
for (const post of page.posts) {
if (post.uri === uri) {
yield post
}
const quotedPost = getEmbeddedPost(post.embed)
if (quotedPost?.uri === uri) {
yield embedViewRecordToPostView(quotedPost)
}
}
}
}
}

View File

@ -36,6 +36,7 @@ import {useTheme} from 'lib/ThemeContext'
import {useSession} from '#/state/session' import {useSession} from '#/state/session'
import {useGetSuggestedFollowersByActor} from '#/state/queries/suggested-follows' import {useGetSuggestedFollowersByActor} from '#/state/queries/suggested-follows'
import {useSearchPostsQuery} from '#/state/queries/search-posts' import {useSearchPostsQuery} from '#/state/queries/search-posts'
import {useActorSearch} from '#/state/queries/actor-search'
import {useActorAutocompleteFn} from '#/state/queries/actor-autocomplete' import {useActorAutocompleteFn} from '#/state/queries/actor-autocomplete'
import {useSetDrawerOpen} from '#/state/shell' import {useSetDrawerOpen} from '#/state/shell'
import {useAnalytics} from '#/lib/analytics/analytics' import {useAnalytics} from '#/lib/analytics/analytics'
@ -278,38 +279,9 @@ function SearchScreenPostResults({query}: {query: string}) {
function SearchScreenUserResults({query}: {query: string}) { function SearchScreenUserResults({query}: {query: string}) {
const {_} = useLingui() const {_} = useLingui()
const [isFetched, setIsFetched] = React.useState(false) const {data: results, isFetched} = useActorSearch(query)
const [results, setResults] = React.useState<
AppBskyActorDefs.ProfileViewBasic[]
>([])
const search = useActorAutocompleteFn()
React.useEffect(() => { return isFetched && results ? (
async function getResults() {
try {
const searchResults = await search({query, limit: 30})
if (searchResults) {
setResults(searchResults)
}
} catch (e: any) {
logger.error(`SearchScreenUserResults: failed to get results`, {
error: e.toString(),
})
} finally {
setIsFetched(true)
}
}
if (query) {
getResults()
} else {
setResults([])
setIsFetched(false)
}
}, [query, search, setResults])
return isFetched ? (
<> <>
{results.length ? ( {results.length ? (
<FlatList <FlatList