blob: 3ee8def37dbaffaa50122a3ed0d44a27ed9c831b [file] [log] [blame]
/**
* 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.
*/
#include <pulsar/Client.h>
#include <gtest/gtest.h>
#include <thread>
#include "HttpHelper.h"
#include "lib/Future.h"
#include "lib/Utils.h"
#include "lib/Latch.h"
#include "lib/LogUtils.h"
#include "lib/ProducerImpl.h"
DECLARE_LOG_OBJECT()
using namespace pulsar;
static const std::string serviceUrl = "pulsar://localhost:6650";
static const std::string adminUrl = "http://localhost:8080/";
TEST(ProducerTest, producerNotInitialized) {
Producer producer;
Message msg = MessageBuilder().setContent("test").build();
ASSERT_EQ(ResultProducerNotInitialized, producer.send(msg));
Promise<Result, MessageId> promise;
producer.sendAsync(msg, WaitForCallbackValue<MessageId>(promise));
MessageId mi;
ASSERT_EQ(ResultProducerNotInitialized, promise.getFuture().get(mi));
ASSERT_EQ(ResultProducerNotInitialized, producer.close());
Promise<bool, Result> promiseClose;
producer.closeAsync(WaitForCallback(promiseClose));
Result result;
promiseClose.getFuture().get(result);
ASSERT_EQ(ResultProducerNotInitialized, result);
ASSERT_TRUE(producer.getTopic().empty());
}
TEST(ProducerTest, exactlyOnceWithProducerNameSpecified) {
Client client(serviceUrl);
std::string topicName = "persistent://public/default/exactlyOnceWithProducerNameSpecified";
Producer producer1;
ProducerConfiguration producerConfiguration1;
producerConfiguration1.setProducerName("p-name-1");
ASSERT_EQ(ResultOk, client.createProducer(topicName, producerConfiguration1, producer1));
Producer producer2;
ProducerConfiguration producerConfiguration2;
producerConfiguration2.setProducerName("p-name-2");
ASSERT_EQ(ResultOk, client.createProducer(topicName, producerConfiguration2, producer2));
Producer producer3;
Result result = client.createProducer(topicName, producerConfiguration2, producer3);
ASSERT_EQ(ResultProducerBusy, result);
}
TEST(ProducerTest, testSynchronouslySend) {
Client client(serviceUrl);
const std::string topic = "ProducerTestSynchronouslySend";
Consumer consumer;
ASSERT_EQ(ResultOk, client.subscribe(topic, "sub-name", consumer));
Producer producer;
ASSERT_EQ(ResultOk, client.createProducer(topic, producer));
MessageId messageId;
ASSERT_EQ(ResultOk, producer.send(MessageBuilder().setContent("hello").build(), messageId));
LOG_INFO("Send message to " << messageId);
Message receivedMessage;
ASSERT_EQ(ResultOk, consumer.receive(receivedMessage, 3000));
LOG_INFO("Received message from " << receivedMessage.getMessageId());
ASSERT_EQ(receivedMessage.getMessageId(), messageId);
ASSERT_EQ(ResultOk, consumer.acknowledge(receivedMessage));
client.close();
}
TEST(ProducerTest, testIsConnected) {
Client client(serviceUrl);
const std::string nonPartitionedTopic =
"testProducerIsConnectedNonPartitioned-" + std::to_string(time(nullptr));
const std::string partitionedTopic =
"testProducerIsConnectedPartitioned-" + std::to_string(time(nullptr));
Producer producer;
ASSERT_FALSE(producer.isConnected());
// ProducerImpl
ASSERT_EQ(ResultOk, client.createProducer(nonPartitionedTopic, producer));
ASSERT_TRUE(producer.isConnected());
ASSERT_EQ(ResultOk, producer.close());
ASSERT_FALSE(producer.isConnected());
int res = makePutRequest(
adminUrl + "admin/v2/persistent/public/default/" + partitionedTopic + "/partitions", "2");
ASSERT_TRUE(res == 204 || res == 409) << "res: " << res;
// PartitionedProducerImpl
ASSERT_EQ(ResultOk, client.createProducer(partitionedTopic, producer));
ASSERT_TRUE(producer.isConnected());
ASSERT_EQ(ResultOk, producer.close());
ASSERT_FALSE(producer.isConnected());
client.close();
}
TEST(ProducerTest, testSendAsyncAfterCloseAsyncWithLazyProducers) {
Client client(serviceUrl);
const std::string partitionedTopic =
"testProducerIsConnectedPartitioned-" + std::to_string(time(nullptr));
int res = makePutRequest(
adminUrl + "admin/v2/persistent/public/default/" + partitionedTopic + "/partitions", "10");
ASSERT_TRUE(res == 204 || res == 409) << "res: " << res;
ProducerConfiguration producerConfiguration;
producerConfiguration.setLazyStartPartitionedProducers(true);
Producer producer;
ASSERT_EQ(ResultOk, client.createProducer(partitionedTopic, producerConfiguration, producer));
Message msg = MessageBuilder().setContent("test").build();
Promise<bool, Result> promiseClose;
producer.closeAsync(WaitForCallback(promiseClose));
Promise<Result, MessageId> promise;
producer.sendAsync(msg, WaitForCallbackValue<MessageId>(promise));
MessageId mi;
ASSERT_EQ(ResultAlreadyClosed, promise.getFuture().get(mi));
Result result;
promiseClose.getFuture().get(result);
ASSERT_EQ(ResultOk, result);
}
TEST(ProducerTest, testSendAsyncCloseAsyncConcurrentlyWithLazyProducers) {
// run sendAsync and closeAsync concurrently and verify that all sendAsync callbacks are called
// and that messages sent after closeAsync is invoked receive ResultAlreadyClosed.
for (int run = 0; run < 20; run++) {
LOG_INFO("Start of run " << run);
Client client(serviceUrl);
const std::string partitionedTopic =
"testProducerIsConnectedPartitioned-" + std::to_string(time(nullptr));
int res = makePutRequest(
adminUrl + "admin/v2/persistent/public/default/" + partitionedTopic + "/partitions", "10");
ASSERT_TRUE(res == 204 || res == 409) << "res: " << res;
ProducerConfiguration producerConfiguration;
producerConfiguration.setLazyStartPartitionedProducers(true);
producerConfiguration.setPartitionsRoutingMode(ProducerConfiguration::UseSinglePartition);
producerConfiguration.setBatchingEnabled(true);
Producer producer;
ASSERT_EQ(ResultOk, client.createProducer(partitionedTopic, producerConfiguration, producer));
int sendCount = 100;
std::vector<Promise<Result, MessageId>> promises(sendCount);
Promise<bool, Result> promiseClose;
// only call closeAsync once at least 10 messages have been sent
Latch sendStartLatch(10);
Latch closeLatch(1);
int closedAt = 0;
std::thread t1([&]() {
for (int i = 0; i < sendCount; i++) {
sendStartLatch.countdown();
Message msg = MessageBuilder().setContent("test").build();
if (closeLatch.getCount() == 0 && closedAt == 0) {
closedAt = i;
LOG_INFO("closedAt set to " << closedAt)
}
producer.sendAsync(msg, WaitForCallbackValue<MessageId>(promises[i]));
std::this_thread::sleep_for(std::chrono::milliseconds(1));
}
});
std::thread t2([&]() {
sendStartLatch.wait(std::chrono::milliseconds(1000));
LOG_INFO("Closing");
producer.closeAsync(WaitForCallback(promiseClose));
LOG_INFO("Close called");
closeLatch.countdown();
Result result;
promiseClose.getFuture().get(result);
ASSERT_EQ(ResultOk, result);
LOG_INFO("Closed");
});
t1.join();
t2.join();
// make sure that all messages after the moment when closeAsync was invoked
// return AlreadyClosed
for (int i = 0; i < sendCount; i++) {
LOG_DEBUG("Checking " << i)
// whether a message was sent successfully or not, it's callback
// must have been invoked
ASSERT_EQ(true, promises[i].isComplete());
MessageId mi;
Result res = promises[i].getFuture().get(mi);
LOG_DEBUG("Result is " << res);
// for the messages sent after closeAsync was invoked, they
// should all return ResultAlreadyClosed
if (i >= closedAt) {
ASSERT_EQ(ResultAlreadyClosed, res);
}
}
client.close();
LOG_INFO("End of run " << run);
}
}
TEST(ProducerTest, testGetNumOfChunks) {
ASSERT_EQ(ProducerImpl::getNumOfChunks(11, 5), 3);
ASSERT_EQ(ProducerImpl::getNumOfChunks(10, 5), 2);
ASSERT_EQ(ProducerImpl::getNumOfChunks(8, 5), 2);
ASSERT_EQ(ProducerImpl::getNumOfChunks(4, 5), 1);
ASSERT_EQ(ProducerImpl::getNumOfChunks(1, 0), 1);
}