Issue with Fetching Data in Next.js (T3 Stack) using Async Params and Suspense Queries
Unanswered
Yellow croaker posted this in #help-forum
Yellow croakerOP
I'm trying to fetch data for my story/[storyId]/page.tsx component in a T3 Stack Next.js app, but I'm running into issues handling the id parameter and using useSuspenseQuery.
First attempt:
Issues:
- id seems to require await, but useSuspenseQuery is meant for client components.
- How should I properly handle params in this setup?
Second Attempt:
- params.storyId is coming in as undefined.
I have a UserStories component that links to each story like this:
How should I properly handle fetching the story data in my story/[storyId]/page.tsx component using the T3 stack? Given that useSuspenseQuery requires a client component and params might be async, what are my best options
First attempt:
'use client'
import { notFound } from "next/navigation";
import { api } from "@/trpc/react";
import StoryDisplay from "@/app/_components/StoryDisplay";
import { use } from "react";
export default function Page({ params }: { params: Promise<{ id: string }> }) {
const { id } = use(params);
const [story] = api.story.getById.useSuspenseQuery({ id });
if (!story) {
notFound();
}
return <StoryDisplay story={story} />;
}
Issues:
- id seems to require await, but useSuspenseQuery is meant for client components.
- How should I properly handle params in this setup?
Second Attempt:
import { notFound } from "next/navigation";
import { api } from "@/trpc/react";
import StoryDisplay from "@/app/_components/StoryDisplay";
interface StoryPageProps {
params: {
storyId: string;
};
}
export default function StoryPage({ params }: StoryPageProps) {
const [story] = api.story.getById.useSuspenseQuery({ id: params.storyId });
if (!story) {
notFound();
}
return <StoryDisplay story={story} />;
}
- params.storyId is coming in as undefined.
I have a UserStories component that links to each story like this:
<Link
href={`/story/${story.id}`}
className="w-full rounded-full bg-white/10 px-4 py-2 text-center font-semibold transition hover:bg-white/20"
>
Read Story
</Link>
How should I properly handle fetching the story data in my story/[storyId]/page.tsx component using the T3 stack? Given that useSuspenseQuery requires a client component and params might be async, what are my best options
9 Replies
In server components you should prefer to fetch data right in the component body and let React know your component is suspending by just awaiting the calls.
instead of
instead of
use(promise)
do await promise
Also, if you make your page a client component now you can to access the searchParams and params via the hooks Next.js provides
But the best practice here would be to keep your pages and layouts always a Server Component.
Params and SearchParams come through props as promises, so you need to await them to access them, and just fetch inside the component, your <Suspense> wrapping your page will trigger as soon as it encounters an await keyword
useParams
and useSearchParams
.But the best practice here would be to keep your pages and layouts always a Server Component.
Params and SearchParams come through props as promises, so you need to await them to access them, and just fetch inside the component, your <Suspense> wrapping your page will trigger as soon as it encounters an await keyword
@LuisLl In server components you should prefer to fetch data right in the component body and let React know your component is suspending by just awaiting the calls.
instead of `use(promise)` do `await promise`
Yellow croakerOP
Okey so i tried this:
But that provides ts error:
Unexpected
And the error on client is:
TypeError: Cannot read properties of undefined (reading 'getById')
import { notFound } from "next/navigation";
import { api } from "@/trpc/react";
import StoryDisplay from "@/app/_components/StoryDisplay";
interface StoryPageProps {
params: {
storyId: string;
};
}
export default async function StoryPage({ params }: StoryPageProps) {
const [story] = await api.story.getById.useSuspenseQuery({ id: params.storyId });
if (!story) {
notFound();
}
return <StoryDisplay story={story} />;
}
But that provides ts error:
Unexpected
await
of a non-Promise (non-"Thenable") valueAnd the error on client is:
TypeError: Cannot read properties of undefined (reading 'getById')
@LuisLl Also, if you make your page a client component now you can to access the searchParams and params via the hooks Next.js provides `useParams` and `useSearchParams`.
But the best practice here would be to keep your pages and layouts always a Server Component.
Params and SearchParams come through props as promises, so you need to await them to access them, and just fetch inside the component, your <Suspense> wrapping your page will trigger as soon as it encounters an **await** keyword
Yellow croakerOP
I would love to keep the page component as server component, becuase it's used only to fetch data. But i would also to love keep my API and trpc in use.
I have used server actions before, but i would maybe want to combine these two?
Or just use the trpc instead.
I have used server actions before, but i would maybe want to combine these two?
Or just use the trpc instead.
Don’t use the
Instead call the regular server query, I’m not a TRPC user but I believe there’s the client queries via hooks and the server queries that don’t need a hook and can simply be awaited.
useSuspenseQuery()
that won’t return what you’re expecting.Instead call the regular server query, I’m not a TRPC user but I believe there’s the client queries via hooks and the server queries that don’t need a hook and can simply be awaited.
@Yellow croaker Okey so i tried this:
ts
import { notFound } from "next/navigation";
import { api } from "@/trpc/react";
import StoryDisplay from "@/app/_components/StoryDisplay";
interface StoryPageProps {
params: {
storyId: string;
};
}
export default async function StoryPage({ params }: StoryPageProps) {
const [story] = await api.story.getById.useSuspenseQuery({ id: params.storyId });
if (!story) {
notFound();
}
return <StoryDisplay story={story} />;
}
But that provides ts error:
Unexpected `await` of a non-Promise (non-"Thenable") value
And the error on client is:
TypeError: Cannot read properties of undefined (reading 'getById')
If this is your page and you’re getting params through props make sure to type them as a promise starting from Next v15.
Yellow croakerOP
Yeah im using nextjs 15. idk this trpc feels a bit confusing, i understand it in some level but it feels a bit more stranger than server actions. Becuase with server action i could just fetch the data i want, but now even the .query does not work.
Like this code hits me error:
Like this code hits me error:
TRPCError: No procedure found on path "story,getById,query"
import { notFound } from "next/navigation";
import { api } from "@/trpc/server";
import StoryDisplay from "@/app/_components/StoryDisplay";
interface StoryPageProps {
params: {
storyId: string;
};
}
export default async function StoryPage({ params }: StoryPageProps) {
const story = await api.story.getById.query({ id: params.storyId });
if (!story) {
notFound();
}
return <StoryDisplay story={story} />;
}
Or am i missing something how the data fetching works with this query thing now?
Tbh I’m not familiar with TRPC but the docs have a full guide on how to make it work with Server Components