Skip to content

Commit

Permalink
fix: fetch timeout has better error UX (#378)
Browse files Browse the repository at this point in the history
* fix: fetch timeout has better error UX

* chore: fix property used to get error name
  • Loading branch information
SgtPooki authored Jul 12, 2023
1 parent ca19310 commit 008f83d
Show file tree
Hide file tree
Showing 6 changed files with 54 additions and 8 deletions.
3 changes: 3 additions & 0 deletions public/locales/en/explore.json
Original file line number Diff line number Diff line change
Expand Up @@ -66,5 +66,8 @@
"paragraph1": "A visual representation of the node and its children."
}
}
},
"errors": {
"BlockFetchTimeoutError": "Failed to fetch content in {timeout}s. Please refresh the page to retry or try a different CID."
}
}
2 changes: 1 addition & 1 deletion src/bundles/explore.js
Original file line number Diff line number Diff line change
Expand Up @@ -52,7 +52,7 @@ const makeBundle = () => {
}
} catch (error) {
console.warn('Failed to resolve path', path, error)
return { path, error: error.toString() }
return { path, error }
}
},
staleAfter: Infinity,
Expand Down
9 changes: 2 additions & 7 deletions src/components/ExplorePage.jsx
Original file line number Diff line number Diff line change
Expand Up @@ -6,6 +6,7 @@ import { connect } from 'redux-bundler-react'

import CidInfo from './cid-info/CidInfo'
import ErrorBoundary from './error/ErrorBoundary'
import { IpldExploreErrorComponent } from './explore/IpldExploreErrorComponent'
import IpldGraph from './graph/LoadableIpldGraph'
import GraphCrumb from './graph-crumb/GraphCrumb'
import ComponentLoader from './loader/ComponentLoader'
Expand Down Expand Up @@ -51,13 +52,7 @@ export class ExplorePage extends React.Component {

<div className='dt-l dt--fixed'>
<div className='dtc-l w-100 w-two-thirds-l pr3-l v-top'>
{error
? (
<div className='bg-red white pa3 lh-copy'>
{error}
</div>
)
: null}
<IpldExploreErrorComponent error={error} />

{targetNode
? (
Expand Down
19 changes: 19 additions & 0 deletions src/components/explore/IpldExploreErrorComponent.tsx
Original file line number Diff line number Diff line change
@@ -0,0 +1,19 @@
import React from 'react'
import { useTranslation } from 'react-i18next'

import type IpldExploreError from '../../lib/errors'

export interface IpldExploreErrorComponentProps {
error?: IpldExploreError
}

export function IpldExploreErrorComponent ({ error }: IpldExploreErrorComponentProps): JSX.Element | null {
const { t } = useTranslation('explore', { keyPrefix: 'errors' })
if (error == null) return null

return (
<div className='bg-red white pa3 lh-copy'>
<div>{error.toString(t)}</div>
</div>
)
}
24 changes: 24 additions & 0 deletions src/lib/errors.ts
Original file line number Diff line number Diff line change
@@ -0,0 +1,24 @@
import { type TFunction } from 'i18next'

export default class IpldExploreError extends Error {
constructor (private readonly options: Record<string, string | number>) {
super()
this.name = this.constructor.name
}

/**
* See IpldExploreError for usage.
* You must pass a t function that is registered with namespace 'explore' and keyPrefix 'errors'.
*
* @param t - the i18next-react t function
* @returns the translated string
* @example
* const {t} = useTranslation('explore', { keyPrefix: 'errors' })
* t('NameOfErrorClassThatExtendsIpldExploreError')
*/
toString (t: TFunction<'translation', undefined, 'translation'>): string {
return t(this.name, this.options)
}
}

export class BlockFetchTimeoutError extends IpldExploreError {}
5 changes: 5 additions & 0 deletions src/lib/get-raw-block.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,6 +3,7 @@ import { type IPFSHTTPClient } from 'kubo-rpc-client'
import { CID } from 'multiformats'
import type { Version as CIDVersion } from 'multiformats/cid'

import { BlockFetchTimeoutError } from './errors'
import getHasherForCode from './hash-importer.js'

async function getCidFromBytes (bytes: Uint8Array, cidVersion: CIDVersion, codecCode: number, multihashCode: number): Promise<CID> {
Expand Down Expand Up @@ -130,6 +131,10 @@ export async function getRawBlock (helia: Helia, kuboClient: IPFSHTTPClient, cid
return rawBlock
} catch (err) {
console.error('unable to get raw block', err)
if (abortController.signal.aborted) {
// if we timed out, we want to throw a timeout error, not a Promise.any error
throw new BlockFetchTimeoutError({ timeout: timeout / 1000 })
}
throw err
}
}

0 comments on commit 008f83d

Please sign in to comment.