如何创建一个能够在 NextJS 13 中获取数据的简单服务器组件?

huangapple go评论84阅读模式
英文:

How do I create a simple server component that can fetch data in NextJS 13?

问题

I am learning NextJS. Currently, I am using NextJS 13 with the new app directory instead of the old pages directory. I have tried various ways of fetching data, but I have not gotten any of them to work. It should be as simple as

src/app/page.tsx:

  1. export interface ToDo {
  2. userId: number;
  3. id: number;
  4. title: string;
  5. completed: boolean;
  6. }
  7. const getToDos = async (): Promise<ToDo[]> => {
  8. const res = await fetch('https://jsonplaceholder.typicode.com/todos/1');
  9. console.log(`Status: ${res.status}`); // prints 'Status: 200'
  10. return res.json();
  11. }
  12. const Home = async () => {
  13. const toDos = await getToDos();
  14. return (
  15. <ul>
  16. {toDos.map((todo) => (
  17. <li key={todo.id}>{todo.title}</li>
  18. ))}
  19. </ul>
  20. );
  21. }
  22. export default Home;

The JSON returned by the endpoint is

  1. {
  2. "userId": 1,
  3. "id": 1,
  4. "title": "delectus aut autem",
  5. "completed": false
  6. }

This is the output. Notice how I am getting both a 200 OK status and an error message:

  1. Warning: Only plain objects can be passed to Client Components from Server Components. Classes or other objects with methods are not supported.
  2. Status: 200
  3. - GET / 200 in 606ms
  4. └──── GET https://jsonplaceholder.typicode.com/todos/1 200 in 67ms (cache: HIT)
  5. - error node_modules/next/dist/compiled/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js (1863:12) @ resolveModelToJSON
  6. - error Error: Functions cannot be passed directly to Client Components unless you explicitly expose it by marking it with "use server".
  7. - error node_modules/next/dist/compiled/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js (1863:12) @ resolveModelToJSON
  8. - error Error: Functions cannot be passed directly to Client Components unless you explicitly expose it by marking it with "use server".

If I add "use server" to the beginning of page.tsx, I get a compilation error:

  1. Server Actions require `experimental.serverActions` option to be enabled in your Next.js config: https://nextjs.org/docs/app/building-your-application/data-fetching/server-actions

Ideally, I would like to statically generate HTML, but for now, I would be grateful for any working fetching code.

Update

One of the previous solutions I tried was to use Tanstack Query. I have now removed all of that code from my layout.tsx. Now I get a different error for this code:

  1. export default async function Home(): Promise<JSX.Element> {
  2. const todos = await getToDos();
  3. console.log(todos);
  4. return (
  5. <ul>
  6. {todos.map((todo) => (
  7. <li key={todo.id}>{todo.title}</li>
  8. ))}
  9. </ul>
  10. );
  11. }

saying

  1. error TypeError: todos.map is not a function
英文:

I am learning NextJS. Currently, I am using NextJS 13 with the new app directory instead of the old pages directory. I have tried various ways of fetching data, but I have not gotten any of them to work. It should be as simple as

src/app/page.tsx:

  1. export interface ToDo {
  2. userId: number;
  3. id: number;
  4. title: string;
  5. completed: boolean;
  6. }
  7. const getToDos = async (): Promise&lt;ToDo[]&gt; =&gt; {
  8. const res = await fetch(&#39;https://jsonplaceholder.typicode.com/todos/1&#39;);
  9. console.log(`Status: ${res.status}`); // prints &#39;Status: 200&#39;
  10. return res.json();
  11. }
  12. const Home = async () =&gt; {
  13. const toDos = await getToDos();
  14. return (
  15. &lt;ul&gt;
  16. {toDos.map((todo) =&gt; (
  17. &lt;li key={todo.id}&gt;{todo.title}&lt;/li&gt;
  18. ))}
  19. &lt;/ul&gt;
  20. );
  21. }
  22. export default Home;

The JSON returned by the endpoint is

  1. {
  2. &quot;userId&quot;: 1,
  3. &quot;id&quot;: 1,
  4. &quot;title&quot;: &quot;delectus aut autem&quot;,
  5. &quot;completed&quot;: false
  6. }

This is the output. Notice how I am getting both a 200 OK status and an error message:

  1. Warning: Only plain objects can be passed to Client Components from Server Components. Classes or other objects with methods are not supported.
  2. &lt;... client={{queryCache: ..., mutationCache: ..., logger: ..., defaultOptions: ..., queryDefaults: ..., mutationDefaults: ..., mountCount: ...}} children=...&gt;
  3. ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  4. Warning: Only plain objects can be passed to Client Components from Server Components. Classes or other objects with methods are not supported.
  5. {queryCache: {listeners: Set, subscribe: function, config: ..., queries: ..., queriesMap: ...}, mutationCache: ..., logger: ..., defaultOptions: ..., queryDefaults: ..., mutationDefaults: ..., mountCount: ...}
  6. ^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
  7. Status: 200
  8. - GET / 200 in 606ms
  9. └──── GET https://jsonplaceholder.typicode.com/todos/1 200 in 67ms (cache: HIT)
  10. - error node_modules/next/dist/compiled/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js (1863:12) @ resolveModelToJSON
  11. - error Error: Functions cannot be passed directly to Client Components unless you explicitly expose it by marking it with &quot;use server&quot;.
  12. {listeners: Set, subscribe: function, config: ..., queries: ..., queriesMap: ...}
  13. ^^^^^^^^
  14. at stringify (&lt;anonymous&gt;)
  15. null
  16. - error node_modules/next/dist/compiled/react-server-dom-webpack/cjs/react-server-dom-webpack-server.edge.development.js (1863:12) @ resolveModelToJSON
  17. - error Error: Functions cannot be passed directly to Client Components unless you explicitly expose it by marking it with &quot;use server&quot;.
  18. {listeners: Set, subscribe: function, config: ..., queries: ..., queriesMap: ...}
  19. ^^^^^^^^
  20. at stringify (&lt;anonymous&gt;)
  21. digest: &quot;501055159&quot;
  22. null

If I add "use server" to the beginning of page.tsx, I get a compilation error:

  1. Server Actions require `experimental.serverActions` option to be enabled in your Next.js config: https://nextjs.org/docs/app/building-your-application/data-fetching/server-actions

Ideally I would like to statically generate HTML, but for now I would be grateful for any working fetching code.

Update

One of the previous solutions I tried was to use Tanstack Query. I have now removed all of that code from my layout.tsx. Now I get a different error for this code:

  1. export default async function Home(): Promise&lt;JSX.Element&gt; {
  2. const todos = await getToDos();
  3. console.log(todos);
  4. return (
  5. &lt;ul&gt;
  6. {todos.map((todo) =&gt; (
  7. &lt;li key={todo.id}&gt;{todo.title}&lt;/li&gt;
  8. ))}
  9. &lt;/ul&gt;
  10. );
  11. }

saying

  1. error TypeError: todos.map is not a function

答案1

得分: 1

似乎您正在尝试将一个来自服务器组件的函数传递给客户端组件,除非它是一个服务器动作,否则这是不可能的,因为服务器动作目前仍处于 alpha 版本,您需要通过在您的 next.config.js 文件中设置 experimental.serverActions 来启用它们:

  1. module.exports = {
  2. experimental: {
  3. serverActions: true,
  4. },
  5. };

这是一个服务器端组件的示例:

  1. import "server-only";
  2. interface ToDo {
  3. userId: number;
  4. id: number;
  5. title: string;
  6. completed: boolean;
  7. }
  8. async function getToDos(): Promise<ToDo[]> {
  9. const res = await fetch("https://jsonplaceholder.typicode.com/todos");
  10. return await res.json(); // 顺便提一下,您漏掉了一个 await
  11. }
  12. export default async function Home(): Promise<JSX.Element> {
  13. const todos = await getToDos();
  14. return (
  15. <ul>
  16. {todos.map((todo) => (
  17. <li key={todo.id}>{todo.title}</li>
  18. ))}
  19. </ul>
  20. );
  21. }

我强烈建议您安装 server-only 包,这样如果您尝试在客户端导入服务器组件,就会抛出错误。

英文:

It seems you are trying to pass a function from a server component to a client component, this is not possible unless it is a server action, since server actions are still a feature currently in alpha, you need to enable them by setting experimental.serverActions in your next.config.js file:

  1. module.exports = {
  2. experimental: {
  3. serverActions: true,
  4. },
  5. };

Here is an example of a server side component:

  1. import &quot;server-only&quot;;
  2. interface ToDo {
  3. userId: number;
  4. id: number;
  5. title: string;
  6. completed: boolean;
  7. }
  8. async function getToDos(): Promise&lt;ToDo[]&gt; {
  9. const res = await fetch(&quot;https://jsonplaceholder.typicode.com/todos&quot;);
  10. return await res.json(); // by the way, you were missing an await here
  11. }
  12. export default async function Home(): Promise&lt;JSX.Element&gt; {
  13. const todos = await getToDos();
  14. return (
  15. &lt;ul&gt;
  16. {todos.map((todo) =&gt; (
  17. &lt;li key={todo.id}&gt;{todo.title}&lt;/li&gt;
  18. ))}
  19. &lt;/ul&gt;
  20. );
  21. }

I strongly recommend you install the server-only package, so an error is thrown if you try to import a server component into the client side.

答案2

得分: 0

正如 @FabioNettis 正确指出的那样,我尝试获取一个单一对象,这不可迭代。我意外地尝试从路径 /todos/1 而不是 /todos 获取。因此,获取代码实际上是正确的,尽管不是很好。良好的错误处理会更容易调试。

英文:

As @FabioNettis correctly pointed out, I was trying to fetch a single object, which is not iterable. I had accidentally tried to fetch from the path /todos/1 rather than /todos. So the fetching code was actually correct, although not great. Good error handling would have made debugging easier.

huangapple
  • 本文由 发表于 2023年6月15日 20:12:08
  • 转载请务必保留本文链接:https://go.coder-hub.com/76482331.html
匿名

发表评论

匿名网友

:?: :razz: :sad: :evil: :!: :smile: :oops: :grin: :eek: :shock: :???: :cool: :lol: :mad: :twisted: :roll: :wink: :idea: :arrow: :neutral: :cry: :mrgreen:

确定