use std::error::Error;
use std::future::Future;
use std::marker::PhantomData;
use std::pin::Pin;
use std::sync::Arc;
use std::task::{Context, Poll};
use std::time::{Duration, Instant};
use futures_channel::oneshot;
use futures_util::FutureExt;
use crate::client::{BrokerAddr, Client, ClientContext, DefaultClientContext, OAuthToken};
use crate::config::{ClientConfig, FromClientConfig, FromClientConfigAndContext, RDKafkaLogLevel};
use crate::consumer::ConsumerGroupMetadata;
use crate::error::{KafkaError, KafkaResult, RDKafkaErrorCode};
use crate::message::{Message, OwnedHeaders, OwnedMessage, Timestamp, ToBytes};
use crate::producer::{
BaseRecord, DeliveryResult, NoCustomPartitioner, Producer, ProducerContext, PurgeConfig,
ThreadedProducer,
};
use crate::statistics::Statistics;
use crate::topic_partition_list::TopicPartitionList;
use crate::util::{AsyncRuntime, DefaultRuntime, IntoOpaque, Timeout};
use super::Partitioner;
#[derive(Debug)]
pub struct FutureRecord<'a, K: ToBytes + ?Sized, P: ToBytes + ?Sized> {
pub topic: &'a str,
pub partition: Option<i32>,
pub payload: Option<&'a P>,
pub key: Option<&'a K>,
pub timestamp: Option<i64>,
pub headers: Option<OwnedHeaders>,
}
impl<'a, K: ToBytes + ?Sized, P: ToBytes + ?Sized> FutureRecord<'a, K, P> {
pub fn to(topic: &'a str) -> FutureRecord<'a, K, P> {
FutureRecord {
topic,
partition: None,
payload: None,
key: None,
timestamp: None,
headers: None,
}
}
fn from_base_record<D: IntoOpaque>(
base_record: BaseRecord<'a, K, P, D>,
) -> FutureRecord<'a, K, P> {
FutureRecord {
topic: base_record.topic,
partition: base_record.partition,
key: base_record.key,
payload: base_record.payload,
timestamp: base_record.timestamp,
headers: base_record.headers,
}
}
pub fn partition(mut self, partition: i32) -> FutureRecord<'a, K, P> {
self.partition = Some(partition);
self
}
pub fn payload(mut self, payload: &'a P) -> FutureRecord<'a, K, P> {
self.payload = Some(payload);
self
}
pub fn key(mut self, key: &'a K) -> FutureRecord<'a, K, P> {
self.key = Some(key);
self
}
pub fn timestamp(mut self, timestamp: i64) -> FutureRecord<'a, K, P> {
self.timestamp = Some(timestamp);
self
}
pub fn headers(mut self, headers: OwnedHeaders) -> FutureRecord<'a, K, P> {
self.headers = Some(headers);
self
}
fn into_base_record<D: IntoOpaque>(self, delivery_opaque: D) -> BaseRecord<'a, K, P, D> {
BaseRecord {
topic: self.topic,
partition: self.partition,
key: self.key,
payload: self.payload,
timestamp: self.timestamp,
headers: self.headers,
delivery_opaque,
}
}
}
#[derive(Clone)]
pub struct FutureProducerContext<C: ClientContext + 'static> {
wrapped_context: C,
}
pub type OwnedDeliveryResult = Result<(i32, i64), (KafkaError, OwnedMessage)>;
impl<C: ClientContext + 'static> ClientContext for FutureProducerContext<C> {
fn enable_refresh_oauth_token(&self) -> bool {
self.wrapped_context.enable_refresh_oauth_token()
}
fn log(&self, level: RDKafkaLogLevel, fac: &str, log_message: &str) {
self.wrapped_context.log(level, fac, log_message);
}
fn stats(&self, statistics: Statistics) {
self.wrapped_context.stats(statistics);
}
fn stats_raw(&self, statistics: &[u8]) {
self.wrapped_context.stats_raw(statistics)
}
fn error(&self, error: KafkaError, reason: &str) {
self.wrapped_context.error(error, reason);
}
fn rewrite_broker_addr(&self, addr: BrokerAddr) -> BrokerAddr {
self.wrapped_context.rewrite_broker_addr(addr)
}
fn generate_oauth_token(
&self,
oauthbearer_config: Option<&str>,
) -> Result<OAuthToken, Box<dyn Error>> {
self.wrapped_context
.generate_oauth_token(oauthbearer_config)
}
}
impl<C, Part> ProducerContext<Part> for FutureProducerContext<C>
where
C: ClientContext + 'static,
Part: Partitioner,
{
type DeliveryOpaque = Box<oneshot::Sender<OwnedDeliveryResult>>;
fn delivery(
&self,
delivery_result: &DeliveryResult<'_>,
tx: Box<oneshot::Sender<OwnedDeliveryResult>>,
) {
let owned_delivery_result = match *delivery_result {
Ok(ref message) => Ok((message.partition(), message.offset())),
Err((ref error, ref message)) => Err((error.clone(), message.detach())),
};
let _ = tx.send(owned_delivery_result); }
}
#[must_use = "Producer polling thread will stop immediately if unused"]
pub struct FutureProducer<C = DefaultClientContext, R = DefaultRuntime, Part = NoCustomPartitioner>
where
Part: Partitioner,
C: ClientContext + 'static,
{
producer: Arc<ThreadedProducer<FutureProducerContext<C>, Part>>,
_runtime: PhantomData<R>,
}
impl<C, R> Clone for FutureProducer<C, R>
where
C: ClientContext + 'static,
{
fn clone(&self) -> FutureProducer<C, R> {
FutureProducer {
producer: self.producer.clone(),
_runtime: PhantomData,
}
}
}
#[async_trait::async_trait]
impl<R> FromClientConfig for FutureProducer<DefaultClientContext, R>
where
R: AsyncRuntime,
{
async fn from_config(
config: &ClientConfig,
) -> KafkaResult<FutureProducer<DefaultClientContext, R>> {
FutureProducer::from_config_and_context(config, DefaultClientContext).await
}
}
#[async_trait::async_trait]
impl<C, R> FromClientConfigAndContext<C> for FutureProducer<C, R>
where
C: ClientContext + 'static,
R: AsyncRuntime,
{
async fn from_config_and_context(
config: &ClientConfig,
context: C,
) -> KafkaResult<FutureProducer<C, R>> {
let future_context = FutureProducerContext {
wrapped_context: context,
};
let threaded_producer =
ThreadedProducer::from_config_and_context(config, future_context).await?;
Ok(FutureProducer {
producer: Arc::new(threaded_producer),
_runtime: PhantomData,
})
}
}
pub struct DeliveryFuture {
rx: oneshot::Receiver<OwnedDeliveryResult>,
}
impl Future for DeliveryFuture {
type Output = Result<OwnedDeliveryResult, oneshot::Canceled>;
fn poll(mut self: Pin<&mut Self>, cx: &mut Context<'_>) -> Poll<Self::Output> {
self.rx.poll_unpin(cx)
}
}
impl<C, R> FutureProducer<C, R>
where
C: ClientContext + 'static,
R: AsyncRuntime,
{
pub async fn send<K, P, T>(
&self,
record: FutureRecord<'_, K, P>,
queue_timeout: T,
) -> OwnedDeliveryResult
where
K: ToBytes + ?Sized,
P: ToBytes + ?Sized,
T: Into<Timeout>,
{
let start_time = Instant::now();
let queue_timeout = queue_timeout.into();
let can_retry = || match queue_timeout {
Timeout::Never => true,
Timeout::After(t) if start_time.elapsed() < t => true,
_ => false,
};
let (tx, rx) = oneshot::channel();
let mut base_record = record.into_base_record(Box::new(tx));
loop {
match self.producer.send(base_record) {
Err((e, record))
if e == KafkaError::MessageProduction(RDKafkaErrorCode::QueueFull)
&& can_retry() =>
{
base_record = record;
R::delay_for(Duration::from_millis(100)).await;
}
Ok(_) => {
break rx.await.expect("producer unexpectedly dropped");
}
Err((e, record)) => {
let owned_message = OwnedMessage::new(
record.payload.map(|p| p.to_bytes().to_vec()),
record.key.map(|k| k.to_bytes().to_vec()),
record.topic.to_owned(),
record
.timestamp
.map_or(Timestamp::NotAvailable, Timestamp::CreateTime),
record.partition.unwrap_or(-1),
0,
record.headers,
);
break Err((e, owned_message));
}
}
}
}
pub fn send_result<'a, K, P>(
&self,
record: FutureRecord<'a, K, P>,
) -> Result<DeliveryFuture, (KafkaError, FutureRecord<'a, K, P>)>
where
K: ToBytes + ?Sized,
P: ToBytes + ?Sized,
{
let (tx, rx) = oneshot::channel();
let base_record = record.into_base_record(Box::new(tx));
self.producer
.send(base_record)
.map(|()| DeliveryFuture { rx })
.map_err(|(e, record)| (e, FutureRecord::from_base_record(record)))
}
pub fn poll<T: Into<Timeout> + Send>(&self, timeout: T) {
self.producer.poll(timeout);
}
}
#[async_trait::async_trait]
impl<C, R, Part> Producer<FutureProducerContext<C>, Part> for FutureProducer<C, R, Part>
where
C: ClientContext + 'static,
R: AsyncRuntime,
Part: Partitioner,
{
fn client(&self) -> &Client<FutureProducerContext<C>> {
self.producer.client()
}
async fn flush<T: Into<Timeout> + Send>(&self, timeout: T) -> KafkaResult<()> {
self.producer.flush(timeout).await
}
fn purge(&self, flags: PurgeConfig) {
self.producer.purge(flags)
}
fn in_flight_count(&self) -> i32 {
self.producer.in_flight_count()
}
async fn init_transactions<T: Into<Timeout> + Send>(&self, timeout: T) -> KafkaResult<()> {
self.producer.init_transactions(timeout).await
}
fn begin_transaction(&self) -> KafkaResult<()> {
self.producer.begin_transaction()
}
async fn send_offsets_to_transaction<T: Into<Timeout> + Send>(
&self,
offsets: &TopicPartitionList,
cgm: &ConsumerGroupMetadata,
timeout: T,
) -> KafkaResult<()> {
self.producer
.send_offsets_to_transaction(offsets, cgm, timeout)
.await
}
async fn commit_transaction<T: Into<Timeout> + Send>(&self, timeout: T) -> KafkaResult<()> {
self.producer.commit_transaction(timeout).await
}
async fn abort_transaction<T: Into<Timeout> + Send>(&self, timeout: T) -> KafkaResult<()> {
self.producer.abort_transaction(timeout).await
}
}
#[cfg(test)]
mod tests {
use super::*;
use crate::config::ClientConfig;
struct TestContext;
impl ClientContext for TestContext {}
impl ProducerContext<NoCustomPartitioner> for TestContext {
type DeliveryOpaque = Box<i32>;
fn delivery(&self, _: &DeliveryResult<'_>, _: Self::DeliveryOpaque) {
unimplemented!()
}
}
#[tokio::test]
async fn test_future_producer_clone() {
let producer = ClientConfig::new()
.create::<FutureProducer>()
.await
.unwrap();
let _producer_clone = producer.clone();
}
#[tokio::test]
async fn test_base_future_topic_send_sync() {
let test_context = TestContext;
let producer = ClientConfig::new()
.create_with_context::<_, FutureProducer<TestContext>>(test_context)
.await
.unwrap();
let _producer_clone = producer.clone();
}
}