How to properly redirect to different page using trpc?

export default function Post() { const { id } = useRouter().query; const postByIdQuery = trpc.post?.byId.useQuery({ userId: session?.user?.id, id: String(id), }); const { data: post } = postByIdQuery if (!post) { return <Error statusCode=404> } return ( <h1>{post.title}</h1> <p>{post.desc}</p> ) } Previously i use the getServersideProps and if the api returns nothing then simply redirect the above method does not render the 404 page for some reason
75 Replies
kacakthegreat
kacakthegreatOP3y ago
Just wondering whether the above method is correct?
Keef
Keef3y ago
purely client side you can hook into the onSuccess/onError prop on ur query and redirect using the next router from useRouter
cje
cje3y ago
use the onError callback that useQuery has
Keef
Keef3y ago
That would work if they’re throwing an error but not if he’s returning nothing right That’s why I mentioned both
cje
cje3y ago
oh yeah just do if (!thing) { throw new TRPCError({ code: NOT_FOUND }) throwing a 404 is the server's responsibility i mean you can redirect to the 404 page from the client arbitrarily, but seems bad
Keef
Keef3y ago
Yeah i agree
kacakthegreat
kacakthegreatOP3y ago
Any solid example can share, sorry im a newbie @cje @CFKeef Thanks guys
cje
cje3y ago
Error Handling | tRPC
Whenever an error occurs in a procedure, tRPC responds to the client with an object that includes an "error" property. This property contains all the information that you need to handle the error in the client.
Keef
Keef3y ago
I can post a direct solution but try using the docs first and if anything I can hold ur hand with questions you may have shyplead
kacakthegreat
kacakthegreatOP3y ago
hahaha @CFKeef im a bit stuck right now
export default createNextApiHandler({
router: appRouter,
createContext,
onError:
env.NODE_ENV === "development"
? ({ path, error }) => {
console.error(`❌ tRPC failed on ${path}: ${error}`);
}
: undefined,
});
export default createNextApiHandler({
router: appRouter,
createContext,
onError:
env.NODE_ENV === "development"
? ({ path, error }) => {
console.error(`❌ tRPC failed on ${path}: ${error}`);
}
: undefined,
});
I don't think i can reroute example
if (error.code === 'NO_CODE') {
// redirect to 404

}
if (error.code === 'NO_CODE') {
// redirect to 404

}
because its a TS file i dont think i can use useRouter or something
Keef
Keef3y ago
The use router is for the client side In your original code block you can pass in a object to the useQuery arguments That let you adjust how your query functions
Keef
Keef3y ago
Use these docs for that It’ll look something like this .useQuery(input,{}) I’m on my phone right now but I can give you a nice formatted code block in a little
kacakthegreat
kacakthegreatOP3y ago
const postByIdQuery = trpc.post?.byId.useQuery({
userId: session?.user?.id,
id: String(id),
}, onError: (error: TError) => {
if (error === "NO_CODE") {
Router.push('/404')
}
});
const postByIdQuery = trpc.post?.byId.useQuery({
userId: session?.user?.id,
id: String(id),
}, onError: (error: TError) => {
if (error === "NO_CODE") {
Router.push('/404')
}
});
Something like the above?
Keef
Keef3y ago
Just wrap it in braces since it’s expecting an object There’s also this nice prop in there “enabled” that will let the query only run when the condition is met which is good to know in general
kacakthegreat
kacakthegreatOP3y ago
Something like this right
const postByIdQuery = trpc.post?.byId.useQuery({
userId: session?.user?.id,
id: String(id),
},{
enabled: true,
onError: (error: any) => {
if (error.code === "NOT_FOUND") {
// send to bug reporting
Router.push("/404");
}
},
});
const postByIdQuery = trpc.post?.byId.useQuery({
userId: session?.user?.id,
id: String(id),
},{
enabled: true,
onError: (error: any) => {
if (error.code === "NOT_FOUND") {
// send to bug reporting
Router.push("/404");
}
},
});
Keef
Keef3y ago
The enabled thing isn’t necessary I’m just mentioning it for ur future I’m just googling to see if trpc lets you redirect to see if you can do it serverside or if clientside is the only way
kacakthegreat
kacakthegreatOP3y ago
I tried to do serverside with getServerSideprops its not working and found a github thread that the only way is clientside
Keef
Keef3y ago
You can type the error as TRPCError btw or see if it’s implicitly typed without the any If so then ye that should work
kacakthegreat
kacakthegreatOP3y ago
kacakthegreat
kacakthegreatOP3y ago
import { TRPCError } from "@trpc/server"
Keef
Keef3y ago
It might have a type already The general rule for typescript is see if it’s implicitly typed already (meaning don’t need to assign it using : <type>) then explicitly type it See if removing the type and hovering gives you a type I avoid explicitly typing so I just assumed trpcerror would work since that’s the type on the server graybobacat sorry
kacakthegreat
kacakthegreatOP3y ago
kacakthegreat
kacakthegreatOP3y ago
No worries man 🙂
Keef
Keef3y ago
Yeah so someone did the hard work for peepoAing it’s implicitly typed and ur good 2 go Are you seeing any issues?
kacakthegreat
kacakthegreatOP3y ago
kacakthegreat
kacakthegreatOP3y ago
I tried to load the page but it does not redirect to 404 for some reason although the error is obvious
Keef
Keef3y ago
Ur conditional is prob failing Check the red squiggly
kacakthegreat
kacakthegreatOP3y ago
Keef
Keef3y ago
Yeah I’d say console.log it and rerun the page to see what’s available for you You can also ctrl click a type to get to its type declaration but that’s a scary typescript world and you might be overwhelmed its err.data.code
kacakthegreat
kacakthegreatOP3y ago
the onError approach has some issue, it will load the page and then only it will trigger the onError which will lead to 404
Keef
Keef3y ago
Yeah thats the downside of clientside
kacakthegreat
kacakthegreatOP3y ago
Oh my
Keef
Keef3y ago
Same with checking user session on client side, it'll have to render the page then redirect if they dont have one for example
kacakthegreat
kacakthegreatOP3y ago
I wish i can do trpc on getServerside
Keef
Keef3y ago
Ill see if i can dig something up since I'm back at my pc but you really dont need to do this
kacakthegreat
kacakthegreatOP3y ago
Okay thanks man im too lazy to rewrite my trpc code to api based
Keef
Keef3y ago
Just render the 404 content in this component when it errors like that I use trpc strictly and dont use the nextapihandler Atleast I haven't found the need to yet
kacakthegreat
kacakthegreatOP3y ago
Do you have any code example on github that i can learn from, i really need to push this project out haha
Keef
Keef3y ago
digging something up
kacakthegreat
kacakthegreatOP3y ago
okay cool thanks alot
Keef
Keef3y ago
GitHub
create-t3-turbo/packages/api at main · t3-oss/create-t3-turbo
Clean and simple starter repo using the T3 Stack along with Expo React Native - create-t3-turbo/packages/api at main · t3-oss/create-t3-turbo
Keef
Keef3y ago
This is a monorepo tho so don't go digging too hard
Keef
Keef3y ago
GitHub
create-t3-turbo/post.ts at main · t3-oss/create-t3-turbo
Clean and simple starter repo using the T3 Stack along with Expo React Native - create-t3-turbo/post.ts at main · t3-oss/create-t3-turbo
Keef
Keef3y ago
GitHub
create-t3-turbo/index.tsx at main · t3-oss/create-t3-turbo
Clean and simple starter repo using the T3 Stack along with Expo React Native - create-t3-turbo/index.tsx at main · t3-oss/create-t3-turbo
kacakthegreat
kacakthegreatOP3y ago
Yeah the api structure is identical to mine im using the t3 boilerplate first time using it, looks promising hence why i tried it out for this new project haha
Keef
Keef3y ago
I enjoy using it a lot but there are some edge cases like this for example So you might have to just render it on the page instead of trying to redirect to a catchall 404
kacakthegreat
kacakthegreatOP3y ago
i see render the 404 component itself
Keef
Keef3y ago
Yep instead of redirecting to a specific page
kacakthegreat
kacakthegreatOP3y ago
Something like this yeah
Keef
Keef3y ago
no the return from the qeury has a couple properties like isLoading,isError you want to tap into and then just conditionallyrender
kacakthegreat
kacakthegreatOP3y ago
ahh got it
Keef
Keef3y ago
I dont think that'll work but I treat my components more akin to a state machine where if its in this state (isError) return <p>somethign blew up</>
const {isError} = ...useQuery();

if(isError) return <p>something blew up</p>
const {isError} = ...useQuery();

if(isError) return <p>something blew up</p>
kacakthegreat
kacakthegreatOP3y ago
Same issue, it will load the page then only it will show a different component but the url stays the same though
/posts/:id
/posts/:id
instead of /404
Keef
Keef3y ago
Yeah thats fine tho
kacakthegreat
kacakthegreatOP3y ago
oh okay
Keef
Keef3y ago
You don't need to redirect in these situations tbh you can but it seems like it is working against you. There are waysto clean up the search bar btw if you wanted to obfuscate this but idk if routing is gonna get involved since I would treat this solution as enough
kacakthegreat
kacakthegreatOP3y ago
but the loading time to transition is slow, maybe i should i add isLoading as well/
Keef
Keef3y ago
Yeah but thats how async works Prisma is also slow as shit when on a cold start so your performance should be fine when people are using the app
kacakthegreat
kacakthegreatOP3y ago
I see i think adding the isLoading did the trick but i prefer the getServersideprops hahaha
Keef
Keef3y ago
Theres a cost to that too 🙂 its all about trade offs honestly thats engineering for u
kacakthegreat
kacakthegreatOP3y ago
if i have to add isLoading on some of the pages, im not sure though haha
Keef
Keef3y ago
Its tedious yeah
kacakthegreat
kacakthegreatOP3y ago
btw i notice the query will keep on running, whenever you click on the page
Keef
Keef3y ago
Thats the stale-while-revalidate (swr) pattern that react query employs to ensure you are generally upto date
kacakthegreat
kacakthegreatOP3y ago
oh okay reading it right now - https://swr.vercel.app/
React Hooks for Data Fetching – SWR
SWR is a React Hooks library for data fetching. SWR first returns the data from cache (stale), then sends the fetch request (revalidate), and finally comes with the up-to-date data again.
Keef
Keef3y ago
Its not a vercel topic
Keef
Keef3y ago
React Conferences by GitNation
YouTube
React Query: It’s Time to Break up with your "Global State”! –Tanne...
Find the latest React.js talks & workshops at https://portal.gitnation.org     🗓 Talk recording from React Summit Remote Edition 2020 Website - https://remote.reactsummit.com/ See other React conferences by GitNation React Summit – https://reactsummit.com React Advanced London – https://reactadvanced.com React Day Berlin – https://reactday.be...
Keef
Keef3y ago
good talk on this idea and REACT specifically
Keef
Keef3y ago
web.dev
Keeping things fresh with stale-while-revalidate
stale-while-revalidate helps developers balance between immediacy—loading cached content right away—and freshness—ensuring updates to the cached content are used in the future.
Keef
Keef3y ago
general
kacakthegreat
kacakthegreatOP3y ago
Thanks for sharing these 🙂
Keef
Keef3y ago
That vercel link gives you an eli5 explanation which works but if you want a deeper understanding those links will help SWR the library is vercel's take on react query
kacakthegreat
kacakthegreatOP3y ago
Oh i see okay i'll watch it i think now i need to add skeleton loader

Did you find this page helpful?