-
Notifications
You must be signed in to change notification settings - Fork 175
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Bridge: rebuild rabbitmq producer on error (#1382)
Previously there were two defects related to RabbitMQ connection handling as a receiver output: 1. When unable to connect to rabbitmq on startup, bridge would exit immediately. 2. When forwarding to rabbitmq resulted in an error, the connection could be left in a bad state and not recover leaving all subsequent deliveries to the node to fail until bridge was restarted. To address both of these issues, `QueueForwarder` now initializes clients lazily, on demand, and will tear down the client when errors are encountered. When the client is discarded, the output will re-initialize another client using the same connection parameters at the time of the next delivery to that output. For webhook receiver and poller inputs, the HTTP client POSTing the input will see a 500 response in the case the publish to rabbit fails and can retry at their convenience. Whenever they do, the rabbitmq producer should either re-initialize (or fail trying). Rinse, repeat.
- Loading branch information
Showing
3 changed files
with
172 additions
and
11 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,4 +1,5 @@ | ||
mod gcp_pubsub_consumer; | ||
mod rabbitmq_consumer; | ||
mod rabbitmq_receiver; | ||
mod redis_stream_consumer; | ||
mod sqs_consumer; |
120 changes: 120 additions & 0 deletions
120
bridge/svix-bridge-plugin-queue/tests/it/rabbitmq_receiver.rs
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,120 @@ | ||
//! Requires a rabbitmq node to be running on localhost:5672 (the default port) and using the | ||
//! default guest/guest credentials. | ||
//! Try using the `testing-docker-compose.yml` in the repo root to get this going. | ||
use std::time::Duration; | ||
|
||
use lapin::{ | ||
options::QueueDeclareOptions, types::FieldTable, Channel, Connection, ConnectionProperties, | ||
Queue, | ||
}; | ||
use serde_json::json; | ||
use svix_bridge_plugin_queue::config::{QueueForwarder, QueueOutputOpts, RabbitMqOutputOpts}; | ||
use svix_bridge_types::{ForwardRequest, ReceiverOutput}; | ||
use tokio::{ | ||
io::copy_bidirectional, | ||
net::{TcpListener, TcpStream, ToSocketAddrs}, | ||
task::JoinHandle, | ||
}; | ||
|
||
async fn declare_queue(name: &str, channel: &Channel) -> Queue { | ||
channel | ||
.queue_declare( | ||
name, | ||
QueueDeclareOptions { | ||
auto_delete: true, | ||
..Default::default() | ||
}, | ||
FieldTable::default(), | ||
) | ||
.await | ||
.unwrap() | ||
} | ||
|
||
async fn mq_connection(uri: &str) -> Connection { | ||
let options = ConnectionProperties::default() | ||
.with_connection_name("test".into()) | ||
.with_executor(tokio_executor_trait::Tokio::current()) | ||
.with_reactor(tokio_reactor_trait::Tokio); | ||
Connection::connect(uri, options).await.unwrap() | ||
} | ||
|
||
const WAIT_MS: u64 = 200; | ||
|
||
/// These tests assume a "vanilla" rabbitmq instance, using the default port, creds, exchange... | ||
const MQ_URI: &str = "amqp://guest:guest@localhost:5672/%2f"; | ||
|
||
/// TCP proxy. Useful for giving us control over the connection to rabbit inside our tests. | ||
async fn proxy( | ||
listener: TcpListener, | ||
server_addr: impl ToSocketAddrs + Clone + Sync + Send + 'static, | ||
) -> Result<JoinHandle<()>, ()> { | ||
let handle = tokio::task::spawn(async move { | ||
while let Ok((mut inbound, _)) = listener.accept().await { | ||
let mut outbound = TcpStream::connect(server_addr.clone()).await.unwrap(); | ||
if let Err(e) = copy_bidirectional(&mut inbound, &mut outbound).await { | ||
eprintln!("Failed to transfer; error={}", e); | ||
} | ||
} | ||
}); | ||
Ok(handle) | ||
} | ||
|
||
#[tokio::test] | ||
async fn test_connection_recovery() { | ||
let mq_conn = mq_connection(MQ_URI).await; | ||
let channel = mq_conn.create_channel().await.unwrap(); | ||
// setup the queue before running the consumer or the consumer will error out | ||
let queue = declare_queue("", &channel).await; | ||
let queue_name = queue.name().as_str(); | ||
|
||
let proxy_listener = TcpListener::bind(("127.0.0.1", 0)).await.unwrap(); | ||
let port = proxy_listener.local_addr().unwrap().port(); | ||
// Start the proxy | ||
let proxy_handle = proxy(proxy_listener, "127.0.0.0:5672").await.unwrap(); | ||
|
||
// Configure the receiver output to connect to the proxy so we can interrupt the connection as needed. | ||
let proxied_mq_uri = format!("amqp://guest:guest@localhost:{port}/%2f"); | ||
|
||
let opts = QueueOutputOpts::RabbitMQ(RabbitMqOutputOpts { | ||
uri: proxied_mq_uri.clone(), | ||
exchange: "".to_string(), | ||
routing_key: queue_name.to_string(), | ||
publish_options: Default::default(), | ||
publish_properties: Default::default(), | ||
}); | ||
|
||
let output = QueueForwarder::from_receiver_output_opts(String::from("test"), opts) | ||
.await | ||
.unwrap(); | ||
|
||
let req = ForwardRequest { | ||
payload: json!({"test": true}), | ||
}; | ||
|
||
assert!( | ||
output.handle(req.clone()).await.is_ok(), | ||
"expected ok when rabbit available" | ||
); | ||
|
||
// Disconnect the proxy | ||
proxy_handle.abort(); | ||
// Sleep a beat to give time for the proxy tear down. | ||
tokio::time::sleep(Duration::from_millis(WAIT_MS)).await; | ||
|
||
assert!( | ||
output.handle(req.clone()).await.is_err(), | ||
"expected err when rabbit unavailable" | ||
); | ||
|
||
// Reconnect the proxy on the same port | ||
let proxy_listener = TcpListener::bind(("127.0.0.1", port)).await.unwrap(); | ||
let proxy_handle = proxy(proxy_listener, "127.0.0.0:5672").await.unwrap(); | ||
|
||
assert!( | ||
output.handle(req.clone()).await.is_ok(), | ||
"expected ok when rabbit available" | ||
); | ||
|
||
proxy_handle.abort(); | ||
} |