[traffic] Implement HttpTraffSource

Signed-off-by: mvglasow <michael -at- vonglasow.com>
This commit is contained in:
mvglasow
2025-07-20 13:14:23 +03:00
parent a20d1453e0
commit 3f58c6ee20
8 changed files with 549 additions and 8 deletions

View File

@@ -5,6 +5,10 @@
#include "geometry/rect2d.hpp"
#include "platform/platform.hpp"
#include <functional>
#include <thread>
#include <vector>
namespace traffxml {
@@ -102,4 +106,202 @@ void MockTraffSource::Poll()
m_nextRequestTime = std::chrono::steady_clock::now() + m_updateInterval;
}
}
TraffResponse HttpPost(std::string const & url, std::string data)
{
platform::HttpClient request(url);
request.SetBodyData(data, "application/xml");
if (!request.RunHttpRequest() || request.ErrorCode() != 200)
{
TraffResponse result;
result.m_status = ResponseStatus::InternalError;
return result;
}
LOG(LDEBUG, ("Got response, status", request.ErrorCode()));
TraffResponse result = ParseResponse(request.ServerResponse());
return result;
}
void HttpTraffSource::Create(TraffSourceManager & manager, std::string const & url)
{
std::unique_ptr<HttpTraffSource> source = std::unique_ptr<HttpTraffSource>(new HttpTraffSource(manager, url));
manager.RegisterSource(std::move(source));
}
HttpTraffSource::HttpTraffSource(TraffSourceManager & manager, std::string const & url)
: TraffSource(manager)
, m_url(url)
{}
void HttpTraffSource::Subscribe(std::set<MwmSet::MwmId> & mwms)
{
std::string data = "<request operation=\"SUBSCRIBE\">\n<filter_list>\n"
+ GetMwmFilters(mwms)
+ "</filter_list>\n"
+ "</request>";
LOG(LDEBUG, ("Sending request:\n", data));
threads::SimpleThread thread([this, data]() {
// TODO sometimes the request gets sent (and processed) twice
TraffResponse response = HttpPost(m_url, data);
OnSubscribeResponse(response);
return;
});
thread.detach();
}
void HttpTraffSource::OnFeedReceived(TraffFeed & feed)
{
m_lastResponseTime = std::chrono::steady_clock::now();
m_nextRequestTime = std::chrono::steady_clock::now() + m_updateInterval;
m_lastAvailability = Availability::IsAvailable;
m_manager.ReceiveFeed(feed);
}
void HttpTraffSource::OnSubscribeResponse(TraffResponse & response)
{
if (response.m_status == ResponseStatus::Ok
|| response.m_status == ResponseStatus::PartiallyCovered)
{
if (response.m_subscriptionId.empty())
LOG(LWARNING, ("Server replied with", response.m_status, "but subscription ID is empty; ignoring"));
else
{
{
std::lock_guard<std::mutex> lock(m_mutex);
m_subscriptionId = response.m_subscriptionId;
// TODO timeout
}
if (response.m_feed && !response.m_feed.value().empty())
OnFeedReceived(response.m_feed.value());
else
Poll();
}
}
else
LOG(LWARNING, ("Subscribe request failed:", response.m_status));
}
void HttpTraffSource::ChangeSubscription(std::set<MwmSet::MwmId> & mwms)
{
std::string data = "<request operation=\"SUBSCRIPTION_CHANGE\" subscription_id=\"" + m_subscriptionId + "\">\n"
+ "<filter_list>\n"
+ GetMwmFilters(mwms)
+ "</filter_list>\n"
+ "</request>";
LOG(LDEBUG, ("Sending request:\n", data));
threads::SimpleThread thread([this, data]() {
TraffResponse response = HttpPost(m_url, data);
OnChangeSubscriptionResponse(response);
return;
});
thread.detach();
}
void HttpTraffSource::OnChangeSubscriptionResponse(TraffResponse & response)
{
if (response.m_status == ResponseStatus::Ok
|| response.m_status == ResponseStatus::PartiallyCovered)
{
if (response.m_feed && !response.m_feed.value().empty())
OnFeedReceived(response.m_feed.value());
else
Poll();
}
else if (response.m_status == ResponseStatus::SubscriptionUnknown)
{
LOG(LWARNING, ("Change Subscription returned", response.m_status, " removing subscription", m_subscriptionId));
{
std::lock_guard<std::mutex> lock(m_mutex);
m_subscriptionId.clear();
}
}
else
LOG(LWARNING, ("Change Subscription request failed:", response.m_status));
}
void HttpTraffSource::Unsubscribe()
{
std::string data;
{
std::lock_guard<std::mutex> lock(m_mutex);
if (m_subscriptionId.empty())
return;
data = "<request operation=\"UNSUBSCRIBE\" subscription_id=\"" + m_subscriptionId + "\"/>";
}
LOG(LDEBUG, ("Sending request:\n", data));
threads::SimpleThread thread([this, data]() {
TraffResponse response = HttpPost(m_url, data);
OnUnsubscribeResponse(response);
return;
});
thread.detach();
}
void HttpTraffSource::OnUnsubscribeResponse(TraffResponse & response)
{
if (response.m_status != ResponseStatus::Ok
&& response.m_status != ResponseStatus::SubscriptionUnknown)
{
LOG(LWARNING, ("Unsubscribe returned", response.m_status, " removing subscription"));
}
{
std::lock_guard<std::mutex> lock(m_mutex);
m_subscriptionId.clear();
}
}
bool HttpTraffSource::IsPollNeeded()
{
// TODO revisit logic
return m_nextRequestTime.load() <= std::chrono::steady_clock::now();
}
void HttpTraffSource::Poll()
{
std::string data;
{
std::lock_guard<std::mutex> lock(m_mutex);
if (m_subscriptionId.empty())
return;
data = "<request operation=\"POLL\" subscription_id=\"" + m_subscriptionId + "\"/>";
}
LOG(LDEBUG, ("Sending request:\n", data));
threads::SimpleThread thread([this, data]() {
// TODO sometimes the request gets sent (and processed) twice
TraffResponse response = HttpPost(m_url, data);
OnPollResponse(response);
return;
});
thread.detach();
}
void HttpTraffSource::OnPollResponse(TraffResponse & response)
{
if (response.m_status == ResponseStatus::Ok)
{
if (response.m_feed && !response.m_feed.value().empty())
OnFeedReceived(response.m_feed.value());
}
else if (response.m_status == ResponseStatus::SubscriptionUnknown)
{
LOG(LWARNING, ("Poll returned", response.m_status, " removing subscription", m_subscriptionId));
{
std::lock_guard<std::mutex> lock(m_mutex);
m_subscriptionId.clear();
}
}
else
LOG(LWARNING, ("Poll returned", response.m_status));
}
} // namespace traffxml