Source code for airflow.providers.google.cloud.hooks.cloud_logging
#
# Licensed to the Apache Software Foundation (ASF) under one
# or more contributor license agreements. See the NOTICE file
# distributed with this work for additional information
# regarding copyright ownership. The ASF licenses this file
# to you 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
#
# http://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.
from __future__ import annotations
from collections.abc import Sequence
from typing import TYPE_CHECKING
from google.cloud.logging_v2.services.config_service_v2 import ConfigServiceV2Client
from google.cloud.logging_v2.types import (
CreateSinkRequest,
DeleteSinkRequest,
GetSinkRequest,
ListSinksRequest,
LogSink,
UpdateSinkRequest,
)
from airflow.providers.google.common.consts import CLIENT_INFO
from airflow.providers.google.common.hooks.base_google import PROVIDE_PROJECT_ID, GoogleBaseHook
if TYPE_CHECKING:
from google.protobuf.field_mask_pb2 import FieldMask
[docs]
class CloudLoggingHook(GoogleBaseHook):
"""
Hook for Google Cloud Logging Log Sinks API.
:param gcp_conn_id: The connection ID to use when fetching connection info.
:param impersonation_chain: Optional service account to impersonate.
"""
def __init__(
self,
gcp_conn_id: str = "google_cloud_default",
impersonation_chain: str | Sequence[str] | None = None,
**kwargs,
) -> None:
super().__init__(gcp_conn_id=gcp_conn_id, impersonation_chain=impersonation_chain, **kwargs)
self._client: ConfigServiceV2Client | None = None
[docs]
def get_conn(self) -> ConfigServiceV2Client:
"""Return the Google Cloud Logging Config client."""
if not self._client:
self._client = ConfigServiceV2Client(credentials=self.get_credentials(), client_info=CLIENT_INFO)
return self._client
[docs]
def get_parent(self, project_id):
return f"projects/{project_id}"
@GoogleBaseHook.fallback_to_default_project_id
[docs]
def create_sink(
self, sink: LogSink | dict, unique_writer_identity: bool = True, project_id: str = PROVIDE_PROJECT_ID
) -> LogSink:
if isinstance(sink, dict):
sink = LogSink(**sink)
request = CreateSinkRequest(
parent=self.get_parent(project_id), sink=sink, unique_writer_identity=unique_writer_identity
)
return self.get_conn().create_sink(request=request)
@GoogleBaseHook.fallback_to_default_project_id
[docs]
def get_sink(self, sink_name: str, project_id: str = PROVIDE_PROJECT_ID) -> LogSink:
request = GetSinkRequest(sink_name=f"projects/{project_id}/sinks/{sink_name}")
return self.get_conn().get_sink(request=request)
@GoogleBaseHook.fallback_to_default_project_id
[docs]
def list_sinks(self, page_size: int | None = None, project_id: str = PROVIDE_PROJECT_ID) -> list[LogSink]:
request = ListSinksRequest(parent=self.get_parent(project_id), page_size=page_size)
return list(self.get_conn().list_sinks(request=request))
@GoogleBaseHook.fallback_to_default_project_id
[docs]
def delete_sink(self, sink_name: str, project_id: str = PROVIDE_PROJECT_ID) -> None:
request = DeleteSinkRequest(sink_name=f"projects/{project_id}/sinks/{sink_name}")
self.get_conn().delete_sink(request=request)
@GoogleBaseHook.fallback_to_default_project_id
[docs]
def update_sink(
self,
sink_name: str,
sink: LogSink | dict,
unique_writer_identity: bool,
update_mask: FieldMask | dict,
project_id: str = PROVIDE_PROJECT_ID,
) -> LogSink:
if isinstance(sink, dict):
sink = LogSink(**sink)
request = UpdateSinkRequest(
sink_name=f"projects/{project_id}/sinks/{sink_name}",
sink=sink,
unique_writer_identity=unique_writer_identity,
update_mask=update_mask,
)
return self.get_conn().update_sink(request=request)