Files
cpp-httplib/docs-src/pages/ja/cookbook/s15-server-logger.md
2026-04-10 18:47:42 -04:00

2.5 KiB

title, order, status
title order status
S15. リクエストをログに記録する 34 draft

サーバーが受け取ったリクエストと返したレスポンスをログに残したいときは、Server::set_logger()を使います。各リクエストの処理が完了するたびに呼ばれるので、アクセスログやメトリクス収集の土台になります。

基本の使い方

svr.set_logger([](const httplib::Request &req, const httplib::Response &res) {
  std::cout << req.remote_addr << " "
            << req.method << " " << req.path
            << " -> " << res.status << std::endl;
});

ログコールバックにはRequestResponseが渡ります。メソッド、パス、ステータスコード、クライアントIP、ヘッダー、ボディなど、好きな情報を取り出せます。

アクセスログ風のフォーマット

Apache / Nginxのアクセスログに似た形式で残す例です。

svr.set_logger([](const auto &req, const auto &res) {
  auto now = std::time(nullptr);
  char timebuf[32];
  std::strftime(timebuf, sizeof(timebuf), "%Y-%m-%d %H:%M:%S",
                std::localtime(&now));

  std::cout << timebuf << " "
            << req.remote_addr << " "
            << "\"" << req.method << " " << req.path << "\" "
            << res.status << " "
            << res.body.size() << "B"
            << std::endl;
});

処理時間を測る

ログで処理時間を出したいときは、pre-routingハンドラで開始時刻をres.user_dataに保存しておき、ロガーで差分を取ります。

svr.set_pre_routing_handler([](const auto &req, auto &res) {
  res.user_data.set("start", std::chrono::steady_clock::now());
  return httplib::Server::HandlerResponse::Unhandled;
});

svr.set_logger([](const auto &req, const auto &res) {
  auto *start = res.user_data.get<std::chrono::steady_clock::time_point>("start");
  auto elapsed = start
    ? std::chrono::duration_cast<std::chrono::milliseconds>(
        std::chrono::steady_clock::now() - *start).count()
    : 0;
  std::cout << req.method << " " << req.path
            << " " << res.status << " " << elapsed << "ms" << std::endl;
});

user_dataの使い方はS12. res.user_dataでハンドラ間データを渡すも参照してください。

Note: ロガーはリクエスト処理と同じスレッドで同期的に呼ばれます。重い処理を直接入れると全体のスループットが落ちるので、必要ならキューに流して非同期で処理しましょう。