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

com.palantir.atlasdb.table.description.Schema Maven / Gradle / Ivy

/*
 * (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.table.description;

import static java.util.stream.Collectors.toSet;

import com.google.common.base.Function;
import com.google.common.base.Preconditions;
import com.google.common.base.Suppliers;
import com.google.common.collect.ArrayListMultimap;
import com.google.common.collect.Collections2;
import com.google.common.collect.ImmutableSortedSet;
import com.google.common.collect.Iterables;
import com.google.common.collect.ListMultimap;
import com.google.common.collect.Multimap;
import com.google.common.collect.Ordering;
import com.palantir.atlasdb.AtlasDbConstants;
import com.palantir.atlasdb.cleaner.api.OnCleanupTask;
import com.palantir.atlasdb.keyvalue.api.Namespace;
import com.palantir.atlasdb.keyvalue.api.TableReference;
import com.palantir.atlasdb.keyvalue.impl.AbstractKeyValueService;
import com.palantir.atlasdb.schema.stream.StreamStoreDefinition;
import com.palantir.atlasdb.table.description.IndexDefinition.IndexType;
import com.palantir.atlasdb.table.description.render.StreamStoreRenderer;
import com.palantir.atlasdb.table.description.render.TableFactoryRenderer;
import com.palantir.atlasdb.table.description.render.TableRenderer;
import com.palantir.atlasdb.table.description.render.TableRendererV2;
import com.palantir.atlasdb.transaction.api.ConflictHandler;
import com.palantir.lock.watch.LockWatchReferences;
import com.palantir.lock.watch.LockWatchReferences.LockWatchReference;
import com.palantir.logsafe.UnsafeArg;
import com.palantir.logsafe.logger.SafeLogger;
import com.palantir.logsafe.logger.SafeLoggerFactory;
import java.io.File;
import java.io.FileWriter;
import java.io.IOException;
import java.nio.charset.StandardCharsets;
import java.util.ArrayList;
import java.util.Collection;
import java.util.HashMap;
import java.util.HashSet;
import java.util.List;
import java.util.Map;
import java.util.Set;
import java.util.function.Supplier;
import java.util.stream.Collectors;
import org.apache.commons.lang3.StringUtils;

/**
 * Defines a schema.
 *
 * A schema consists of table definitions and indexes.
 *
 * Schema objects can be used for creating/dropping tables within key values
 * stores, as well as compiling automatically generated code for accessing
 * tables in a type-safe fashion.
 */
@SuppressWarnings("checkstyle:Indentation")
public class Schema {
    private static final SafeLogger log = SafeLoggerFactory.get(Schema.class);

    private final String name;
    private final String packageName;
    private final Namespace namespace;
    private final OptionalType optionalType;
    private boolean ignoreTableNameLengthChecks = false;

    private final Multimap> cleanupTasks = ArrayListMultimap.create();
    private final Map tableDefinitions = new HashMap<>();
    private final Map indexDefinitions = new HashMap<>();
    private final List streamStoreRenderers = new ArrayList<>();
    private final Set lockWatches = new HashSet<>();

    // N.B., the following is a list multimap because we want to preserve order
    // for code generation purposes.
    private final ListMultimap indexesByTable = ArrayListMultimap.create();

    /** Creates a new schema, using Guava Optionals. */
    public Schema(Namespace namespace) {
        this(null, null, namespace);
    }

    /** Creates a new schema, using Guava Optionals. */
    public Schema() {
        this(null, null, Namespace.DEFAULT_NAMESPACE);
    }

    /** Creates a new schema, using Guava Optionals. */
    public Schema(String name, String packageName, Namespace namespace) {
        this(name, packageName, namespace, OptionalType.GUAVA);
    }

    public Schema(String name, String packageName, Namespace namespace, OptionalType optionalType) {
        this.name = name;
        this.packageName = packageName;
        this.namespace = namespace;
        this.optionalType = optionalType;
    }

    public String getName() {
        return name;
    }

    public void addTableDefinition(String tableName, TableDefinition definition) {
        Preconditions.checkArgument(
                !tableDefinitions.containsKey(tableName) && !indexDefinitions.containsKey(tableName),
                "Table already defined: %s",
                tableName);
        Preconditions.checkArgument(Schemas.isTableNameValid(tableName), "Invalid table name %s", tableName);
        validateTableNameLength(tableName);
        if (definition.enableCaching) {
            com.palantir.logsafe.Preconditions.checkArgument(
                    definition.conflictHandler == ConflictHandler.SERIALIZABLE_CELL,
                    "Caching can only be enabled with the SERIALIZABLE_CELL conflict handler.");
            lockWatches.add(LockWatchReferences.entireTable(
                    TableReference.create(namespace, tableName).getQualifiedName()));
        }
        tableDefinitions.put(tableName, definition);
    }

    public void addDefinitionsForTables(Iterable tableNames, TableDefinition definition) {
        for (String t : tableNames) {
            addTableDefinition(t, definition);
        }
    }

    public TableDefinition getTableDefinition(TableReference tableRef) {
        return tableDefinitions.get(tableRef.getTableName());
    }

    public Map getAllTablesAndIndexMetadata() {
        Map ret = new HashMap<>();
        for (Map.Entry e : tableDefinitions.entrySet()) {
            ret.put(TableReference.create(namespace, e.getKey()), e.getValue().toTableMetadata());
        }
        for (Map.Entry e : indexDefinitions.entrySet()) {
            ret.put(
                    TableReference.create(namespace, e.getKey()),
                    e.getValue().toIndexMetadata(e.getKey()).getTableMetadata());
        }
        return ret;
    }

    public Set getAllIndexes() {
        return indexDefinitions.keySet().stream()
                .map(table -> TableReference.create(namespace, table))
                .collect(toSet());
    }

    public Set getAllTables() {
        return tableDefinitions.keySet().stream()
                .map(table -> TableReference.create(namespace, table))
                .collect(toSet());
    }

    public void addIndexDefinition(String idxName, IndexDefinition definition) {
        validateIndex(idxName, definition);
        String indexName = Schemas.appendIndexSuffix(idxName, definition).getQualifiedName();
        validateTableNameLength(indexName);
        indexesByTable.put(definition.getSourceTable(), indexName);
        indexDefinitions.put(indexName, definition);
    }

    private void validateTableNameLength(String idxName) {
        if (!ignoreTableNameLengthChecks) {
            String internalTableName =
                    AbstractKeyValueService.internalTableName(TableReference.create(namespace, idxName));
            List kvsExceeded = new ArrayList<>();

            if (internalTableName.length() > AtlasDbConstants.CASSANDRA_TABLE_NAME_CHAR_LIMIT) {
                kvsExceeded.add(CharacterLimitType.CASSANDRA);
            }
            if (internalTableName.length() > AtlasDbConstants.POSTGRES_TABLE_NAME_CHAR_LIMIT) {
                kvsExceeded.add(CharacterLimitType.POSTGRES);
            }
            Preconditions.checkArgument(
                    kvsExceeded.isEmpty(),
                    "Internal table name %s is too long, known to exceed character limits for "
                            + "the following KVS: %s. If using a table prefix, please ensure that the concatenation "
                            + "of the prefix with the internal table name is below the KVS limit. "
                            + "If running only against a different KVS, set the ignoreTableNameLength flag.",
                    idxName,
                    StringUtils.join(kvsExceeded, ", "));
        }
    }

    /**
     * Adds the given stream store to your schema.
     *
     * @param streamStoreDefinition You probably want to use a @{StreamStoreDefinitionBuilder} for convenience.
     */
    public void addStreamStoreDefinition(StreamStoreDefinition streamStoreDefinition) {
        streamStoreDefinition.getTables().forEach(this::addTableDefinition);
        StreamStoreRenderer renderer = streamStoreDefinition.getRenderer(packageName, name);
        Multimap> streamStoreCleanupTasks =
                streamStoreDefinition.getCleanupTasks(packageName, name, renderer, namespace);

        cleanupTasks.putAll(streamStoreCleanupTasks);
        streamStoreRenderers.add(renderer);
    }

    private void validateIndex(String idxName, IndexDefinition definition) {
        for (IndexType type : IndexType.values()) {
            Preconditions.checkArgument(
                    !idxName.endsWith(type.getIndexSuffix()),
                    "Index name cannot end with '%s'.",
                    type.getIndexSuffix());
            String indexName = idxName + type.getIndexSuffix();
            com.palantir.logsafe.Preconditions.checkArgument(
                    !tableDefinitions.containsKey(indexName) && !indexDefinitions.containsKey(indexName),
                    "Table already defined.");
        }
        com.palantir.logsafe.Preconditions.checkArgument(
                tableDefinitions.containsKey(definition.getSourceTable()), "Index source table undefined.");
        Preconditions.checkArgument(Schemas.isTableNameValid(idxName), "Invalid table name %s", idxName);
        com.palantir.logsafe.Preconditions.checkArgument(
                !tableDefinitions
                                .get(definition.getSourceTable())
                                .toTableMetadata()
                                .getColumns()
                                .hasDynamicColumns()
                        || !definition.getIndexType().equals(IndexType.CELL_REFERENCING),
                "Cell referencing indexes not implemented for tables with dynamic columns.");
    }

    public IndexDefinition getIndex(TableReference indexRef) {
        return indexDefinitions.get(indexRef.getTableName());
    }

    /**
     * Performs some basic checks on this schema to check its validity.
     */
    public void validate() {
        // Try converting to metadata to see if any validation logic throws.
        for (Map.Entry entry : tableDefinitions.entrySet()) {
            try {
                entry.getValue().validate();
            } catch (Exception e) {
                log.error("Failed to validate table {}.", UnsafeArg.of("tableName", entry.getKey()), e);
                throw e;
            }
        }

        for (Map.Entry indexEntry : indexDefinitions.entrySet()) {
            IndexDefinition def = indexEntry.getValue();
            try {
                def.toIndexMetadata(indexEntry.getKey()).getTableMetadata();
                def.validate();
            } catch (Exception e) {
                log.error("Failed to validate index {}.", UnsafeArg.of("indexName", indexEntry.getKey()), e);
                throw e;
            }
        }

        for (Map.Entry e : indexesByTable.entries()) {
            TableMetadata tableMetadata = tableDefinitions.get(e.getKey()).toTableMetadata();

            Collection rowNames = Collections2.transform(
                    tableMetadata.getRowMetadata().getRowParts(), NameComponentDescription::getComponentName);

            IndexMetadata indexMetadata = indexDefinitions.get(e.getValue()).toIndexMetadata(e.getValue());
            for (IndexComponent c :
                    Iterables.concat(indexMetadata.getRowComponents(), indexMetadata.getColumnComponents())) {
                if (c.rowComponentName != null) {
                    com.palantir.logsafe.Preconditions.checkArgument(
                            rowNames.contains(c.rowComponentName),
                            "In index, a componentFromRow must reference an existing row component");
                }
            }

            if (indexMetadata.getColumnNameToAccessData() != null) {
                com.palantir.logsafe.Preconditions.checkArgument(
                        tableMetadata.getColumns().getDynamicColumn() == null,
                        "Indexes accessing columns not supported for tables with dynamic columns.");
                Collection columnNames = Collections2.transform(
                        tableMetadata.getColumns().getNamedColumns(), NamedColumnDescription::getLongName);
                com.palantir.logsafe.Preconditions.checkArgument(
                        columnNames.contains(indexMetadata.getColumnNameToAccessData()),
                        "In index, a component derived from column must reference an existing column");
            }

            if (indexMetadata.getIndexType().equals(IndexType.CELL_REFERENCING)) {
                com.palantir.logsafe.Preconditions.checkArgument(
                        ConflictHandler.RETRY_ON_WRITE_WRITE.equals(tableMetadata.getConflictHandler()),
                        "Nonadditive indexes require write-write conflicts on their tables");
            }
        }
    }

    public Map getTableDefinitions() {
        return tableDefinitions.entrySet().stream()
                .collect(Collectors.toMap(e -> TableReference.create(namespace, e.getKey()), Map.Entry::getValue));
    }

    public Set getLockWatches() {
        return lockWatches;
    }

    public Map getIndexDefinitions() {
        return indexDefinitions.entrySet().stream()
                .collect(Collectors.toMap(e -> TableReference.create(namespace, e.getKey()), Map.Entry::getValue));
    }

    public Namespace getNamespace() {
        return namespace;
    }

    /**
     * Performs code generation.
     *
     * @param srcDir root source directory where code generation is performed.
     */
    @SuppressWarnings("DangerousIdentityKey")
    public void renderTables(File srcDir) throws IOException {
        com.palantir.logsafe.Preconditions.checkNotNull(name, "schema name not set");
        com.palantir.logsafe.Preconditions.checkNotNull(packageName, "package name not set");

        TableRenderer tableRenderer = new TableRenderer(packageName, namespace, optionalType);
        TableRendererV2 tableRendererV2 = new TableRendererV2(packageName, namespace);
        for (Map.Entry entry : tableDefinitions.entrySet()) {
            String rawTableName = entry.getKey();
            TableDefinition table = entry.getValue();
            ImmutableSortedSet.Builder indices = ImmutableSortedSet.orderedBy(
                    Ordering.natural().onResultOf((Function) IndexMetadata::getIndexName));
            if (table.getGenericTableName() != null) {
                com.palantir.logsafe.Preconditions.checkState(
                        !indexesByTable.containsKey(rawTableName), "Generic tables cannot have indices");
            } else {
                for (String indexName : indexesByTable.get(rawTableName)) {
                    indices.add(indexDefinitions.get(indexName).toIndexMetadata(indexName));
                }
            }
            emit(
                    srcDir,
                    tableRenderer.render(rawTableName, table, indices.build()),
                    packageName,
                    tableRenderer.getClassName(rawTableName, table));
            if (table.hasV2TableEnabled()) {
                emit(
                        srcDir,
                        tableRendererV2.render(rawTableName, table),
                        packageName,
                        tableRendererV2.getClassName(rawTableName, table));
            }
        }
        for (StreamStoreRenderer renderer : streamStoreRenderers) {
            emit(srcDir, renderer.renderStreamStore(), renderer.getPackageName(), renderer.getStreamStoreClassName());
            emit(
                    srcDir,
                    renderer.renderIndexCleanupTask(),
                    renderer.getPackageName(),
                    renderer.getIndexCleanupTaskClassName());
            emit(
                    srcDir,
                    renderer.renderMetadataCleanupTask(),
                    renderer.getPackageName(),
                    renderer.getMetadataCleanupTaskClassName());
        }
        TableFactoryRenderer tableFactoryRenderer =
                TableFactoryRenderer.of(name, packageName, namespace, tableDefinitions);
        emit(
                srcDir,
                tableFactoryRenderer.render(),
                tableFactoryRenderer.getPackageName(),
                tableFactoryRenderer.getClassName());
    }

    private void emit(File srcDir, String code, String packName, String className) throws IOException {
        File outputDir = new File(srcDir, packName.replace(".", "/"));
        File outputFile = new File(outputDir, className + ".java");

        // create paths if they don't exist
        outputDir.mkdirs();
        outputFile = outputFile.getAbsoluteFile();
        outputFile.createNewFile();
        FileWriter os = null;
        try {
            os = new FileWriter(outputFile, StandardCharsets.UTF_8);
            os.write(code);
        } finally {
            if (os != null) {
                os.close();
            }
        }
    }

    // Cannot be removed, as it is used by the large internal product
    public void addCleanupTask(String rawTableName, OnCleanupTask task) {
        addCleanupTask(rawTableName, Suppliers.ofInstance(task));
    }

    public void addCleanupTask(String rawTableName, Supplier task) {
        cleanupTasks.put(rawTableName, task);
    }

    public Multimap getCleanupTasksByTable() {
        Multimap ret = ArrayListMultimap.create();
        for (Map.Entry> e : cleanupTasks.entries()) {
            ret.put(TableReference.create(namespace, e.getKey()), e.getValue().get());
        }
        return ret;
    }

    public void ignoreTableNameLengthChecks() {
        ignoreTableNameLengthChecks = true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy