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

de.bwaldvogel.mongo.backend.postgresql.index.PostgresUniqueIndex Maven / Gradle / Ivy

The newest version!
package de.bwaldvogel.mongo.backend.postgresql.index;

import java.sql.Connection;
import java.sql.PreparedStatement;
import java.sql.ResultSet;
import java.sql.SQLException;
import java.sql.Types;
import java.util.LinkedHashMap;
import java.util.List;
import java.util.Map;
import java.util.stream.Collectors;

import org.slf4j.Logger;
import org.slf4j.LoggerFactory;

import de.bwaldvogel.mongo.MongoCollection;
import de.bwaldvogel.mongo.backend.Index;
import de.bwaldvogel.mongo.backend.IndexKey;
import de.bwaldvogel.mongo.backend.KeyValue;
import de.bwaldvogel.mongo.backend.Missing;
import de.bwaldvogel.mongo.backend.Utils;
import de.bwaldvogel.mongo.backend.postgresql.PostgresqlBackend;
import de.bwaldvogel.mongo.backend.postgresql.PostgresqlCollection;
import de.bwaldvogel.mongo.backend.postgresql.PostgresqlUtils;
import de.bwaldvogel.mongo.bson.Document;
import de.bwaldvogel.mongo.exception.DuplicateKeyError;
import de.bwaldvogel.mongo.exception.KeyConstraintError;
import de.bwaldvogel.mongo.exception.MongoServerException;

public class PostgresUniqueIndex extends Index {

    private static final Logger log = LoggerFactory.getLogger(PostgresUniqueIndex.class);

    private final PostgresqlBackend backend;
    private final String databaseName;
    private final String collectionName;
    private final String fullCollectionName;

    public PostgresUniqueIndex(PostgresqlBackend backend, String databaseName, String collectionName, String indexName,
                               List keys, boolean sparse) {
        super(indexName, keys, sparse);
        this.backend = backend;
        this.databaseName = databaseName;
        this.collectionName = collectionName;
        this.fullCollectionName = PostgresqlCollection.getQualifiedTablename(databaseName, collectionName);
    }

    private String getIndexName() {
        return collectionName + "_" + getName();
    }

    public void initialize() {
        String columns = keyColumns(getKeys());
        String sql = "CREATE UNIQUE INDEX IF NOT EXISTS \"" + getIndexName() + "\" ON " + fullCollectionName + " (" + columns + ")";
        try (Connection connection = backend.getConnection();
             PreparedStatement stmt = connection.prepareStatement(sql)) {
            stmt.executeUpdate();
        } catch (SQLException e) {
            if (PostgresqlUtils.isErrorDuplicateKey(e)) {
                throw new DuplicateKeyError(databaseName + "." + collectionName, getName() + " dup key: " + e.getMessage());
            } else {
                throw new MongoServerException("failed to create " + this + " on " + fullCollectionName, e);
            }
        }
    }

    @Override
    public void drop() {
        log.debug("Dropping index '{}'", getIndexName());
        String sql = "DROP INDEX \"" + databaseName + "\".\"" + getIndexName() + "\"";
        try (Connection connection = backend.getConnection();
             PreparedStatement stmt = connection.prepareStatement(sql)) {
            stmt.executeUpdate();
        } catch (SQLException e) {
            throw new MongoServerException("failed to drop " + this + " on " + fullCollectionName, e);
        }
    }

    private static String keyColumns(List keys) {
        return keys.stream()
            .map(key -> "(" + PostgresqlUtils.toDataKey(key.getKey()) + ") " + (key.isAscending() ? "ASC" : "DESC"))
            .collect(Collectors.joining(", "));
    }

    @Override
    public void checkAdd(Document document, MongoCollection collection) {
        checkDocument(document, collection, "add");
    }

    private void checkDocument(Document document, MongoCollection collection, String operation) {
        Map keyValues = getKeyValueMap(document);
        String sql = createSelectStatement(keyValues);
        try (Connection connection = backend.getConnection();
             PreparedStatement stmt = connection.prepareStatement(sql)) {
            fillStrings(stmt, keyValues);
            try (ResultSet resultSet = stmt.executeQuery()) {
                if (resultSet.next()) {
                    throw new DuplicateKeyError(this, collection, getKeys(), new KeyValue(keyValues.values()));
                }
            }
        } catch (SQLException e) {
            throw new MongoServerException("failed to " + operation + " document to " + fullCollectionName, e);
        }
    }

    private Map getKeyValueMap(Document document) {
        Map result = new LinkedHashMap<>();
        for (String key : keys()) {
            Object value = Utils.getSubdocumentValueCollectionAware(document, key);
            if (value instanceof Missing) {
                value = null;
            }
            result.put(key, value);
        }
        return result;
    }

    @Override
    public void add(Document document, Long position, MongoCollection collection) {
    }

    @Override
    public Long remove(Document document) {
        return getPosition(document);
    }

    private void fillStrings(PreparedStatement statement, Map keyValues) throws SQLException {
        int pos = 1;
        for (Object keyValue : keyValues.values()) {
            if (keyValue != null) {
                String queryValue = PostgresqlUtils.toQueryValue(keyValue);
                statement.setString(pos++, queryValue);
            } else if (isSparse()) {
                statement.setNull(pos++, Types.VARCHAR);
            }
        }
    }

    String createSelectStatement(Map keyValues) {
        return "SELECT id FROM " + fullCollectionName + " WHERE " +
            keyValues.entrySet().stream()
                .map(entry -> {
                    String dataKey = PostgresqlUtils.toDataKey(entry.getKey());
                    if (!isSparse() && entry.getValue() == null) {
                        return dataKey + " IS NULL";
                    } else if (entry.getValue() instanceof Number) {
                        return "CASE json_typeof(" + dataKey.replace("->>", "->") + ")"
                            + " WHEN 'number' THEN (" + dataKey + ")::numeric = ?::numeric"
                            + " ELSE FALSE"
                            + " END";
                    } else {
                        return dataKey + " = ?";
                    }
                })
                .collect(Collectors.joining(" AND "));
    }

    @Override
    public boolean canHandle(Document query) {
        return false;
    }

    @Override
    public Iterable getPositions(Document query) {
        return null;
    }

    @Override
    public long getCount() {
        return 0;
    }

    @Override
    public boolean isEmpty() {
        return false;
    }

    @Override
    public long getDataSize() {
        return 0;
    }

    @Override
    public void checkUpdate(Document oldDocument, Document newDocument, MongoCollection collection) {
        if (!nullAwareEqualsKeys(oldDocument, newDocument)) {
            checkDocument(newDocument, collection, "update");
        }
    }

    @Override
    public void updateInPlace(Document oldDocument, Document newDocument, Long position, MongoCollection collection) throws KeyConstraintError {
    }

    @Override
    public Long getPosition(Document document) {
        Map keyValues = getKeyValueMap(document);
        String sql = createSelectStatement(keyValues);
        try (Connection connection = backend.getConnection();
             PreparedStatement stmt = connection.prepareStatement(sql)) {
            fillStrings(stmt, keyValues);
            try (ResultSet resultSet = stmt.executeQuery()) {
                if (!resultSet.next()) {
                    return null;
                }
                long position = resultSet.getLong("id");
                if (resultSet.next()) {
                    throw new MongoServerException("got more than one id");
                }
                return Long.valueOf(position);
            }
        } catch (SQLException e) {
            throw new MongoServerException("failed to get document position from " + fullCollectionName, e);
        }
    }

    @Override
    public boolean isUnique() {
        return true;
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy