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

io.helidon.dbclient.health.DbClientHealthCheck Maven / Gradle / Ivy

/*
 * Copyright (c) 2019, 2023 Oracle and/or its affiliates.
 *
 * 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.helidon.dbclient.health;

import java.time.Duration;
import java.util.HashMap;
import java.util.Map;
import java.util.concurrent.TimeUnit;

import io.helidon.common.config.Config;
import io.helidon.dbclient.DbClient;
import io.helidon.health.HealthCheck;
import io.helidon.health.HealthCheckResponse;

/**
 * Database health check.
 */
public abstract class DbClientHealthCheck implements HealthCheck {

    /* Local logger instance. */
    private static final System.Logger LOGGER = System.getLogger(DbClientHealthCheck.class.getName());

    /* Default hHealth check timeout in seconds (to wait for statement execution response). */
    private static final int DEFAULT_TIMEOUT_SECONDS = 10;
    private final Duration timeout;

    /**
     * Create a health check with configured settings for the database.
     * This health check will execute health check as defined in provided {@link Config} node.
     *
     * @param dbClient database client used to execute health check statement
     * @param config   {@link Config} node with health check configuration
     * @return health check that can be used with health services
     */
    public static DbClientHealthCheck create(DbClient dbClient, Config config) {
        return builder(dbClient).config(config).build();
    }

    /**
     * A fluent API builder to create a fully customized database health check.
     *
     * @param dbClient database
     * @return a new builder
     */
    public static Builder builder(DbClient dbClient) {
        return new Builder(dbClient);
    }

    private final DbClient dbClient;
    private final String name;

    private DbClientHealthCheck(Builder builder) {
        this.dbClient = builder.database;
        this.name = builder.name();
        this.timeout = Duration.of(builder.timeoutDuration, builder.timeoutUnit.toChronoUnit());
    }

    /**
     * Execute the ping statement.
     */
    protected abstract void execPing();

    @Override
    public HealthCheckResponse call() {
        HealthCheckResponse.Builder builder = HealthCheckResponse.builder();
        try {
            execPing();
            builder.status(HealthCheckResponse.Status.UP);
        } catch (Throwable e) {
            builder.status(HealthCheckResponse.Status.DOWN);
            builder.detail("ErrorMessage", e.getMessage());
            builder.detail("ErrorClass", e.getClass().getName());
            LOGGER.log(System.Logger.Level.TRACE,
                    () -> String.format("Database %s is not responding: %s", dbClient.dbType(), e.getMessage()), e);
        }

        return builder.build();
    }

    protected DbClient dbClient() {
        return dbClient;
    }

    // Getters for local usage and jUnit tests

    @Override
    public String name() {
        return name;
    }

    Duration timeout() {
        return timeout;
    }

    /**
     * Database health check which calls DBClient's {@code namedDml} method.
     */
    static final class DbClientHealthCheckAsNamedDml extends DbClientHealthCheck {

        /* Name of the statement. */
        private final String statementName;

        private DbClientHealthCheckAsNamedDml(Builder builder) {
            super(builder);
            this.statementName = builder.statementName();
            LOGGER.log(System.Logger.Level.TRACE, "Created an instance of DbClientHealthCheckAsNamedDml");
        }

        @Override
        protected void execPing() {
            dbClient().execute().namedDml(statementName);
        }

        // Getter for jUnit tests
        String statementName() {
            return statementName;
        }

    }

    /**
     * Database health check which calls DBClient's {@code dml} method.
     */
    static final class DbClientHealthCheckAsDml extends DbClientHealthCheck {

        /* Custom statement. */
        private final String statement;

        private DbClientHealthCheckAsDml(Builder builder) {
            super(builder);
            this.statement = builder.statement();
            LOGGER.log(System.Logger.Level.TRACE, "Created an instance of DbClientHealthCheckAsDml");
        }

        @Override
        protected void execPing() {
            dbClient().execute().dml(statement);
        }

        // Getter for jUnit tests
        String statement() {
            return statement;
        }

    }

    /**
     * Database health check which calls DBClient's {@code namedQuery} method.
     */
    static final class DbClientHealthCheckAsNamedQuery extends DbClientHealthCheck {

        /* Name of the statement. */
        private final String statementName;

        private DbClientHealthCheckAsNamedQuery(Builder builder) {
            super(builder);
            this.statementName = builder.statementName();
            LOGGER.log(System.Logger.Level.TRACE, "Created an instance of DbClientHealthCheckAsNamedQuery");
        }

        @Override
        protected void execPing() {
            dbClient().execute().namedQuery(statementName).forEach(it -> {});
        }

        // Getter for jUnit tests
        String statementName() {
            return statementName;
        }

    }

    /**
     * Database health check which calls DBClient's {@code query} method.
     */
    static final class DbClientHealthCheckAsQuery extends DbClientHealthCheck {

        /* Custom statement. */
        private final String statement;

        private DbClientHealthCheckAsQuery(Builder builder) {
            super(builder);
            this.statement = builder.statement();
            LOGGER.log(System.Logger.Level.TRACE, "Created an instance of DbClientHealthCheckAsQuery");
        }

        @Override
        protected void execPing() {
            dbClient().execute().query(statement).forEach(it -> {});
        }

        // Getter for jUnit tests
        String statement() {
            return statement;
        }

    }

    /**
     * Fluent API builder for {@link DbClientHealthCheck}.
     * Default health check setup will call named DML statement with name {@code ping}.
     * This named DML statement shall be configured in {@code statements} section
     * of the DBClient configuration file.
     */
    public static final class Builder implements io.helidon.common.Builder {

        // Name of Config node with custom health check name.
        private static final String CONFIG_NAME = "name";
        // Name of Config node with statement type.
        private static final String CONFIG_TYPE = "type";
        // Name of Config node with statement to be executed in database.
        private static final String CONFIG_STMT = "statement";
        // Name of Config node with statement name to be executed in database.
        private static final String CONFIG_STMT_NAME = "statementName";
        // Name of Config node with timeout duration value.
        private static final String CONFIG_TIMEOUT_DURATION = "timeout";
        // Name of Config node with timeout units.
        private static final String CONFIG_TIMEOUT_UNIT = "timeUnit";

        // Size of TimeUnit enum.
        private static final int TIME_UNIT_SIZE = TimeUnit.values().length;
        // Configuration strings resolver based on TimeUnit enum content.
        private static final Map NAME_TO_TIME_UNIT = new HashMap<>(TIME_UNIT_SIZE);

        // Initialize TimeUnit resolver content.
        static {
            for (TimeUnit value : TimeUnit.values()) {
                NAME_TO_TIME_UNIT.put(value.name().toLowerCase(), value);
            }
        }

        // Helidon database client.
        private final DbClient database;
        // Health check name.
        private String name;
        // Health check timeout length (to wait for statement execution response).
        private long timeoutDuration;
        // Health check timeout units (to wait for statement execution response).
        private TimeUnit timeoutUnit;

        // Those two boolean variables define 4 ways of query execution:
        //
        // +-------------------+----------+--------+------------+-------+
        // | DbExecute         | namedDML | dml    | namedQuery | query |
        // +-------------------+----------+--------+------------+-------+
        // | isDML             | true     | true   | false      | false |
        // | isNamedStatement  | true     | false | true       | false |
        // +-------------------+----------+--------+------------+-------+
        // The best performance optimized solution seems to be polymorphism for part of check method.

        // Health check statement is DML when {@code true} and query when {@code false}.
        private boolean isDML;
        // Whether to use named statement or statement passed as an argument.
        // Holds information about latest statement definition.
        private boolean isNamedStatement;

        // Name of the statement.
        private String statementName;
        // Custom statement string.
        private String statement;

        private Builder(DbClient database) {
            this.database = database;
            this.name = database.dbType();
            this.timeoutDuration = DEFAULT_TIMEOUT_SECONDS;
            this.timeoutUnit = TimeUnit.SECONDS;
            this.isDML = false;
            this.isNamedStatement = true;
            this.statementName = null;
            this.statement = null;
        }

        // Defines polymorphism for ping statement execution based on isDML and isNamedStatement values.
        // Default health check is to call DBClient's ping method (when no customization is set).
        @Override
        public DbClientHealthCheck build() {
            // Statement defined as name in statements config node
            if (isNamedStatement) {
                // Statement null check is required just here because isNamedStatement is set to true by default.
                if (statementName == null) {
                    throw new HealthCheckBuilderException(
                            "No statement name or statement custom string was defined.");
                }
                return isDML
                        ? new DbClientHealthCheckAsNamedDml(this)
                        : new DbClientHealthCheckAsNamedQuery(this);
                // Statement defined as custom string
            } else {
                return isDML
                        ? new DbClientHealthCheckAsDml(this)
                        : new DbClientHealthCheckAsQuery(this);
            }
        }

        /**
         * Customized name of the health check.
         * Default uses {@link io.helidon.dbclient.DbClient#dbType()}.
         *
         * @param name name of the health check
         * @return updated builder instance
         */
        public Builder name(String name) {
            this.name = name;
            return this;
        }

        /**
         * Set health check parameters using {@link Config} node.
         * 

* Configuration node expected content:

    *
  • name: custom health check name.
  • *
  • type: statement type (default value is {@code query}).
  • *
  • statement: statement to be executed in database.
  • *
  • statementName: name of statement to be executed in database. Statement * with given name must exist in statements {@link Config} node.
  • *
  • timeout: timeout value.
  • *
  • timeUnit: units of timeout value (default value is {@code seconds}).
* Only one of statement and statementName parameters is allowed. * * @param config {@link Config} instance with health check parameters * @return updated builder instance */ public Builder config(Config config) { config.get(CONFIG_NAME) .as(String.class) .ifPresent(checkName -> this.name = checkName); // Lambda expressions require final variables but code needs it's value to be changed. // Statement definition: // - false: not set // - true: statement string already set final boolean[] stmtDef = new boolean[]{false}; config.get(CONFIG_STMT) .as(String.class) .ifPresent(stmt -> { stmtDef[0] = true; // Statement definition as statement string this.isNamedStatement = false; this.statement = stmt; }); config.get(CONFIG_STMT_NAME) .as(String.class) .ifPresent(stmtName -> { if (stmtDef[0]) { // Collision with statement definition as statement string throw new HealthCheckBuilderException( String.format( "Duplicate statement definition in health check config: " + "statement \"%s\" and statement name %s", this.statement, stmtName)); } this.isNamedStatement = true; this.statementName = stmtName; }); config.get(CONFIG_TYPE) .as(String.class) .ifPresent(type -> { HealthCheckStMtType stmtType = HealthCheckStMtType.nameToType(type); if (stmtType == null) { throw new HealthCheckBuilderException( String.format("Unknown statement type: %s", type)); } this.isDML = switch (stmtType) { case DML -> true; case QUERY -> false; }; }); try { config.get(CONFIG_TIMEOUT_DURATION) .as(Long.class) .ifPresent(duration -> this.timeoutDuration = duration); // Number conversion may fail } catch (Throwable t) { throw new HealthCheckBuilderException( String.format("Could not set timeout duration: %s", t.getMessage()), t); } config.get(CONFIG_TIMEOUT_UNIT) .as(String.class) .ifPresent(tmUnit -> { final TimeUnit timeUnit = NAME_TO_TIME_UNIT.get(tmUnit.toLowerCase()); if (timeUnit == null) { throw new HealthCheckBuilderException( String.format("Unknown timeout unit name: %s", tmUnit)); } this.timeoutUnit = timeUnit; }); return this; } /** * Set health check statement type to query. * Allows to override value set in {@link Config} node. * Default health check statement type is query. * * @return updated builder instance */ public Builder query() { this.isDML = false; return this; } /** * Set health check statement type to DML. * Allows to override value set in {@link Config} node. * Default health check statement type is query. * * @return updated builder instance */ public Builder dml() { this.isDML = true; return this; } /** * Set custom statement name. * Default statement name value is {@code ping}. * * @param name custom statement name. * @return updated builder instance */ public Builder statementName(String name) { if (statement != null) { throw new HealthCheckBuilderException( "Can't use both statementName and statement methods in a single builder instance!"); } this.isNamedStatement = true; this.statementName = name; return this; } /** * Set custom statement. * * @param statement custom statement name. * @return updated builder instance */ public Builder statement(String statement) { if (statementName != null) { throw new HealthCheckBuilderException( "Can't use both statementName and statement methods in a single builder instance!"); } this.isNamedStatement = false; this.statement = statement; return this; } /** * Set custom timeout to wait for statement execution response. * Default value is {@code 10} seconds. * * @param duration the maximum time to wait for statement execution response * @param timeUnit the time unit of the timeout argument * @return updated builder instance * @deprecated use {@link #timeout(Duration)} instead */ @Deprecated(since = "4.0.0") public Builder timeout(long duration, TimeUnit timeUnit) { this.timeoutDuration = duration; this.timeoutUnit = timeUnit; return this; } /** * Set custom timeout to wait for statement execution response. * Default value is {@code 10} seconds. * * @param timeout the maximum time to wait for statement execution response * @return updated builder instance */ public Builder timeout(Duration timeout) { this.timeoutDuration = timeout.toNanos(); this.timeoutUnit = TimeUnit.NANOSECONDS; return this; } // Getters for local usage and jUnit tests String name() { return name; } long timeoutDuration() { return timeoutDuration; } TimeUnit timeoutUnit() { return timeoutUnit; } boolean isDML() { return isDML; } boolean isNamedStatement() { return isNamedStatement; } String statementName() { return statementName; } String statement() { return statement; } } }




© 2015 - 2025 Weber Informatics LLC | Privacy Policy