All Downloads are FREE. Search and download functionalities are using the official Maven repository.

feast.common.logging.interceptors.GrpcMessageInterceptor Maven / Gradle / Ivy

There is a newer version: 0.26.3
Show newest version
/*
 * SPDX-License-Identifier: Apache-2.0
 * Copyright 2018-2019 The Feast Authors
 *
 * 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.
 */
package feast.common.logging.interceptors;

import com.google.protobuf.Empty;
import com.google.protobuf.Message;
import feast.common.auth.config.SecurityProperties;
import feast.common.auth.config.SecurityProperties.AuthenticationProperties;
import feast.common.auth.utils.AuthUtils;
import feast.common.logging.AuditLogger;
import feast.common.logging.config.LoggingProperties;
import feast.common.logging.entry.MessageAuditLogEntry;
import io.grpc.ForwardingServerCall.SimpleForwardingServerCall;
import io.grpc.ForwardingServerCallListener.SimpleForwardingServerCallListener;
import io.grpc.Metadata;
import io.grpc.ServerCall;
import io.grpc.ServerCall.Listener;
import io.grpc.ServerCallHandler;
import io.grpc.ServerInterceptor;
import io.grpc.Status;
import java.util.Map;
import org.slf4j.event.Level;
import org.springframework.beans.factory.annotation.Autowired;
import org.springframework.lang.Nullable;
import org.springframework.security.core.Authentication;
import org.springframework.security.core.context.SecurityContextHolder;
import org.springframework.stereotype.Component;

/**
 * GrpcMessageInterceptor intercepts a GRPC calls to log handling of GRPC messages to the Audit Log.
 * Intercepts the incoming and outgoing messages logs them to the audit log, together with method
 * name and assumed authenticated identity (if authentication is enabled). NOTE:
 * GrpcMessageInterceptor assumes that all service calls are unary (ie single request/response).
 */
@Component
public class GrpcMessageInterceptor implements ServerInterceptor {
  private SecurityProperties securityProperties;
  private LoggingProperties loggingProperties;

  /**
   * Construct GrpcMessageIntercetor.
   *
   * @param loggingProperties properties used to configure logging interceptor.
   * @param securityProperties If provided, will output the subject claim specified in
   *     securityProperties as identity in {@link MessageAuditLogEntry} instead.
   */
  @Autowired
  public GrpcMessageInterceptor(
      LoggingProperties loggingProperties, @Nullable SecurityProperties securityProperties) {
    this.securityProperties = securityProperties;
    this.loggingProperties = loggingProperties;
  }

  @Override
  public  Listener interceptCall(
      ServerCall call, Metadata headers, ServerCallHandler next) {
    // Disable the message logging interceptor entirely if message logging is disabled.
    if (!loggingProperties.getAudit().getMessageLogging().isEnabled()) {
      return next.startCall(call, headers);
    }

    MessageAuditLogEntry.Builder entryBuilder = MessageAuditLogEntry.newBuilder();
    // default response/request message to empty proto in log entry.
    // request could be empty when the client closes the connection before sending a request
    // message.
    // response could be unset when the service encounters an error when processsing the service
    // call.
    entryBuilder.setRequest(Empty.newBuilder().build());
    entryBuilder.setResponse(Empty.newBuilder().build());

    // Unpack service & method name from call
    // full method name is in format ./
    String fullMethodName = call.getMethodDescriptor().getFullMethodName();
    entryBuilder.setService(
        fullMethodName.substring(fullMethodName.lastIndexOf(".") + 1, fullMethodName.indexOf("/")));
    entryBuilder.setMethod(fullMethodName.substring(fullMethodName.indexOf("/") + 1));

    // Attempt Extract current authenticated identity.
    Authentication authentication = SecurityContextHolder.getContext().getAuthentication();
    String identity = (authentication != null) ? getIdentity(authentication) : "";
    entryBuilder.setIdentity(identity);

    // Register forwarding call to intercept outgoing response and log to audit log
    call =
        new SimpleForwardingServerCall(call) {
          @Override
          public void sendMessage(RespT message) {
            // 2. Track the response & Log entry to audit logger
            super.sendMessage(message);
            entryBuilder.setResponse((Message) message);
          }

          @Override
          public void close(Status status, Metadata trailers) {
            super.close(status, trailers);
            // 3. Log the message log entry to the audit log
            Level logLevel = (status.isOk()) ? Level.INFO : Level.ERROR;
            entryBuilder.setStatusCode(status.getCode());
            AuditLogger.logMessage(logLevel, entryBuilder);
          }
        };

    ServerCall.Listener listener = next.startCall(call, headers);
    return new SimpleForwardingServerCallListener(listener) {
      @Override
      // Register listener to intercept incoming request messages and log to audit log
      public void onMessage(ReqT message) {
        super.onMessage(message);
        // 1. Track the request.
        entryBuilder.setRequest((Message) message);
      }
    };
  }

  /**
   * Extract current authenticated identity from given {@link Authentication}. Extracts subject
   * claim if specified in AuthorizationProperties, otherwise returns authentication subject.
   */
  private String getIdentity(Authentication authentication) {
    // use subject claim as identity if set in security authorization properties
    if (securityProperties != null) {
      Map options = securityProperties.getAuthentication().getOptions();
      if (options.containsKey(AuthenticationProperties.SUBJECT_CLAIM)) {
        try {
          return AuthUtils.getSubjectFromAuth(
              authentication, options.get(AuthenticationProperties.SUBJECT_CLAIM));
        } catch (IllegalStateException e) {
          // could not extract claim, revert to authenticated name.
          return authentication.getName();
        }
      }
    }
    return authentication.getName();
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy