Upgrade to Pro
— share decks privately, control downloads, hide ads and more …
Speaker Deck
Features
Speaker Deck
PRO
Sign in
Sign up for free
Search
Search
PHP で負荷試験のシナリオを書きたい!ので amphp を使って自作した件
Search
Sponsored
·
Your Podcast. Everywhere. Effortlessly.
Share. Educate. Inspire. Entertain. You do you. We'll handle the rest.
→
Masaru Yamagishi
September 16, 2023
Programming
2
1.1k
PHP で負荷試験のシナリオを書きたい!ので amphp を使って自作した件
PHP カンファレンス沖縄 2023 登壇資料です。
Masaru Yamagishi
September 16, 2023
Tweet
Share
More Decks by Masaru Yamagishi
See All by Masaru Yamagishi
Babylon.js 勉強会 vol.4 JAPAN 活動紹介
myamagishi
0
120
PSR-15 はあなたのための ものではない? - phpcon2024
myamagishi
0
980
3D ブラウザゲーム開発を始めたい人のために準備している話
myamagishi
1
200
「DI」と仲良くなる
myamagishi
6
3.5k
大解剖!amphpを使って非同期 PHP を実現しよう!
myamagishi
1
4k
秒間 10,000 リクエストを "簡単に"いなすゲームサーバーを Laravel で作る設計
myamagishi
19
15k
xR グラスが普及した新時代を妄想する - XRKaigi 2022
myamagishi
0
290
Reflection を使いこなして、 オブジェクトを型安全に マッピングしよう!
myamagishi
2
930
世界最速? で PHP8 Native Framework 作った
myamagishi
1
850
Other Decks in Programming
See All in Programming
Amazon Bedrockを活用したRAGの品質管理パイプライン構築
tosuri13
5
830
要求定義・仕様記述・設計・検証の手引き - 理論から学ぶ明確で統一された成果物定義
orgachem
PRO
1
280
Sekiban + Microsoft Orleans のアクターをAWS対応しました / Sekiban + Microsoft Orleans actors are now supported on AWS.
tomohisa
0
110
CSC307 Lecture 05
javiergs
PRO
0
500
AWS re:Invent 2025参加 直前 Seattle-Tacoma Airport(SEA)におけるハードウェア紛失インシデントLT
tetutetu214
2
120
20260127_試行錯誤の結晶を1冊に。著者が解説 先輩データサイエンティストからの指南書 / author's_commentary_ds_instructions_guide
nash_efp
1
1k
AI時代の認知負荷との向き合い方
optfit
0
180
そのAIレビュー、レビューしてますか? / Are you reviewing those AI reviews?
rkaga
6
4.6k
Fluid Templating in TYPO3 14
s2b
0
140
コマンドとリード間の連携に対する脅威分析フレームワーク
pandayumi
1
470
Rust 製のコードエディタ “Zed” を使ってみた
nearme_tech
PRO
0
240
ぼくの開発環境2026
yuzneri
0
270
Featured
See All Featured
Bootstrapping a Software Product
garrettdimon
PRO
307
120k
Art, The Web, and Tiny UX
lynnandtonic
304
21k
How to Ace a Technical Interview
jacobian
281
24k
[SF Ruby Conf 2025] Rails X
palkan
1
770
Agile Actions for Facilitating Distributed Teams - ADO2019
mkilby
0
120
We Analyzed 250 Million AI Search Results: Here's What I Found
joshbly
1
770
Bioeconomy Workshop: Dr. Julius Ecuru, Opportunities for a Bioeconomy in West Africa
akademiya2063
PRO
1
60
Ethics towards AI in product and experience design
skipperchong
2
200
Visual Storytelling: How to be a Superhuman Communicator
reverentgeek
2
440
The Mindset for Success: Future Career Progression
greggifford
PRO
0
240
The browser strikes back
jonoalderson
0
440
brightonSEO & MeasureFest 2025 - Christian Goodrich - Winning strategies for Black Friday CRO & PPC
cargoodrich
3
110
Transcript
PHP で負荷試験のシナリオを書きたい!の で amphp を使って自作した件 2023/09/16 PHP カンファレンス沖縄 やまゆ
セッション概要 jMeter, Locust, k6, ... 負荷試験ツールは色々ありますが、どれも PHP でシナリオを書くことができません。 PHP のコードは
PHP でシナリオ も書いてしまった方が一貫性があって良いですよね? PHP 8.1 から Fiber が入ったことで、 PHP でも非同期・並行的な処理が簡単に実装 出来るようになりました。 PHP によるシナリオツール、 ないなら作ろう精神 で、自作することにしました。 Fiber ままで並行処理を作るのは大変なので、今回は amphp を使って簡単に distributed なシステムを構築します。 今回は何故自作したのか、どのように並行処理を実装したのかを紹介します。 ターゲット 負荷試験したい方 PHP で並行処理ってどうやるの?という方
赤魔道士系エンジニア ㈱インフィニットループ at 札幌/仙台 やまゆ この画像は自撮りでも いつも使っているアイコンでも構いません ☕
負荷試験
負荷試験 大規模 Web サービスのリリース・大規模アプデに欠かせない事前テスト 想定された Monthly Active User(MAU) から仮定する Request
per sec(rps) が安定 して稼働するか、負荷をかけ続けて試験すること ↓やらないとどうなる? - 事前スケーリングのサイズが決まらない - リリース時に🔥炎上🔥する
負荷試験内容 1. MAU 定義 2. シナリオ定義 3. rps 定義 4.
負荷シナリオプログラム作成 5. 実施・監視・適宜最適化 6. レポート
負荷試験定義例 - 事前のアンケートや類似事例から MAU を検討 - 想定されたシナリオをいくつか作成 - トップページ ->
リンク回遊 -> ユーザー登録 -> プロフィール編集 -> リンク回遊 - トップページ -> ユーザー登録 -> 課金 -> リンク回遊 - ソシャゲ: ユーザー登録 -> ログイン -> チュートリアル -> コアサイクル - シナリオから「何秒おきにどんな API 順で呼ばれるか」表を作成 - 同時に、シナリオ表をプログラム化(機械的に呼べるようにする) - MAUとシナリオ表から「1秒間に何リクエスト送られるか」を推測
負荷試験定義例: rps - 仮に 20 万 MAU とする - 仮定として、全体の
50 %(10万) が最大の同時接続数とする - シナリオ表から、平均的な rps が 0.2 であるとする - 100,000 * 0.2 = 20,000 rps - このサービスに安定・継続して耐えるべき値は 20,000 rps と仮定
Next action? 20,000 rps を耐えられる構成を考える そもそも 20,000 rps がどのリソースにどれくらいの負荷を掛けるのか、計測する必要 がある
ということで機械的に計測するための負荷試験ツールの出番!
負荷をかける (HTTP リクエストする) ツール
Apache jMeter https://jmeter.apache.org/
Locust https://locust.io/
Locust Scenario(Python)
k6 https://k6.io/
k6 Scenario(JavaScript like)
PHP なくない?
じゃあ作るか
heavyrain
heavyrain - PHP Loadtest/Stresstest tool
heavyrain - PHP Loadtest/Stresstest tool
Why? - Locust 使っていたが python かぁ...となる - k6 もなんか変な JavaScript
だしなぁ...となる - jMeter…となる - PHP でシナリオ書けるのが一番楽やろ - 今は PHP でも Fiber 等で並行処理強くなってきたし - 自分が一番欲しい(普通にプロジェクトの負荷試験に使いたい) - ないなら作ろう精神
同期的だと無理
同期処理シーケンス
非同期処理シーケンス
非同期処理における並行 & 並列 非同期処理には種類がある - 並行(Concurrency) - 並列(Parallelism) 言語の仕組みによって並行性が高かったり並列性が高かったりする 言語だけでなくドキュメントでもこの表現は揺れているので、今回の資料では
- 並行=シングルスレッドでうまいこと非同期する - 並列=マルチスレッド・マルチプロセスで非同期する とする。
PHP は(基本的に)シングルスレッド なので「並列」ではなく「並行」が近い マルチプロセスは可能だが、プロセス間通信はちょっと大変 マルチスレッドは extension を使えば可能(zts じゃないとダメ)
JavaScript も(基本的に)シングルスレッド
Fiber
Fiber https://www.php.net/manual/ja/language.fibers.php サードパーティ extension なしで「非同期・並行処理」を実現できる API PHP 8.1 から利用可能 シンプルで低レベルの
API を提供している
None
シンプルすぎて わからん 🤔
Revolt
Revolt イベントループを管理してくれる低レベル API ライブラリ - Defer: go の defer みたいに、イテレーションの最後に
- Delay: x 秒後に - Repeat: x 秒おきに - Stream readable: ストリームが読み込めるようになったら - Stream writable: ストリームに書きこめるようになったら - Signal: プロセスシグナルを OS から受け取ったら
None
シンプルすぎて わからん 🤔
amphp
amphp Revolt をベースとした高レベル非同期 API を提供するライブラリ群 File, Log, MySQL, Redis などの
I/O 処理を非同期化し ブロックしない形で実行出来るライブラリがある HTTP Server を動かすこともできる
None
async await
async? 「将来どこかで終わるけど、今はまだ終わってないと思う」という表明的なもの (既に終わっている時もある) 色んな言語で実装されているが、 PHP はライブラリレベルで実装 `function async(\Closure $callback): Future`
Future だったり Promise だったりする(概念が違うが似た系統)
await? async なクロージャが完了するまで待つ `$results = await($futures)` 複数のクロージャが全て完了するまで待つことも出来る JavaScript の変遷をたどると分かりやすい SoftwareDesign
2023年9月号に詳しく載ってます! (他人の記事を勝手に宣伝) https://gihyo.jp/magazine/SD/archive/2023/202309
heavyrain
heavyrain での非同期処理 - ほとんどの処理は HTTP レスポンスが来るまでの待機時間 - 同期処理すると、 CPU が暇
- 暇している間に別のリクエストを実行したい - そこで「非同期」処理 - シングルスレッドで「同時に x リクエスト」出来るようにする!
目指せ「1,000 並行」 extension なしだと「1,024 ファイルディスクリプタ」が限界値 ディスクリプタ数が増えると線形に遅くなるので実際そこまではいかなさそう extension を入れるともっと増やせる(最大はエフェメラルポート数くらい?) 現状だと amphp/ext-uv
が比較的アクティブにメンテされている node.js でも使われていた libuv を使った PHP 拡張
まずは「 HTTP シナリオ」を書けること リクエストとレスポンスの連なり(シナリオ)を表現できる必要がある POST /register -> POST /login ->
GET /me -> POST /post -> … - HTTP リクエストを送る - 前のレスポンスのデータを使う - 途中暗号化や認証を行えるようにする(ミドルウェア) - 1人のユーザーの一連のシナリオをスクリプトに変換する
None
ClientInterface - 簡単に HTTP リクエストを実行出来る API を定義 - PSR-7 互換
- Locust や k6 などを参考にした
AssertableResponseInterface - assertion も出来るし json body も取得出来る便利な Response API -
これも PSR-7 互換
MiddlewareInterface - リクエスト間で共通の処理を行える API - これも PSR-7 互換 - 暗号化(ボディ改変)・認証(ヘッダ改変)などを行える
ExecutorInterface - 実際に「シナリオを無限に実行し続ける」 API - OS Signal などで途中でキャンセル出来る - ここで「非同期」を発動!
AmphpExecutor 非同期処理のコア部分の実装 ループでたくさん並行に 実行していく TODO は色々あるが ベースは大体これ
Heavyrain の未来 - 「シナリオテスター」として 1 並列でシナリオが一通り動く - Ramp-up(ちょっとずつ同接が増えていく) の仕組み -
数万 rps, 数十万 rps の負荷をかけれるようになる - 複数マシンを使った 分散負荷かけ環境 - OpenAPI 3.0 から ClientInterface 実装を自動生成 - HARReporter によるシナリオ可視化 - Locust のような Web GUI - Amphp の詳細は PHPCon 2023(東京) で話します!!
次は東京で で会いましょう!
北海道でも 会いましょう!
https://www.infiniteloop.co.jp