Günlük kaydı ve izleme, uygulama performansını anlamanıza ve optimize etmenize, ayrıca hataları ve sistemle ilgili sorunları teşhis etmenize yardımcı olmak için birlikte çalışır. Teknik destek gerektiğinde API çağrısı günlüklerini sağlayabilmek için tüm API çağrıları için özet günlüklerini ve başarısız API çağrıları için ayrıntılı günlükleri etkinleştirmeniz gerekir.
İstemci kitaplığı günlüğü
Google Ads API istemci kitaplıklarında yerleşik günlük kaydı bulunur. Platforma özel günlük kaydı ayrıntıları için tercih ettiğiniz istemci kitaplığındaki günlük kaydı belgelerine bakın.
Dil | Kılavuz |
---|---|
Java | Java için günlük kaydı belgeleri |
.NET | .NET için günlük kaydı belgeleri |
PHP | PHP için günlük kaydı belgeleri |
Python | Python için Logging belgeleri |
Ruby | Ruby için günlük kaydı belgeleri |
Perl | Perl için günlük kaydı belgeleri |
Günlük biçimi
Google Ads API istemci kitaplıkları, her API çağrısı için ayrıntılı bir günlük ve özet günlük oluşturur. Ayrıntılı günlük, API çağrısının tüm ayrıntılarını içerirken özet günlük, API çağrısının en az ayrıntısını içerir. Her günlük türüne ilişkin bir örnek gösterilir. Günlükler, okunabilirlik için kısaltılmış ve biçimlendirilmiştir.
Özet günlüğü
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.")
Ayrıntılı günlük
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----------------
İstemci kitaplığı kullanmıyorsam ne olur?
İstemci kitaplığı kullanmıyorsanız giden ve gelen API çağrılarının ayrıntılarını yakalamak için kendi günlük kaydınızı uygulayın. En azından request-id
yanıt başlığının değerini günlüğe kaydetmeniz gerekir. Bu değer daha sonra gerektiğinde teknik destek ekipleri ile paylaşılabilir.
Buluta kaydetme
Uygulamanızla ilgili günlükleri ve performans metriklerini yakalamak için kullanabileceğiniz birçok araç vardır. Örneğin, Google Cloud Logging'i kullanarak performans metriklerini Google Cloud projenize kaydedebilirsiniz. Bu sayede, kaydedilen metriklerden yararlanmak için Google Cloud Monitoring'de kontrol panelleri ve uyarılar ayarlamak mümkün olur.
Cloud Logging, Perl hariç desteklenen tüm Google Ads API istemci kitaplığı dilleri için istemci kitaplıkları sunar. Bu nedenle, çoğu durumda doğrudan istemci kitaplığı entegrasyonunuzdan Cloud Logging ile günlük kaydı oluşturmak mümkündür. Cloud Logging, Perl dahil diğer diller için de REST API sunar.
Google Ads API istemci kitaplığından Cloud Logging'e veya başka bir araca günlük kaydetmek için birkaç seçenek vardır. Her seçeneğin uygulama süresi, karmaşıklığı ve performansı açısından kendine özgü avantajları ve dezavantajları vardır. Hangi çözümü uygulayacağınıza karar vermeden önce bu ödünler hakkında dikkatlice düşünün.
1. seçenek: Arka plan işleminden buluta yerel günlükler yazma
Günlük yapılandırmanızı değiştirerek istemci kitaplığı günlüklerini makinenizdeki yerel bir dosyaya yazabilirsiniz. Günlükler yerel bir dosyaya çıktı olarak verildikten sonra, günlükleri toplayıp buluta göndermek için bir arka plan programı ayarlayabilirsiniz.
Bu yaklaşımın sınırlamalarından biri, bazı performans metriklerinin varsayılan olarak yakalanmamasıdır. İstemci kitaplığı günlükleri, istek ve yanıt nesnelerindeki ayrıntıları içerir. Bu nedenle, gecikme metrikleri de günlüklerde yer almaz. Ancak bu metriklerin de günlüğe kaydedilmesi için ek değişiklikler yapılabilir.
2. seçenek: Uygulamanızı Compute Engine'de çalıştırın ve İşlem Aracısı'nı yükleyin
Uygulamanız Compute Engine'de çalışıyorsa İşlem Aracısı'nı yükleyerek günlüklerinizi Google Cloud Logging'e gönderebilirsiniz. Ops Agent, varsayılan olarak gönderilen metrikler ve günlüklerin yanı sıra uygulama günlüklerinizi Cloud Logging'e gönderecek şekilde yapılandırılabilir.
Uygulamanız zaten bir Google Cloud ortamında çalışıyorsa veya uygulamanızı Google Cloud'a taşımayı düşünüyorsanız bu seçeneği değerlendirmeniz önerilir.
3. seçenek: Uygulama kodunuzda günlüğe kaydetmeyi uygulayın
Doğrudan uygulama kodundan günlük kaydı oluşturma işlemi iki şekilde yapılabilir:
Metrik hesaplamalarını ve günlük ifadelerini kodunuzdaki geçerli her yere dahil etme. Bu seçenek, kapsam ve bakım maliyetlerinin minimum düzeyde olacağı daha küçük kod tabanları için daha uygundur.
Günlük kaydı arayüzü uygulama. Uygulama mantığı, uygulamanın farklı parçaları aynı temel sınıftan devralacak şekilde soyutlanabiliyorsa günlük kaydı mantığı bu temel sınıfta uygulanabilir. Bu seçenek, genellikle uygulama koduna günlük ifadeleri eklemeye kıyasla daha kolay bakım ve ölçeklendirme sağladığı için tercih edilir. Daha büyük kod tabanlarında bu çözümün sürdürülebilirliği ve ölçeklenebilirliği daha da önem kazanır.
Bu yaklaşımın bir sınırlaması, tam istek ve yanıt günlüklerinin uygulama kodundan kullanılamamasıdır. Tam istek ve yanıt nesnelerine gRPC araya girenlerinden erişilebilir. Yerleşik istemci kitaplığı günlüğe kaydetme özelliği, istek ve yanıt günlüklerini bu şekilde alır. Hata durumunda, istisna nesnesinde ek bilgiler bulunabilir ancak uygulama mantığı içindeki başarılı yanıtlar için daha az ayrıntı kullanılabilir. Örneğin, çoğu durumda başarılı bir isteğin istek kimliğine Google Ads API yanıt nesnelerinden erişilemez.
4. seçenek: Özel bir gRPC günlük kaydı araya giricisi uygulama
gRPC, istemci ile sunucu arasında iletilirken istek ve yanıt nesnelerine erişebilen tek bileşenli ve akış arayıcılarını destekler. Google Ads API istemci kitaplıkları, yerleşik günlük kaydı desteği sunmak için gRPC araya girenlerini kullanır. Benzer şekilde, istek ve yanıt nesnelerine erişmek, günlük kaydı ve izleme amacıyla bilgi ayıklamak ve bu verileri istediğiniz konuma yazmak için özel bir gRPC araya girme işlevi uygulayabilirsiniz.
Burada sunulan diğer çözümlerden farklı olarak, özel bir gRPC önleyici uygulamak, her istekte istek ve yanıt nesnelerini yakalama ve isteğin ayrıntılarını yakalamak için ek mantık uygulama esnekliği sağlar. Örneğin, özel araya girme işlevinin içinde performans zamanlama mantığı uygulayarak bir isteğin geçen süresini hesaplayabilir, ardından Google Cloud Monitoring'de gecikme izleme için kullanılabilir hale getirmek üzere metriği Google Cloud Logging'e kaydedebilirsiniz.
Python'da özel Google Cloud Logging önleyici
Bu çözümü göstermek için Python'da özel günlük kaydı araya giricisi örneği yazdık. Özel araya girme işlevi oluşturulur ve hizmet istemcisine iletilir. Ardından, her hizmet yöntemi çağrısında geçen istek ve yanıt nesnelerine erişir, bu nesnelerdeki verileri işler ve verileri Google Cloud Logging'e gönderir.
İstek ve yanıt nesnelerinden gelen verilere ek olarak, örnekte isteğin geçen süresini ve izleme amacıyla yararlı olacak diğer bazı meta verileri (ör. isteğin başarılı olup olmadığı) yakalamak için bazı ek mantıklar uygulanır. Bu bilgilerin hem genel olarak izleme hem de özellikle Google Cloud Logging ve Google Cloud Monitoring'i birleştirme açısından nasıl faydalı olabileceği hakkında daha fazla bilgi için İzleme kılavuzu'na bakın.
# 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 time from typing import Any, Callable, Dict, Optional from google.cloud import logging as google_cloud_logging from grpc._interceptor import _ClientCallDetails from google.ads.googleads.interceptors import LoggingInterceptor 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: str): """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: google_cloud_logging.Client = google_cloud_logging.Client() self.logger: google_cloud_logging.Logger = logging_client.logger("cloud_logging") self.rpc_start: float self.rpc_end: float def log_successful_request( self, method: str, customer_id: Optional[str], metadata_json: str, request_id: str, request: Any, # google.ads.googleads.vX.services.types.SearchGoogleAdsRequest or SearchGoogleAdsStreamRequest trailing_metadata_json: str, response: Any, # grpc.Call or grpc.Future ) -> None: """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: Any = 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: float = (self.rpc_end - self.rpc_start) * 1000 debug_log: Dict[str, Any] = { "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: Dict[str, Any] = { "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: str, customer_id: Optional[str], metadata_json: str, request_id: str, request: Any, # google.ads.googleads.vX.services.types.SearchGoogleAdsRequest or SearchGoogleAdsStreamRequest trailing_metadata_json: str, response: Any, # grpc.Call or grpc.Future ) -> None: """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: Any = self._get_error_from_response(response) exception_str: str = self._parse_exception_to_str(exception) fault_message: str = self._get_fault_message(exception) info_log: Dict[str, Any] = { "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: Dict[str, Any] = { "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: Callable[[_ClientCallDetails, Any], Any], # Any is request type client_call_details: _ClientCallDetails, request: Any, # google.ads.googleads.vX.services.types.SearchGoogleAdsRequest ) -> Any: # grpc.Call or grpc.Future """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: Any) -> None: # response_future is grpc.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: Any = continuation(client_call_details, request) # response is grpc.Call or grpc.Future 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: Callable[[_ClientCallDetails, Any], Any], # Any is request type client_call_details: _ClientCallDetails, request: Any, # google.ads.googleads.vX.services.types.SearchGoogleAdsStreamRequest ) -> Any: # grpc.Call or grpc.Future """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: Any) -> None: # response_future is grpc.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: Any = continuation(client_call_details, request) # response is grpc.Call or grpc.Future # 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