アプリ
Next.js で Squid を使用する場合、アプリケーションのサーバー側とクライアント側の両方で Squid client にアクセスできます。サーバーでは、初期ペイロードの一部としてデータをクエリできます。クライアントでは、query、mutation を実行したり、リアルタイムのデータ更新をストリーミングで受け取ったりできます。
App Router では、use client と use server ディレクティブを使って、クライアントでレンダリングされるコンポーネントとサーバーでレンダリングされるコンポーネントを区別できます。React Server Components (RSCs) は、レンダリング前に(データのクエリのような)非同期処理を実行できる点が特徴です。このチュートリアルでは、まずクライアントで Squid を使い始め、その後 React Server Components を使う流れに進みます。
まず app/layout.tsx で、children を SquidContextProvider でラップします。プレースホルダーを Squid の設定オプションに置き換えてください。これらの値は Squid Console または .env ファイルで確認できます。.env ファイルは Squid backend の作成 時に自動生成され、backend ディレクトリ内にあります。
import './globals.css';
import type { Metadata } from 'next';
import { Inter } from 'next/font/google';
import { SquidContextProvider } from '@squidcloud/react';
const inter = Inter({ subsets: ['latin'] });
export const metadata: Metadata = {
title: 'Create Next App',
description: 'Generated by create next app',
};
export default function RootLayout({ children }: { children: React.ReactNode }) {
return (
<html lang="en">
<body className={inter.className}>
<SquidContextProvider
options={{
appId: 'YOUR_APP_ID',
region: 'YOUR_REGION',
environmentId: 'dev',
squidDeveloperId: 'YOUR_SQUID_DEVELOPER_ID',
}}
>
{children}
</SquidContextProvider>
</body>
</html>
);
}
users をクエリする
次に、ユーザー一覧を表示する新しいコンポーネントを作成します。app ディレクトリ配下に新しい components ディレクトリと、users.tsx という新しいファイルを作成してください。新しいファイルに以下のコードを追加します。
'use client';
import { useCollection, useQuery } from '@squidcloud/react';
type User = {
id: string;
};
export default function Users() {
const collection = useCollection<User>('users');
const { loading, data, error } = useQuery(collection.query().dereference());
if (loading) {
return <div className="flex flex-col items-center justify-center min-h-screen">Loading...</div>;
}
if (error) {
return <div className="flex flex-col items-center justify-center min-h-screen">{error.message}</div>;
}
return (
<div className="flex flex-col items-center justify-center min-h-screen">
<span>Users</span>
<ul>
{data.map((user) => (
<li key={user.id}>{user.id}</li>
))}
</ul>
</div>
);
}
Users コンポーネントをレンダリングするために、app/page.tsx のコードをすべて次の内容に置き換えます。
import Users from '@/components/users';
export default function Home() {
return <Users />;
}
Squid は標準で組み込み database を提供します。この例では、Users クライアントコンポーネント内で useQuery hook を使い、database の users collection に対するクエリを実行します。hook はクエリと boolean を受け取り、その boolean はテーブルのライブ更新に subscribe するかどうかを示します。query().dereference() を使うと、クエリの raw data が返ります。
Web アプリでは “Users” の見出しが表示されますが、ユーザーがいません! まだ collection にユーザーを insert していないためです。
“Loading…” や error message が表示される場合は、Squid backend を起動したか確認してください。tutorial-backend に移動して squid start を実行します。Running the project を参照してください。
ユーザーを insert する
database にユーザーを insert する関数をトリガーする button コンポーネントを追加します。components/users.tsx に以下を追加してください。
import { useCollection, useQuery } from "@squidcloud/react";
...
export default function Users() {
...
const insertUser = async () => {
await collection.doc().insert({
id: crypto.randomUUID(),
});
}
...
return (
<div className="flex flex-col items-center justify-center min-h-screen">
<button onClick={insertUser}>Insert</button>
<span>Users</span>
<ul>
{data.map((user) => (
<li key={user.id}>{user.id}</li>
))}
</ul>
</div>
);
Web アプリに Insert ボタンが表示されます。ボタンをクリックするとランダムな ID を持つユーザーが insert されます。database クエリはライブ更新に subscribe されているため、ボタンをクリックした直後にユーザー ID がユーザー一覧に表示されます。collection.doc().insert(...) を実行すると、ユーザーデータがアプリケーションの組み込み database に永続化されるため、ページをリフレッシュしてもユーザー一覧は保持されます。
サーバーでクエリを実行する
ページをリフレッシュすると、ユーザー一覧が表示される前に Loading…* のインジケーターが短時間表示されます。これは Users コンポーネントがクライアントコンポーネントであり、クライアント側で users をクエリするのに少し時間がかかるためです。Next.js App Router では、このクエリを React Server Component 内で実行し、ページロード時にそのデータをサーバーからクライアントへ渡せます。
app/page.tsx で Squid query を直接実行して初期 user data を取得し、その users のリストを Users コンポーネントの初期レンダリングに渡します。
import Users from '@/components/users';
import { Squid } from '@squidcloud/client';
type User = {
id: string;
};
export default async function Home() {
const squid = Squid.getInstance({
appId: 'YOUR_APP_ID',
region: 'YOUR_REGION',
environmentId: 'dev',
squidDeveloperId: 'YOUR_SQUID_DEVELOPER_ID',
});
const users = await squid.collection<User>('users').query().dereference().snapshot();
return <Users users={users} />;
}
アプリの初期設定では、クライアント側で SquidContextProvider により Squid を初期化しました。この Squid インスタンスは Home ページ内からはアクセスできません。React Server Components は React Context にアクセスできないためです。代わりに、Squid React SDK をインストールすると自動的にインストールされる @squidcloud/client package を使って、別のインスタンスを作成する必要があります。コードの重複を減らすために、Squid options を取得する共有 utility を作ることを推奨します。
utils というフォルダを作成し、squid.ts というファイルを追加します。新しいファイルに以下のコードを追加してください。
import { SquidOptions } from '@squidcloud/client';
export function getOptions(): SquidOptions {
return {
appId: 'YOUR_APP_ID',
region: 'YOUR_REGION',
environmentId: 'dev',
squidDeveloperId: 'YOUR_SQUID_DEVELOPER_ID',
};
}
これで options={...} はどこでも options={getOptions()} に置き換えられます。
app/layout.tsx で getOptions 関数を import し、SquidContextProvider の options として渡します。
import { getOptions } from "@/utils/squid";
...
export default function RootLayout({
children,
}: {
children: React.ReactNode;
}) {
return (
<html lang="en">
<body className={inter.className}>
<SquidContextProvider options={getOptions()}>
{children}
</SquidContextProvider>
</body>
</html>
);
}
サーバーでクエリした users にアクセスできるようになったので、Users コンポーネントの useQuery hook を更新して初期値を受け取れるようにします。
...
export default function Users({ users }: { user: Array<User> }) {
...
const { loading, data, error } = useQuery(
collection.query().dereference(),
{ initialData: users }
);
...
if (loading && !data.length) {
return (
<div className="flex flex-col items-center justify-center min-h-screen">
Loading...
</div>
);
}
...
}
このコードブロックでは 2 つの変更を行っています。
useQueryhook に初期データとしてusersのリストを渡します。これにより hook から返される初期dataが空配列ではなくユーザー一覧になります。- Loading… の条件を「データが存在するか」を確認するように更新します。デフォルトでは、
useQueryに初期値を渡しても、クライアントでデータのクエリが成功するまではloadingはtrueのままです。dataの存在をチェックすることで、クライアント側のクエリがまだ loading 中でも、サーバーからのユーザー一覧をレンダリングできます。
これらの変更により、ページをリフレッシュしても Loading… インジケーターは表示されなくなります。代わりに、ページロード直後からユーザー一覧が表示されます。
withServerQuery を使って重複を最小化する
ここまでで、React Server Component でデータをクエリし、それをクライアントコンポーネントに渡して初期クエリ値として使う方法を学びました。しかし、クエリのロジックが 2 箇所(Home の React Server Component と Users のクライアントコンポーネント)に存在していることに気づくでしょう。これは、片方のクエリを変更してもう片方を更新し忘れるなど、保守が難しくなる場合があります。
重複を避けるために、Squid React SDK には withServerQuery 関数が用意されています。この hook は、サーバー側でデータをクエリし、それをクライアントコンポーネントへ渡す処理を担います。
Home ページを更新して新しい withServerQuery 関数を使用します。この関数は 3 つの引数を取ります。
- クエリデータを受け取るクライアントコンポーネント
- 実行するクエリ
- クエリ更新に subscribe するかどうか
この関数は、Users コンポーネントをレンダリングするために使える Higher Order Component を生成します。app/page.tsx を更新して新しい関数を含めてください。
import Users from '@/components/users';
import { Squid } from '@squidcloud/client';
import { getOptions } from '@/utils/squid';
import { withServerQuery } from '@squidcloud/react';
type User = {
id: string;
};
export default async function Home() {
const squid = Squid.getInstance(getOptions());
const UsersWithQuery = withServerQuery(
Users,
squid.collection<User>('users').query().dereference(),
true
);
return <UsersWithQuery />;
}
この新しい関数に合わせて、Users コンポーネント側にもいくつか変更が必要です。
- コンポーネントから
useQueryhook を削除します。データはwithServerQueryのラッパー関数から供給されるようになります。 usersprop をdataにリネームします。withServerQueryはラップしたクライアントコンポーネントにdataprop を渡します。- prop の型を
WithQueryProps<User>に更新します。これは実質的に{ data: Array<User> }を表すラッパーです。 if (loading) {...}とif (error) {...}の条件分岐を削除します。コンポーネントがレンダリングされる前にデータがロードされるため、不要になります。
その結果、新しい components/users.tsx は次のようになります。
'use client';
import { useCollection, WithQueryProps } from '@squidcloud/react';
type User = {
id: string;
};
export default function Users({ data }: WithQueryProps<User>) {
const collection = useCollection<User>('users');
const insertUser = async () => {
await collection.doc().insert({
id: crypto.randomUUID(),
});
};
return (
<div className="flex flex-col items-center justify-center min-h-screen">
<button onClick={insertUser}>Insert</button>
<span>Users</span>
<ul>
{data.map((user) => (
<li key={user.id}>{user.id}</li>
))}
</ul>
</div>
);
}
Next.js アプリをリロードしてください。Loading… インジケーターもデータのちらつきもなく、すべての users が表示されます。さらに Insert をクリックすると、ユーザー一覧が引き続き動的に更新されます。
クエリの subscribe の動作を試したい場合は、withServerQuery 関数内の true を false に変更してみてください。この場合でもロード時にはユーザーデータが表示されますが、ユーザーを insert してもユーザー一覧はライブ更新されません(ページをリロードするとユーザーが表示されます)。これは、クライアントコンポーネント内で変更に subscribe しなくなるためであり、実際 Squid query はクライアントではまったく実行されなくなります。
変更に subscribe しない場合、実質的にサーバー側だけで Squid を使っていることになります。これは完全に有効なユースケースで、特にリアルタイムのデータ更新が不要な場合に適しています。
サーバーでデータを insert する
サーバーでのクエリに加えて、router handlers や server actions 内でも Squid を使用できます。サーバーからユーザーを insert してみましょう。
Route handlers
app/api/insert/route.ts 配下に新しい router handler を作成し、この route を次のコードに置き換えます。
import { getOptions } from '@/utils/squid';
import { Squid } from '@squidcloud/client';
import { NextResponse } from 'next/server';
type User = {
id: string;
};
export async function POST() {
const squid = Squid.getInstance(getOptions());
const user = {
id: crypto.randomUUID(),
};
await squid.collection<User>('users').doc().insert(user);
return NextResponse.json(user);
}
このコードが Users コンポーネント内の insertUser 関数とよく似ていることに気づくでしょう。どちらも本質的には同じ目的(組み込み database にユーザーを作成する)を果たしますが、今度はクライアントではなくサーバーで実行されます。
クライアントからこの関数を呼び出すために、insertUser 関数を次のように更新します。
export default function Home(...) {
...
const insertUser = async () => {
await fetch("api/insert", { method: "POST" });
};
...
}
“Insert” ボタンをクリックすると、サーバーからユーザーが insert されるようになります。
Optimistic updates
ボタンをクリックしてから、insert されたユーザーがユーザー一覧に表示されるまでに少し遅延があることに気づくでしょう。これは、Squid がクライアントで optimistic updates を自動的に扱う仕組みによるものです。
insertUser のクライアント側実装では、insert がクライアントで直接行われます。この場合 Squid は optimistic に insert を行うため、insert リクエストがまだ処理中であっても新しいユーザーが即座に表示されます。そして、何らかの理由で insert が失敗した場合、Squid は optimistic insert を rollback します。
サーバーから insert する場合、optimistic updates の恩恵は失われます。一般に、Squid を API routes 内で使って insert することもできますが、ユーザー体験としてはクライアントから直接 insert と update を行う方が良いケースが多いです。
Server actions
Route Handler に加えて、App Router では実験的な Server Actions をサポートしています。Server Actions を使うと、サーバー上で動的に実行できる関数をユーザーが書けます。
Server Actions を有効にするため、next.config.js を更新します。
module.exports = {
experimental: {
serverActions: true,
},
};
actions という新しいフォルダを作成し、insert.tsx というファイルを追加して、次のコードを書きます。use server ディレクティブは、このファイルが Server Action を表すことを示します。
'use server';
import { Squid } from '@squidcloud/client';
import { getOptions } from '@/utils/squid';
type User = {
id: string;
};
export default async function insertUser() {
const squid = Squid.getInstance(getOptions());
await squid.collection<User>('users').doc().insert({
id: crypto.randomUUID(),
});
}
クライアントでは、Server Actions は import して form を submit することで呼び出せます。先ほど作成した insertUser action を呼び出すには、Users コンポーネントを次のように更新します。
"use client";
import insertAction from '@/actions/insert';
...
export default function Users({ data }: WithQueryProps<User>) {
...
return (
<div className="flex flex-col items-center justify-center min-h-screen">
<form action={insertAction}>
<button type="submit">Insert</button>
</form>
<span>Users</span>
<ul>
{data.map((user) => (
<li key={user.id}>{user.id}</li>
))}
</ul>
</div>
);
}
Server Actions はサーバー側で実行されるため、optimistic updates はサポートされません。