I/O Adventure Google Cloud アーキテクチャ
data:image/s3,"s3://crabby-images/aad07/aad07fda00349fcdcae0d4ab246e67d0163e007d" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/A9YucjGLu5HUHVf.max-2600x2600.png"
Google Cloud Japan Team
※この投稿は米国時間 2022 年 7 月 7 日に、Google Cloud blog に投稿されたものの抄訳です。
2020 年以来、主催者、講演者、参加者の誰もが、生活や仕事に対するアプローチを新たに考え直す中で、会議の多くはすべて、または一部がオンラインで行われるようになっています。Google I/O Adventure では、現実世界のイベントの良い部分をデジタルの世界に持ち込んだ、バーチャルな会議を体験できます。
I/O Adventure では、イベント参加者はプロダクトデモの視聴、Google 社員や他の参加者とのチャット、バーチャル アイテムの取得、デベロッパー コミュニティとの交流、個人用アバターの作成、イースター エッグの発見ができます。
この投稿では、Google Cloud を使用して I/O Adventure エクスペリエンスをどのように実現しているかについて詳しく説明します。
フロントエンドは、参加者が一種の「オフライン モード」で、I/O Adventure を単独で楽しむのに十分な静的アセットで構成されています。
data:image/s3,"s3://crabby-images/54945/549453a9acd905d62cd1ddd2c0a2e2ccec3eff06" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image9_OWYUq4i.max-2000x2000.png"
data:image/s3,"s3://crabby-images/54945/549453a9acd905d62cd1ddd2c0a2e2ccec3eff06" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image9_OWYUq4i.max-2000x2000.png"
ブラウザのグラフィックスとアニメーションは、一般的なライブラリ(React、PixieJS、GSAP、Spine)を使用してレンダリングされています。
もし、I/O Adventure で提供されるのが、静的なアセットと外部リソースへのリンクだけを含む「オフライン モード」にすぎないのであれば、バックエンドは最小限のウェブサーバーで十分です。
data:image/s3,"s3://crabby-images/cff28/cff28f648025073c98d72610ef2eb675ad5d24bf" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image3_UJzRcZX.max-1300x1300.png"
data:image/s3,"s3://crabby-images/cff28/cff28f648025073c98d72610ef2eb675ad5d24bf" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image3_UJzRcZX.max-1300x1300.png"
言うまでもなく、他の参加者と同じ世界に浸り、テキスト チャットやボイスチャットで交流する方がより楽しめます。
data:image/s3,"s3://crabby-images/0508d/0508d08e7ce5c9f32b5d1fd2d9b55ccfcdefd975" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image13_kzjfKuT.max-2000x2000.png"
data:image/s3,"s3://crabby-images/0508d/0508d08e7ce5c9f32b5d1fd2d9b55ccfcdefd975" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image13_kzjfKuT.max-2000x2000.png"
このマルチプレーヤー型のオンライン エクスペリエンスを実現するには、ゲームサーバーを Google Kubernetes Engine(GKE)のステートフル Pod としてデプロイした、より高機能なバックエンドが必要でした。
data:image/s3,"s3://crabby-images/0164d/0164d4fe3a819cd3d75d6e624d22a41077619fe2" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image2_r5xdMWJ.max-1400x1400.png"
data:image/s3,"s3://crabby-images/0164d/0164d4fe3a819cd3d75d6e624d22a41077619fe2" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image2_r5xdMWJ.max-1400x1400.png"
この会議の世界地図は大きく、下の図のように 12 種類のゾーンがあります。
data:image/s3,"s3://crabby-images/57a7e/57a7e21c496408c9a7609b0c61290b3968664253" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image1_4gA6Akt.max-2000x2000.png"
data:image/s3,"s3://crabby-images/57a7e/57a7e21c496408c9a7609b0c61290b3968664253" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image1_4gA6Akt.max-2000x2000.png"
この地図の各ゾーンでは、それぞれ独立した Pod が使用されています。
data:image/s3,"s3://crabby-images/46771/46771292ad543ffeb50455b8a8395421afd3f342" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image4_j4AGmIA.max-2000x2000.png"
data:image/s3,"s3://crabby-images/46771/46771292ad543ffeb50455b8a8395421afd3f342" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image4_j4AGmIA.max-2000x2000.png"
つまり、ゾーン(仮想世界での所在地)に応じて、任意の参加者が単一のゲームサーバーに接続されます。
トラフィック量が多いときは、各ゾーンに複数あるシャードの一つに参加者がディスパッチされます。
data:image/s3,"s3://crabby-images/7f836/7f83647d1a12991ea35a4269a4385eceea79f52c" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image10_2BOAedO.max-2000x2000.png"
data:image/s3,"s3://crabby-images/7f836/7f83647d1a12991ea35a4269a4385eceea79f52c" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image10_2BOAedO.max-2000x2000.png"
I/O 2022 では、イベント開始前に多くのシャード サーバーを立ち上げ、必要なときにすぐに使用できるように、オーバープロビジョニングすることにしました。スケーラビリティに関する戦略は、容量のしきい値に達するまで参加者をシャードに接続し、その後、次の空のシャードを使い始めるというシンプルなものでした。
シャード サーバーはステートフルです。それぞれが独立して小さな世界を動かしていて、他のシャードとの通信は最小限に抑えられています。シャードの状態(出席者の現在位置など)は、シャード サーバーの実行ファイルによってメモリ内に維持され、Go で記述されます。
シャード サーバーは、一部の情報(例: 特定のシャードに接続している参加者の人数)を中央サーバーと共有し、中央サーバーは新しい参加者をシャードにルーティングする役割を担っています。この情報は、グローバルな Memorystore for Redis インスタンスに保持されます。
data:image/s3,"s3://crabby-images/7cae1/7cae1fb8bf18559733eba985f1dcbf4fdc10be4a" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image6_r7s7our.max-2000x2000.png"
data:image/s3,"s3://crabby-images/7cae1/7cae1fb8bf18559733eba985f1dcbf4fdc10be4a" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image6_r7s7our.max-2000x2000.png"
参加者のクライアント ブラウザがシャードに割り当てられると、シャード サーバーと WebSocket 接続を確立して、エクスペリエンス全体を通じて双方向で通信を行い、参加者のアクションを送信し、環境状態の更新情報を受信します。
data:image/s3,"s3://crabby-images/d788d/d788d6589a086369a48aef44b3200144c983eb64" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image8_DCSQsnb.max-1300x1300.png"
data:image/s3,"s3://crabby-images/d788d/d788d6589a086369a48aef44b3200144c983eb64" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image8_DCSQsnb.max-1300x1300.png"
各 GKE ノードには、音声サーバーとの通信に使用されるローカル Redis インスタンスがあります。
data:image/s3,"s3://crabby-images/42e62/42e62e389b21255e93a4ea3414a43f9a0c914e17" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image7_MzPHOlo.max-1800x1800.png"
data:image/s3,"s3://crabby-images/42e62/42e62e389b21255e93a4ea3414a43f9a0c914e17" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image7_MzPHOlo.max-1800x1800.png"
data:image/s3,"s3://crabby-images/1c844/1c844910b98b0f316ffffaed6a2f01d64fbaf9d2" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image14_3sJJxL3.max-1700x1700.png"
data:image/s3,"s3://crabby-images/1c844/1c844910b98b0f316ffffaed6a2f01d64fbaf9d2" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image14_3sJJxL3.max-1700x1700.png"
アーキテクチャを簡素化するため、すべてのサーバーは同じ Google Cloud リージョン(us-central1)に配置されています。この設計を選択することで、すべてのサーバー コンポーネント間で低レイテンシの通信が実現します。また、ヨーロッパ、アフリカ、アジア、オセアニア、南米の参加者は、遠く離れた海外のサーバーに接続することになりますが、他の参加者の動きやテキスト チャットのメッセージのやり取りなどで発生するレイテンシは最大で数百ミリ秒と許容範囲内なので、問題ありません。
data:image/s3,"s3://crabby-images/69f04/69f0463176d556eeb16cac742a7111eafb64c011" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image11_TmUMnR8.max-1900x1900.png"
data:image/s3,"s3://crabby-images/69f04/69f0463176d556eeb16cac742a7111eafb64c011" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image11_TmUMnR8.max-1900x1900.png"
I/O Adventure にアクセスするには、参加者は Google アカウントでログインする必要があります。これには、Firebase Authentication サービスを使用しています。すべてのアバターは、帽子、肌の色、ハンド アクセサリーなどをカスタマイズできます。カスタマイズされたアバターの特徴や、ゲームの進行状況、完了したクエストなどを含む、参加者のプロフィールは、Firestore ドキュメント データベースに保存されます。また参加者は任意で、Google デベロッパー プロフィールを自分のアバターにリンクさせてバッジ関連の情報を提供し、他の参加者に示すことも可能です。
data:image/s3,"s3://crabby-images/a786a/a786afa239b8a1527ce75013abfc34b89957c9a7" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image12_gSTzDjp.max-1300x1300.png"
data:image/s3,"s3://crabby-images/a786a/a786afa239b8a1527ce75013abfc34b89957c9a7" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image12_gSTzDjp.max-1300x1300.png"
I/O Adventure のコアサーバーやコンポーネントに加え、この会議では以下のような動的要素も活用されています。
技術セッション(YouTube)
Flutter コーディング チャレンジ(DartPad)
ゲーム内エクスペリエンス(マテリアル デザイン、Google Cloud、Google Pay など向け)
他のコンテンツ(インタラクティブな体験、Codelab、ドキュメント)を新しいブラウザタブで開くリンク
こうしたインテグレーションの多くは、I/O Adventure のフロントエンド(つまりブラウザ)で直接処理され、コアサーバーのアーキテクチャから切り離されています。
data:image/s3,"s3://crabby-images/8fef4/8fef463e68298e9fe8f227ebbc2f8db50e9e1f61" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image5_7cCz65c.max-2000x2000.png"
data:image/s3,"s3://crabby-images/8fef4/8fef463e68298e9fe8f227ebbc2f8db50e9e1f61" alt="https://storage.googleapis.com/gweb-cloudblog-publish/images/image5_7cCz65c.max-2000x2000.png"
まとめ
I/O Adventure のウェブ エクスペリエンスの構築は、Google 社員である Tom Greenaway と Google I/O チームが主導し、Set Snail スタジオの才能あるデザイナーとデベロッパーによって実現した並々ならぬ努力の賜です。
結果は上々でした。サーバーはすべて Google Cloud 上で完全にホストされていて、負荷に適切に対応し、ソーシャル メディアの反応は非常に好意的なものでした。人々は、バーチャルな形であっても、アイテムが大好きだということがわかりました。
Google Cloud デベロッパー アドボケイト Valentin Deleplace