ロギング

ロギングとモニタリングは連携して機能し、コンテンツの理解と最適化に役立ちます。 診断を行うことができるほか、エラーやシステム関連の診断を行う サポートします。すべての API 呼び出しとログに対して概要ログをオンにし、 失敗した API 呼び出しの詳細ログ(API を提供できるようにするため) テクニカル サポートが必要な場合に通話ログを確認できます。

クライアント ライブラリのロギング

Google Ads API のクライアント ライブラリにはロギング機能が組み込まれています。プラットフォーム固有の ロギングの詳細については、クライアント ライブラリ内のロギングに関するドキュメントをご覧ください。 あります。

言語 ガイド
Java Java 用の Logging ドキュメント
.NET .NET のロギング ドキュメント
PHP PHP 用のロギング ドキュメント
Python Python 用の Logging のドキュメント
Ruby Ruby のロギング ドキュメント
Perl Perl のロギング ドキュメント

ログ形式

Google Ads API のクライアント ライブラリによって、詳細ログ概要が生成されます。 log を記録します。詳細なログには、インフラストラクチャの 概要ログには API 呼び出しの最小限の詳細しか含まれません。 各タイプのログの例を示しており、ログは省略してフォーマットされています。 渡します。

概要ログ

GoogleAds.SummaryRequestLogs Warning: 1 : [2023-09-15 19:58:39Z] -
Request made: Host: , Method: /google.ads.googleads.v14.services.GoogleAdsService/SearchStream,
ClientCustomerID: 5951878031, RequestID: hELhBPNlEDd8mWYcZu7b8g,
IsFault: True, FaultMessage: Status(StatusCode="InvalidArgument",
Detail="Request contains an invalid argument.")

詳細なログ

GoogleAds.DetailedRequestLogs Verbose: 1 : [2023-11-02 21:09:36Z] -
---------------BEGIN API CALL---------------

Request
-------

Method Name: /google.ads.googleads.v14.services.GoogleAdsService/SearchStream
Host:
Headers: {
  "x-goog-api-client": "gl-dotnet/5.0.0 gapic/17.0.1 gax/4.2.0 grpc/2.46.3 gccl/3.0.1 pb/3.21.5",
  "developer-token": "REDACTED",
  "login-customer-id": "1234567890",
  "x-goog-request-params": "customer_id=4567890123"
}

{ "customerId": "4567890123", "query": "SELECT ad_group_criterion.type FROM
  ad_group_criterion WHERE ad_group.status IN(ENABLED, PAUSED) AND
  campaign.status IN(ENABLED, PAUSED) ", "summaryRowSetting": "NO_SUMMARY_ROW" }

Response
--------
Headers: {
  "date": "Thu, 02 Nov 2023 21:09:35 GMT",
  "alt-svc": "h3-29=\":443\"; ma=2592000"
}

{
  "results": [ {
    "adGroupCriterion": {
      "resourceName": "customers/4567890123/adGroupCriteria/456789456789~123456123467",
      "type": "KEYWORD"
    } }, {
    "adGroupCriterion": {
      "resourceName": "customers/4567890123/adGroupCriteria/456789456789~56789056788",
      "type": "KEYWORD"
    } } ],
    "fieldMask": "adGroupCriterion.type", "requestId": "VsJ4F00ew6s9heHvAJ-abw"
}
----------------END API CALL----------------

クライアント ライブラリを使用しない場合はどうすればよいですか?

クライアント ライブラリを使用しない場合は、独自のロギングを実装して、 詳細情報が含まれます。少なくとも、 request-id レスポンス ヘッダーの値。この値は テクニカル サポート チーム(必要に応じて)。

クラウドへのロギング

Google Cloud のログやパフォーマンス指標をキャプチャするために使用できるツールは数多くあります。 説明します。たとえば、Google Cloud Logging を使用して、 パフォーマンス指標を Google Cloud プロジェクトに追加します。これにより、 Google Cloud Monitoring でダッシュボードとアラートを設定可能 ログの指標を活用できます

Cloud Logging には、サポートされているすべての Google Ads API クライアント用のクライアント ライブラリが用意されています。 ライブラリ言語に対応していないため、ほとんどの場合、 クライアント ライブラリ統合から直接 Cloud Logging にアクセスできます。その他の言語の場合 Cloud Logging では REST API も提供されています。

Cloud Logging やその他のツールにログを記録する方法はいくつかあり、 Google Ads API クライアント ライブラリです。どの方法にも、移行に必要な時間のトレードオフが伴います。 実装、複雑さ、パフォーマンスです。これらのトレードオフについて慎重に検討する 決定を下します

オプション 1: バックグラウンド プロセスからローカルログをクラウドに書き込む

クライアント ライブラリのログをマシンのローカル ファイルに書き込むには、 ロギング構成を変更します。ログをローカル ファイルに出力した後は、 ログを収集してクラウドに送信するデーモンをセットアップします。

この方法の制限の一つは、一部のパフォーマンス指標が デフォルトでキャプチャされます。クライアント ライブラリのログには、リクエストと そのため、追加の変更を行わない限りレイテンシ指標は含まれません。 ログに記録されます。

オプション 2: Compute Engine でアプリケーションを実行し、Ops エージェントをインストールする

アプリケーションが Compute Engine で実行されている場合は、 Ops エージェントをインストールして、ログを Google Cloud Logging に送信する。Ops アプリケーション ログを Cloud に送信するようにエージェントを構成可能デフォルトで送信される指標とログに加えて)ロギング。

アプリケーションがすでに Google Cloud 環境で実行されている場合や、 Google Cloud へのアプリケーションの移行を検討している場合、 検討する必要があります

オプション 3: アプリケーション コードにロギングを実装する

アプリケーション コードから直接ロギングするには、次の 2 つの方法があります。

  1. すべてのログに指標計算とログ ステートメントを組み込んで コード内の該当する場所に配置してください。この方法は、小規模なプロジェクトや 変更の範囲とメンテナンス費用は、コードベースで 最小限に抑える必要があります。

  2. ロギング インターフェースを実装する。アプリケーション ロジックを抽象化し、 アプリケーションのさまざまな部分が同じベースから継承されるようにするため、 その基本クラスにロギング ロジックを実装できます。このオプションは プロセス全体を通して、ログ ステートメントを組み込むことよりも、一般に好まれる 管理とスケーリングが容易です。より大きい このソリューションの保守性とスケーラビリティが 向上します

この方法の制限の一つは、リクエストとレスポンスの完全なログが アプリケーション コードからはアクセスできません。完全なリクエスト オブジェクトとレスポンス オブジェクトは、 gRPC インターセプタからアクセスされる組み込みのクライアント ライブラリは、 logging はリクエストとレスポンスのログを取得します。エラーが発生した場合は 例外オブジェクトに格納できる情報はありますが、 正常なレスポンスのためにアプリロジック内で使用できます。たとえば、 ほとんどの場合、成功したリクエストのリクエスト ID は、 Google Ads API レスポンス オブジェクト。

オプション 4: カスタム gRPC ロギング インターセプタを実装する

gRPC は単一およびストリーミングのインターセプタをサポートしており、 リクエスト オブジェクトとレスポンス オブジェクトがクライアントとサーバーの間で渡されるときに使用されます。「 Google Ads API クライアント ライブラリは、gRPC インターセプタを使用して組み込みロギングを提供します。 サポート。同様に、カスタム gRPC インターセプタを実装して、 リクエスト オブジェクトとレスポンス オブジェクト、ロギングとモニタリングのための情報の抽出 そのデータを任意のロケーションに書き込むことができます。

ここで紹介する他のソリューションとは異なり、カスタム gRPC の実装 インターセプタを使用すると、ネットワーク上のリクエスト / レスポンス オブジェクトを柔軟にキャプチャできます。 リクエストの詳細を取得するための追加のロジックを実装できます。 たとえば、次を実装してリクエストの経過時間を計算できます。 カスタム インターセプタ自体の内部パフォーマンス タイミング ロジックを作成し、 指標を Google Cloud Logging に追加してレイテンシ モニタリングに使用できるようにする モニタリングできます。

Python でのカスタム Google Cloud Logging インターセプタ

このソリューションのデモを行うために、カスタム ロギングの例を 使用します。カスタム インターセプタが作成され、 サービスクライアントです次に、このバケットが渡された request オブジェクトと response オブジェクトにアクセスし、 これらのオブジェクトのデータを処理して Google Cloud Logging にデータを送信します。

request オブジェクトと response オブジェクトから取得されるデータに加えて、 サンプルには、ジョブの経過時間をキャプチャする追加のロジックが実装されています。 モニタリングに役立つその他のメタデータも リクエストが成功したかどうかなどの 分析情報が含まれますこの方法の詳細については、 モニタリングに役立つ場合もあれば、特に Google Cloud Logging と Google Cloud Monitoring の組み合わせについては、Monitoring の ガイドをご覧ください。

# Copyright 2022 Google LLC
#
# Licensed under the Apache License, Version 2.0 (the "License");
# you may not use this file except in compliance with the License.
# You may obtain a copy of the License at
#
#     https://www.apache.org/licenses/LICENSE-2.0
#
# Unless required by applicable law or agreed to in writing, software
# distributed under the License is distributed on an "AS IS" BASIS,
# WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
# See the License for the specific language governing permissions and
# limitations under the License.
"""A custom gRPC Interceptor that logs requests and responses to Cloud Logging.

The custom interceptor object is passed into the get_service method of the
GoogleAdsClient. It intercepts requests and responses, parses them into a
human readable structure and logs them using the logging service instantiated
within the class (in this case, a Cloud Logging client).
"""

import logging
import time

from google.cloud import logging
from grpc import UnaryUnaryClientInterceptor, UnaryStreamClientInterceptor

from google.ads.googleads.interceptors import LoggingInterceptor, mask_message


class CloudLoggingInterceptor(LoggingInterceptor):
    """An interceptor that logs rpc request and response details to Google Cloud Logging.

    This class inherits logic from the LoggingInterceptor, which simplifies the
    implementation here. Some logic is required here in order to make the
    underlying logic work -- comments make note of this where applicable.
    NOTE: Inheriting from the LoggingInterceptor class could yield unexpected side
    effects. For example, if the LoggingInterceptor class is updated, this class would
    inherit the updated logic, which could affect its functionality. One option to avoid
    this is to inherit from the Interceptor class instead, and selectively copy whatever
    logic is needed from the LoggingInterceptor class."""

    def __init__(self, api_version):
        """Initializer for the CloudLoggingInterceptor.

        Args:
            api_version: a str of the API version of the request.
        """
        super().__init__(logger=None, api_version=api_version)
        # Instantiate the Cloud Logging client.
        logging_client = logging.Client()
        self.logger = logging_client.logger("cloud_logging")

    def log_successful_request(
        self,
        method,
        customer_id,
        metadata_json,
        request_id,
        request,
        trailing_metadata_json,
        response,
    ):
        """Handles logging of a successful request.

        Args:
            method: The method of the request.
            customer_id: The customer ID associated with the request.
            metadata_json: A JSON str of initial_metadata.
            request_id: A unique ID for the request provided in the response.
            request: An instance of a request proto message.
            trailing_metadata_json: A JSON str of trailing_metadata.
            response: A grpc.Call/grpc.Future instance.
        """
        # Retrieve and mask the RPC result from the response future.
        # This method is available from the LoggingInterceptor class.
        # Ensure self._cache is set in order for this to work.
        # The response result could contain up to 10,000 rows of data,
        # so consider truncating this value before logging it, to save
        # on data storage costs and maintain readability.
        result = self.retrieve_and_mask_result(response)

        # elapsed_ms is the approximate elapsed time of the RPC, in milliseconds.
        # There are different ways to define and measure elapsed time, so use
        # whatever approach makes sense for your monitoring purposes.
        # rpc_start and rpc_end are set in the intercept_unary_* methods below.
        elapsed_ms = (self.rpc_end - self.rpc_start) * 1000

        debug_log = {
            "method": method,
            "host": metadata_json,
            "request_id": request_id,
            "request": str(request),
            "headers": trailing_metadata_json,
            "response": str(result),
            "is_fault": False,
            "elapsed_ms": elapsed_ms,
        }
        self.logger.log_struct(debug_log, severity="DEBUG")

        info_log = {
            "customer_id": customer_id,
            "method": method,
            "request_id": request_id,
            "is_fault": False,
            # Available from the Interceptor class.
            "api_version": self._api_version,
        }
        self.logger.log_struct(info_log, severity="INFO")

    def log_failed_request(
        self,
        method,
        customer_id,
        metadata_json,
        request_id,
        request,
        trailing_metadata_json,
        response,
    ):
        """Handles logging of a failed request.

        Args:
            method: The method of the request.
            customer_id: The customer ID associated with the request.
            metadata_json: A JSON str of initial_metadata.
            request_id: A unique ID for the request provided in the response.
            request: An instance of a request proto message.
            trailing_metadata_json: A JSON str of trailing_metadata.
            response: A JSON str of the response message.
        """
        exception = self._get_error_from_response(response)
        exception_str = self._parse_exception_to_str(exception)
        fault_message = self._get_fault_message(exception)

        info_log = {
            "method": method,
            "endpoint": self.endpoint,
            "host": metadata_json,
            "request_id": request_id,
            "request": str(request),
            "headers": trailing_metadata_json,
            "exception": exception_str,
            "is_fault": True,
        }
        self.logger.log_struct(info_log, severity="INFO")

        error_log = {
            "method": method,
            "endpoint": self.endpoint,
            "request_id": request_id,
            "customer_id": customer_id,
            "is_fault": True,
            "fault_message": fault_message,
        }
        self.logger.log_struct(error_log, severity="ERROR")

    def intercept_unary_unary(self, continuation, client_call_details, request):
        """Intercepts and logs API interactions.

        Overrides abstract method defined in grpc.UnaryUnaryClientInterceptor.

        Args:
            continuation: a function to continue the request process.
            client_call_details: a grpc._interceptor._ClientCallDetails
                instance containing request metadata.
            request: a SearchGoogleAdsRequest or SearchGoogleAdsStreamRequest
                message class instance.

        Returns:
            A grpc.Call/grpc.Future instance representing a service response.
        """
        # Set the rpc_end value to current time when RPC completes.
        def update_rpc_end(response_future):
            self.rpc_end = time.perf_counter()

        # Capture precise clock time to later calculate approximate elapsed
        # time of the RPC.
        self.rpc_start = time.perf_counter()

        # The below call is REQUIRED.
        response = continuation(client_call_details, request)

        response.add_done_callback(update_rpc_end)

        self.log_request(client_call_details, request, response)

        # The below return is REQUIRED.
        return response

    def intercept_unary_stream(
        self, continuation, client_call_details, request
    ):
        """Intercepts and logs API interactions for Unary-Stream requests.

        Overrides abstract method defined in grpc.UnaryStreamClientInterceptor.

        Args:
            continuation: a function to continue the request process.
            client_call_details: a grpc._interceptor._ClientCallDetails
                instance containing request metadata.
            request: a SearchGoogleAdsRequest or SearchGoogleAdsStreamRequest
                message class instance.

        Returns:
            A grpc.Call/grpc.Future instance representing a service response.
        """

        def on_rpc_complete(response_future):
            self.rpc_end = time.perf_counter()
            self.log_request(client_call_details, request, response_future)

        # Capture precise clock time to later calculate approximate elapsed
        # time of the RPC.
        self.rpc_start = time.perf_counter()

        # The below call is REQUIRED.
        response = continuation(client_call_details, request)

        # Set self._cache to the cache on the response wrapper in order to
        # access the streaming logs. This is REQUIRED in order to log streaming
        # requests.
        self._cache = response.get_cache()

        response.add_done_callback(on_rpc_complete)

        # The below return is REQUIRED.
        return response