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
安全にプロセスを停止するためにシグナル制御を学ぼう!
Search
Sponsored
·
Ship Features Fearlessly
Turn features on and off without deploys. Used by thousands of Ruby developers.
→
yamamoto-hiroya
March 23, 2023
Technology
2.3k
0
Share
Embed
Copy iframe code
Copy JS code
Copy link
Start on current slide
安全にプロセスを停止するためにシグナル制御を学ぼう!
yamamoto-hiroya
March 23, 2023
More Decks by yamamoto-hiroya
See All by yamamoto-hiroya
AUTO_INCREMENTのIDカラムがオーバーフローしたらどうなるの?実例から学ぶDB設計の注意点
yamamotohiroya
1
940
パフォーマンスを改善するには仕様変更が1番はやい
yamamotohiroya
17
8.1k
プルリクサイズが大きいと警告してくれる君を作りました!
yamamotohiroya
1
550
カンファレンスはフィードバックが大事
yamamotohiroya
1
150
ISUCONのススメ
yamamotohiroya
0
980
Other Decks in Technology
See All in Technology
社内 AI エージェント Synapse と セマンティックレイヤーの育て方
hiroakis
1
1.5k
製造業のクラウド活用最適解〜AI,DXを加速するデータ基盤の作り方〜
hamadakoji
0
430
脆弱性対応、どこで線を引くか
rymiyamoto
0
280
ポケモンの型をTypeScriptの型システムで表現してみた
subroh0508
0
360
手塩にかけりゃいいってもんじゃない
ming_ayami
0
130
2026 TECHFRESH 畢業分享會 - 開發日常大解密!從領域驅動到企業級上線
line_developers_tw
PRO
0
570
AI駆動開発が変える、大規模開発の前提 ーHuman in the Loop から Human on the Loop へ / AIE2026
visional_engineering_and_design
30
23k
FDE という解 ― 暗黙知と明示知をつなぐ、伴走型エンジニアリング ―
otanet
0
110
AWSシリコン最前線 〜AI時代のチップ選択を読み解く〜
htokoyo
2
350
ブロックチェーン / Blockchain
ks91
PRO
0
120
Building applications in the Gemini API family.
line_developers_tw
PRO
0
2.6k
フロンティアAIのゲート化と地政学リスク
nagatsu
0
110
Featured
See All Featured
Practical Orchestrator
shlominoach
191
11k
Primal Persuasion: How to Engage the Brain for Learning That Lasts
tmiket
0
360
The Curse of the Amulet
leimatthew05
1
13k
Reflections from 52 weeks, 52 projects
jeffersonlam
356
21k
Joys of Absence: A Defence of Solitary Play
codingconduct
1
390
Applied NLP in the Age of Generative AI
inesmontani
PRO
4
2.3k
The Impact of AI in SEO - AI Overviews June 2024 Edition
aleyda
5
1.1k
How to Think Like a Performance Engineer
csswizardry
28
2.6k
Noah Learner - AI + Me: how we built a GSC Bulk Export data pipeline
techseoconnect
PRO
0
200
Creating an realtime collaboration tool: Agile Flush - .NET Oxford
marcduiker
35
2.5k
Claude Code のすすめ
schroneko
67
230k
Rails Girls Zürich Keynote
gr2m
96
14k
Transcript
安全にプロセスを停止するために シグナル制御を学ぼう! NE株式会社 やまもとひろや
先に注意 • 本スライドにはサンプルコードがいくつか登場し ます。 • 文字が小さく読みにくい場合があります。 • 気になる方はお手元でスライドを開いて視聴す ることをオススメします。 2
自己紹介 • NE株式会社のやまもとひろやです。 • PHPerKaigi2022 ◦ ベストフィードバッカー殿堂入りしました。 ◦ PHPerチャレンジは4位でした。 •
プロポーザル採択のレギュラーセッションは初です。 • PHP歴約10年です。 • 右の写真はPHPerKaigi2021の ベストフィードバッカー賞でもらった Oculus Quest 2です。 • よろしくおねがいします! 3
アジェンダ • ターゲット • 「プロセス」とは • 「シグナル」とは • 「シグナル制御」とは •
「安全に」とは • 活用例 • まとめ 4
ターゲット • PHPの処理を ◦ Ctrl+Cで止めたことがある人 ◦ killコマンドで止めたことがある人 • PHPの処理が途中で死んで困ったことがある人 •
プロセス/シグナルが何となくしか分からない人 • 同僚や後輩に「シグナル制御って何?」って聞かれた時に答えに困る人 • 是非持ち帰って共有してください! 5
それではスタート! 6
プロセスはすぐ死ぬ 7 よーし今から処理するぞー! プロセス君
プロセスはすぐ死ぬ 8 よーし今から処理するぞー! プロセス君 シグナル君 あ、処理やめてください
プロセスはすぐ死ぬ 9 よーし今から処理するぞー! プロセス君 シグナル君 やっぱり処理やーめた あ、処理やめてください
プロセスはすぐ死ぬ 10 よーし今から処理するぞー! プロセス君 シグナル君 やっぱり処理やーめた あ、処理やめてください • どこまで処理されたか分からない! •
最初から処理をやり直さないと! • ゴミデータが残ってしまった!
• 本講演においてはPHPの処理のことをプロセスと呼びます。 • 例えば以下のtest.phpを実行したとして、psコマンドに載ってくるものがプロセスです。 プロセスってなに? test.php <?php while(1){} 11 $
ps PID TTY TIME CMD 63512 ttys001 0:03.34 php test.php
シグナルって何? • 本講演においてはプロセスに対して与える命令のことをシグナルと呼びます。 • 例えば先程のtest.phpの実行を停止するためにCtrl+Cで止めたとして、内部的にはプロセ スに対してSIGINTというシグナルが送られて停止しています。 • Ctrl+CがSIGINTというのは以下で確認ができます。 ◦ ^CがCtrl+Cの意味
◦ intrがSIGINTの意味 12 $ stty -a cchars: discard = ^O; dsusp = ^Y; eof = ^D; eol = <undef>; eol2 = <undef>; erase = ^?; intr = ^C; kill = ^U; lnext = ^V; min = 1; quit = ^\; reprint = ^R; start = ^Q; status = ^T; stop = ^S; susp = ^Z; time = 0; werase = <undef>;
どういう時にシグナルが送られてくるの? • PHPスクリプト実行中にCtrl+C • PHPスクリプトのプロセスに対してkillコマンド • PHPが動いているDockerコンテナに対してdocker stop • PHPが動いているAWS/コンテナに対してデプロイ/コンテナ切替など
• etc 13
シグナル制御ってなに? 14 よーし今から処理するぞー! プロセス君
シグナル制御ってなに? 15 よーし今から処理するぞー! プロセス君 シグナル君 あ、処理やめてください
シグナル制御ってなに? 16 よーし今から処理するぞー! プロセス君 シグナル君 ちょっと待ってくださいね。 この処理が終わってから停止しますね。(キリッ) あ、処理やめてください
シグナル制御ってなに? • 本講演においてはプロセスにシグナルが送られてきた際、処理を終了する前に何らかの処 理を実施することをシグナル制御(シグナルハンドリング)と呼びます。 • 例えば先程の無限ループにプロセスにて終了する前に echo ”hoge”;する、みたいなことが できるようになるのがシグナル制御です。 17
「安全に」とは • 本講演においては「後片付け・後処理をしたうえで終了すること」を安全にプロセスを停止す る、と定義します。 • 例えば以下のようなステータスをデータ管理しているとして • 10: 未着手→20: 着手中→30:
終了 • これらは一連の処理で30になり20で止まることが基本的にないものの場合、異常終了した 際には10に戻して終了する、などが安全に処理を終了するということになります。 18
シグナル制御のコード解説 19
シグナル制御 20 • pcntl_async_signals(true); ◦ シグナルハンドラを有効にする。 ◦ とりあえず最初に宣言しておけば OK。 •
pcntl_signal(SIGINT, "hoge_handler"); ◦ シグナルハンドラの設定をする。 ◦ SIGINTが来たらhoge_handlerというハンドラ (関数)を実行するという意味。 • hoge_handler ◦ 今回の制御処理。以降ハンドラと呼ぶ。 ◦ hogeを出力して処理を終了する。
シグナル制御 21 • pcntl_async_signals(true); ◦ シグナルハンドラを有効にする。 ◦ とりあえず最初に宣言しておけば OK。 •
pcntl_signal(SIGINT, "hoge_handler"); ◦ シグナルハンドラの設定をする。 ◦ SIGINTが来たらhoge_handlerというハンドラ (関数)を実行するという意味。 • hoge_handler ◦ 今回の制御処理。以降ハンドラと呼ぶ。 ◦ hogeを出力して処理を終了する。
シグナル制御 22 • pcntl_async_signals(true); ◦ シグナルハンドラを有効にする。 ◦ とりあえず最初に宣言しておけば OK。 •
pcntl_signal(SIGINT, "hoge_handler"); ◦ シグナルハンドラの設定をする。 ◦ SIGINTが来たらhoge_handlerというハンドラ (関数)を実行するという意味。 • hoge_handler ◦ 今回の制御処理。以降ハンドラと呼ぶ。 ◦ hogeを出力して処理を終了する。
pcntl_signalについて • 第1引数signal: シグナルの種類(1とか2とか) • 第2引数handler: 後述 • 第3引数restart_syscalls: ※説明を省略
23
pcntl_signalについて • 引数に渡すhandlerについて • 公式リファレンス 24 • signo, siginfoが送られてくるか受け取らなくても良い •
returnはvoidでなくても良い(何でも良い)
余談: callableについて • callableについて ◦ 関数名そのまま渡す場合 ▪ 単純に文字列で’hoge’を渡せば良い ◦ staticメソッドを渡す場合
▪ 配列でarray(‘class_name’, ‘method_name’) ▪ 文字列で’class_name::method_name’ ◦ インスタンスメソッドを渡す場合 ▪ array($obj, ‘method_name’) ▪ array($this, ‘method_name’) ◦ その他selfやparentも使える 25
シグナルの種類 $ kill -l 1) SIGHUP 2) SIGINT 3) SIGQUIT
4) SIGILL 5) SIGTRAP 6) SIGABRT 7) SIGEMT 8) SIGFPE 9) SIGKILL 10) SIGBUS 11) SIGSEGV 12) SIGSYS 13) SIGPIPE 14) SIGALRM 15) SIGTERM 16) SIGURG 17) SIGSTOP 18) SIGTSTP 19) SIGCONT 20) SIGCHLD 21) SIGTTIN 22) SIGTTOU 23) SIGIO 24) SIGXCPU 25) SIGXFSZ 26) SIGVTALRM 27) SIGPROF 28) SIGWINCH 29) SIGINFO 30) SIGUSR1 31) SIGUSR2 26
よく見るシグナル • SIGINT: キーボードからの割り込みシグナル( Ctrl+C) • SIGTERM: 終了シグナル(killコマンドのデフォルト) • SIGKILL:
強制終了シグナル • それぞれ意味が異なる 27
シグナルの種類 28 • SIGKILLは制御できません。 ◦ これができてしまうと永遠にkillできないプロセスができあがります。
安全にするための 事例/活用例紹介 29
活用例(データをキレイにしてから終了する) 30 通常パターン プロセス停止パターン プロセスを制御せずに停止した場合 20で止まる可能性がある。 DBレコードの場合はシグナル制御ではなくトランザクション -ロールバックの 方が良い。
活用例(データをキレイにしてから終了する) 31 通常パターン プロセス停止パターン プロセスを制御せずに停止した場合 20で止まる可能性がある。 DBレコードの場合はシグナル制御ではなくトランザクション -ロールバックの 方が良い。
活用例(データをキレイにしてから終了する) 32 通常パターン プロセス停止パターン プロセスを制御せずに停止した場合 20で止まる可能性がある。 DBレコードの場合はシグナル制御ではなくトランザクション -ロールバックの 方が良い。
活用例(どこまで終わったかを通知してから終了する) 33 通常パターン プロセス停止パターン 通知の部分はメールとか slack通知とかのイメージです
活用例(処理が終わるのを待ってから終了する) 34
活用例(受けたシグナルによって処理を分岐) 35
活用例(受けたシグナルによって処理を分岐) 36 kill -2(SIGINT)を送る シグナルを検知してプロセスが停止する様子
活用例(受けたシグナルによって処理を分岐) 37 kill -15(SIGTERM)を送る シグナルを検知してもプロセスが停止しない様子
活用例(受けたシグナルによって処理を分岐) 38 kill -3(SIGQUIT)を送る 意図しないシグナルなので例外が起こる様子
NEでの活用事例 • AWSへコンテナをデプロイ • デプロイするとコンテナが停止=バッチ処理がデプロイのタイミングで終了してしまう • タイミングによっては意図しない状態になってしまう • その度にエンジニアが裏から対応する必要があった •
何とかバッチを「安全に」停止させられないものか … • ↓ • シグナル制御しよう! 39
NEでの活用例(before) デプロイ時にすぐにコンテナが停止し、中のプロセスが停止してしまう。 バッチが途中で死んでしまうため中途半端なデータができてしまう可能性があった。 40 バッチ ver1.0 デプロイ 命令 バッチ ver2.0
NEでの活用例(after) バッチを最後まで走らせてからコンテナを閉じるように制御を入れた。 その結果処理途中で死ぬことがなくなりデータの整合性が保たれるようになり 安全にデプロイできるようになった。 41 バッチ ver1.0 デプロイ 命令 バッチ
ver2.0
その他時間が余った時用 • PHPの中身について ◦ https://github.com/php/php-src/blob/php-8.2.3/ext/pcntl/pcntl.c#L593 ◦ この辺に定義されてる ◦ 試しにsignoに1より小さい数を渡すと以下のエラーが出た ◦
a ◦ 42
その他時間が余った時用 • SIGUSR1, SIGUSR2について • これらはユーザーが自分で定義して使うことのできるシグナルです。 • SIGINTでもSIGTERMでもないんだよなーという場合に、ユーザー固有の用途で自由に使 うことができます。 43
参考 • PHPとシグナル その裏側 ◦ めちゃくちゃ参考になります。 44
付録 • サンプルコード ◦ https://github.com/yamamoto-hiroya/phperkaigi2023 45
まとめ • 本講演では安全にプロセスを停止するためのシグナル制御を解説しました。 • 仕組みを理解して正しく使っていきましょう。 • Discordやtwitterでフィードバックお待ちしております! • @HiroyaYamamoto1 46
47