Skip to content

Commit 54af2dd

Browse files
authoredFeb 28, 2025··
fix(core): batch requests for archived releases view (#8779)
1 parent 4945f0a commit 54af2dd

File tree

3 files changed

+117
-37
lines changed

3 files changed

+117
-37
lines changed
 

‎packages/sanity/src/core/releases/i18n/resources.ts

+8-2
Original file line numberDiff line numberDiff line change
@@ -130,8 +130,6 @@ const releasesLocaleStrings = {
130130
/** Title for dialog for discarding a version of a document */
131131
'discard-version-dialog.title': 'Yes, discard version',
132132

133-
/** Text for when documents of a release are loading */
134-
'document-loading': 'Loading documents',
135133
/** Label for when a document in a release has multiple validation warnings */
136134
'document-validation.error_other': '{{count}} validation errors',
137135
/** Label for when a document in a release has a single validation warning */
@@ -162,6 +160,14 @@ const releasesLocaleStrings = {
162160
/** Label text for the loading state whilst release is being loaded */
163161
'loading-release': 'Loading release',
164162

163+
/** Text for when documents of a release are loading */
164+
'loading-release-documents': 'Loading documents',
165+
/** Title text for when loading documents on a release failed */
166+
'loading-release-documents.error.title': 'Something went wrong',
167+
/** Description text for when loading documents on a release failed */
168+
'loading-release-documents.error.description':
169+
"We're unable to load the documents for this release. Please try again later.",
170+
165171
/** Label for the release menu */
166172
'menu.label': 'Release menu',
167173
/** Tooltip for the release menu */

‎packages/sanity/src/core/releases/tool/detail/ReleaseDetail.tsx

+43-5
Original file line numberDiff line numberDiff line change
@@ -1,4 +1,6 @@
1-
import {Card, Container, Flex, Heading, Stack} from '@sanity/ui'
1+
import {ErrorOutlineIcon} from '@sanity/icons'
2+
import {Box, Card, Container, Flex, Heading, Stack, Text} from '@sanity/ui'
3+
import {motion} from 'framer-motion'
24
import {useMemo, useRef, useState} from 'react'
35
import {useRouter} from 'sanity/router'
46

@@ -19,18 +21,22 @@ import {ReleaseSummary} from './ReleaseSummary'
1921
import {useBundleDocuments} from './useBundleDocuments'
2022

2123
export type ReleaseInspector = 'activity'
24+
const MotionCard = motion(Card)
2225

2326
export const ReleaseDetail = () => {
2427
const router = useRouter()
2528
const [inspector, setInspector] = useState<ReleaseInspector | undefined>(undefined)
2629
const {t} = useTranslation(releasesLocaleNamespace)
27-
2830
const {releaseId: releaseIdRaw}: ReleasesRouterState = router.state
2931
const releaseId = decodeURIComponent(releaseIdRaw || '')
3032
const {data, loading} = useActiveReleases()
3133
const {data: archivedReleases} = useArchivedReleases()
3234

33-
const {loading: documentsLoading, results} = useBundleDocuments(releaseId)
35+
const {
36+
loading: documentsLoading,
37+
results,
38+
error: bundleDocumentsError,
39+
} = useBundleDocuments(releaseId)
3440
const releaseEvents = useReleaseEvents(releaseId)
3541

3642
const documentIds = results.map((result) => result.document?._id)
@@ -43,8 +49,33 @@ export const ReleaseDetail = () => {
4349
const scrollContainerRef = useRef<HTMLDivElement | null>(null)
4450

4551
const detailContent = useMemo(() => {
52+
if (bundleDocumentsError) {
53+
return (
54+
<Box padding={3}>
55+
<MotionCard
56+
initial={{opacity: 0}}
57+
animate={{opacity: 1}}
58+
tone="critical"
59+
padding={4}
60+
radius={4}
61+
>
62+
<Flex gap={3}>
63+
<Text size={1}>
64+
<ErrorOutlineIcon />
65+
</Text>
66+
<Stack space={4}>
67+
<Text size={1} weight="semibold">
68+
{t('loading-release-documents.error.title')}
69+
</Text>
70+
<Text size={1}>{t('loading-release-documents.error.description')}</Text>
71+
</Stack>
72+
</Flex>
73+
</MotionCard>
74+
</Box>
75+
)
76+
}
4677
if (documentsLoading) {
47-
return <LoadingBlock title={t('document-loading')} />
78+
return <LoadingBlock title={t('loading-release-documents')} showText />
4879
}
4980
if (!releaseInDetail) return null
5081

@@ -56,7 +87,14 @@ export const ReleaseDetail = () => {
5687
scrollContainerRef={scrollContainerRef}
5788
/>
5889
)
59-
}, [releaseInDetail, documentsLoading, history.documentsHistory, results, t])
90+
}, [
91+
bundleDocumentsError,
92+
documentsLoading,
93+
releaseInDetail,
94+
results,
95+
history.documentsHistory,
96+
t,
97+
])
6098

6199
if (loading) {
62100
return (

‎packages/sanity/src/core/releases/tool/detail/useBundleDocuments.ts

+66-30
Original file line numberDiff line numberDiff line change
@@ -9,10 +9,14 @@ import {useMemo} from 'react'
99
import {useObservable} from 'react-rx'
1010
import {combineLatest, from, type Observable, of} from 'rxjs'
1111
import {
12+
catchError,
13+
concatMap,
1214
distinctUntilChanged,
15+
expand,
1316
filter,
1417
map,
1518
mergeMap,
19+
reduce,
1620
startWith,
1721
switchAll,
1822
switchMap,
@@ -48,7 +52,11 @@ export interface DocumentInRelease {
4852
}
4953
}
5054

51-
type ReleaseDocumentsObservableResult = Observable<{loading: boolean; results: DocumentInRelease[]}>
55+
type ReleaseDocumentsObservableResult = Observable<{
56+
loading: boolean
57+
results: DocumentInRelease[]
58+
error: Error | null
59+
}>
5260

5361
const getActiveReleaseDocumentsObservable = ({
5462
schema,
@@ -153,7 +161,10 @@ const getActiveReleaseDocumentsObservable = ({
153161
})),
154162
)
155163
}),
156-
map((results) => ({loading: false, results})),
164+
map((results) => ({loading: false, results, error: null})),
165+
catchError((error) => {
166+
return of({loading: false, results: [], error})
167+
}),
157168
)
158169
}
159170

@@ -172,49 +183,73 @@ const getPublishedArchivedReleaseDocumentsObservable = ({
172183
const observableClient = client.observable
173184
const dataset = client.config().dataset
174185

175-
if (!release.finalDocumentStates?.length) return of({loading: false, results: []})
186+
if (!release.finalDocumentStates?.length) return of({loading: false, results: [], error: null})
176187

177-
return from(release.finalDocumentStates || []).pipe(
178-
mergeMap(({id: documentId}) => {
179-
const document$ = observableClient
180-
.request<{documents: DocumentInRelease['document'][]}>({
181-
url: `/data/history/${dataset}/documents/${documentId}?lastRevision=true`,
182-
})
183-
.pipe(map(({documents: [document]}) => document))
188+
function batchRequestDocumentFromHistory(startIndex: number) {
189+
const finalIndex = startIndex + 10
190+
return observableClient
191+
.request<{documents: DocumentInRelease['document'][]}>({
192+
url: `/data/history/${dataset}/documents/${release.finalDocumentStates
193+
?.slice(startIndex, finalIndex)
194+
.map((d) => d.id)
195+
.join(',')}?lastRevision=true`,
196+
})
197+
.pipe(map(({documents}) => ({documents, finalIndex})))
198+
}
199+
200+
const documents$ = batchRequestDocumentFromHistory(0).pipe(
201+
expand((response) => {
202+
if (release.finalDocumentStates && response.finalIndex < release.finalDocumentStates.length) {
203+
// Continue with next batch
204+
return batchRequestDocumentFromHistory(response.finalIndex)
205+
}
206+
// End recursion by emitting an empty observable
207+
return of()
208+
}),
209+
reduce(
210+
(documents: DocumentInRelease['document'][], batch) => documents.concat(batch.documents),
211+
[],
212+
),
213+
)
184214

185-
const previewValues$ = document$.pipe(
186-
switchMap((document) => {
215+
return documents$.pipe(
216+
mergeMap((documents) => {
217+
return from(documents).pipe(
218+
concatMap((document) => {
187219
const schemaType = schema.get(document._type)
188220
if (!schemaType) {
189221
throw new Error(`Schema type not found for document type ${document._type}`)
190222
}
191-
192-
return documentPreviewStore.observeForPreview(document, schemaType).pipe(
223+
const previewValues$ = documentPreviewStore.observeForPreview(document, schemaType).pipe(
193224
take(1),
194225
map(({snapshot}) => ({
195226
isLoading: false,
196227
values: snapshot,
197228
})),
198229
startWith({isLoading: true, values: {}}),
230+
filter(({isLoading}) => !isLoading),
199231
)
200-
}),
201-
filter(({isLoading}) => !isLoading),
202-
)
203232

204-
return combineLatest([document$, previewValues$]).pipe(
205-
map(([document, previewValues]) => ({
206-
document,
207-
previewValues,
208-
memoKey: uuid(),
209-
validation: {validation: [], hasError: false, isValidating: false},
233+
return previewValues$.pipe(
234+
map((previewValues) => ({
235+
document,
236+
previewValues,
237+
memoKey: uuid(),
238+
validation: {validation: [], hasError: false, isValidating: false},
239+
})),
240+
)
241+
}),
242+
toArray(),
243+
map((results) => ({
244+
loading: false,
245+
results,
246+
error: null,
210247
})),
211248
)
212249
}),
213-
toArray(),
214-
map((results) => ({
215-
loading: false,
216-
results,
217-
})),
250+
catchError((error) => {
251+
return of({loading: false, results: [], error})
252+
}),
218253
)
219254
}
220255

@@ -257,12 +292,13 @@ const getReleaseDocumentsObservable = ({
257292
releaseId,
258293
})
259294
}),
260-
startWith({loading: true, results: []}),
295+
startWith({loading: true, results: [], error: null}),
261296
)
262297

263298
export function useBundleDocuments(releaseId: string): {
264299
loading: boolean
265300
results: DocumentInRelease[]
301+
error: null | Error
266302
} {
267303
const documentPreviewStore = useDocumentPreviewStore()
268304
const {getClient, i18n} = useSource()
@@ -282,5 +318,5 @@ export function useBundleDocuments(releaseId: string): {
282318
[schema, documentPreviewStore, getClient, releaseId, i18n, releasesState$],
283319
)
284320

285-
return useObservable(releaseDocumentsObservable, {loading: true, results: []})
321+
return useObservable(releaseDocumentsObservable, {loading: true, results: [], error: null})
286322
}

0 commit comments

Comments
 (0)
Please sign in to comment.