forked from organicmaps/organicmaps
[partners_api] booking api hotelAvailability method implementation
This commit is contained in:
parent
7585a2c1e1
commit
d24ac54087
5 changed files with 257 additions and 55 deletions
|
@ -6,26 +6,29 @@
|
|||
#include "coding/url_encode.hpp"
|
||||
|
||||
#include "base/get_time.hpp"
|
||||
#include "base/gmtime.hpp"
|
||||
#include "base/logging.hpp"
|
||||
#include "base/thread.hpp"
|
||||
|
||||
#include "std/initializer_list.hpp"
|
||||
#include "std/iomanip.hpp"
|
||||
#include "std/iostream.hpp"
|
||||
#include "std/sstream.hpp"
|
||||
#include "std/utility.hpp"
|
||||
#include <ctime>
|
||||
#include <iomanip>
|
||||
#include <iostream>
|
||||
#include <iterator>
|
||||
#include <sstream>
|
||||
#include <utility>
|
||||
|
||||
#include "3party/jansson/myjansson.hpp"
|
||||
|
||||
#include "private.h"
|
||||
|
||||
namespace
|
||||
{
|
||||
using namespace platform;
|
||||
using namespace booking;
|
||||
using namespace std;
|
||||
using namespace std::chrono;
|
||||
|
||||
string const kBookingApiBaseUrl = "https://distribution-xml.booking.com/json/bookings";
|
||||
namespace
|
||||
{
|
||||
string const kBookingApiBaseUrlV1 = "https://distribution-xml.booking.com/json/bookings";
|
||||
string const kBookingApiBaseUrlV2 = "https://distribution-xml.booking.com/2.0/json/";
|
||||
string const kExtendedHotelInfoBaseUrl = "https://hotels.maps.me/getDescription";
|
||||
string const kPhotoOriginalUrl = "http://aff.bstatic.com/images/hotel/max500/";
|
||||
string const kPhotoSmallUrl = "http://aff.bstatic.com/images/hotel/max300/";
|
||||
|
@ -44,18 +47,20 @@ bool RunSimpleHttpRequest(bool const needAuth, string const & url, string & resu
|
|||
result = request.ServerResponse();
|
||||
return true;
|
||||
}
|
||||
|
||||
return false;
|
||||
}
|
||||
|
||||
string MakeApiUrl(string const & func, initializer_list<pair<string, string>> const & params)
|
||||
string MakeApiUrl(string const & baseUrl, string const & func,
|
||||
vector<pair<string, string>> const & params)
|
||||
{
|
||||
ASSERT_NOT_EQUAL(params.size(), 0, ());
|
||||
|
||||
ostringstream os;
|
||||
if (!g_BookingUrlForTesting.empty())
|
||||
os << g_BookingUrlForTesting << "." << func << "?";
|
||||
os << g_BookingUrlForTesting << func << "?";
|
||||
else
|
||||
os << kBookingApiBaseUrl << "." << func << "?";
|
||||
os << baseUrl << func << "?";
|
||||
|
||||
bool firstParam = true;
|
||||
for (auto const & param : params)
|
||||
|
@ -74,6 +79,33 @@ string MakeApiUrl(string const & func, initializer_list<pair<string, string>> co
|
|||
return os.str();
|
||||
}
|
||||
|
||||
string MakeApiUrlV1(string const & func, vector<pair<string, string>> const & params)
|
||||
{
|
||||
return MakeApiUrl(kBookingApiBaseUrlV1, "." + func, params);
|
||||
}
|
||||
|
||||
string MakeApiUrlV2(string const & func, vector<pair<string, string>> const & params)
|
||||
{
|
||||
return MakeApiUrl(kBookingApiBaseUrlV2, "/" + func, params);
|
||||
}
|
||||
|
||||
std::string FormatTime(system_clock::time_point p)
|
||||
{
|
||||
time_t t = duration_cast<seconds>(p.time_since_epoch()).count();
|
||||
ostringstream os;
|
||||
os << put_time(std::gmtime(&t), "%Y-%m-%d");
|
||||
return os.str();
|
||||
}
|
||||
|
||||
string FormatByComma(vector<string> const & src)
|
||||
{
|
||||
ostringstream os;
|
||||
ostream_iterator<string> outIt(os, ",");
|
||||
copy(src.cbegin(), src.cend(), outIt);
|
||||
|
||||
return os.str();
|
||||
}
|
||||
|
||||
void ClearHotelInfo(HotelInfo & info)
|
||||
{
|
||||
info.m_hotelId.clear();
|
||||
|
@ -238,26 +270,56 @@ void FillPriceAndCurrency(string const & src, string const & currency, string &
|
|||
}
|
||||
}
|
||||
}
|
||||
|
||||
void FillHotelIds(string const & src, vector<uint64_t> & result)
|
||||
{
|
||||
my::Json root(src.c_str());
|
||||
auto const resultsArray = json_object_get(root.get(), "result");
|
||||
|
||||
auto const size = json_array_size(resultsArray);
|
||||
if (size == 0)
|
||||
return;
|
||||
|
||||
result.resize(size);
|
||||
for (size_t i = 0; i < size; ++i)
|
||||
{
|
||||
auto const obj = json_array_get(resultsArray, i);
|
||||
FromJSONObject(obj, "hotel_id", result[i]);
|
||||
}
|
||||
}
|
||||
} // namespace
|
||||
|
||||
namespace booking
|
||||
{
|
||||
vector<pair<string, string>> AvailabilityParams::Get() const
|
||||
{
|
||||
vector<pair<string, string>> result;
|
||||
|
||||
result.push_back({"hotel_ids", FormatByComma(m_hotelIds)});
|
||||
result.push_back({"checkin", FormatTime(m_checkin)});
|
||||
result.push_back({"checkout", FormatTime(m_checkout)});
|
||||
|
||||
for (size_t i = 0; i < m_rooms.size(); ++i)
|
||||
result.push_back({"room" + to_string(i+1), m_rooms[i]});
|
||||
|
||||
if (m_minReviewScore != 0.0)
|
||||
result.push_back({"min_review_score", std::to_string(m_minReviewScore)});
|
||||
|
||||
if (!m_stars.empty())
|
||||
result.push_back({"stars", FormatByComma(m_stars)});
|
||||
|
||||
return result;
|
||||
}
|
||||
|
||||
// static
|
||||
bool RawApi::GetHotelAvailability(string const & hotelId, string const & currency, string & result)
|
||||
{
|
||||
char dateArrival[12]{};
|
||||
char dateDeparture[12]{};
|
||||
|
||||
system_clock::time_point p = system_clock::from_time_t(time(nullptr));
|
||||
tm arrival = my::GmTime(system_clock::to_time_t(p));
|
||||
tm departure = my::GmTime(system_clock::to_time_t(p + hours(24)));
|
||||
strftime(dateArrival, sizeof(dateArrival), "%Y-%m-%d", &arrival);
|
||||
strftime(dateDeparture, sizeof(dateDeparture), "%Y-%m-%d", &departure);
|
||||
|
||||
string url = MakeApiUrl("getHotelAvailability", {{"hotel_ids", hotelId},
|
||||
{"currency_code", currency},
|
||||
{"arrival_date", dateArrival},
|
||||
{"departure_date", dateDeparture}});
|
||||
string url = MakeApiUrlV1("getHotelAvailability", {{"hotel_ids", hotelId},
|
||||
{"currency_code", currency},
|
||||
{"arrival_date", FormatTime(p)},
|
||||
{"departure_date", FormatTime(p + hours(24))}});
|
||||
return RunSimpleHttpRequest(true, url, result);
|
||||
}
|
||||
|
||||
|
@ -269,6 +331,14 @@ bool RawApi::GetExtendedInfo(string const & hotelId, string const & lang, string
|
|||
return RunSimpleHttpRequest(false, os.str(), result);
|
||||
}
|
||||
|
||||
// static
|
||||
bool RawApi::HotelAvailability(AvailabilityParams const & params, string & result)
|
||||
{
|
||||
string url = MakeApiUrlV2("hotelAvailability", params.Get());
|
||||
|
||||
return RunSimpleHttpRequest(true, url, result);
|
||||
}
|
||||
|
||||
string Api::GetBookHotelUrl(string const & baseUrl) const
|
||||
{
|
||||
ASSERT(!baseUrl.empty(), ());
|
||||
|
@ -363,6 +433,34 @@ void Api::GetHotelInfo(string const & hotelId, string const & lang, GetHotelInfo
|
|||
});
|
||||
}
|
||||
|
||||
void Api::GetHotelAvailability(AvailabilityParams const & params,
|
||||
GetHotelAvailabilityCallback const & fn)
|
||||
{
|
||||
GetPlatform().RunOnNetworkThread([params, fn]()
|
||||
{
|
||||
std::vector<uint64_t> result;
|
||||
string httpResult;
|
||||
if (!RawApi::HotelAvailability(params, httpResult))
|
||||
{
|
||||
fn(result);
|
||||
return;
|
||||
}
|
||||
|
||||
try
|
||||
{
|
||||
FillHotelIds(httpResult, result);
|
||||
}
|
||||
catch (my::Json::Exception const & e)
|
||||
{
|
||||
LOG(LINFO, (httpResult));
|
||||
LOG(LERROR, (e.Msg()));
|
||||
result.clear();
|
||||
}
|
||||
|
||||
fn(result);
|
||||
});
|
||||
}
|
||||
|
||||
void SetBookingUrlForTesting(string const & url)
|
||||
{
|
||||
g_BookingUrlForTesting = url;
|
||||
|
|
|
@ -2,76 +2,111 @@
|
|||
|
||||
#include "platform/safe_callback.hpp"
|
||||
|
||||
#include "std/chrono.hpp"
|
||||
#include "std/function.hpp"
|
||||
#include "std/shared_ptr.hpp"
|
||||
#include "std/string.hpp"
|
||||
#include "std/vector.hpp"
|
||||
#include <chrono>
|
||||
#include <functional>
|
||||
#include <memory>
|
||||
#include <string>
|
||||
#include <vector>
|
||||
|
||||
namespace booking
|
||||
{
|
||||
struct HotelPhotoUrls
|
||||
{
|
||||
string m_small;
|
||||
string m_original;
|
||||
std::string m_small;
|
||||
std::string m_original;
|
||||
};
|
||||
|
||||
struct HotelReview
|
||||
{
|
||||
/// An issue date.
|
||||
time_point<system_clock> m_date;
|
||||
std::chrono::time_point<std::chrono::system_clock> m_date;
|
||||
/// Author's hotel evaluation.
|
||||
float m_score = 0.0;
|
||||
/// Review author name.
|
||||
string m_author;
|
||||
std::string m_author;
|
||||
/// Review text. There can be either one or both positive/negative review.
|
||||
string m_pros;
|
||||
string m_cons;
|
||||
std::string m_pros;
|
||||
std::string m_cons;
|
||||
};
|
||||
|
||||
struct HotelFacility
|
||||
{
|
||||
string m_type;
|
||||
string m_name;
|
||||
std::string m_type;
|
||||
std::string m_name;
|
||||
};
|
||||
|
||||
struct HotelInfo
|
||||
{
|
||||
string m_hotelId;
|
||||
std::string m_hotelId;
|
||||
|
||||
string m_description;
|
||||
vector<HotelPhotoUrls> m_photos;
|
||||
vector<HotelFacility> m_facilities;
|
||||
vector<HotelReview> m_reviews;
|
||||
std::string m_description;
|
||||
std::vector<HotelPhotoUrls> m_photos;
|
||||
std::vector<HotelFacility> m_facilities;
|
||||
std::vector<HotelReview> m_reviews;
|
||||
float m_score = 0.0;
|
||||
uint32_t m_scoreCount = 0;
|
||||
};
|
||||
|
||||
/// Params for checking availability of hotels.
|
||||
/// [m_hotelIds], [m_checkin], [m_checkout], [m_rooms] are required.
|
||||
struct AvailabilityParams
|
||||
{
|
||||
using Time = std::chrono::system_clock::time_point;
|
||||
|
||||
std::vector<std::pair<std::string, std::string>> Get() const;
|
||||
|
||||
/// Limit the result list to the specified hotels where they have availability for the
|
||||
/// specified guests and dates.
|
||||
std::vector<std::string> m_hotelIds;
|
||||
/// The arrival date. Must be within 360 days in the future and in the format yyyy-mm-dd.
|
||||
Time m_checkin;
|
||||
/// The departure date. Must be later than [m_checkin]. Must be between 1 and 30 days after
|
||||
/// [m_checkin]. Must be within 360 days in the future and in the format yyyy-mm-dd.
|
||||
Time m_checkout;
|
||||
/// Each room is s comma separated array of guests to stay in this room where "A" represents an
|
||||
/// adult and an integer represents a child. eg room1=A,A,4 would be a room with 2 adults and 1
|
||||
/// four year-old child. Child age numbers are 0..17.
|
||||
std::vector<std::string> m_rooms;
|
||||
/// Show only hotels with review_score >= that. min_review_score should be in the range 1 to 10.
|
||||
/// Values are rounded down: min_review_score 7.8 will result in properties with review scores
|
||||
/// of 7 and up.
|
||||
double m_minReviewScore = {};
|
||||
/// Limit to hotels with the given number(s) of stars. Supported values 1-5.
|
||||
vector<std::string> m_stars;
|
||||
};
|
||||
|
||||
class RawApi
|
||||
{
|
||||
public:
|
||||
static bool GetHotelAvailability(string const & hotelId, string const & currency, string & result);
|
||||
static bool GetExtendedInfo(string const & hotelId, string const & lang, string & result);
|
||||
// Booking Api v1 methods:
|
||||
static bool GetHotelAvailability(std::string const & hotelId, std::string const & currency, std::string & result);
|
||||
static bool GetExtendedInfo(std::string const & hotelId, std::string const & lang, std::string & result);
|
||||
// Booking Api v2 methods:
|
||||
static bool HotelAvailability(AvailabilityParams const & params, std::string & result);
|
||||
};
|
||||
|
||||
using GetMinPriceCallback = platform::SafeCallback<void(string const & hotelId, string const & price, string const & currency)>;
|
||||
using GetMinPriceCallback = platform::SafeCallback<void(std::string const & hotelId, std::string const & price, std::string const & currency)>;
|
||||
using GetHotelInfoCallback = platform::SafeCallback<void(HotelInfo const & hotelInfo)>;
|
||||
using GetHotelAvailabilityCallback = platform::SafeCallback<void(std::vector<uint64_t> hotelIds)>;
|
||||
|
||||
class Api
|
||||
{
|
||||
public:
|
||||
string GetBookHotelUrl(string const & baseUrl) const;
|
||||
string GetDescriptionUrl(string const & baseUrl) const;
|
||||
string GetHotelReviewsUrl(string const & hotelId, string const & baseUrl) const;
|
||||
string GetSearchUrl(string const & city, string const & name) const;
|
||||
// Real-time information methods (used for retriving rapidly changing information).
|
||||
// These methods send requests directly to Booking.
|
||||
void GetMinPrice(string const & hotelId, string const & currency, GetMinPriceCallback const & fn);
|
||||
std::string GetBookHotelUrl(std::string const & baseUrl) const;
|
||||
std::string GetDescriptionUrl(std::string const & baseUrl) const;
|
||||
std::string GetHotelReviewsUrl(std::string const & hotelId, std::string const & baseUrl) const;
|
||||
std::string GetSearchUrl(std::string const & city, std::string const & name) const;
|
||||
/// Real-time information methods (used for retriving rapidly changing information).
|
||||
/// These methods send requests directly to Booking.
|
||||
void GetMinPrice(std::string const & hotelId, std::string const & currency, GetMinPriceCallback const & fn);
|
||||
|
||||
// Static information methods (use for information that can be cached).
|
||||
// These methods use caching server to prevent Booking from being ddossed.
|
||||
void GetHotelInfo(string const & hotelId, string const & lang, GetHotelInfoCallback const & fn);
|
||||
/// Static information methods (use for information that can be cached).
|
||||
/// These methods use caching server to prevent Booking from being ddossed.
|
||||
void GetHotelInfo(std::string const & hotelId, std::string const & lang, GetHotelInfoCallback const & fn);
|
||||
|
||||
void GetHotelAvailability(AvailabilityParams const & params,
|
||||
GetHotelAvailabilityCallback const & fn);
|
||||
};
|
||||
|
||||
void SetBookingUrlForTesting(string const & url);
|
||||
void SetBookingUrlForTesting(std::string const & url);
|
||||
} // namespace booking
|
||||
|
|
|
@ -6,6 +6,8 @@
|
|||
|
||||
#include "base/scope_guard.hpp"
|
||||
|
||||
#include <chrono>
|
||||
|
||||
using namespace partners_api;
|
||||
|
||||
namespace
|
||||
|
@ -26,6 +28,20 @@ UNIT_TEST(Booking_GetExtendedInfo)
|
|||
TEST(!result.empty(), ());
|
||||
}
|
||||
|
||||
UNIT_TEST(Booking_HotelAvailability)
|
||||
{
|
||||
booking::AvailabilityParams params;
|
||||
params.m_hotelIds = {"98251"};
|
||||
params.m_rooms = {"A,A"};
|
||||
params.m_checkin = std::chrono::system_clock::now() + std::chrono::hours(24);
|
||||
params.m_checkout = std::chrono::system_clock::now() + std::chrono::hours(24 * 7);
|
||||
params.m_stars = {"4", "5"};
|
||||
string result;
|
||||
TEST(booking::RawApi::HotelAvailability(params, result), ());
|
||||
TEST(!result.empty(), ());
|
||||
LOG(LINFO, (result));
|
||||
}
|
||||
|
||||
UNIT_CLASS_TEST(AsyncGuiThread, Booking_GetMinPrice)
|
||||
{
|
||||
booking::SetBookingUrlForTesting("http://localhost:34568/booking/min_price");
|
||||
|
@ -110,4 +126,31 @@ UNIT_CLASS_TEST(AsyncGuiThread, GetHotelInfo)
|
|||
TEST_EQUAL(info.m_facilities.size(), 7, ());
|
||||
TEST_EQUAL(info.m_reviews.size(), 4, ());
|
||||
}
|
||||
|
||||
UNIT_CLASS_TEST(AsyncGuiThread, GetHotelAvailability)
|
||||
{
|
||||
booking::SetBookingUrlForTesting("http://localhost:34568/booking/min_price");
|
||||
MY_SCOPE_GUARD(cleanup, []() { booking::SetBookingUrlForTesting(""); });
|
||||
|
||||
booking::AvailabilityParams params;
|
||||
params.m_hotelIds = {"77615", "10623"};
|
||||
params.m_rooms = {"A,A"};
|
||||
params.m_checkin = std::chrono::system_clock::now() + std::chrono::hours(24);
|
||||
params.m_checkout = std::chrono::system_clock::now() + std::chrono::hours(24 * 7);
|
||||
params.m_stars = {"4"};
|
||||
booking::Api api;
|
||||
std::vector<uint64_t> result;
|
||||
|
||||
api.GetHotelAvailability(params, [&result](std::vector<uint64_t> const & r)
|
||||
{
|
||||
result = r;
|
||||
testing::Notify();
|
||||
});
|
||||
testing::Wait();
|
||||
|
||||
TEST_EQUAL(result.size(), 3, ());
|
||||
TEST_EQUAL(result[0], 10623, ());
|
||||
TEST_EQUAL(result[1], 10624, ());
|
||||
TEST_EQUAL(result[2], 10625, ());
|
||||
}
|
||||
}
|
||||
|
|
|
@ -143,6 +143,7 @@ class ResponseProvider:
|
|||
"/partners/price": self.partners_price,
|
||||
"/booking/min_price": self.partners_minprice,
|
||||
"/booking/min_price.getHotelAvailability": self.partners_minprice,
|
||||
"/booking/min_price/hotelAvailability": self.partners_hotel_availability,
|
||||
"/partners/taxi_info": self.partners_yandex_taxi_info,
|
||||
"/partners/get-offers-in-bbox/": self.partners_rent_nearby,
|
||||
}[url]()
|
||||
|
@ -223,6 +224,9 @@ class ResponseProvider:
|
|||
def partners_minprice(self):
|
||||
return Payload(jsons.PARTNERS_MINPRICE)
|
||||
|
||||
def partners_hotel_availability(self):
|
||||
return Payload(jsons.HOTEL_AVAILABILITY)
|
||||
|
||||
def partners_yandex_taxi_info(self):
|
||||
return Payload(jsons.PARTNERS_TAXI_INFO)
|
||||
|
||||
|
|
|
@ -186,6 +186,28 @@ PARTNERS_TIME = """
|
|||
}
|
||||
"""
|
||||
|
||||
HOTEL_AVAILABILITY = """
|
||||
{
|
||||
"result": [
|
||||
{
|
||||
"hotel_currency_code": "EUR",
|
||||
"hotel_id": 10623,
|
||||
"price": 801
|
||||
},
|
||||
{
|
||||
"hotel_currency_code": "USD",
|
||||
"hotel_id": 10624,
|
||||
"price": 802
|
||||
},
|
||||
{
|
||||
"hotel_currency_code": "RUR",
|
||||
"hotel_id": 10625,
|
||||
"price": 803
|
||||
}
|
||||
]
|
||||
}
|
||||
"""
|
||||
|
||||
PARTNERS_MINPRICE = """
|
||||
[
|
||||
{
|
||||
|
|
Loading…
Add table
Reference in a new issue