تعمل ميزتا التسجيل والمراقبة معًا لمساعدتك في فهم أداء التطبيق وتحسينه، بالإضافة إلى تشخيص الأخطاء والمشاكل المتعلقة بالنظام. عليك تفعيل سجلات الملخّص لجميع طلبات البيانات من واجهة برمجة التطبيقات والسجلات التفصيلية لطلبات البيانات من واجهة برمجة التطبيقات التي تعذّر تنفيذها حتى تتمكّن من تقديم سجلات طلبات البيانات من واجهة برمجة التطبيقات عند الحاجة إلى الدعم الفني.
تسجيل مكتبة العميل
تتضمّن مكتبات عميل Google Ads API ميزة تسجيل البيانات المدمجة. للحصول على تفاصيل التسجيل الخاصة بالنظام الأساسي، يُرجى الرجوع إلى مستندات التسجيل ضمن مكتبة البرامج التي تختارها.
اللغة | الدليل |
---|---|
Java | تسجيل المستندات في Java |
NET. | مستندات التسجيل في .NET |
PHP | مستندات التسجيل للغة PHP |
Python | مستندات التسجيل في Python |
Ruby | تسجيل المستندات لـ Ruby |
Perl | تسجيل المستندات في Perl |
تنسيق السجلّ
تنشئ مكتبات برامج Google Ads 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 Logging لتسجيل مقاييس الأداء في مشروع Google Cloud. ويتيح ذلك إعداد لوحات بيانات وتنبيهات في خدمة "المراقبة في Google Cloud" للاستفادة من المقاييس المسجّلة.
توفّر خدمة Cloud Logging مكتبات برامج لجميع لغات مكتبات برامج Google Ads API المتوافقة باستثناء Perl، لذا في معظم الحالات، يمكن تسجيل البيانات باستخدام Cloud Logging مباشرةً من عملية دمج مكتبة البرامج. بالنسبة إلى اللغات الأخرى، بما في ذلك Perl، توفّر Cloud Logging أيضًا REST API.
تتوفّر بعض الخيارات لتسجيل البيانات في Cloud Logging أو أداة أخرى من مكتبة برامج Google Ads API. ويأتي كل خيار مع مزايا وعيوب من حيث الوقت اللازم للتنفيذ والتعقيد والأداء. يجب التفكير مليًا في هذه المفاضلات قبل اتّخاذ قرار بشأن الحلّ الذي سيتم تنفيذه.
الخيار 1: كتابة السجلّات المحلية إلى السحابة الإلكترونية من عملية في الخلفية
يمكن كتابة سجلات مكتبة البرامج للعملاء في ملف محلي على جهازك من خلال تعديل إعدادات التسجيل. بعد إخراج السجلّات إلى ملف محلي، يمكنك إعداد برنامج خفي لجمع السجلّات وإرسالها إلى السحابة الإلكترونية.
أحد القيود المفروضة على هذا الأسلوب هو أنّه لن يتم تلقائيًا تسجيل بعض مقاييس الأداء. تتضمّن سجلّات مكتبة البرامج للعميل تفاصيل من عناصر الطلب والاستجابة، لذا لن يتم تضمين مقاييس وقت الاستجابة ما لم يتم إجراء تغييرات إضافية لتسجيلها أيضًا.
الخيار 2: تشغيل تطبيقك على Compute Engine وتثبيت Ops Agent
إذا كان تطبيقك يعمل على Compute Engine، يمكنك إرسال سجلّاتك إلى Google Cloud Logging من خلال تثبيت Ops Agent. يمكن ضبط Ops Agent لإرسال سجلّات تطبيقك إلى Cloud Logging، بالإضافة إلى المقاييس والسجلّات التي يتم إرسالها تلقائيًا.
إذا كان تطبيقك يعمل حاليًا في بيئة Google Cloud، أو إذا كنت تفكّر في نقل تطبيقك إلى Google Cloud، سيكون هذا خيارًا رائعًا.
الخيار 3: تنفيذ تسجيل الدخول في الرمز البرمجي لتطبيقك
يمكن تسجيل الدخول مباشرةً من رمز التطبيق بإحدى الطريقتَين التاليتَين:
تضمين عمليات حساب المقاييس وبيانات السجلّ في كل موقع جغرافي ذي صلة في الرمز يكون هذا الخيار أكثر جدوى لقواعد الرموز الأصغر، حيث يكون نطاق هذا التغيير وتكاليف صيانته في الحد الأدنى.
تنفيذ واجهة تسجيل إذا كان من الممكن تجريد منطق التطبيق بحيث ترث أجزاء مختلفة من التطبيق من فئة أساسية نفسها، يمكن تنفيذ منطق التسجيل في تلك الفئة الأساسية. ويُفضّل عمومًا استخدام هذا الخيار بدلاً من تضمين عبارات السجلّ في جميع أنحاء رمز التطبيق، لأنّه أسهل في الصيانة والتوسيع. بالنسبة إلى قواعد الرموز البرمجية الأكبر حجمًا، تزداد أهمية قابلية الصيانة وقابلية التوسّع لهذا الحل.
أحد القيود المفروضة على هذا الأسلوب هو عدم توفّر سجلّات الطلبات والردود الكاملة من رمز التطبيق. يمكن الوصول إلى عناصر الطلب والاستجابة الكاملة من خلال أدوات اعتراض gRPC، وهي الطريقة التي تحصل بها ميزة تسجيل الدخول المضمّنة في مكتبة العميل على سجلّات الطلبات والاستجابات. في حال حدوث خطأ، قد تتوفّر معلومات إضافية في عنصر الاستثناء، ولكن تتوفّر تفاصيل أقل للاستجابات الناجحة ضمن منطق التطبيق. على سبيل المثال، في معظم الحالات، لا يمكن الوصول إلى معرّف الطلب لطلب ناجح من عناصر الاستجابة في Google Ads API.
الخيار 4: تنفيذ أداة اعتراض مخصّصة لتسجيل gRPC
يتوافق gRPC مع برامج اعتراضية أحادية ومتدفقة يمكنها الوصول إلى عناصر الطلب والاستجابة أثناء انتقالها بين العميل والخادم. تستخدِم مكتبات عملاء Google Ads API برامج اعتراض gRPC لتوفير ميزة تسجيل البيانات المضمّنة. وبالمثل، يمكنك تنفيذ أداة اعتراض gRPC مخصّصة للوصول إلى عناصر الطلب والاستجابة، واستخراج المعلومات لأغراض التسجيل والمراقبة، وكتابة هذه البيانات في الموقع الجغرافي الذي تختاره.
على عكس بعض الحلول الأخرى المعروضة هنا، يمنحك تنفيذ أداة اعتراض مخصّصة لبروتوكول gRPC المرونة في تسجيل عناصر الطلب والاستجابة في كل طلب، وتنفيذ منطق إضافي لتسجيل تفاصيل الطلب. على سبيل المثال، يمكنك احتساب الوقت المنقضي لطلب ما من خلال تنفيذ منطق توقيت الأداء ضمن أداة الاعتراض المخصّصة نفسها، ثم تسجيل المقياس في Google Cloud Logging لإتاحته لتتبُّع وقت الاستجابة ضمن Google Cloud Monitoring.
أداة اعتراض مخصّصة لخدمة "التسجيل في السحابة الإلكترونية" من Google Cloud في Python
لتوضيح هذا الحل، كتبنا مثالاً على أداة اعتراض مخصّصة لتسجيل البيانات في لغة Python. يتم إنشاء أداة الاعتراض المخصّصة وتمريرها إلى عميل الخدمة. بعد ذلك، يصل إلى عناصر الطلب والاستجابة التي تمر عبر كل استدعاء لطريقة الخدمة، ويعالج البيانات من هذه العناصر، ويرسل البيانات إلى Google Cloud Logging.
بالإضافة إلى البيانات التي تأتي من عناصر الطلب والاستجابة، يتضمّن المثال بعض المنطق الإضافي لتسجيل الوقت المنقضي للطلب وبعض البيانات الوصفية الأخرى التي ستكون مفيدة لأغراض المراقبة، مثل ما إذا كان الطلب ناجحًا أم لا. لمزيد من المعلومات حول كيفية الاستفادة من هذه المعلومات، سواء بشكل عام لأغراض المراقبة أو بشكل خاص عند الجمع بين Google Cloud Logging وGoogle Cloud 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