/**
* 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.pgevent;
import java.io.InvalidClassException;
import java.sql.DriverManager;
import java.util.Properties;
import javax.sql.DataSource;
import com.impossibl.postgres.api.jdbc.PGConnection;
import com.impossibl.postgres.jdbc.PGDataSource;
import org.apache.camel.Consumer;
import org.apache.camel.Processor;
import org.apache.camel.Producer;
import org.apache.camel.impl.DefaultEndpoint;
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.URISupport;
import org.slf4j.Logger;
import org.slf4j.LoggerFactory;
/**
* The pgevent component allows for producing/consuming PostgreSQL events related to the LISTEN/NOTIFY commands.
*
* This requires using PostgreSQL 8.3 or newer.
*/
@UriEndpoint(firstVersion = "2.15.0", scheme = "pgevent", title = "PostgresSQL Event", syntax = "pgevent:host:port/database/channel", consumerClass = PgEventConsumer.class, label = "database,sql")
public class PgEventEndpoint extends DefaultEndpoint {
private static final Logger LOG = LoggerFactory.getLogger(PgEventEndpoint.class);
private static final String FORMAT1 = "^pgevent://([^:]*):(\\d+)/(\\w+)/(\\w+).*$";
private static final String FORMAT2 = "^pgevent://([^:]+)/(\\w+)/(\\w+).*$";
private static final String FORMAT3 = "^pgevent:///(\\w+)/(\\w+).*$";
private static final String FORMAT4 = "^pgevent:(\\w+)/(\\w+)/(\\w+).*$";
@UriPath(defaultValue = "localhost")
private String host = "localhost";
@UriPath(defaultValue = "5432")
private Integer port = 5432;
@UriPath @Metadata(required = "true")
private String database;
@UriPath @Metadata(required = "true")
private String channel;
@UriParam(defaultValue = "postgres", label = "security", secret = true)
private String user = "postgres";
@UriParam(label = "security", secret = true)
private String pass;
@UriParam
private DataSource datasource;
private final String uri;
private PGConnection dbConnection;
public PgEventEndpoint(String uri, PgEventComponent component) {
super(uri, component);
this.uri = uri;
parseUri();
}
public PgEventEndpoint(String uri, PgEventComponent component, DataSource dataSource) {
super(uri, component);
this.uri = uri;
this.datasource = dataSource;
parseUri();
}
public final PGConnection initJdbc() throws Exception {
PGConnection conn;
Properties props = new Properties();
props.putAll(URISupport.parseQuery(uri));
if (this.getDatasource() != null) {
conn = (PGConnection) this.getDatasource().getConnection();
} else {
// ensure we can load the class
getCamelContext().getClassResolver().resolveMandatoryClass("com.impossibl.postgres.jdbc.PGDriver");
conn = (PGConnection) DriverManager.getConnection("jdbc:pgsql://" + this.getHost() + ":" + this.getPort() + "/" + this.getDatabase(), this.getUser(), this.getPass());
}
return conn;
}
/**
* Parse the provided URI and extract available parameters
*
* @throws IllegalArgumentException if there is an error in the parameters
*/
protected final void parseUri() throws IllegalArgumentException {
LOG.info("URI: " + uri);
if (uri.matches(FORMAT1)) {
LOG.info("FORMAT1");
String[] parts = uri.replaceFirst(FORMAT1, "$1:$2:$3:$4").split(":");
host = parts[0];
port = Integer.parseInt(parts[1]);
database = parts[2];
channel = parts[3];
} else if (uri.matches(FORMAT2)) {
LOG.info("FORMAT2");
String[] parts = uri.replaceFirst(FORMAT2, "$1:$2:$3").split(":");
host = parts[0];
port = 5432;
database = parts[1];
channel = parts[2];
} else if (uri.matches(FORMAT3)) {
LOG.info("FORMAT3");
String[] parts = uri.replaceFirst(FORMAT3, "$1:$2").split(":");
host = "localhost";
port = 5432;
database = parts[0];
channel = parts[1];
} else if (uri.matches(FORMAT4)) {
LOG.info("FORMAT4");
String[] parts = uri.replaceFirst(FORMAT4, "$1:$2").split(":");
database = parts[0];
channel = parts[1];
} else {
throw new IllegalArgumentException("The provided URL does not match the acceptable patterns.");
}
}
@Override
public Producer createProducer() throws Exception {
validateInputs();
return new PgEventProducer(this);
}
private void validateInputs() throws InvalidClassException, IllegalArgumentException {
if (getChannel() == null || getChannel().length() == 0) {
throw new IllegalArgumentException("A required parameter was not set when creating this Endpoint (channel)");
}
if (datasource != null) {
LOG.debug("******Datasource detected*****");
if (!PGDataSource.class.isInstance(datasource)) {
throw new InvalidClassException("The datasource passed to the "
+ "pgevent component is NOT a PGDataSource class from the"
+ "pgjdbc-ng library. See: https://github.com/impossibl/pgjdbc-ng");
}
} else {
if (user == null) {
throw new IllegalArgumentException("A required parameter was "
+ "not set when creating this Endpoint (pgUser or pgDataSource)");
}
}
}
@Override
public Consumer createConsumer(Processor processor) throws Exception {
validateInputs();
PgEventConsumer consumer = new PgEventConsumer(this, processor);
configureConsumer(consumer);
return consumer;
}
@Override
public boolean isSingleton() {
return true;
}
public String getHost() {
return host;
}
/**
* To connect using hostname and port to the database.
*/
public void setHost(String host) {
this.host = host;
}
public Integer getPort() {
return port;
}
/**
* To connect using hostname and port to the database.
*/
public void setPort(Integer port) {
this.port = port;
}
public String getDatabase() {
return database;
}
/**
* The database name
*/
public void setDatabase(String database) {
this.database = database;
}
public String getChannel() {
return channel;
}
/**
* The channel name
*/
public void setChannel(String channel) {
this.channel = channel;
}
public String getUser() {
return user;
}
/**
* Username for login
*/
public void setUser(String user) {
this.user = user;
}
public String getPass() {
return pass;
}
/**
* Password for login
*/
public void setPass(String pass) {
this.pass = pass;
}
public DataSource getDatasource() {
return datasource;
}
/**
* To connect using the given {@link javax.sql.DataSource} instead of using hostname and port.
*/
public void setDatasource(DataSource datasource) {
this.datasource = datasource;
}
}