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

akka.grpc.interop.AkkaGrpcServerJava Maven / Gradle / Ivy

/*
 * Copyright (C) 2018-2023 Lightbend Inc. 
 */

package akka.grpc.interop;

import akka.actor.ActorSystem;
import akka.http.javadsl.Http;
import akka.http.javadsl.HttpsConnectionContext;
import akka.http.javadsl.ServerBinding;
import akka.http.javadsl.model.HttpRequest;
import akka.http.javadsl.model.HttpResponse;
import akka.http.javadsl.model.StatusCodes;
import akka.http.javadsl.settings.ServerSettings;
import akka.japi.function.Function;
import akka.stream.Materializer;
import akka.stream.SystemMaterializer;
import akka.util.ByteString;
import com.typesafe.config.ConfigFactory;
import io.grpc.internal.testing.TestUtils;
import io.grpc.testing.integration.TestService;
import scala.Function2;
import scala.Tuple2;

import javax.net.ssl.KeyManagerFactory;
import javax.net.ssl.SSLContext;
import java.io.FileInputStream;
import java.io.InputStream;
import java.nio.file.Files;
import java.nio.file.Paths;
import java.security.KeyFactory;
import java.security.KeyStore;
import java.security.PrivateKey;
import java.security.SecureRandom;
import java.security.cert.Certificate;
import java.security.cert.CertificateFactory;
import java.security.spec.PKCS8EncodedKeySpec;
import java.util.Arrays;
import java.util.Iterator;
import java.util.concurrent.CompletableFuture;
import java.util.concurrent.CompletionStage;

/**
 * Glue code to start a gRPC server based on the akka-grpc Java API to test against
 */
public class AkkaGrpcServerJava extends GrpcServer> {
  private final Function2>> handlerFactory;

  public AkkaGrpcServerJava(Function2>> handlerFactory) {
    this.handlerFactory = handlerFactory;
  }

  public Tuple2 start(String[] args) throws Exception {
    ActorSystem sys = ActorSystem.create(
      "akka-grpc-server-java",
      ConfigFactory.parseString("akka.http.server.enable-http2 = on"));
    Materializer mat = SystemMaterializer.get(sys).materializer();

    Function> testService = handlerFactory.apply(mat, sys);

    Function> handler = req -> {
      Iterator segmentIterator = req.getUri().pathSegments().iterator();
      if (segmentIterator.hasNext()) {
        if (segmentIterator.next().equals(TestService.name)) {
          return testService.apply(req);
        } else {
          return CompletableFuture.completedFuture(HttpResponse.create().withStatus(StatusCodes.NOT_FOUND));
        }
      } else {
        return CompletableFuture.completedFuture(HttpResponse.create().withStatus(StatusCodes.NOT_FOUND));
      }
    };
    ServerSettings serverSettings = ServerSettings.create(sys);

    CompletionStage binding;
    if (Arrays.asList(args).contains("--use_tls=false")) {
      binding = Http.get(sys).newServerAt("127.0.0.1", 0)
              .withMaterializer(mat)
              .withSettings(serverSettings)
              .bind(handler);
    } else {
      binding = Http.get(sys).newServerAt("127.0.0.1", 0)
              .withMaterializer(mat)
              .withSettings(serverSettings)
              .enableHttps(serverHttpContext())
              .bind(handler);
    }

    ServerBinding serverBinding = binding.toCompletableFuture().get();
    return new Tuple2<>(sys, serverBinding);
  }

  @Override
  public int getPort(Tuple2 binding) {
    return binding._2.localAddress().getPort();
  }

  public void stop(Tuple2 binding) throws Exception {
    binding._2.unbind().toCompletableFuture().get();
    binding._1.terminate();
    binding._1.getWhenTerminated().toCompletableFuture().get();
  }

  private HttpsConnectionContext serverHttpContext() throws Exception {
    String keyEncoded = new String(Files.readAllBytes(Paths.get(TestUtils.loadCert("server1.key").getAbsolutePath())), "UTF-8")
      .replace("-----BEGIN PRIVATE KEY-----\n", "")
      .replace("-----END PRIVATE KEY-----\n", "")
      .replace("\n", "");

    byte[] decodedKey = ByteString.fromString(keyEncoded).decodeBase64().toArray();

    PKCS8EncodedKeySpec spec = new PKCS8EncodedKeySpec(decodedKey);

    KeyFactory kf = KeyFactory.getInstance("RSA");
    PrivateKey privateKey = kf.generatePrivate(spec);

    CertificateFactory fact = CertificateFactory.getInstance("X.509");
    InputStream is = new FileInputStream(TestUtils.loadCert("server1.pem"));
    Certificate cer = fact.generateCertificate(is);

    KeyStore ks = KeyStore.getInstance("PKCS12");
    ks.load(null);
    ks.setKeyEntry("private", privateKey, new char[]{}, new Certificate[] { cer });

    KeyManagerFactory keyManagerFactory = KeyManagerFactory.getInstance("SunX509");
    keyManagerFactory.init(ks, null);

    SSLContext context = SSLContext.getInstance("TLS");
    context.init(keyManagerFactory.getKeyManagers(), null, new SecureRandom());

    return HttpsConnectionContext.httpsServer(context);
  }

}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy