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

io.grpc.testing.integration.AltsHandshakerTestService Maven / Gradle / Ivy

There is a newer version: 1.68.0
Show newest version
/*
 * Copyright 2021 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.testing.integration;

import static com.google.common.base.Preconditions.checkState;
import static io.grpc.alts.internal.HandshakerReq.ReqOneofCase.CLIENT_START;
import static io.grpc.alts.internal.HandshakerReq.ReqOneofCase.NEXT;
import static io.grpc.alts.internal.HandshakerReq.ReqOneofCase.SERVER_START;

import com.google.protobuf.ByteString;
import io.grpc.alts.internal.HandshakerReq;
import io.grpc.alts.internal.HandshakerResp;
import io.grpc.alts.internal.HandshakerResult;
import io.grpc.alts.internal.HandshakerServiceGrpc.HandshakerServiceImplBase;
import io.grpc.alts.internal.Identity;
import io.grpc.alts.internal.RpcProtocolVersions;
import io.grpc.alts.internal.RpcProtocolVersions.Version;
import io.grpc.stub.StreamObserver;
import java.util.Random;
import java.util.logging.Level;
import java.util.logging.Logger;

/**
 * A fake HandshakeService for ALTS integration testing in non-gcp environments.
 * */
public class AltsHandshakerTestService extends HandshakerServiceImplBase {
  private static final Logger log = Logger.getLogger(AltsHandshakerTestService.class.getName());

  private final Random random = new Random();
  private static final int FIXED_LENGTH_OUTPUT = 16;
  private final ByteString fakeOutput = data(FIXED_LENGTH_OUTPUT);
  private final ByteString secret = data(128);
  private State expectState = State.CLIENT_INIT;

  @Override
  public StreamObserver doHandshake(
      final StreamObserver responseObserver) {
    return new StreamObserver() {
      @Override
      public void onNext(HandshakerReq value) {
        log.log(Level.FINE, "request received: " + value);
        synchronized (AltsHandshakerTestService.this) {
          switch (expectState) {
            case CLIENT_INIT:
              checkState(CLIENT_START.equals(value.getReqOneofCase()));
              HandshakerResp initClient = HandshakerResp.newBuilder()
                  .setOutFrames(fakeOutput)
                  .build();
              log.log(Level.FINE, "init client response " + initClient);
              responseObserver.onNext(initClient);
              expectState = State.SERVER_INIT;
              break;
            case SERVER_INIT:
              checkState(SERVER_START.equals(value.getReqOneofCase()));
              HandshakerResp initServer = HandshakerResp.newBuilder()
                  .setBytesConsumed(FIXED_LENGTH_OUTPUT)
                  .setOutFrames(fakeOutput)
                  .build();
              log.log(Level.FINE, "init server response" + initServer);
              responseObserver.onNext(initServer);
              expectState = State.CLIENT_FINISH;
              break;
            case CLIENT_FINISH:
              checkState(NEXT.equals(value.getReqOneofCase()));
              HandshakerResp resp = HandshakerResp.newBuilder()
                  .setResult(getResult())
                  .setBytesConsumed(FIXED_LENGTH_OUTPUT)
                  .setOutFrames(fakeOutput)
                  .build();
              log.log(Level.FINE, "client finished response " + resp);
              responseObserver.onNext(resp);
              expectState = State.SERVER_FINISH;
              break;
            case SERVER_FINISH:
              resp = HandshakerResp.newBuilder()
                  .setResult(getResult())
                  .setBytesConsumed(FIXED_LENGTH_OUTPUT)
                  .build();
              log.log(Level.FINE, "server finished response " + resp);
              responseObserver.onNext(resp);
              expectState = State.CLIENT_INIT;
              break;
            default:
              throw new RuntimeException("unknown state");
          }
        }
      }

      @Override
      public void onError(Throwable t) {
        log.log(Level.INFO, "onError " + t);
      }

      @Override
      public void onCompleted() {
        responseObserver.onCompleted();
      }
    };
  }

  private HandshakerResult getResult() {
    return HandshakerResult.newBuilder().setApplicationProtocol("grpc")
        .setRecordProtocol("ALTSRP_GCM_AES128_REKEY")
        .setKeyData(secret)
        .setMaxFrameSize(131072)
        .setPeerIdentity(Identity.newBuilder()
            .setServiceAccount("[email protected]")
            .build())
        .setPeerRpcVersions(RpcProtocolVersions.newBuilder()
            .setMaxRpcVersion(Version.newBuilder()
                .setMajor(2).setMinor(1)
                .build())
            .setMinRpcVersion(Version.newBuilder()
                .setMajor(2).setMinor(1)
                .build())
            .build())
        .build();
  }

  private ByteString data(int len) {
    byte[] k = new byte[len];
    random.nextBytes(k);
    return ByteString.copyFrom(k);
  }

  private enum State {
    CLIENT_INIT,
    SERVER_INIT,
    CLIENT_FINISH,
    SERVER_FINISH
  }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy