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

org.firebirdsql.testcontainers.FirebirdContainer Maven / Gradle / Ivy

Go to download

A module for Testcontainers.org to provide lightweight, throwaway instances of Firebird for JUnit tests.

There is a newer version: 1.5.0
Show newest version
package org.firebirdsql.testcontainers;

import lombok.extern.slf4j.Slf4j;
import org.testcontainers.containers.JdbcDatabaseContainer;
import org.testcontainers.utility.DockerImageName;

import javax.crypto.Cipher;
import java.security.NoSuchAlgorithmException;
import java.time.ZoneId;

@Slf4j
public class FirebirdContainer> extends JdbcDatabaseContainer {

    public static final String NAME = "firebird";
    public static final String ALTERNATE_NAME = "firebirdsql";
    public static final String IMAGE = "jacobalberty/firebird";
    static final DockerImageName DEFAULT_IMAGE_NAME = DockerImageName.parse(IMAGE);
    public static final String DEFAULT_TAG = "3.0.7";

    public static final Integer FIREBIRD_PORT = 3050;
    private static final String FIREBIRD_SYSDBA = "sysdba";
    private static final int ARC4_REQUIRED_BITS = 160;
    private static final String CONNECTION_PROPERTY_AUTH_PLUGINS = "authPlugins";

    private String databaseName = "test";
    private String username = "test";
    private String password = "test";
    private boolean enableLegacyClientAuth;
    private String timeZone = ZoneId.systemDefault().getId();
    private boolean enableWireCrypt;
    private String sysdbaPassword;

    /**
     * Creates a Firebird container with the default image ({@link #IMAGE} and {@link #DEFAULT_TAG}).
     *
     * @deprecated Use explicit image using {@link #FirebirdContainer(DockerImageName)}
     */
    @Deprecated
    public FirebirdContainer() {
        this(DEFAULT_IMAGE_NAME.withTag(DEFAULT_TAG));
    }

    /**
     * Creates a Firebird container with an image name (e.g. {@code "jacobalberty/firebird:3.0.7"}.
     *
     * @param dockerImageName Image name
     */
    public FirebirdContainer(String dockerImageName) {
        this(DockerImageName.parse(dockerImageName));
    }

    /**
     * Creates a Firebird container with a parsed image name.
     *
     * @param dockerImageName Parse image name
     */
    public FirebirdContainer(DockerImageName dockerImageName) {
        super(dockerImageName);

        dockerImageName.assertCompatibleWith(DEFAULT_IMAGE_NAME);

        addExposedPort(FIREBIRD_PORT);
    }

    @Override
    protected void configure() {
        addEnv("TZ", timeZone);
        addEnv("FIREBIRD_DATABASE", databaseName);

        if (FIREBIRD_SYSDBA.equalsIgnoreCase(username)) {
            addEnv("ISC_PASSWORD", password);
        } else {
            addEnv("FIREBIRD_USER", username);
            addEnv("FIREBIRD_PASSWORD", password);
            if (sysdbaPassword != null) {
                addEnv("ISC_PASSWORD", sysdbaPassword);
            }
        }

        if (enableLegacyClientAuth) {
            addEnv("EnableLegacyClientAuth", "true");
            if (!urlParameters.containsKey(CONNECTION_PROPERTY_AUTH_PLUGINS)) {
                // Allow legacy auth with Jaybird 4, while also allowing Srp256 and Srp
                withUrlParam(CONNECTION_PROPERTY_AUTH_PLUGINS, "Srp256,Srp,Legacy_Auth");
            }
        }

        if (enableWireCrypt) {
            addEnv("EnableWireCrypt", "true");
        } else if (!isWireEncryptionSupported()) {
            log.warn("Java Virtual Machine does not support wire protocol encryption requirements. " +
                "Downgrading to EnableWireCrypt = true. To fix this, configure the JVM with unlimited strength Cryptographic Jurisdiction Policy.");
            addEnv("EnableWireCrypt", "true");
        }
    }

    @Override
    public String getDriverClassName() {
        return "org.firebirdsql.jdbc.FBDriver";
    }

    @Override
    public String getJdbcUrl() {
        String additionalUrlParams = constructUrlParameters("?", "&");
        return "jdbc:firebirdsql://" + getHost() + ":" + getMappedPort(FIREBIRD_PORT)
                + "/" + getDatabaseName() + additionalUrlParams;
    }

    @Override
    public String getDatabaseName() {
        if (isRunning()) {
            if (isFirebird25Image()) {
                // The 2.5 images of jacobalberty/firebird require an absolute path to access the database
                // Provide this value only when the container is running
                String databasePath = getEnvMap().getOrDefault("DBPATH", "/firebird/data");
                return databasePath + "/" + databaseName;
            }
        }
        return databaseName;
    }

    private boolean isFirebird25Image() {
        DockerImageName imageName = new DockerImageName(getDockerImageName());
        return imageName.getUnversionedPart().equals(IMAGE) && imageName.getVersionPart().startsWith("2.5");
    }

    @Override
    public String getUsername() {
        return username;
    }

    @Override
    public String getPassword() {
        return password;
    }

    @Override
    protected String getTestQueryString() {
        return "select 1 from RDB$DATABASE";
    }

    @Override
    public SELF withDatabaseName(final String databaseName) {
        this.databaseName = databaseName;
        return self();
    }

    @Override
    public SELF withUsername(final String username) {
        this.username = username;
        return self();
    }

    @Override
    public SELF withPassword(final String password) {
        this.password = password;
        return self();
    }

    /**
     * Enables legacy authentication plugin ({@code legacy_auth}) and use it as the default.
     *
     * @return this container
     */
    public SELF withEnableLegacyClientAuth() {
        this.enableLegacyClientAuth = true;
        return self();
    }

    /**
     * Relax wireCrypt setting from Required to Enabled.
     *
     * @return this container
     */
    public SELF withEnableWireCrypt() {
        this.enableWireCrypt = true;
        return self();
    }

    /**
     * Set the time zone of the image, defaults to the JVM default zone.
     *
     * @param timeZone Time zone name (prefer long names like Europe/Amsterdam)
     * @return this container
     */
    public SELF withTimeZone(final String timeZone) {
        this.timeZone = timeZone;
        return self();
    }

    /**
     * Set the sysdba password.
     * 

* If {@code username} is {@code "sysdba"} (case insensitive), then {@code password} is used instead. *

* * @param sysdbaPassword Sysdba password * @return this container */ public SELF withSysdbaPassword(final String sysdbaPassword) { this.sysdbaPassword = sysdbaPassword; return self(); } @Override protected void waitUntilContainerStarted() { getWaitStrategy().waitUntilReady(this); super.waitUntilContainerStarted(); } /** * Checks if the JVM meets the Jaybird (Firebird JDBC driver) requirements for encrypted connections. *

* Specifically, this checks if the ARC4 cipher can be used with 160 bit keys. *

* * @return {@code true} if Jaybird will be able to support encrypted connections on this JVM. */ public static boolean isWireEncryptionSupported() { try { return Cipher.getMaxAllowedKeyLength("ARC4") >= ARC4_REQUIRED_BITS; } catch (NoSuchAlgorithmException e) { log.error("Cipher not found, JVM doesn't support encryption requirements", e); return false; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy