2019-03-20 22:29:02 +01:00
|
|
|
/*
|
|
|
|
* IXRedisClient.cpp
|
|
|
|
* Author: Benjamin Sergeant
|
|
|
|
* Copyright (c) 2019 Machine Zone, Inc. All rights reserved.
|
|
|
|
*/
|
|
|
|
|
|
|
|
#include "IXRedisClient.h"
|
|
|
|
#include <ixwebsocket/IXSocketFactory.h>
|
|
|
|
#include <ixwebsocket/IXSocket.h>
|
|
|
|
|
2019-04-23 02:24:01 +02:00
|
|
|
#include <iostream>
|
2019-03-20 22:29:02 +01:00
|
|
|
#include <sstream>
|
|
|
|
#include <iomanip>
|
|
|
|
#include <vector>
|
|
|
|
#include <cstring>
|
|
|
|
|
|
|
|
namespace ix
|
|
|
|
{
|
|
|
|
bool RedisClient::connect(const std::string& hostname, int port)
|
|
|
|
{
|
|
|
|
bool tls = false;
|
|
|
|
std::string errorMsg;
|
|
|
|
_socket = createSocket(tls, errorMsg);
|
|
|
|
|
|
|
|
if (!_socket)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
std::string errMsg;
|
|
|
|
return _socket->connect(hostname, port, errMsg, nullptr);
|
|
|
|
}
|
|
|
|
|
2019-03-26 17:33:22 +01:00
|
|
|
bool RedisClient::auth(const std::string& password,
|
|
|
|
std::string& response)
|
|
|
|
{
|
|
|
|
response.clear();
|
|
|
|
|
|
|
|
if (!_socket) return false;
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "AUTH ";
|
|
|
|
ss << password;
|
|
|
|
ss << "\r\n";
|
|
|
|
|
|
|
|
bool sent = _socket->writeBytes(ss.str(), nullptr);
|
|
|
|
if (!sent)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pollResult = _socket->isReadyToRead(-1);
|
|
|
|
if (pollResult == PollResultType::Error)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto lineResult = _socket->readLine(nullptr);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
|
|
|
|
|
|
|
response = line;
|
|
|
|
return lineValid;
|
|
|
|
}
|
|
|
|
|
2019-04-23 02:24:01 +02:00
|
|
|
std::string RedisClient::writeString(const std::string& str)
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "$";
|
|
|
|
ss << str.size();
|
|
|
|
ss << "\r\n";
|
|
|
|
ss << str;
|
|
|
|
ss << "\r\n";
|
|
|
|
|
|
|
|
return ss.str();
|
|
|
|
}
|
2019-03-26 17:33:22 +01:00
|
|
|
|
2019-03-20 22:29:02 +01:00
|
|
|
bool RedisClient::publish(const std::string& channel,
|
2019-04-23 02:24:01 +02:00
|
|
|
const std::string& message,
|
|
|
|
std::string& errMsg)
|
2019-03-20 22:29:02 +01:00
|
|
|
{
|
2019-04-23 02:24:01 +02:00
|
|
|
errMsg.clear();
|
|
|
|
|
|
|
|
if (!_socket)
|
|
|
|
{
|
|
|
|
errMsg = "socket is not initialized";
|
|
|
|
return false;
|
|
|
|
}
|
2019-03-20 22:29:02 +01:00
|
|
|
|
|
|
|
std::stringstream ss;
|
2019-04-23 02:24:01 +02:00
|
|
|
ss << "*3\r\n";
|
|
|
|
ss << writeString("PUBLISH");
|
|
|
|
ss << writeString(channel);
|
|
|
|
ss << writeString(message);
|
2019-03-20 22:29:02 +01:00
|
|
|
|
|
|
|
bool sent = _socket->writeBytes(ss.str(), nullptr);
|
|
|
|
if (!sent)
|
|
|
|
{
|
2019-04-23 02:24:01 +02:00
|
|
|
errMsg = "Cannot write bytes to socket";
|
2019-03-20 22:29:02 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto pollResult = _socket->isReadyToRead(-1);
|
|
|
|
if (pollResult == PollResultType::Error)
|
|
|
|
{
|
2019-04-23 02:24:01 +02:00
|
|
|
errMsg = "Error while polling for result";
|
2019-03-20 22:29:02 +01:00
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
auto lineResult = _socket->readLine(nullptr);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
|
|
|
|
2019-04-23 02:24:01 +02:00
|
|
|
// A successful response starts with a :
|
|
|
|
if (line.empty() || line[0] != ':')
|
|
|
|
{
|
|
|
|
errMsg = line;
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-20 22:29:02 +01:00
|
|
|
return lineValid;
|
|
|
|
}
|
|
|
|
|
|
|
|
//
|
|
|
|
// FIXME: we assume that redis never return errors...
|
|
|
|
//
|
|
|
|
bool RedisClient::subscribe(const std::string& channel,
|
2019-03-26 17:33:22 +01:00
|
|
|
const OnRedisSubscribeResponseCallback& responseCallback,
|
2019-03-20 22:29:02 +01:00
|
|
|
const OnRedisSubscribeCallback& callback)
|
|
|
|
{
|
|
|
|
if (!_socket) return false;
|
|
|
|
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << "SUBSCRIBE ";
|
|
|
|
ss << channel;
|
|
|
|
ss << "\r\n";
|
|
|
|
|
|
|
|
bool sent = _socket->writeBytes(ss.str(), nullptr);
|
|
|
|
if (!sent)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
|
|
|
// Wait 1s for the response
|
|
|
|
auto pollResult = _socket->isReadyToRead(-1);
|
|
|
|
if (pollResult == PollResultType::Error)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-26 17:33:22 +01:00
|
|
|
// build the response as a single string
|
|
|
|
std::stringstream oss;
|
|
|
|
|
2019-03-20 22:29:02 +01:00
|
|
|
// Read the first line of the response
|
|
|
|
auto lineResult = _socket->readLine(nullptr);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
2019-03-26 17:33:22 +01:00
|
|
|
oss << line;
|
2019-03-20 22:29:02 +01:00
|
|
|
|
|
|
|
if (!lineValid) return false;
|
|
|
|
|
|
|
|
// There are 5 items for the subscribe repply
|
|
|
|
for (int i = 0; i < 5; ++i)
|
|
|
|
{
|
|
|
|
auto lineResult = _socket->readLine(nullptr);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
2019-03-26 17:33:22 +01:00
|
|
|
oss << line;
|
2019-03-20 22:29:02 +01:00
|
|
|
|
|
|
|
if (!lineValid) return false;
|
|
|
|
}
|
|
|
|
|
2019-03-26 17:33:22 +01:00
|
|
|
responseCallback(oss.str());
|
|
|
|
|
2019-03-20 22:29:02 +01:00
|
|
|
// Wait indefinitely for new messages
|
|
|
|
while (true)
|
|
|
|
{
|
|
|
|
// Wait until something is ready to read
|
|
|
|
auto pollResult = _socket->isReadyToRead(-1);
|
|
|
|
if (pollResult == PollResultType::Error)
|
|
|
|
{
|
|
|
|
return false;
|
|
|
|
}
|
|
|
|
|
2019-03-21 02:34:24 +01:00
|
|
|
// The first line of the response describe the return type,
|
2019-03-20 22:29:02 +01:00
|
|
|
// => *3 (an array of 3 elements)
|
|
|
|
auto lineResult = _socket->readLine(nullptr);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
|
|
|
|
|
|
|
if (!lineValid) return false;
|
|
|
|
|
|
|
|
int arraySize;
|
|
|
|
{
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << line.substr(1, line.size()-1);
|
|
|
|
ss >> arraySize;
|
|
|
|
}
|
|
|
|
|
|
|
|
// There are 6 items for each received message
|
|
|
|
for (int i = 0; i < arraySize; ++i)
|
|
|
|
{
|
|
|
|
auto lineResult = _socket->readLine(nullptr);
|
|
|
|
auto lineValid = lineResult.first;
|
|
|
|
auto line = lineResult.second;
|
|
|
|
|
|
|
|
if (!lineValid) return false;
|
|
|
|
|
|
|
|
// Messages are string, which start with a string size
|
|
|
|
// => $7 (7 bytes)
|
|
|
|
int stringSize;
|
|
|
|
std::stringstream ss;
|
|
|
|
ss << line.substr(1, line.size()-1);
|
|
|
|
ss >> stringSize;
|
|
|
|
|
|
|
|
auto readResult = _socket->readBytes(stringSize, nullptr, nullptr);
|
|
|
|
if (!readResult.first) return false;
|
|
|
|
|
|
|
|
if (i == 2)
|
|
|
|
{
|
|
|
|
// The message is the 3rd element.
|
|
|
|
callback(readResult.second);
|
|
|
|
}
|
|
|
|
|
|
|
|
// read last 2 bytes (\r\n)
|
|
|
|
char c;
|
|
|
|
_socket->readByte(&c, nullptr);
|
|
|
|
_socket->readByte(&c, nullptr);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
return true;
|
|
|
|
}
|
|
|
|
}
|