I want to implement GetServerSideProps with SWR, but i cant stop the clientside from fetching even though the data is loaded from the server side.
Here's a simplified example
We fetch a wordlist from the api with a given wordlength in the getServerSide/getStaticProps and SSR/SSG build the page for fast delivery
The user interacts with the page causing
iterateWordLength()
to firenow we go fetch an updated wordlist with
mutate
What's happening: SWR is firing twice on load
- Serverside runs just once ✅
- Wordlist useEffect fires - get the word "foo" from my artificial getServerSideProps and render ✅
- WordClientSideFetcher fires ❌ (it shouldnt cause we already have the data)
Wordlist changed
fires again ❌
i am not using useSWRImmutable because my fetcher and GET params will change based on the state of another state variable
I've set all the revalidate options false... why is it firing twice?
Simplified Index.js
function Index( { fallback } ) {
const [wordLength, setWordLength] = useState(DEFAULT_WORD_LENGTH);
const { data: wordlist, mutate } = useSWR('/api/wordlist', clientSideFetcher, {
fallbackData: fallback['/api/wordlist'],
revalidateIfStale: false, // all set to false to stop that clientSideFetcher
revalidateOnMount: false,
revalidateOnFocus: false
})
}
useEffect(()=>{
// wordlength iterated
mutate() // updates wordlist
}, [wordLength])
useEffect(()=>{
console.log("### Wordlist changed")
setWord( generateRandomWord() )
},[wordlist])
const iterateWordlength = () => { setWordlength(wordLength+1)}
export async function staticFetcher() {
const res = await fetch(API_URL)
const data = await res.json()
console.log(`### staticFetcher fetched ${data.length} words`)
return new Wordlist(...data)
}
export async function getServerSideProps(context) {
const staticWordlist = await staticFetcher(context)
.catch( err => {
return { notFound: true }
})
const props = {
fallback: {
'/api/wordlist' : ["foo"] // temp replace the actual array
}
}
return { props }
}