/**
* 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.mybatis;
import java.io.IOException;
import org.apache.camel.Component;
import org.apache.camel.Consumer;
import org.apache.camel.Processor;
import org.apache.camel.Producer;
import org.apache.camel.impl.DefaultPollingEndpoint;
import org.apache.camel.spi.Metadata;
import org.apache.camel.spi.UriEndpoint;
import org.apache.camel.spi.UriParam;
import org.apache.camel.spi.UriPath;
import org.apache.camel.util.ObjectHelper;
import org.apache.ibatis.session.ExecutorType;
import org.apache.ibatis.session.SqlSessionFactory;
/**
* Performs a query, poll, insert, update or delete in a relational database using MyBatis.
*/
@UriEndpoint(firstVersion = "2.7.0", scheme = "mybatis", title = "MyBatis", syntax = "mybatis:statement", consumerClass = MyBatisConsumer.class, label = "database,sql")
public class MyBatisEndpoint extends DefaultPollingEndpoint {
@UriPath @Metadata(required = "true")
private String statement;
@UriParam(label = "producer")
private StatementType statementType;
@UriParam(label = "consumer", description = "Enables or disables transaction. If enabled then if processing an exchange failed then the consumer"
+ "break out processing any further exchanges to cause a rollback eager.")
private boolean transacted;
@UriParam(label = "consumer", defaultValue = "0")
private int maxMessagesPerPoll;
@UriParam
private String outputHeader;
@UriParam(label = "consumer")
private String inputHeader;
@UriParam(label = "consumer", optionalPrefix = "consumer.")
private String onConsume;
@UriParam(label = "consumer", optionalPrefix = "consumer.", defaultValue = "true")
private boolean useIterator = true;
@UriParam(label = "consumer", optionalPrefix = "consumer.")
private boolean routeEmptyResultSet;
@UriParam(label = "consumer,advanced")
private MyBatisProcessingStrategy processingStrategy = new DefaultMyBatisProcessingStrategy();
@UriParam(label = "producer", defaultValue = "SIMPLE")
private ExecutorType executorType;
public MyBatisEndpoint() {
}
public MyBatisEndpoint(String endpointUri, Component component, String statement) {
super(endpointUri, component);
this.statement = statement;
}
public Producer createProducer() throws Exception {
ObjectHelper.notNull(statementType, "statementType", this);
ObjectHelper.notNull(statement, "statement", this);
return new MyBatisProducer(this);
}
public Consumer createConsumer(Processor processor) throws Exception {
ObjectHelper.notNull(statement, "statement", this);
MyBatisConsumer consumer = new MyBatisConsumer(this, processor);
consumer.setMaxMessagesPerPoll(getMaxMessagesPerPoll());
consumer.setOnConsume(getOnConsume());
consumer.setUseIterator(isUseIterator());
consumer.setRouteEmptyResultSet(isRouteEmptyResultSet());
configureConsumer(consumer);
return consumer;
}
public boolean isSingleton() {
return true;
}
@Override
public MyBatisComponent getComponent() {
return (MyBatisComponent) super.getComponent();
}
public SqlSessionFactory getSqlSessionFactory() throws IOException {
return getComponent().getSqlSessionFactory();
}
public String getStatement() {
return statement;
}
/**
* The statement name in the MyBatis XML mapping file which maps to the query, insert, update or delete operation you wish to evaluate.
*/
public void setStatement(String statement) {
this.statement = statement;
}
public StatementType getStatementType() {
return statementType;
}
/**
* Mandatory to specify for the producer to control which kind of operation to invoke.
*/
public void setStatementType(StatementType statementType) {
this.statementType = statementType;
}
public ExecutorType getExecutorType() {
return executorType;
}
/**
* The executor type to be used while executing statements.
* <ul>
* <li>simple - executor does nothing special.</li>
* <li>reuse - executor reuses prepared statements.</li>
* <li>batch - executor reuses statements and batches updates.</li>
* </ul>
*/
public void setExecutorType(ExecutorType executorType) {
this.executorType = executorType;
}
public void setExecutorType(String executorType) {
this.executorType = ExecutorType.valueOf(executorType.toUpperCase());
}
public boolean isTransacted() {
return transacted;
}
/**
* Enables or disables transaction. If enabled then if processing an exchange failed then the consumer
+ break out processing any further exchanges to cause a rollback eager
*/
public void setTransacted(boolean transacted) {
this.transacted = transacted;
}
public MyBatisProcessingStrategy getProcessingStrategy() {
return processingStrategy;
}
/**
* To use a custom MyBatisProcessingStrategy
*/
public void setProcessingStrategy(MyBatisProcessingStrategy processingStrategy) {
this.processingStrategy = processingStrategy;
}
public int getMaxMessagesPerPoll() {
return maxMessagesPerPoll;
}
/**
* This option is intended to split results returned by the database pool into the batches and deliver them in multiple exchanges.
* This integer defines the maximum messages to deliver in single exchange. By default, no maximum is set.
* Can be used to set a limit of e.g. 1000 to avoid when starting up the server that there are thousands of files.
* Set a value of 0 or negative to disable it.
*/
public void setMaxMessagesPerPoll(int maxMessagesPerPoll) {
this.maxMessagesPerPoll = maxMessagesPerPoll;
}
public String getOutputHeader() {
return outputHeader;
}
/**
* Store the query result in a header instead of the message body.
* By default, outputHeader == null and the query result is stored in the message body,
* any existing content in the message body is discarded.
* If outputHeader is set, the value is used as the name of the header to store the
* query result and the original message body is preserved. Setting outputHeader will
* also omit populating the default CamelMyBatisResult header since it would be the same
* as outputHeader all the time.
*/
public void setOutputHeader(String outputHeader) {
this.outputHeader = outputHeader;
}
public String getInputHeader() {
return inputHeader;
}
/**
* User the header value for input parameters instead of the message body.
* By default, inputHeader == null and the input parameters are taken from the message body.
* If outputHeader is set, the value is used and query parameters will be taken from the
* header instead of the body.
*/
public void setInputHeader(String inputHeader) {
this.inputHeader = inputHeader;
}
public String getOnConsume() {
return onConsume;
}
/**
* Statement to run after data has been processed in the route
*/
public void setOnConsume(String onConsume) {
this.onConsume = onConsume;
}
public boolean isUseIterator() {
return useIterator;
}
/**
* Process resultset individually or as a list
*/
public void setUseIterator(boolean useIterator) {
this.useIterator = useIterator;
}
public boolean isRouteEmptyResultSet() {
return routeEmptyResultSet;
}
/**
* Whether allow empty resultset to be routed to the next hop
*/
public void setRouteEmptyResultSet(boolean routeEmptyResultSet) {
this.routeEmptyResultSet = routeEmptyResultSet;
}
}