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

io.trino.tests.product.launcher.env.common.Standard Maven / Gradle / Ivy

There is a newer version: 451
Show newest version
/*
 * 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.trino.tests.product.launcher.env.common;

import com.google.common.collect.ImmutableMap;
import com.google.inject.Inject;
import io.airlift.log.Logger;
import io.trino.tests.product.launcher.docker.DockerFiles;
import io.trino.tests.product.launcher.env.Debug;
import io.trino.tests.product.launcher.env.DockerContainer;
import io.trino.tests.product.launcher.env.Environment;
import io.trino.tests.product.launcher.env.EnvironmentConfig;
import io.trino.tests.product.launcher.env.EnvironmentContainers;
import io.trino.tests.product.launcher.env.ServerPackage;
import io.trino.tests.product.launcher.env.Tracing;
import io.trino.tests.product.launcher.env.jdk.JdkProvider;
import io.trino.tests.product.launcher.testcontainers.PortBinder;
import org.testcontainers.containers.startupcheck.IsRunningStartupCheckStrategy;
import org.testcontainers.containers.wait.strategy.WaitAllStrategy;

import java.io.File;
import java.io.IOException;
import java.io.UncheckedIOException;
import java.net.URI;
import java.nio.file.Files;
import java.nio.file.Path;
import java.nio.file.attribute.FileAttribute;
import java.nio.file.attribute.PosixFilePermission;
import java.nio.file.attribute.PosixFilePermissions;
import java.time.Duration;
import java.util.Set;
import java.util.function.Function;
import java.util.regex.Matcher;
import java.util.regex.Pattern;

import static com.google.common.base.Preconditions.checkArgument;
import static com.google.common.net.UrlEscapers.urlFragmentEscaper;
import static io.trino.tests.product.launcher.env.EnvironmentContainers.COORDINATOR;
import static io.trino.tests.product.launcher.env.EnvironmentContainers.OPENTRACING_COLLECTOR;
import static io.trino.tests.product.launcher.env.EnvironmentContainers.TESTS;
import static io.trino.tests.product.launcher.env.EnvironmentContainers.WORKER;
import static io.trino.tests.product.launcher.env.EnvironmentContainers.WORKER_NTH;
import static io.trino.tests.product.launcher.testcontainers.PortBinder.unsafelyExposePort;
import static java.lang.Integer.parseInt;
import static java.lang.String.format;
import static java.nio.charset.StandardCharsets.UTF_8;
import static java.util.Objects.requireNonNull;
import static org.testcontainers.containers.BindMode.READ_ONLY;
import static org.testcontainers.containers.wait.strategy.Wait.forHealthcheck;
import static org.testcontainers.containers.wait.strategy.Wait.forLogMessage;
import static org.testcontainers.utility.MountableFile.forHostPath;

public final class Standard
        implements EnvironmentExtender
{
    private static final Logger log = Logger.get(Standard.class);
    private static final int COLLECTOR_UI_PORT = 16686;
    private static final int COLLECTOR_GRPC_PORT = 4317;

    public static final String CONTAINER_HEALTH_D = "/etc/health.d/";
    public static final String CONTAINER_CONF_ROOT = "/docker/presto-product-tests/";
    public static final String CONTAINER_TRINO_ETC = CONTAINER_CONF_ROOT + "conf/presto/etc";
    public static final String CONTAINER_TRINO_JVM_CONFIG = CONTAINER_TRINO_ETC + "/jvm.config";
    public static final String CONTAINER_TRINO_ACCESS_CONTROL_PROPERTIES = CONTAINER_TRINO_ETC + "/access-control.properties";
    public static final String CONTAINER_TRINO_CONFIG_PROPERTIES = CONTAINER_TRINO_ETC + "/config.properties";
    /**
     * @deprecated please use {@link EnvironmentContainers#configureTempto} instead.
     */
    public static final String CONTAINER_TEMPTO_PROFILE_CONFIG = "/docker/presto-product-tests/conf/tempto/tempto-configuration-profile-config-file.yaml";

    private final DockerFiles dockerFiles;
    private final PortBinder portBinder;

    private final String imagesVersion;
    private final JdkProvider jdkProvider;
    private final File serverPackage;
    private final boolean debug;
    private final boolean tracing;

    @Inject
    public Standard(
            DockerFiles dockerFiles,
            PortBinder portBinder,
            EnvironmentConfig environmentConfig,
            @ServerPackage File serverPackage,
            JdkProvider jdkProvider,
            @Debug boolean debug,
            @Tracing boolean tracing)
    {
        this.dockerFiles = requireNonNull(dockerFiles, "dockerFiles is null");
        this.portBinder = requireNonNull(portBinder, "portBinder is null");
        this.imagesVersion = environmentConfig.getImagesVersion();
        this.jdkProvider = requireNonNull(jdkProvider, "jdkProvider is null");
        this.serverPackage = requireNonNull(serverPackage, "serverPackage is null");
        this.debug = debug;
        this.tracing = tracing;
        checkArgument(serverPackage.getName().endsWith(".tar.gz"), "Currently only server .tar.gz package is supported");
    }

    @Override
    public void extendEnvironment(Environment.Builder builder)
    {
        builder.addStartupLogs(() -> "Trino running with JDK: " + jdkProvider.getDescription());
        if (tracing) {
            DockerContainer tracingContainer = createTracingCollector();
            builder.addContainer(tracingContainer);
            builder.addLogsTransformer(addTracingLink(tracingContainer));
            builder.addStartupLogs(() -> "Tracing collector available at http://localhost:" + tracingContainer.getMappedPort(COLLECTOR_UI_PORT));
        }

        builder.addContainers(createTrinoCoordinator(), createTestsContainer());
        // default catalogs copied from /docker/presto-product-tests
        builder.addConnector("blackhole");
        builder.addConnector("jmx");
        builder.addConnector("system");
        builder.addConnector("tpch");
    }

    private Function addTracingLink(DockerContainer tracingContainer)
    {
        Pattern pattern = Pattern.compile(".*TIMELINE: Query ([a-z0-9_]+) ::.*");

        return line -> {
            Matcher matcher = pattern.matcher(line);
            if (matcher.matches()) {
                String queryId = matcher.group(1);
                String query = "{\"trino.query_id\": \"%s\"}".formatted(queryId);
                URI searchLink = URI.create("http://localhost:%d/search?operation=query&service=trino&tags=%s".formatted(
                        tracingContainer.getMappedPort(COLLECTOR_UI_PORT),
                        urlFragmentEscaper().escape(query)));
                return line + " :: TRACING :: " + searchLink;
            }
            return line;
        };
    }

    private DockerContainer createTracingCollector()
    {
        DockerContainer container = new DockerContainer("jaegertracing/all-in-one:latest", OPENTRACING_COLLECTOR)
                .withEnv(ImmutableMap.of(
                        "COLLECTOR_OTLP_ENABLED", "true",
                        "SPAN_STORAGE_TYPE", "badger",
                        "GOMAXPROCS", "2"))
                .withNetworkAliases(OPENTRACING_COLLECTOR + ".docker.cluster")
                .withCreateContainerCmdModifier(createContainerCmd -> {
                    createContainerCmd.getHostConfig().withMemory(2 * 1024 * 1024 * 1024L); // 2GB limit
                });

        portBinder.exposePort(container, COLLECTOR_UI_PORT); // UI port
        portBinder.exposePort(container, COLLECTOR_GRPC_PORT);  // OpenTelemetry over gRPC
        return container;
    }

    @SuppressWarnings("resource")
    private DockerContainer createTrinoCoordinator()
    {
        DockerContainer container =
                createTrinoContainer(dockerFiles, serverPackage, jdkProvider, debug, tracing, "ghcr.io/trinodb/testing/centos7-oj17:" + imagesVersion, COORDINATOR)
                        .withCopyFileToContainer(forHostPath(dockerFiles.getDockerFilesHostPath("common/standard/access-control.properties")), CONTAINER_TRINO_ACCESS_CONTROL_PROPERTIES)
                        .withCopyFileToContainer(forHostPath(dockerFiles.getDockerFilesHostPath("common/standard/config.properties")), CONTAINER_TRINO_CONFIG_PROPERTIES);

        portBinder.exposePort(container, 8080); // Trino default port
        return container;
    }

    @SuppressWarnings("resource")
    private DockerContainer createTestsContainer()
    {
        return new DockerContainer("ghcr.io/trinodb/testing/centos7-oj17:" + imagesVersion, TESTS)
                .withCopyFileToContainer(forHostPath(dockerFiles.getDockerFilesHostPath()), "/docker/presto-product-tests")
                .withCommand("bash", "-xeuc", "echo 'No command provided' >&2; exit 69")
                .waitingFor(new WaitAllStrategy()) // don't wait
                .withStartupCheckStrategy(new IsRunningStartupCheckStrategy());
    }

    @SuppressWarnings("resource")
    public static DockerContainer createTrinoContainer(DockerFiles dockerFiles, File serverPackage, JdkProvider jdkProvider, boolean debug, boolean tracing, String dockerImageName, String logicalName)
    {
        DockerContainer container = new DockerContainer(dockerImageName, logicalName)
                .withNetworkAliases(logicalName + ".docker.cluster")
                .withExposedLogPaths("/var/trino/var/log", "/var/log/container-health.log")
                .withCopyFileToContainer(forHostPath(dockerFiles.getDockerFilesHostPath()), "/docker/presto-product-tests")
                .withCopyFileToContainer(forHostPath(dockerFiles.getDockerFilesHostPath("conf/presto/etc/jvm.config")), CONTAINER_TRINO_JVM_CONFIG)
                .withCopyFileToContainer(forHostPath(dockerFiles.getDockerFilesHostPath("health-checks/trino-health-check.sh")), CONTAINER_HEALTH_D + "trino-health-check.sh")
                // the server package is hundreds MB and file system bind is much more efficient
                .withFileSystemBind(serverPackage.getPath(), "/docker/presto-server.tar.gz", READ_ONLY)
                .withEnv("JAVA_HOME", jdkProvider.getJavaHome())
                .withCommand("/docker/presto-product-tests/run-presto.sh")
                .withStartupCheckStrategy(new IsRunningStartupCheckStrategy())
                .waitingForAll(forLogMessage(".*======== SERVER STARTED ========.*", 1), forHealthcheck())
                .withStartupTimeout(Duration.ofMinutes(5));
        if (debug) {
            enableTrinoJavaDebugger(container);
            enableTrinoJmxRmi(container);
        }
        else {
            container.withHealthCheck(dockerFiles.getDockerFilesHostPath("health-checks/health.sh"));
        }

        if (tracing) {
            enableTrinoTracing(container);
        }

        return jdkProvider.applyTo(container);
    }

    private static void enableTrinoJavaDebugger(DockerContainer dockerContainer)
    {
        String logicalName = dockerContainer.getLogicalName();

        int debugPort;
        if (logicalName.equals(COORDINATOR)) {
            debugPort = 5005;
        }
        else if (logicalName.equals(WORKER)) {
            debugPort = 5009;
        }
        else if (logicalName.startsWith(WORKER_NTH)) {
            int workerNumber = parseInt(logicalName.substring(WORKER_NTH.length()));
            debugPort = 5008 + workerNumber;
        }
        else {
            throw new IllegalStateException("Cannot enable Java debugger for: " + logicalName);
        }

        enableTrinoJavaDebugger(dockerContainer, logicalName, debugPort);
    }

    private static void enableTrinoJavaDebugger(DockerContainer container, String containerName, int debugPort)
    {
        log.info("Enabling Java debugger for container: '%s' on port %d", containerName, debugPort);

        try {
            FileAttribute> rwx = PosixFilePermissions.asFileAttribute(PosixFilePermissions.fromString("rwxrwxrwx"));
            Path script = Files.createTempFile("enable-java-debugger", ".sh", rwx);
            script.toFile().deleteOnExit();
            Files.writeString(
                    script,
                    format(
                            "#!/bin/bash\n" +
                                    "printf '%%s\\n' '%s' >> '%s'\n",
                            "-agentlib:jdwp=transport=dt_socket,server=y,suspend=n,address=0.0.0.0:" + debugPort,
                            CONTAINER_TRINO_JVM_CONFIG),
                    UTF_8);
            container.withCopyFileToContainer(forHostPath(script), "/docker/presto-init.d/enable-java-debugger.sh");

            // expose debug port unconditionally when debug is enabled
            unsafelyExposePort(container, debugPort);
        }
        catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

    private static void enableTrinoTracing(DockerContainer container)
    {
        log.info("Enabling tracing collection for container %s", container.getLogicalName());

        try {
            FileAttribute> rwx = PosixFilePermissions.asFileAttribute(PosixFilePermissions.fromString("rwxrwxrwx"));
            Path script = Files.createTempFile("enable-tracing", ".sh", rwx);
            script.toFile().deleteOnExit();
            Files.writeString(
                    script,
                    """
                    #!/bin/bash
                    echo 'tracing.enabled=true' >> '%1$s'
                    echo 'tracing.exporter.endpoint=http://opentracing-collector.docker.cluster:%2$d' >> '%1$s'
                    """.formatted(CONTAINER_TRINO_CONFIG_PROPERTIES, COLLECTOR_GRPC_PORT),
                    UTF_8);
            container.withCopyFileToContainer(forHostPath(script), "/docker/presto-init.d/enable-tracing.sh");
        }
        catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }

    private static void enableTrinoJmxRmi(DockerContainer dockerContainer)
    {
        String logicalName = dockerContainer.getLogicalName();

        int jmxPort;
        if (logicalName.equals(COORDINATOR)) {
            jmxPort = 6005;
        }
        else if (logicalName.equals(WORKER)) {
            jmxPort = 6009;
        }
        else if (logicalName.startsWith(WORKER_NTH)) {
            int workerNumber = parseInt(logicalName.substring(WORKER_NTH.length()));
            jmxPort = 6008 + workerNumber;
        }
        else {
            throw new IllegalStateException("Cannot enable Java JMX RMI for: " + logicalName);
        }

        enableTrinoJmxRmi(dockerContainer, logicalName, jmxPort);
    }

    private static void enableTrinoJmxRmi(DockerContainer container, String containerName, int jmxPort)
    {
        log.info("Enabling Java JMX RMI for container: '%s' on port %d", containerName, jmxPort);

        try {
            FileAttribute> rwx = PosixFilePermissions.asFileAttribute(PosixFilePermissions.fromString("rwxrwxrwx"));
            Path script = Files.createTempFile("enable-java-jmx-rmi", ".sh", rwx);
            script.toFile().deleteOnExit();
            Files.writeString(
                    script,
                        """
                                #!/bin/bash
                                printf '%%s\\n' '-Dcom.sun.management.jmxremote=true' >> '%2$s'
                                printf '%%s\\n' '-Dcom.sun.management.jmxremote.port=%1$s' >> '%2$s'
                                printf '%%s\\n' '-Dcom.sun.management.jmxremote.rmi.port=%1$s' >> '%2$s'
                                printf '%%s\\n' '-Dcom.sun.management.jmxremote.authenticate=false' >> '%2$s'
                                printf '%%s\\n' '-Djava.rmi.server.hostname=0.0.0.0' >> '%2$s'
                                printf '%%s\\n' '-Dcom.sun.management.jmxremote.ssl=false' >> '%2$s'
                                printf '%%s\\n' '-XX:FlightRecorderOptions=stackdepth=256' >> '%2$s'
                                """.formatted(Integer.toString(jmxPort), CONTAINER_TRINO_JVM_CONFIG),
                    UTF_8);
            container.withCopyFileToContainer(forHostPath(script), "/docker/presto-init.d/enable-java-jmx-rmi.sh");

            // expose JMX port unconditionally when debug is enabled
            unsafelyExposePort(container, jmxPort);
        }
        catch (IOException e) {
            throw new UncheckedIOException(e);
        }
    }
}




© 2015 - 2024 Weber Informatics LLC | Privacy Policy