Skip to content

Commit

Permalink
Merge pull request #78 from svix/jplatte/redis-legacy
Browse files Browse the repository at this point in the history
redis: Add non-stream fallback implementation
  • Loading branch information
svix-jplatte authored Apr 30, 2024
2 parents 12f0e4b + 8907130 commit f50cf3d
Show file tree
Hide file tree
Showing 8 changed files with 898 additions and 294 deletions.
4 changes: 2 additions & 2 deletions omniqueue/Cargo.toml
Original file line number Diff line number Diff line change
Expand Up @@ -27,7 +27,7 @@ serde = "1.0.196"
serde_json = "1"
svix-ksuid = { version = "0.8.0", optional = true }
thiserror = "1"
time = { version = "0.3.34", optional = true }
time = "0.3.34"
tokio = { version = "1", features = ["rt", "sync", "time"] }
tracing = "0.1"

Expand All @@ -45,7 +45,7 @@ in_memory = []
gcp_pubsub = ["dep:futures-util", "dep:google-cloud-googleapis", "dep:google-cloud-pubsub"]
rabbitmq = ["dep:futures-util", "dep:lapin"]
# Generate message IDs for queue items. Likely not needed outside of Svix.
rabbitmq-with-message-ids = ["rabbitmq", "dep:time", "dep:svix-ksuid"]
rabbitmq-with-message-ids = ["rabbitmq", "dep:svix-ksuid"]
redis = ["dep:bb8", "dep:bb8-redis", "dep:redis", "dep:svix-ksuid"]
redis_cluster = ["redis", "redis/cluster-async"]
sqs = ["dep:aws-config", "dep:aws-sdk-sqs"]
Expand Down
196 changes: 196 additions & 0 deletions omniqueue/src/backends/redis/fallback.rs
Original file line number Diff line number Diff line change
@@ -0,0 +1,196 @@
//! Implementation of the main queue using two lists instead of redis streams,
//! for compatibility with redis versions older than 6.2.0.
use std::time::Duration;

use async_trait::async_trait;
use bb8::ManageConnection;
use redis::AsyncCommands;
use svix_ksuid::{KsuidLike as _, KsuidMs};
use time::OffsetDateTime;
use tracing::{error, trace};

use super::{from_key, to_key, RawPayload, RedisConnection, RedisConsumer, RedisProducer};
use crate::{queue::Acker, Delivery, QueueError, Result};

pub(super) async fn send_raw<R: RedisConnection>(
producer: &RedisProducer<R>,
payload: &[u8],
) -> Result<()> {
producer
.redis
.get()
.await
.map_err(QueueError::generic)?
.lpush(&producer.queue_key, to_key(payload))
.await
.map_err(QueueError::generic)
}

pub(super) async fn receive<R: RedisConnection>(consumer: &RedisConsumer<R>) -> Result<Delivery> {
let res = receive_with_timeout(consumer, Duration::ZERO).await?;
res.ok_or_else(|| QueueError::Generic("No data".into()))
}

pub(super) async fn receive_all<R: RedisConnection>(
consumer: &RedisConsumer<R>,
deadline: Duration,
_max_messages: usize,
) -> Result<Vec<Delivery>> {
// FIXME: Run up to max_messages RPOPLPUSH'es until there is a null reply?
let delivery = receive_with_timeout(consumer, deadline).await?;
Ok(delivery.into_iter().collect())
}

async fn receive_with_timeout<R: RedisConnection>(
consumer: &RedisConsumer<R>,
timeout: Duration,
) -> Result<Option<Delivery>> {
let key: Option<Vec<u8>> = consumer
.redis
.get()
.await
.map_err(QueueError::generic)?
.brpoplpush(
&consumer.queue_key,
&consumer.processing_queue_key,
// The documentation at https://redis.io/docs/latest/commands/brpoplpush/ does not
// state what unit the timeout is, but `BLPOP` and `BLMPOP` have similar timeout
// parameters that are documented as being seconds.
timeout.as_secs_f64(),
)
.await
.map_err(QueueError::generic)?;

key.map(|key| make_delivery(consumer, &key)).transpose()
}

fn make_delivery<R: RedisConnection>(consumer: &RedisConsumer<R>, key: &[u8]) -> Result<Delivery> {
let (_, payload) = from_key(key)?;

Ok(Delivery {
payload: Some(payload.to_owned()),
acker: Box::new(RedisFallbackAcker {
redis: consumer.redis.clone(),
processing_queue_key: consumer.processing_queue_key.clone(),
key: key.to_owned(),
already_acked_or_nacked: false,
}),
})
}

struct RedisFallbackAcker<M: ManageConnection> {
redis: bb8::Pool<M>,
processing_queue_key: String,
key: RawPayload,

already_acked_or_nacked: bool,
}

#[async_trait]
impl<R: RedisConnection> Acker for RedisFallbackAcker<R> {
async fn ack(&mut self) -> Result<()> {
if self.already_acked_or_nacked {
return Err(QueueError::CannotAckOrNackTwice);
}

self.already_acked_or_nacked = true;

self.redis
.get()
.await
.map_err(QueueError::generic)?
.lrem(&self.processing_queue_key, 1, &self.key)
.await
.map_err(QueueError::generic)?;

Ok(())
}

async fn nack(&mut self) -> Result<()> {
if self.already_acked_or_nacked {
return Err(QueueError::CannotAckOrNackTwice);
}

self.already_acked_or_nacked = true;

Ok(())
}
}

pub(super) async fn add_to_main_queue(
keys: &[Vec<u8>],
main_queue_name: &str,
conn: &mut impl AsyncCommands,
) -> Result<()> {
let new_keys = keys
.iter()
.map(|x| regenerate_key(x))
.collect::<Result<Vec<_>>>()?;
let _: () = conn
.lpush(main_queue_name, new_keys)
.await
.map_err(QueueError::generic)?;
Ok(())
}

pub(super) async fn background_task_processing<R: RedisConnection>(
pool: bb8::Pool<R>,
queue_key: String,
processing_queue_key: String,
ack_deadline_ms: i64,
) -> Result<()> {
// FIXME: ack_deadline_ms should be unsigned
let ack_deadline = Duration::from_millis(ack_deadline_ms as _);
loop {
if let Err(err) =
reenqueue_timed_out_messages(&pool, &queue_key, &processing_queue_key, ack_deadline)
.await
{
error!("{err}");
tokio::time::sleep(Duration::from_millis(500)).await;
continue;
}
}
}

async fn reenqueue_timed_out_messages<R: RedisConnection>(
pool: &bb8::Pool<R>,
queue_key: &str,
processing_queue_key: &str,
ack_deadline: Duration,
) -> Result<(), Box<dyn std::error::Error + Send + Sync>> {
const BATCH_SIZE: isize = 50;

let mut conn = pool.get().await?;

let keys: Vec<RawPayload> = conn.lrange(processing_queue_key, 0, 1).await?;

// If the key is older than now, it means we should be processing keys
let validity_limit = KsuidMs::new(Some(OffsetDateTime::now_utc() - ack_deadline), None)
.to_string()
.into_bytes();

if !keys.is_empty() && keys[0] <= validity_limit {
let keys: Vec<RawPayload> = conn.lrange(processing_queue_key, 0, BATCH_SIZE).await?;
for key in keys {
if key <= validity_limit {
// We use LREM to be sure we only delete the keys we should be deleting
trace!("Pushing back overdue task to queue");
let refreshed_key = regenerate_key(&key)?;
let _: () = conn.rpush(queue_key, &refreshed_key).await?;
let _: () = conn.lrem(processing_queue_key, 1, &key).await?;
}
}
} else {
// Sleep before attempting to fetch again if nothing was found
tokio::time::sleep(Duration::from_millis(500)).await;
}

Ok(())
}

fn regenerate_key(key: &[u8]) -> Result<RawPayload> {
let (_, payload) = from_key(key)?;
Ok(to_key(payload))
}
Loading

0 comments on commit f50cf3d

Please sign in to comment.