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

io.helidon.dbclient.mongodb.MongoDbStatementQuery Maven / Gradle / Ivy

There is a newer version: 4.1.4
Show newest version
/*
 * Copyright (c) 2019, 2020 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.mongodb;

import java.util.concurrent.CompletableFuture;
import java.util.logging.Logger;

import io.helidon.common.reactive.Multi;
import io.helidon.common.reactive.Single;
import io.helidon.dbclient.DbClientServiceContext;
import io.helidon.dbclient.DbRow;
import io.helidon.dbclient.DbStatementQuery;
import io.helidon.dbclient.DbStatementType;
import io.helidon.dbclient.common.DbStatementContext;

import com.mongodb.reactivestreams.client.FindPublisher;
import com.mongodb.reactivestreams.client.MongoCollection;
import com.mongodb.reactivestreams.client.MongoDatabase;
import org.bson.Document;

/**
 * Implementation of a query for MongoDB.
 */
class MongoDbStatementQuery extends MongoDbStatement> implements DbStatementQuery {
    private static final Logger LOGGER = Logger.getLogger(MongoDbStatementQuery.class.getName());

    MongoDbStatementQuery(MongoDatabase db, DbStatementContext statementContext) {
        super(db, statementContext);
    }

    @Override
    protected Multi doExecute(Single dbContextFuture,
                                     CompletableFuture statementFuture,
                                     CompletableFuture queryFuture) {

        dbContextFuture.exceptionally(throwable -> {
            statementFuture.completeExceptionally(throwable);
            queryFuture.completeExceptionally(throwable);
            return null;
        });

        String statement = build();
        MongoDbStatement.MongoStatement stmt;

        try {
            stmt = queryOrCommand(statement);
        } catch (Exception e) {
            return Multi.error(e);
        }

        if (stmt.getOperation() != MongoDbStatement.MongoOperation.QUERY) {
            if (stmt.getOperation() == MongoDbStatement.MongoOperation.COMMAND) {
                return MongoDbCommandExecutor.executeCommand(this,
                                                             dbContextFuture,
                                                             statementFuture,
                                                             queryFuture);
            } else {
                return Multi.error(new UnsupportedOperationException(
                        String.format("Operation %s is not supported by query", stmt.getOperation().toString())));
            }
        }
        // we reach here only for query statements

        // final variable required for lambda
        MongoDbStatement.MongoStatement usedStatement = stmt;
        return Single.create(dbContextFuture)
                .flatMap(it -> callStatement(usedStatement, statementFuture, queryFuture));
    }

    private MongoStatement queryOrCommand(String statement) {
        try {
            return new MongoDbStatement.MongoStatement(DbStatementType.QUERY, READER_FACTORY, statement);
        } catch (IllegalStateException e) {
            // maybe this is a command?
            try {
                return new MongoDbStatement.MongoStatement(DbStatementType.COMMAND, READER_FACTORY, statement);
            } catch (IllegalStateException ignored) {
                // we want to report the original exception
                throw e;
            }
        }
    }

    private Multi callStatement(MongoDbStatement.MongoStatement mongoStmt,
                                       CompletableFuture statementFuture,
                                       CompletableFuture queryFuture) {

        final MongoCollection mc = db()
                .getCollection(mongoStmt.getCollection());
        final Document query = mongoStmt.getQuery();
        final Document projection = mongoStmt.getProjection();
        LOGGER.fine(() -> String.format(
                "Query: %s, Projection: %s", query.toString(), (projection != null ? projection : "N/A")));
        FindPublisher publisher = noTx()
                ? mc.find(query)
                : mc.find(txManager().tx(), query);
        if (projection != null) {
            publisher = publisher.projection(projection);
        }

        return Multi.create(new MongoDbRows<>(clientContext(),
                                            publisher,
                                            this,
                                            DbRow.class,
                                            statementFuture,
                                            queryFuture)
                                  .publisher());
    }
}




© 2015 - 2025 Weber Informatics LLC | Privacy Policy