へっぽこプログラマーの備忘録
プログラムを中心とした個人的なメモ用のブログです。 タイトルは迷走中。
内容の保証はできませんのであしからずご了承ください。
menu
keyboard_arrow_up
Top
search
close
home
ホーム
computer
PC一般
construction
開発環境・ツール
code
プログラミング
home
ホーム
computer
PC一般
construction
開発環境・ツール
code
プログラミング
Home
›
Python
›
[Python] マルチプロセスにおけるロギング (QueueHandler, QueueListener)
2023/04/04
[Python] マルチプロセスにおけるロギング (QueueHandler, QueueListener)
update
event_note
label
Python
SyslogHandler で UDP でログを記録していたのですが、マルチプロセスでは Socket 通信のところでエラーが発生しました。 マルチプロセスでのロギングでは、`QueueHandler` と `QueueListner` を使ってログを一箇所に集約して書き込むようにする必要があります。
## 環境 - Python 3.8.10 ## 基本的な書き方 とりあえずマルチプロセスは置いておいて、`QueueHandler` と `QueueListener` の基本的な使い方です。 ```py import os from multiprocessing import Queue from logging import StreamHandler, handlers, basicConfig, getLogger, Formatter, DEBUG # 標準出力の設定 stream_handler = StreamHandler() stream_handler.setFormatter(Formatter('%(asctime)s [%(levelname)s] [%(name)s] %(message)s')) # Syslog 出力の設定 syslog_handler = handlers.SysLogHandler(address=('localhost', 514)) syslog_handler.setFormatter(Formatter(f'{os.environ.get("HOSTNAME")} ris-main: %(message)s')) # ファイル出力の設定 file_handler = handlers.TimedRotatingFileHandler('log/sample.log') file_handler.setFormatter(Formatter(fmt='%(asctime)s.%(msecs)03d [%(levelname)s] [%(name)s] %(message)s', datefmt='%Y-%m-%d %H:%M:%S')) # キューハンドラの設定 que = Queue() queue_handler = handlers.QueueHandler(que) basicConfig(level=DEBUG, handlers=[queue_handler]) # リスナーの作成と開始 queue_listener = handlers.QueueListener(que, *[stream_handler, syslog_handler, file_handler]) queue_listener.start() logger = getLogger(__name__) logger.info('this is info') logger.debug('this is debug') logger.critical('this is critical') queue_listener.stop() ``` `logger` を使って書いたログは `QueueHandler` によりキューに入れられます。 `QueueListner` はキューに溜まったログを取り出して、`StreamHandler` `SyslogHandler` `TimedRotatingFileHandler` で記録されます。 尚、上記は `basicConfig` を使って設定していますので、ライブラリなどのロギングにも影響します(対処方法は後述)。 ### マルチプロセスで試してみた 上記を踏まえ、プロセスを3つ立ち上げ、各プロセス100回ログを記録させてみたサンプルが以下です。 ```py import os from multiprocessing import Process, Queue from logging import StreamHandler, handlers, basicConfig, getLogger, Formatter, DEBUG def hoge(index): logger = getLogger(__name__) for count in range(100): logger.info(f'({index}) this is info') logger.debug(f'({index}) this is debug') logger.critical(f'({index}) this is critical') # 標準出力の設定 stream_handler = StreamHandler() stream_handler.setFormatter(Formatter('%(asctime)s [%(levelname)s] [%(name)s] %(message)s')) # Syslog 出力の設定 syslog_handler = handlers.SysLogHandler(address=('localhost', 514)) syslog_handler.setFormatter(Formatter(f'{os.environ.get("HOSTNAME")} ris-main: %(message)s')) # ファイル出力の設定 file_handler = handlers.TimedRotatingFileHandler('log/sample.log') file_handler.setFormatter(Formatter(fmt='%(asctime)s.%(msecs)03d [%(levelname)s] [%(name)s] %(message)s', datefmt='%Y-%m-%d %H:%M:%S')) # キューハンドラの設定 que = Queue() queue_handler = handlers.QueueHandler(que) basicConfig(level=DEBUG, handlers=[queue_handler]) # リスナーの作成と開始 queue_listener = handlers.QueueListener(que, *[stream_handler, syslog_handler, file_handler]) queue_listener.start() # プロセスの立ち上げ processes = [] for i in range(3): p = Process(target=hoge, args=(i,)) p.start() processes.append(p) # プロセスの終了を待つ for process in processes: try: process.join() except KeyboardInterrupt: process.join(1) queue_listener.stop() ``` 問題なくログが記録できていることが確認できました。 ## basicConfig を使わずに設定する 上述したように、`basicConfig` を使うとライブラリなどの全てのモジュールに影響するので、`logger` に対して設定を行うようにします。 ただ、そうなると都度設定を行う必要が出てくるので、以下のページを参考に共通の関数を作成しました。 - http://joemphilips.com/post/python_logging/ ただし、QueueHandler を使う場合、キューを渡してやる必要があるので、それはグローバル変数として用意することにしました。 ここらへん、もっと良い実装方法はないかなぁとは思いますが・・・。 以下、サンプルです。 ```py # mylogger.py import multiprocessing import logging import logging.handlers log_queue = multiprocessing.Queue() def getLogger(modname) -> logging.Logger: """ 共通のロガーを作成するための関数 logging.getLogger と簡単に差し替えられるように同じ関数名としておく """ # キューハンドラの設定 queue_handler = logging.handlers.QueueHandler(log_queue) logger = logging.getLogger(modname) logger.addHandler(queue_handler) logger.setLevel(logging.DEBUG) return logger ``` ```py # main.py import os from logging import handlers, StreamHandler, Formatter, DEBUG, INFO from mylogger import getLogger, log_queue # 標準出力の設定 stream_handler = StreamHandler() stream_handler.setFormatter(Formatter('%(asctime)s [%(levelname)s] [%(name)s] %(message)s')) stream_handler.setLevel(DEBUG) # Syslog 出力の設定 syslog_handler = handlers.SysLogHandler(address=('localhost', 514)) syslog_handler.setFormatter(Formatter(f'{os.environ.get("HOSTNAME")} ris-main: %(message)s')) syslog_handler.setLevel(INFO) # ファイル出力の設定 file_handler = handlers.TimedRotatingFileHandler('log/sample.log') file_handler.setFormatter(Formatter(fmt='%(asctime)s.%(msecs)03d [%(levelname)s] [%(name)s] %(message)s', datefmt='%Y-%m-%d %H:%M:%S')) file_handler.setLevel(INFO) # リスナーの作成と開始 queue_listener = handlers.QueueListener( log_queue, *[stream_handler, syslog_handler, file_handler], respect_handler_level=True) queue_listener.start() logger = getLogger(__name__) logger.info('this is info') logger.debug('this is debug') logger.critical('this is critical') queue_listener.stop() ``` また、ここでは標準出力とそれ以外で、出力するログレベルを変えています。 この場合、`QueueHandler` の `respect_handler_level` を `True` にして、各ハンドラー側のレベルを優先するように設定してやる必要があります。
## 参考 URL - https://docs.python.org/ja/3.8/library/logging.handlers.html - https://ikatakos.com/pot/programming/python/packages/multiprocessing/queuehandler - https://rob-blackbourn.medium.com/how-to-use-python-logging-queuehandler-with-dictconfig-1e8b1284e27a - http://joemphilips.com/post/python_logging/
tweet
facebook
Pocket
B!
はてブ
LINE
chevron_left
chevron_right
Translate
Popular Posts
Labels
.NET Core
31
.NET Framework
17
.NET Standard
2
AdminLTE
1
AI
1
Apache
3
AppVeyor
2
AsciiDoc
7
ASP.NET Core
55
Atom
4
AWS
5
AWS Cloud9
4
blockdiag
1
Blogger
13
Bootstrap
3
C/C++
6
C#
106
CentOS
3
Chrome
1
Chronograf
3
chrony
1
Codecov
1
CSS
1
Docker
82
DokuWiki
4
Doxygen
1
draw.io
1
EasyTag
1
Electron
1
Electron.NET
2
Entity Framework Core
9
Excel
2
FFmpeg
3
Firefox
6
Flask
1
Git
19
GitBook
4
GitBucket
7
GitHub
7
GitLab
39
Go
1
Google
1
Google Cloud Platform
1
Grafana
13
GStreamer
2
HTML
5
IIS
8
InfluxDB
14
JavaScript
15
Jekyll
4
Jenkins
7
Linux
34
Log4View
1
MahApps.Metro
3
MaterialDesignInXamlToolkit
1
MkDocs
2
MongoDB
5
MVC
1
MVVM
6
nginx
3
NLog
3
Node.js
8
npm
1
NVIDIA
3
onvif
1
OpenAPI
2
OpenCV
4
OpenSSL
3
OpenVINO
2
ownCloud
2
pandas
1
Pine Script
1
PlantUML
5
Portainer
3
PowerShell
8
Prism
2
PySide
19
Python
87
PyTorch
1
RabbitVCS
1
Razor
3
redis
1
Redmine
33
Redoc
1
remark.js
2
rocketchat
10
Ruby
3
scikit-learn
1
shotcut
1
SignalR
1
Slack
1
Socket.IO
1
SonarQube
5
Sphinx
10
SQL Server
5
SQLite
1
StableDiffusion
2
Subversion
2
Swagger
1
Swarmpit
1
Syslog
3
Telegraf
6
Tesseract
3
TestLink
2
Tomcat
2
TortoiseGit
11
TortoiseSVN
2
Trading View
1
Traefik
3
Travis CI
1
Ubuntu
31
Visual Studio
39
Visual Studio Code
10
VSCode
8
Vue.js
8
Windows
62
Windows 10
5
Windows ADK
1
Windows API
2
Windows Embedded
4
wkhtmltopdf
2
Word
3
WPF
12
WSL
5
WSL2
5
Xamarin
1
xUnit
5
yaml
1
yolo
1
アプリケーション
1
デザインパターン
1
テスト
1
バッチファイル
2
プログラミング
4
ライセンス
1
暗号資産(仮想通貨)
1
英語
2
確定申告
1
機械学習
1
強化学習
1
雑記
1
書籍
1
数学
1
正規表現
1
動画編集
1
Blog Archive
►
2025
(1)
►
3月
(1)
►
2024
(21)
►
12月
(3)
►
9月
(5)
►
8月
(1)
►
7月
(2)
►
6月
(1)
►
4月
(2)
►
3月
(1)
►
2月
(5)
►
1月
(1)
▼
2023
(30)
►
12月
(3)
►
11月
(5)
►
10月
(2)
►
9月
(1)
►
8月
(2)
►
7月
(4)
►
6月
(2)
►
5月
(3)
▼
4月
(2)
[Python] SyslogHandler でログを送信したときに `Bad file descr...
[Python] マルチプロセスにおけるロギング (QueueHandler, QueueListe...
►
3月
(2)
►
2月
(3)
►
1月
(1)
►
2022
(105)
►
12月
(5)
►
11月
(1)
►
10月
(3)
►
9月
(5)
►
8月
(7)
►
7月
(6)
►
6月
(13)
►
5月
(9)
►
4月
(15)
►
3月
(11)
►
2月
(14)
►
1月
(16)
►
2021
(85)
►
12月
(11)
►
11月
(6)
►
10月
(4)
►
9月
(10)
►
8月
(8)
►
7月
(4)
►
6月
(18)
►
5月
(7)
►
4月
(8)
►
3月
(2)
►
2月
(2)
►
1月
(5)
►
2020
(56)
►
12月
(1)
►
11月
(3)
►
10月
(3)
►
9月
(3)
►
8月
(3)
►
7月
(7)
►
6月
(7)
►
5月
(2)
►
4月
(6)
►
3月
(6)
►
2月
(3)
►
1月
(12)
►
2019
(92)
►
12月
(13)
►
11月
(9)
►
10月
(3)
►
9月
(2)
►
8月
(3)
►
7月
(5)
►
6月
(11)
►
5月
(6)
►
4月
(17)
►
3月
(9)
►
2月
(6)
►
1月
(8)
►
2018
(100)
►
12月
(1)
►
11月
(11)
►
10月
(8)
►
9月
(6)
►
8月
(10)
►
7月
(10)
►
6月
(8)
►
5月
(9)
►
4月
(8)
►
3月
(14)
►
2月
(4)
►
1月
(11)
►
2017
(117)
►
12月
(14)
►
11月
(20)
►
10月
(17)
►
9月
(19)
►
8月
(10)
►
7月
(8)
►
6月
(3)
►
5月
(6)
►
4月
(5)
►
3月
(2)
►
2月
(8)
►
1月
(5)
►
2016
(91)
►
12月
(5)
►
11月
(9)
►
10月
(11)
►
9月
(9)
►
8月
(6)
►
7月
(14)
►
6月
(14)
►
5月
(11)
►
4月
(10)
►
3月
(2)
►
2015
(23)
►
12月
(4)
►
11月
(2)
►
10月
(8)
►
9月
(8)
►
7月
(1)
►
2013
(3)
►
11月
(1)
►
9月
(1)
►
7月
(1)
►
2012
(2)
►
7月
(1)
►
6月
(1)
►
2011
(1)
►
9月
(1)
►
2009
(1)
►
7月
(1)
►
2008
(2)
►
11月
(1)
►
7月
(1)
►
2007
(3)
►
10月
(3)