17
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?

Medley(メドレー)Advent Calendar 2024

Day 7

【GraphQL】そのApollo ClientのキャッシュIDは一意ですか?

Last updated at Posted at 2024-12-06

はじめに

こちらの記事は株式会社メドレーのAdvent Calendar(2024)の7日目の記事です!

今年度から株式会社メドレーにて新卒エンジニアとして働いている@shiromieです。
私が携わっているプロダクトでは
フロントエンド: TypeScript × React × Apollo Client(GraphQL)
バックエンド : TypeScript × NestJS × Apollo Server(GraphQL)
という技術スタックを主に採用しています。

その中で、直近の業務においてApollo Clientの「キャッシュIDの一意性」を考慮しなければならないケースに何度か遭遇しました。

そこで本記事では、Apollo ClientにおけるキャッシュIDの一意性がなぜ重要かについて、具体例を挙げながら詳しく解説したいと思います。

本記事の構成

  • Apollo Clientとは?
  • Apollo Clientのキャッシュとは?
  • キャッシュIDはどのように使われるのか?
  • なぜ一意なキャッシュIDが重要か?
  • 一意なキャッシュIDを担保するには?
    • 方法① キャッシュキーとして扱うフィールドをカスタムで設定する
    • 方法② サーバーサイドで生成した一意なIDをクライアントサイドで用いる
  • 終わりに
  • あとがき

対象読者

  • 業務でApollo Clientを使用している方
  • これからApollo Clientを使用する方
  • Apollo Clientのキャッシュを活用したい方

Apollo Clientとは?

Apollo ClientはGraphQL APIとフロントエンドを接続するためのライブラリで、以下のような特徴を持っています。

  • TypeScriptとの親和性が高く、型安全なデータ取得が可能
  • キャッシュ機能により、データの再利用を効率化しパフォーマンスを向上させる
  • クエリ、ミューテーション、サブスクリプションを通じてリアクティブなデータ管理を実現する

またバックエンドがGraphQLサーバーである場合、Apollo Clientはクエリを送信し結果をキャッシュに保存することで、フロントエンドの負担軽減も可能にします。

Apollo Clientのキャッシュとは?

Apollo ClientではクエリのレスポンスデータをInMemoryCacheに保存し、次回以降の同様のクエリに対してキャッシュデータを優先的に使用(※一部のfetch-policyを除く)します。

その仕組みは公式ドキュメントにあるイメージ図が非常に明快であるため、引用した以下の画像を参照下さい。

  • Bookがcacheに存在しなかった場合
    スクリーンショット 2024-11-24 13.03.32.png

  • Bookがcacheに存在した場合
    スクリーンショット 2024-11-24 13.02.36.png

上記の画像は、以下の公式ドキュメントより引用しました。

キャッシュIDはどのように使われるのか?

Apollo Clientではキャッシュデータを識別する際、デフォルトでは__typenameidを基に生成されたキャッシュキーを使用します。

例えば、以下のようなGraphQL Typeを定義する場合、

"""
ある特定の生徒における特定の講義の受講状況
"""
type StudentLecturesAttendance {
  """
  講義のID
  """
  id: ID!

  """
  生徒のID
  """
  studentId: String!

  """
  講義のタイトル
  """
  title: String!
  
  """
  生徒がこの講義を完了したかどうか
  """
  isCompleted: Boolean!
}

type Student {
  id: ID!
  name: String!
}

type Query {  
  studentLecturesAttendance(input: StudentLecturesAttendanceInput!): [StudentLecturesAttendance!]!
}

input StudentLecturesAttendanceInput {
  """
  生徒のID
  """
  studentId: ID!

  """
  講義のID
  """
  lectureId: ID
}

StudentLecturesAttendanceのキャッシュIDはStudentLecturesAttendance:101といったものになります。

※ この例では不適切なスキーマ定義を意図的に使用しています。本来、StudentLecturesAttendanceは「生徒と講義の組み合わせ」によって一意に識別されるべきで、idに講義のID(Lectureid)を使用して定義するべきではありません。故にこの設計は実際の運用には適していないため、あくまでサンプルとしてご覧ください。

なぜ一意なキャッシュIDが重要か?

キャッシュIDの一意性が担保されない場合、予期せぬキャッシュの上書きが発生します。
例えば、

生徒A:

{
  "studentLecturesAttendance": {
   "id": 101,
   "studentId": "01HKVP5MQ3ZARPW63J8Z6EJ84V",
   "title": "講義A",
   "isCompleted": true
  }
}

生徒B:

{
  "studentLecturesAttendance": {
   "id": 101,
   "studentId": "0000003BW9BSS9VAY4KDND9BP7",
   "title": "講義A",
   "isCompleted": false
  }
}

とする場合、これらはどちらもキャッシュIDはStudentLecturesAttendance:101として保存されます。

Apollo Clientでは新しいクエリ結果をキャッシュに保存する際、既存のキャッシュキーに対するエントリが存在すればそのデータを上書きするため、このようなIDの定義では生徒AとBのキャッシュのデータに競合が生じてしまいます。

一意なキャッシュIDを担保するには?

  • 生徒に対して受けるべき講義を割り当て、「各生徒が講義を受講完了しているか?」を先生が確認する
  • 生徒の一覧を表示し、各生徒を示すコンポーネント内のボタンをクリックすると、その生徒が受講した講義の受講状況を取得する

上記の仕様を満たす画面の実装を考えます。

コード例
type Student = {
  id: string;
  name: string;
};

type Lecture = {
  id: number;
  title: string;
  isCompleted: boolean;
};

function StudentsList({ lectureId }: { lectureId: string }) {
  // Apollo ClientのuseQueryにより、生徒一覧を取得
  const { data } = useQuery(ListLectureAssignedStudentsDocument, {
    fetchPolicy: 'cache-and-network',
    variables: {
      lectureId,
    },
  });

  const students = data?.listLectureAssignedStudents || [];

  return (
    <ul>
      {students.map((student) => (
        <ListElement key={student.id} student={student} lectureId={lectureId}/>
      ))}
    </ul>
  );
}

/**
 * 各生徒を表示するコンポーネント
 *
 * ボタンをクリックすると、その生徒に割り当てられた講義を取得する
 * 講義には、その生徒が講義を受講完了したか示す`isCompleted`が含まれる
 */
function ListElement({ student, lectureId }: { student: Student; lectureId: number }) {
  // Apollo ClientのuseLazyQueryにより、受講状況を取得
  const [fetchStudentLecturesAttendance, { data }] = useLazyQuery(
    StudentLecturesAttendanceDocument
  );

  const handleFetchAttendance = () => {
    fetchStudentLecturesAttendance({
      variables: {
        input: {
          lectureId,
          studentId: student.id,
        },
      },
    });
  };

  const lecture = data?.studentLecturesAttendance;

  return (
    <li>
      <div>生徒名: {student.name}</div>
      <button onClick={handleFetchAttendance}>受講状況を取得する</button>

      {lecture !== undefined && (
        <div> {lecture.title} - {lecture.isCompleted ? '完了' : '未完了'} </div>
      )}
    </li>
  );
}

また受講状況を取得する際に実行するクライアントサイドでのqueryであるGetStudentLecturesAttendanceは以下のように定義します。

  query GetStudentLecturesAttendance($input: StudentLecturesAttendanceInput!) {
    studentLecturesAttendance(input: $input) { 
      id 
      title
      isCompleted
    }
  }

方法① キャッシュキーとして扱うフィールドをカスタムで設定する

StudentLecturesAttendanceidが講義のID(lectureId)のみにより決定される場合、このidは講義そのものを識別するもので属する生徒(studentId)の情報を含んでいないため不適切となります。画面上では、生徒Aの受講状況を取得後に生徒Bの受講状況を取得すると、既に取得し表示されていた生徒Aの情報が生徒Bの情報に書き換わってしまうという問題が発生します。

したがって、StudentLecturesAttendanceidlectureIdから生成するのではなく、studentIdも考慮した一意性のあるものに変更する必要があります。

しかし既存のデータ構造の変更が難しい場合、短期的な解決策としてidを変更するのではなく、Apollo Clientのキャッシュキーとして扱うフィールドをカスタムで設定するという方法があります。

Apollo Clientのキャッシュキーはデフォルトでidのみを参照するようになっていますが、これをStudentLecturesAttendanceのキャッシュキーのみ、id(lectureId)、studentIdの2つがキーとなるように設定します。

export const apolloClient = new ApolloClient({
  cache: new InMemoryCache({
    possibleTypes: generatedIntrospection.possibleTypes,
    typePolicies: {
      StudentLecturesAttendance: {
        keyFields: ['id', 'studentId'], // カスタムキーを定義
      },
    },
  }),
});

この設定により、キャッシュIDが以下のように変更され一意性が担保されます。

StudentLecturesAttendance:{"id":101,"studentId":"01HKVP5MQ3ZARPW63J8Z6EJ84V"}
StudentLecturesAttendance:{"id":101,"studentId":"0000003BW9BSS9VAY4KDND9BP7"}

またカスタムキーが有効に機能するよう、GetStudentLecturesAttendanceをリクエストする場合は必ずstudentIdを呼ぶように変更します。

  query GetStudentLecturesAttendance($input: StudentLecturesAttendanceInput!) {
    studentLecturesAttendance(input: $input) {
      id 
      studentId # 追加
      title
      isCompleted
    }
  }

これにより異なるstudentIdに基づくデータが個別にキャッシュされ、上書き問題が解消されます。

しかしこの解決策は長期視点で考えた場合、あまり良い方法とは呼べません。なぜなら、今後GetStudentLecturesAttendanceをリクエストする際はidstudentIdの2つを常に必ず呼び出す必要があることを開発者に周知する必要があり、意図しない実装ミスを生み出す要因になるためです。

方法② サーバーサイドで生成した一意なIDをクライアントサイドで用いる

より良い解決策として、サーバーサイドで生成した一意なIDをクライアントサイドで用いる方法があります。以下にその実装例を示します。

export async function listLectures(studentId: string): ListLecturesPayload {
  const lectures = await getLectures(studentId)

  return lectures.map((lecture) => ({ id: `student_${stuedentId}_lecture_${lecture.id}`}))
}

上記のようにstudentIdと組み合わせたIDをStudentLecturesAttendanceのキャッシュIDとして使用することで、同じIDを持つ講義でも別々のキャッシュとして保存することが可能です。これは方法①のようにカスタムキーを全て指定して呼び出す必要がない(※idは常に指定するべき)ため、保守性の高い解決策と言えます。

また、重要なのはキャッシュオブジェクトの同一性がIDによって担保されることであり、クライアントサイドからの入力でフィールドの値が変わる場合はその入力に起因したIDを生成する(例: 検索条件をハッシュ化する)といったことも必要になるかもしれません。

終わりに

以上になります。
この記事では、Apollo ClientのキャッシュIDの一意性の重要性と担保する方法について、具体例を元に解説しました。データ競合やキャッシュの上書き問題に直面した際の参考になれば幸いです。

Medley Advent Calendar 2024、明日は @Layzie さんです!
ぜひお楽しみに!

あとがき

株式会社メドレーではエンジニアを絶賛大募集中です。
興味のある方は、ぜひ以下のリンクから詳細をご覧ください。

皆さまのご応募を心よりお待ちしております!

17
2
0

Register as a new user and use Qiita more conveniently

  1. You get articles that match your needs
  2. You can efficiently read back useful information
  3. You can use dark theme
What you can do with signing up
17
2

Delete article

Deleted articles cannot be recovered.

Draft of this article would be also deleted.

Are you sure you want to delete this article?