io.vertx.camel.impl.CamelBridgeImpl Maven / Gradle / Ivy
/*
* Copyright (c) 2011-2015 The original author or authors
* ------------------------------------------------------
* All rights reserved. This program and the accompanying materials
* are made available under the terms of the Eclipse Public License v1.0
* and Apache License v2.0 which accompanies this distribution.
*
* The Eclipse Public License is available at
* http://www.eclipse.org/legal/epl-v10.html
*
* The Apache License v2.0 is available at
* http://www.opensource.org/licenses/apache2.0.php
*
* You may elect to redistribute this code under either of these licenses.
*/
package io.vertx.camel.impl;
import java.util.ArrayList;
import java.util.List;
import java.util.Objects;
import io.vertx.camel.CamelBridge;
import io.vertx.camel.CamelBridgeOptions;
import io.vertx.camel.CamelMapping;
import io.vertx.camel.InboundMapping;
import io.vertx.camel.OutboundMapping;
import io.vertx.core.AsyncResult;
import io.vertx.core.Handler;
import io.vertx.core.Vertx;
import io.vertx.core.eventbus.MessageConsumer;
import io.vertx.core.impl.logging.Logger;
import io.vertx.core.impl.logging.LoggerFactory;
import org.apache.camel.CamelContext;
import org.apache.camel.Consumer;
import org.apache.camel.Endpoint;
import org.apache.camel.ExtendedStartupListener;
import org.apache.camel.Producer;
/**
* The implementation of the camel bridge.
*
* @author Clement Escoffier
*/
public class CamelBridgeImpl implements CamelBridge {
private final CamelContext camel;
private final List camelConsumers = new ArrayList<>();
private final List camelProducers = new ArrayList<>();
private final List vertxConsumers = new ArrayList<>();
private static final Logger LOGGER = LoggerFactory.getLogger(CamelBridgeImpl.class);
private final Vertx vertx;
/**
* Creates an instance of the bridge.
*
* @param vertx the Vert.x instance, must not be {@code null}
* @param options the bridge configuration, must not be {@code null}
*/
public CamelBridgeImpl(Vertx vertx, CamelBridgeOptions options) {
Objects.requireNonNull(vertx);
Objects.requireNonNull(options);
this.camel = options.getCamelContext();
Objects.requireNonNull(camel);
this.vertx = vertx;
// validate all endpoints eager so we get errors early
options.getInboundMappings().forEach(this::validate);
options.getOutboundMappings().forEach(this::validate);
try {
// setup the inbound and outbound bridge after camel has been started (so all camel components are started)
this.camel.addStartupListener(new ExtendedStartupListener() {
public void onCamelContextFullyStarted(CamelContext context, boolean alreadyStarted) throws Exception {
for (InboundMapping inbound : options.getInboundMappings()) {
// camel -> vert.x
createInboundBridge(vertx, inbound);
}
for (OutboundMapping outbound : options.getOutboundMappings()) {
// vert.x -> camel
createOutboundBridge(vertx, outbound);
}
}
@Override
public void onCamelContextStarted(CamelContext context, boolean alreadyStarted) throws Exception {
// noop
}
});
} catch (Exception e) {
throw new IllegalStateException("Error preparing Camel with vert.x", e);
}
}
private void createOutboundBridge(Vertx vertx, OutboundMapping outbound) {
Endpoint endpoint = validate(outbound);
Producer producer;
try {
producer = endpoint.createProducer();
camelProducers.add(producer);
} catch (Exception e) {
throw new IllegalStateException("The endpoint " + outbound.getUri() + " does not support producers", e);
}
LOGGER.debug("Creating Vert.x message consumer for " + outbound.getUri() + " receiving messages from "
+ outbound.getAddress());
vertxConsumers.add(vertx.eventBus().consumer(outbound.getAddress(),
new FromVertxToCamelProducer(vertx, producer, outbound, outbound.isBlocking(), outbound.getWorkerExecutor())));
LOGGER.info("Created Vert.x message consumer for " + outbound.getUri() + " receiving messages from "
+ outbound.getAddress());
}
private void createInboundBridge(Vertx vertx, InboundMapping inbound) {
Endpoint endpoint = validate(inbound);
try {
LOGGER.debug("Creating camel consumer for " + inbound.getUri() + " sending messages to " + inbound.getAddress());
camelConsumers.add(endpoint.createConsumer(new CamelToVertxProcessor(vertx, inbound)));
} catch (Exception e) {
throw new IllegalStateException("The endpoint " + inbound.getUri() + " does not support consumers", e);
}
LOGGER.info("Created camel consumer for " + inbound.getUri() + " sending messages to " + inbound.getAddress());
}
private Endpoint validate(CamelMapping mapping) {
Objects.requireNonNull(mapping.getAddress(), "The vert.x event bus address must not be `null`");
Objects.requireNonNull(mapping.getUri(), "The endpoint uri must not be `null`");
Endpoint endpoint = camel.getEndpoint(mapping.getUri());
Objects.requireNonNull(endpoint, "Cannot find the endpoint " + mapping.getUri() + " in the camel context");
return endpoint;
}
@Override
public CamelBridge start() {
return start(null);
}
@Override
public CamelBridge start(Handler> completed) {
vertx.executeBlocking(
future -> {
camelConsumers.stream().forEach(c -> {
try {
c.start();
} catch (Exception e) {
future.fail(e);
}
});
camelProducers.stream().forEach(c -> {
try {
c.start();
} catch (Exception e) {
future.fail(e);
}
});
future.complete();
},
completed
);
return this;
}
@Override
public CamelBridge stop() {
return stop(null);
}
@Override
public CamelBridge stop(Handler> completed) {
vertx.executeBlocking(
future -> {
camelConsumers.stream().forEach(c -> {
try {
c.stop();
} catch (Exception e) {
future.fail(e);
}
});
camelProducers.stream().forEach(c -> {
try {
c.stop();
} catch (Exception e) {
future.fail(e);
}
});
vertxConsumers.stream().forEach(MessageConsumer::unregister);
future.complete();
},
completed
);
return this;
}
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy