org.apache.camel.component.couchbase.CouchbaseConsumer Maven / Gradle / Ivy
/*
* Licensed to the Apache Software Foundation (ASF) under one or more
* contributor license agreements. See the NOTICE file distributed with
* this work for additional information regarding copyright ownership.
* The ASF licenses this file to You 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 org.apache.camel.component.couchbase;
import java.util.concurrent.locks.Lock;
import java.util.concurrent.locks.ReentrantLock;
import com.couchbase.client.core.deps.com.fasterxml.jackson.databind.JsonNode;
import com.couchbase.client.java.Bucket;
import com.couchbase.client.java.Collection;
import com.couchbase.client.java.Scope;
import com.couchbase.client.java.view.ViewOptions;
import com.couchbase.client.java.view.ViewOrdering;
import com.couchbase.client.java.view.ViewResult;
import com.couchbase.client.java.view.ViewRow;
import org.apache.camel.Exchange;
import org.apache.camel.Processor;
import org.apache.camel.resume.ResumeAware;
import org.apache.camel.resume.ResumeStrategy;
import org.apache.camel.support.DefaultScheduledPollConsumer;
import org.apache.camel.support.resume.ResumeStrategyHelper;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
import static org.apache.camel.component.couchbase.CouchbaseConstants.COUCHBASE_RESUME_ACTION;
import static org.apache.camel.component.couchbase.CouchbaseConstants.HEADER_DESIGN_DOCUMENT_NAME;
import static org.apache.camel.component.couchbase.CouchbaseConstants.HEADER_ID;
import static org.apache.camel.component.couchbase.CouchbaseConstants.HEADER_KEY;
import static org.apache.camel.component.couchbase.CouchbaseConstants.HEADER_VIEWNAME;
public class CouchbaseConsumer extends DefaultScheduledPollConsumer implements ResumeAware {
private static final Logger LOG = LoggerFactory.getLogger(CouchbaseConsumer.class);
private final Lock lock = new ReentrantLock();
private final CouchbaseEndpoint endpoint;
private final Bucket bucket;
private final Collection collection;
private ViewOptions viewOptions;
private ResumeStrategy resumeStrategy;
public CouchbaseConsumer(CouchbaseEndpoint endpoint, Bucket client, Processor processor) {
super(endpoint, processor);
this.bucket = client;
this.endpoint = endpoint;
Scope scope;
if (endpoint.getScope() != null) {
scope = client.scope(endpoint.getScope());
} else {
scope = client.defaultScope();
}
if (endpoint.getCollection() != null) {
this.collection = scope.collection(endpoint.getCollection());
} else {
this.collection = client.defaultCollection();
}
}
@Override
protected void doInit() {
this.viewOptions = ViewOptions.viewOptions();
int limit = endpoint.getLimit();
if (limit > 0) {
viewOptions.limit(limit);
}
int skip = endpoint.getSkip();
if (skip > 0) {
viewOptions.skip(skip);
}
if (endpoint.isDescending()) {
viewOptions.order(ViewOrdering.DESCENDING);
}
String rangeStartKey = endpoint.getRangeStartKey();
String rangeEndKey = endpoint.getRangeEndKey();
if (rangeStartKey == null || rangeStartKey.isEmpty() || rangeEndKey == null || rangeEndKey.isEmpty()) {
return;
}
viewOptions.startKey(rangeStartKey).endKey(rangeEndKey);
}
@Override
protected void doStart() throws Exception {
super.doStart();
ResumeStrategyHelper.resume(getEndpoint().getCamelContext(), this, resumeStrategy, COUCHBASE_RESUME_ACTION);
}
@Override
protected void doStop() throws Exception {
super.doStop();
if (bucket != null) {
bucket.core().shutdown();
}
}
@Override
protected int poll() throws Exception {
lock.lock();
try {
ViewResult result = bucket.viewQuery(endpoint.getDesignDocumentName(), endpoint.getViewName(), this.viewOptions);
// okay we have some response from CouchBase so lets mark the consumer as ready
forceConsumerAsReady();
if (LOG.isTraceEnabled()) {
LOG.trace("ViewResponse = {}", result);
}
String consumerProcessedStrategy = endpoint.getConsumerProcessedStrategy();
for (ViewRow row : result.rows()) {
Object doc;
String id = row.id().get();
if (endpoint.isFullDocument()) {
doc = CouchbaseCollectionOperation.getDocument(collection, id, endpoint.getQueryTimeout());
} else {
doc = row.valueAs(Object.class);
}
String key = row.keyAs(JsonNode.class).get().asText();
String designDocumentName = endpoint.getDesignDocumentName();
String viewName = endpoint.getViewName();
Exchange exchange = createExchange(false);
try {
exchange.getIn().setBody(doc);
exchange.getIn().setHeader(HEADER_ID, id);
exchange.getIn().setHeader(HEADER_KEY, key);
exchange.getIn().setHeader(HEADER_DESIGN_DOCUMENT_NAME, designDocumentName);
exchange.getIn().setHeader(HEADER_VIEWNAME, viewName);
if ("delete".equalsIgnoreCase(consumerProcessedStrategy)) {
if (LOG.isTraceEnabled()) {
LOG.trace("Deleting doc with ID {}", id);
}
CouchbaseCollectionOperation.removeDocument(collection, id, endpoint.getWriteQueryTimeout(),
endpoint.getProducerRetryPause());
} else if ("filter".equalsIgnoreCase(consumerProcessedStrategy)) {
if (LOG.isTraceEnabled()) {
LOG.trace("Filtering out ID {}", id);
}
// add filter for already processed docs
} else {
LOG.trace("No strategy set for already processed docs, beware of duplicates!");
}
logDetails(id, doc, key, designDocumentName, viewName, exchange);
getProcessor().process(exchange);
} catch (Exception e) {
this.getExceptionHandler().handleException("Error processing exchange.", exchange, e);
} finally {
releaseExchange(exchange, false);
}
}
return result.rows().size();
} finally {
lock.unlock();
}
}
private void logDetails(String id, Object doc, String key, String designDocumentName, String viewName, Exchange exchange) {
if (LOG.isTraceEnabled()) {
LOG.trace("Created exchange = {}", exchange);
LOG.trace("Added Document in body = {}", doc);
LOG.trace("Adding to Header");
LOG.trace("ID = {}", id);
LOG.trace("Key = {}", key);
LOG.trace("Design Document Name = {}", designDocumentName);
LOG.trace("View Name = {}", viewName);
}
}
@Override
public ResumeStrategy getResumeStrategy() {
return resumeStrategy;
}
@Override
public void setResumeStrategy(ResumeStrategy resumeStrategy) {
this.resumeStrategy = resumeStrategy;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy