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

io.grpc.okhttp.OkHttpFrameLogger Maven / Gradle / Ivy

/*
 * Copyright 2019 The gRPC 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
 *
 *     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.
 */

package io.grpc.okhttp;

import static com.google.common.base.Preconditions.checkNotNull;

import com.google.common.annotations.VisibleForTesting;
import io.grpc.okhttp.internal.framed.ErrorCode;
import io.grpc.okhttp.internal.framed.Header;
import io.grpc.okhttp.internal.framed.Settings;
import java.util.EnumMap;
import java.util.List;
import java.util.logging.Level;
import java.util.logging.Logger;
import okio.Buffer;
import okio.ByteString;

class OkHttpFrameLogger {
  private static final int BUFFER_LENGTH_THRESHOLD = 64;
  private final Logger logger;
  private final Level level;

  OkHttpFrameLogger(Level level, Class clazz) {
    this(level, Logger.getLogger(clazz.getName()));
  }

  @VisibleForTesting
  OkHttpFrameLogger(Level level, Logger logger) {
    this.level = checkNotNull(level, "level");
    this.logger = checkNotNull(logger, "logger");
  }

  private static String toString(Settings settings) {
    EnumMap map = new EnumMap<>(SettingParams.class);
    for (SettingParams p : SettingParams.values()) {
      // Only log set parameters.
      if (settings.isSet(p.getBit())) {
        map.put(p, settings.get(p.getBit()));
      }
    }
    return map.toString();
  }

  private static String toString(Buffer buf) {
    if (buf.size() <= BUFFER_LENGTH_THRESHOLD) {
      // Log the entire buffer.
      return buf.snapshot().hex();
    }

    // Otherwise just log the first 64 bytes.
    int length = (int) Math.min(buf.size(), BUFFER_LENGTH_THRESHOLD);
    return buf.snapshot(length).hex() + "...";
  }

  private boolean isEnabled() {
    return logger.isLoggable(level);
  }

  void logData(Direction direction, int streamId, Buffer data, int length, boolean endStream) {
    if (isEnabled()) {
      logger.log(
          level,
          direction
              + " DATA: streamId="
              + streamId
              + " endStream="
              + endStream
              + " length="
              + length
              + " bytes="
              + toString(data));
    }
  }

  void logHeaders(Direction direction, int streamId, List
headers, boolean endStream) { if (isEnabled()) { logger.log( level, direction + " HEADERS: streamId=" + streamId + " headers=" + headers + " endStream=" + endStream); } } public void logPriority( Direction direction, int streamId, int streamDependency, int weight, boolean exclusive) { if (isEnabled()) { logger.log( level, direction + " PRIORITY: streamId=" + streamId + " streamDependency=" + streamDependency + " weight=" + weight + " exclusive=" + exclusive); } } void logRstStream(Direction direction, int streamId, ErrorCode errorCode) { if (isEnabled()) { logger.log( level, direction + " RST_STREAM: streamId=" + streamId + " errorCode=" + errorCode); } } void logSettingsAck(Direction direction) { if (isEnabled()) { logger.log(level, direction + " SETTINGS: ack=true"); } } void logSettings(Direction direction, Settings settings) { if (isEnabled()) { logger.log(level, direction + " SETTINGS: ack=false settings=" + toString(settings)); } } void logPing(Direction direction, long data) { if (isEnabled()) { logger.log(level, direction + " PING: ack=false bytes=" + data); } } void logPingAck(Direction direction, long data) { if (isEnabled()) { logger.log(level, direction + " PING: ack=true bytes=" + data); } } void logPushPromise( Direction direction, int streamId, int promisedStreamId, List
headers) { if (isEnabled()) { logger.log( level, direction + " PUSH_PROMISE: streamId=" + streamId + " promisedStreamId=" + promisedStreamId + " headers=" + headers); } } void logGoAway(Direction direction, int lastStreamId, ErrorCode errorCode, ByteString debugData) { if (isEnabled()) { logger.log( level, direction + " GO_AWAY: lastStreamId=" + lastStreamId + " errorCode=" + errorCode + " length=" + debugData.size() + " bytes=" + toString(new Buffer().write(debugData))); } } void logWindowsUpdate(Direction direction, int streamId, long windowSizeIncrement) { if (isEnabled()) { logger.log( level, direction + " WINDOW_UPDATE: streamId=" + streamId + " windowSizeIncrement=" + windowSizeIncrement); } } enum Direction { INBOUND, OUTBOUND } // Note the set bits in OkHttp's Settings are different from HTTP2 Specifications. private enum SettingParams { HEADER_TABLE_SIZE(1), ENABLE_PUSH(2), MAX_CONCURRENT_STREAMS(4), MAX_FRAME_SIZE(5), MAX_HEADER_LIST_SIZE(6), INITIAL_WINDOW_SIZE(7); private final int bit; SettingParams(int bit) { this.bit = bit; } public int getBit() { return this.bit; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy