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

com.palantir.atlasdb.keyvalue.dbkvs.impl.postgres.PostgresDdlTable Maven / Gradle / Ivy

There is a newer version: 0.1193.0
Show newest version
/*
 * (c) Copyright 2018 Palantir Technologies Inc. All rights reserved.
 *
 * 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 com.palantir.atlasdb.keyvalue.dbkvs.impl.postgres;

import com.google.common.annotations.VisibleForTesting;
import com.google.common.collect.Iterables;
import com.google.errorprone.annotations.CompileTimeConstant;
import com.palantir.atlasdb.AtlasDbConstants;
import com.palantir.atlasdb.keyvalue.api.TableReference;
import com.palantir.atlasdb.keyvalue.dbkvs.PostgresDdlConfig;
import com.palantir.atlasdb.keyvalue.dbkvs.impl.CaseSensitivity;
import com.palantir.atlasdb.keyvalue.dbkvs.impl.ConnectionSupplier;
import com.palantir.atlasdb.keyvalue.dbkvs.impl.DbDdlTable;
import com.palantir.atlasdb.keyvalue.dbkvs.impl.DbKvs;
import com.palantir.atlasdb.keyvalue.dbkvs.impl.TableValueStyle;
import com.palantir.atlasdb.keyvalue.dbkvs.impl.oracle.PrimaryKeyConstraintNames;
import com.palantir.exception.PalantirSqlException;
import com.palantir.logsafe.Arg;
import com.palantir.logsafe.SafeArg;
import com.palantir.logsafe.UnsafeArg;
import com.palantir.logsafe.exceptions.SafeRuntimeException;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import com.palantir.nexus.db.sql.AgnosticResultRow;
import com.palantir.nexus.db.sql.AgnosticResultSet;
import com.palantir.nexus.db.sql.ExceptionCheck;
import java.util.Arrays;
import java.util.concurrent.Semaphore;
import java.util.function.Predicate;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import org.slf4j.helpers.MessageFormatter;

public class PostgresDdlTable implements DbDdlTable {
    @SuppressWarnings({"ConsistentLoggerName", "PreferSafeLogger"})
    private static final Logger oldLog = LoggerFactory.getLogger(PostgresDdlTable.class);

    private static final SafeLogger log = SafeLoggerFactory.get(PostgresDdlTable.class);
    private static final int POSTGRES_NAME_LENGTH_LIMIT = 63;
    public static final int ATLASDB_POSTGRES_TABLE_NAME_LIMIT =
            POSTGRES_NAME_LENGTH_LIMIT - AtlasDbConstants.PRIMARY_KEY_CONSTRAINT_PREFIX.length();
    private static final String FAILED_TO_CREATE_TABLE_MESSAGE = "Failed to create table {}."
            + " The table name is longer than the postgres limit of {} characters."
            + " Attempted to truncate the name but the truncated table name or truncated primary"
            + " key constraint name already exists. Please ensure all your table names have unique"
            + " first {} characters.";

    private final TableReference tableName;
    private final ConnectionSupplier conns;
    private final PostgresDdlConfig config;
    private final Semaphore compactionSemaphore = new Semaphore(1);

    public PostgresDdlTable(TableReference tableName, ConnectionSupplier conns, PostgresDdlConfig config) {
        this.tableName = tableName;
        this.conns = conns;
        this.config = config;
    }

    @Override
    public void create(byte[] tableMetadata) {
        if (conns.get()
                .selectExistsUnregisteredQuery(
                        "SELECT 1 FROM " + config.metadataTable().getQualifiedName() + " WHERE table_name = ?",
                        tableName.getQualifiedName())) {
            return;
        }

        String prefixedTableName = prefixedTableName();
        try {
            conns.get()
                    .executeUnregisteredQuery(String.format(
                                    "CREATE TABLE %s ("
                                            + "  row_name   BYTEA NOT NULL,"
                                            + "  col_name   BYTEA NOT NULL,"
                                            + "  ts         INT8 NOT NULL,"
                                            + "  val        BYTEA,"
                                            + "  CONSTRAINT %s PRIMARY KEY (row_name, col_name, ts) ",
                                    prefixedTableName, PrimaryKeyConstraintNames.get(prefixedTableName))
                            + ")");
            logDebugOrTrace("created postgres table", UnsafeArg.of("tableName", prefixedTableName));
        } catch (PalantirSqlException e) {
            if (!e.getMessage().contains("already exists")) {
                log.error("Error occurred trying to create the table", e);
                throw e;
            } else if (prefixedTableName.length() > ATLASDB_POSTGRES_TABLE_NAME_LIMIT) {
                log.error(
                        FAILED_TO_CREATE_TABLE_MESSAGE,
                        UnsafeArg.of("prefixedTableName", prefixedTableName),
                        SafeArg.of("limit", ATLASDB_POSTGRES_TABLE_NAME_LIMIT),
                        SafeArg.of("limit2", ATLASDB_POSTGRES_TABLE_NAME_LIMIT),
                        e);
                String exceptionMsg = MessageFormatter.arrayFormat(FAILED_TO_CREATE_TABLE_MESSAGE, new Object[] {
                            prefixedTableName, ATLASDB_POSTGRES_TABLE_NAME_LIMIT, ATLASDB_POSTGRES_TABLE_NAME_LIMIT
                        })
                        .getMessage();
                throw new RuntimeException(exceptionMsg, e);
            } else {
                logDebugOrTrace(
                        "attempted to create postgres table that already existed",
                        UnsafeArg.of("tableName", prefixedTableName));
            }
        }

        ignoringError(
                () -> conns.get()
                        .insertOneUnregisteredQuery(
                                String.format(
                                        "INSERT INTO %s (table_name, table_size) VALUES (?, ?)",
                                        config.metadataTable().getQualifiedName()),
                                tableName.getQualifiedName(),
                                TableValueStyle.RAW.getId()),
                ExceptionCheck::isUniqueConstraintViolation);
    }

    @Override
    public void drop() {
        drop(CaseSensitivity.CASE_SENSITIVE);
    }

    @Override
    public void drop(CaseSensitivity referenceCaseSensitivity) {
        if (referenceCaseSensitivity == CaseSensitivity.CASE_INSENSITIVE) {
            throw new UnsupportedOperationException(
                    "AtlasDB does not currently support case insensitive drop table commands for Postgres");
        }

        String prefixedTableName = prefixedTableName();
        executeIgnoringError("DROP TABLE " + prefixedTableName, "does not exist");
        conns.get()
                .executeUnregisteredQuery(
                        String.format(
                                "DELETE FROM %s WHERE table_name = ?",
                                config.metadataTable().getQualifiedName()),
                        tableName.getQualifiedName());
        logDebugOrTrace("deleted postgres table", UnsafeArg.of("tableName", prefixedTableName));
    }

    @Override
    public void truncate() {
        conns.get().executeUnregisteredQuery("TRUNCATE TABLE " + prefixedTableName());
    }

    @Override
    public void checkDatabaseVersion() {
        AgnosticResultSet result = conns.get().selectResultSetUnregisteredQuery("SHOW server_version");
        String version = result.get(0).getString("server_version");
        PostgresVersionCheck.checkDatabaseVersion(version, oldLog);
    }

    @Override
    public void compactInternally(boolean unused) {
        if (compactionSemaphore.tryAcquire()) {
            try {
                if (shouldRunCompaction()) {
                    runCompactOnTable();
                }
            } finally {
                compactionSemaphore.release();
            }
        }
    }

    @VisibleForTesting
    boolean shouldRunCompaction() {
        long compactIntervalMillis = config.compactInterval().toMilliseconds();
        return compactIntervalMillis <= 0 || getMillisSinceLastCompact() >= compactIntervalMillis;
    }

    /**
     * Returns the number of milliseconds since the last compaction, or Long.MAX_VALUE if
     * compaction has never run.
     */
    private long getMillisSinceLastCompact() {
        AgnosticResultSet rs = conns.get()
                .selectResultSetUnregisteredQuery(
                        "SELECT FLOOR(EXTRACT(EPOCH FROM GREATEST( "
                                + "  last_vacuum, last_autovacuum, last_analyze, last_autoanalyze"
                                + "))*1000) AS last, "
                                + "FLOOR(EXTRACT(EPOCH FROM CURRENT_TIMESTAMP)*1000) AS current "
                                + "FROM pg_stat_user_tables WHERE relname = ?",
                        prefixedTableName());

        AgnosticResultRow row = Iterables.getOnlyElement(rs.rows());

        // last could be null if vacuum has never run
        long last = row.getLong("last", -1);
        if (last == -1) {
            return Long.MAX_VALUE;
        }
        long current = row.getLong("current");
        return current - last;
    }

    private void runCompactOnTable() {
        // VACUUM FULL is /really/ what we want here, but it takes out a table lock
        conns.get().executeUnregisteredQuery("VACUUM ANALYZE " + prefixedTableName());
    }

    private String prefixedTableName() {
        return config.tablePrefix() + DbKvs.internalTableName(tableName);
    }

    private void executeIgnoringError(String sql, String errorToIgnore) {
        ignoringError(() -> conns.get().executeUnregisteredQuery(sql), e -> e.getMessage()
                .contains(errorToIgnore));
    }

    private static void ignoringError(Runnable fn, Predicate shouldIgnoreException) {
        try {
            fn.run();
        } catch (PalantirSqlException e) {
            if (!shouldIgnoreException.test(e)) {
                log.error("Error occurred trying to run function", e);
                throw e;
            }
        }
    }

    private static void logDebugOrTrace(@CompileTimeConstant String message, Arg... args) {
        if (log.isTraceEnabled()) {
            log.trace(message, Arrays.asList(args), new SafeRuntimeException("provided for stack trace"));
        } else if (log.isDebugEnabled()) {
            log.debug(message, Arrays.asList(args));
        }
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy