[core] cross reference api + tests + framework integration

This commit is contained in:
Arsentiy Milchakov 2019-05-21 20:19:28 +03:00 committed by mpimenov
parent 98045752e4
commit fa59cb2a6a
20 changed files with 825 additions and 9 deletions

View file

@ -20,6 +20,12 @@ public:
m_os << DebugPrint(t);
}
template <typename T>
void operator()(T const & t, T const & /* with optional value */, char const * name = nullptr)
{
operator()(t, name);
}
std::string ToString() const { return m_name + " [" + m_os.str() + "]"; }
private:

View file

@ -21,6 +21,7 @@ protected:
SPONSORED_CHECKER(IsBookingChecker, "booking");
SPONSORED_CHECKER(IsOpentableChecker, "opentable");
SPONSORED_CHECKER(IsHolidayChecker, "holiday");
SPONSORED_CHECKER(IsCrossReferenceCityChecker, "cross_reference");
#undef SPONSORED_CHECKER
} // namespace ftypes

View file

@ -16,6 +16,8 @@ set(
benchmark_tools.cpp
booking_availability_filter.cpp
booking_availability_filter.hpp
cross_reference_delegate.cpp
cross_reference_delegate.hpp
booking_filter.hpp
booking_filter_cache.cpp
booking_filter_cache.hpp

View file

@ -0,0 +1,31 @@
#include "map/cross_reference_delegate.hpp"
#include "search/city_finder.hpp"
#include "indexer/data_source.hpp"
#include "indexer/ftypes_sponsored.hpp"
CrossReferenceDelegate::CrossReferenceDelegate(DataSource const & dataSource,
search::CityFinder & cityFinder)
: m_dataSource(dataSource)
, m_cityFinder(cityFinder)
{
}
std::string CrossReferenceDelegate::GetCityOsmId(m2::PointD const & point)
{
auto const featureId = m_cityFinder.GetCityFeatureID(point);
if (!featureId.IsValid())
return {};
FeaturesLoaderGuard guard(m_dataSource, featureId.m_mwmId);
auto feature = guard.GetOriginalFeatureByIndex(featureId.m_index);
if (!feature)
return {};
if (ftypes::IsCrossReferenceCityChecker::Instance()(*feature))
return feature->GetMetadata().Get(feature::Metadata::FMD_SPONSORED_ID);
return {};
}

View file

@ -0,0 +1,26 @@
#pragma once
#include "partners_api/cross_reference_api.hpp"
#include "geometry/point2d.hpp"
#include <string>
class DataSource;
namespace search
{
class CityFinder;
}
class CrossReferenceDelegate : public cross_reference::Api::Delegate
{
public:
CrossReferenceDelegate(DataSource const & dataSource, search::CityFinder & cityFinder);
std::string GetCityOsmId(m2::PointD const & point) override;
private:
DataSource const & m_dataSource;
search::CityFinder & m_cityFinder;
};

View file

@ -1,6 +1,7 @@
#include "map/framework.hpp"
#include "map/benchmark_tools.hpp"
#include "map/chart_generator.hpp"
#include "map/cross_reference_delegate.hpp"
#include "map/displayed_categories_modifiers.hpp"
#include "map/everywhere_search_params.hpp"
#include "map/ge0_parser.hpp"
@ -575,6 +576,9 @@ Framework::Framework(FrameworkParams const & params)
GetPowerManager().Subscribe(this);
GetPowerManager().Load();
m_crossReferenceApi->SetDelegate(make_unique<CrossReferenceDelegate>(m_model.GetDataSource(),
*m_cityFinder));
}
Framework::~Framework()
@ -639,6 +643,15 @@ locals::Api * Framework::GetLocalsApi(platform::NetworkPolicy const & policy)
return nullptr;
}
cross_reference::Api * Framework::GetCrossReferenceApi(platform::NetworkPolicy const & policy) const
{
ASSERT(m_crossReferenceApi, ());
if (policy.CanUse())
return m_crossReferenceApi.get();
return nullptr;
}
void Framework::ShowNode(storage::CountryId const & countryId)
{
StopLocationFollow();
@ -955,6 +968,10 @@ void Framework::FillInfoFromFeatureType(FeatureType & ft, place_page::Info & inf
{
info.SetSponsoredType(place_page::SponsoredType::Holiday);
}
else if (ftypes::IsCrossReferenceCityChecker::Instance()(ft))
{
info.SetSponsoredType(SponsoredType::CrossReference);
}
FillLocalExperts(ft, info);
FillDescription(ft, info);
@ -1481,6 +1498,8 @@ void Framework::EnterForeground()
m_trafficManager.OnEnterForeground();
m_routingManager.SetAllowSendingPoints(true);
m_crossReferenceApi->OnEnterForeground();
}
void Framework::InitCountryInfoGetter()

View file

@ -62,6 +62,7 @@
#include "partners_api/banner.hpp"
#include "partners_api/booking_api.hpp"
#include "partners_api/cross_reference_api.hpp"
#include "partners_api/locals_api.hpp"
#include "partners_api/taxi_engine.hpp"
@ -229,6 +230,7 @@ protected:
unique_ptr<booking::Api> m_bookingApi = make_unique<booking::Api>();
unique_ptr<locals::Api> m_localsApi = make_unique<locals::Api>();
unique_ptr<cross_reference::Api> m_crossReferenceApi = make_unique<cross_reference::Api>();
df::DrapeApi m_drapeApi;
@ -276,6 +278,7 @@ public:
booking::Api const * GetBookingApi(platform::NetworkPolicy const & policy) const;
taxi::Engine * GetTaxiEngine(platform::NetworkPolicy const & policy);
locals::Api * GetLocalsApi(platform::NetworkPolicy const & policy);
cross_reference::Api * GetCrossReferenceApi(platform::NetworkPolicy const & policy) const;
ugc::Api * GetUGCApi() { return m_ugcApi.get(); }
ugc::Api const * GetUGCApi() const { return m_ugcApi.get(); }

View file

@ -45,7 +45,8 @@ enum class SponsoredType
Booking,
Opentable,
Partner,
Holiday
Holiday,
CrossReference,
};
enum class LocalAdsStatus

View file

@ -23,10 +23,14 @@ namespace
auto constexpr kMapObjectEventsExpirePeriod = std::chrono::hours(24 * 30 * 3);
auto constexpr kEventCooldown = std::chrono::seconds(2);
std::array<std::string, 7> const kMapEventSupportedTypes = {{"amenity-bar", "amenity-cafe",
"amenity-pub", "amenity-restaurant",
"amenity-fast_food", "amenity-biergarden",
"shop-bakery"}};
std::array<std::string, 16> const kMapEventSupportedTypes = {{"amenity-bar", "amenity-cafe",
"amenity-pub", "amenity-restaurant",
"amenity-fast_food", "amenity-biergarden",
"shop-bakery", "tourism-hotel",
"tourism-apartment", "tourism-camp_site",
"tourism-chalet", "tourism-guest_house",
"tourism-hostel", "tourism-motel",
"tourism-resort", "sponsored-booking"}};
void Load(Info & info)
{
@ -343,9 +347,10 @@ void Eye::RegisterMapObjectEvent(MapObject const & mapObject, MapObject::Event::
MapObject result = mapObject;
MapObject::Event event;
auto const eventTime = Clock::now();
event.m_type = type;
event.m_userPos = userPos;
event.m_eventTime = Clock::now();
event.m_eventTime = eventTime;
bool found = false;
bool duplication = false;
@ -378,6 +383,14 @@ void Eye::RegisterMapObjectEvent(MapObject const & mapObject, MapObject::Event::
mapObjects.Add(result);
}
if (type == MapObject::Event::Type::BookingBook ||
type == MapObject::Event::Type::BookingMore ||
type == MapObject::Event::Type::BookingReviews ||
type == MapObject::Event::Type::BookingDetails)
{
editableInfo->m_crossReferences.m_transitionToBookingTime = eventTime;
}
if (!SaveLastMapObjectEvent(result))
return;
@ -391,6 +404,26 @@ void Eye::RegisterMapObjectEvent(MapObject const & mapObject, MapObject::Event::
});
}
void Eye::RegisterCrossReferenceAfterBookingShown()
{
auto const info = m_info.Get();
auto editableInfo = std::make_shared<Info>(*info);
auto const now = Clock::now();
editableInfo->m_crossReferences.m_lastTimeShownAfterBooking = now;
if (!Save(editableInfo))
return;
GetPlatform().RunTask(Platform::Thread::Gui, [this, now]
{
for (auto subscriber : m_subscribers)
{
subscriber->OnCrossReferenceAfterBookingShown(now);
}
});
}
// Eye::Event methods ------------------------------------------------------------------------------
// static
void Eye::Event::TipClicked(Tip::Type type, Tip::Event event)
@ -470,4 +503,13 @@ void Eye::Event::MapObjectEvent(MapObject const & mapObject, MapObject::Event::T
Instance().RegisterMapObjectEvent(mapObject, type, userPos);
});
}
// static
void Eye::Event::CrossReferenceAfterBookingShown()
{
GetPlatform().RunTask(Platform::Thread::File, []
{
Instance().RegisterCrossReferenceAfterBookingShown();
});
}
} // namespace eye

View file

@ -24,6 +24,7 @@ public:
virtual void OnDiscoveryItemClicked(Discovery::Event event) {}
virtual void OnLayerShown(Layer const & layer) {}
virtual void OnMapObjectEvent(MapObject const & poi) {}
virtual void OnCrossReferenceAfterBookingShown(Time const & time) {}
};
// Note This class IS thread-safe.
@ -47,6 +48,7 @@ public:
static void LayerShown(Layer::Type type);
static void MapObjectEvent(MapObject const & mapObject, MapObject::Event::Type type,
m2::PointD const & userPos);
static void CrossReferenceAfterBookingShown();
};
static Eye & Instance();
@ -75,6 +77,7 @@ private:
void RegisterLayerShown(Layer::Type type);
void RegisterMapObjectEvent(MapObject const & mapObject, MapObject::Event::Type type,
m2::PointD const & userPos);
void RegisterCrossReferenceAfterBookingShown();
base::AtomicSharedPtr<Info> m_info;
std::vector<Subscriber *> m_subscribers;

View file

@ -183,7 +183,11 @@ public:
AddToBookmark,
UgcEditorOpened,
UgcSaved,
RouteToCreated
RouteToCreated,
BookingBook,
BookingMore,
BookingReviews,
BookingDetails,
};
DECLARE_VISITOR(visitor(m_type, "type"), visitor(m_userPos, "user_pos"),
@ -254,6 +258,17 @@ private:
m2::RectD m_limitRect;
};
struct CrossReferences
{
CrossReferences() = default;
DECLARE_VISITOR_AND_DEBUG_PRINT(CrossReferences,
visitor(m_transitionToBookingTime, "transitionToBookingTime"),
visitor(m_lastTimeShownAfterBooking, "lastTimeShownAfterBooking"))
Time m_transitionToBookingTime;
Time m_lastTimeShownAfterBooking;
};
using MapObjects = m4::Tree<MapObject>;
struct InfoV0
@ -262,7 +277,8 @@ struct InfoV0
DECLARE_VISITOR_AND_DEBUG_PRINT(InfoV0, visitor(m_booking, "booking"),
visitor(m_bookmarks, "bookmarks"),
visitor(m_discovery, "discovery"), visitor(m_layers, "layers"),
visitor(m_tips, "tips"))
visitor(m_tips, "tips"),
visitor(m_crossReferences, CrossReferences(), "crossReferences"))
Booking m_booking;
Bookmarks m_bookmarks;
@ -270,6 +286,7 @@ struct InfoV0
Layers m_layers;
Tips m_tips;
MapObjects m_mapObjects;
CrossReferences m_crossReferences;
};
using Info = InfoV0;
@ -330,6 +347,10 @@ inline std::string DebugPrint(MapObject::Event::Type const & type)
case MapObject::Event::Type::UgcEditorOpened: return "UgcEditorOpened";
case MapObject::Event::Type::UgcSaved: return "UgcSaved";
case MapObject::Event::Type::RouteToCreated: return "RouteToCreated";
case MapObject::Event::Type::BookingBook: return "BookingBook";
case MapObject::Event::Type::BookingMore: return "BookingMore";
case MapObject::Event::Type::BookingReviews: return "BookingReviews";
case MapObject::Event::Type::BookingDetails: return "BookingDetails";
}
}
} // namespace eye

View file

@ -54,8 +54,13 @@ Info MakeDefaultInfoForTesting()
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().push_back(eventInfo);
eventInfo.m_eventTime = Time(std::chrono::hours(100000));
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::BookingMore;
info.m_mapObjects.Add(poi);
info.m_crossReferences.m_lastTimeShownAfterBooking = Time(std::chrono::hours(100000));
return info;
}

View file

@ -26,7 +26,7 @@ class ScopedEyeForTesting
{
public:
ScopedEyeForTesting() { EyeForTesting::ResetEye(); }
~ScopedEyeForTesting() { EyeForTesting::ResetEye(); }
virtual ~ScopedEyeForTesting() { EyeForTesting::ResetEye(); }
private:
Platform::ThreadRunner m_runner;

View file

@ -16,6 +16,8 @@ set(
booking_block_params.cpp
booking_block_params.hpp
booking_params_base.hpp
cross_reference_api.cpp
cross_reference_api.hpp
facebook_ads.cpp
facebook_ads.hpp
google_ads.cpp

View file

@ -0,0 +1,241 @@
#include "cross_reference_api.hpp"
#include "metrics/eye.hpp"
#include "platform/http_client.hpp"
#include "platform/platform.hpp"
#include "platform/preferred_languages.hpp"
#include "platform/settings.hpp"
#include "base/assert.hpp"
#include <algorithm>
#include <chrono>
#include <utility>
#include "3party/jansson/myjansson.hpp"
using namespace cross_reference;
using namespace std::chrono;
namespace
{
constexpr minutes kMinMinutesCountAfterBooking = minutes(5);
constexpr minutes kMaxMinutesCountAfterBooking = minutes(60);
constexpr hours kShowCrossReferenceNotRaterThan = hours(24);
std::array<std::string, 9> const kSupportedBookingTypes = {{"tourism-hotel", "tourism-apartment",
"tourism-camp_site", "tourism-chalet",
"tourism-guest_house", "tourism-hostel",
"tourism-motel", "tourism-resort",
"sponsored-booking"}};
bool NeedToShowImpl(eye::Eye::InfoType const & eyeInfo)
{
auto const timeSinceLastShown =
eye::Clock::now() - eyeInfo->m_crossReferences.m_lastTimeShownAfterBooking;
auto const timeSinceLastTransitionToBooking =
eye::Clock::now() - eyeInfo->m_crossReferences.m_transitionToBookingTime;
return timeSinceLastTransitionToBooking >= kMinMinutesCountAfterBooking ||
timeSinceLastTransitionToBooking <= kMaxMinutesCountAfterBooking ||
timeSinceLastShown > kShowCrossReferenceNotRaterThan;
}
void ParseCityGallery(std::string const & src, cross_reference::CityGallery & result)
{
base::Json root(src.c_str());
auto const dataArray = json_object_get(root.get(), "data");
auto const size = json_array_size(dataArray);
result.reserve(size);
for (size_t i = 0; i < size; ++i)
{
cross_reference::CityGalleryItem item;
auto const obj = json_array_get(dataArray, i);
FromJSONObject(obj, "name", item.m_name);
FromJSONObject(obj, "url", item.m_url);
auto const imageUrlObj = json_object_get(obj, "image_url");
if (!json_is_null(imageUrlObj))
FromJSON(imageUrlObj, item.m_imageUrl);
FromJSONObject(obj, "access", item.m_access);
auto const tierObj = json_object_get(obj, "tier");
if (!json_is_null(tierObj))
FromJSON(tierObj, item.m_tier);
auto const authorObj = json_object_get(obj, "author");
FromJSONObject(authorObj, "key_id", item.m_author.m_id);
FromJSONObject(authorObj, "name", item.m_author.m_name);
auto const luxCategoryObj = json_object_get(obj, "lux_category");
auto const luxCategoryNameobj = json_object_get(luxCategoryObj, "name");
if (!json_is_null(luxCategoryNameobj))
FromJSON(luxCategoryNameobj, item.m_luxCategory.m_name);
FromJSONObject(luxCategoryObj, "color", item.m_luxCategory.m_color);
result.emplace_back(std::move(item));
}
}
std::string MakeCityGalleryUrl(std::string const & baseUrl, std::string const & osmId,
std::string const & lang)
{
return baseUrl + osmId + "/?lang=" + lang;
}
void GetCrossReferenceCityGalleryImpl(std::string const & baseUrl, std::string const & osmId,
CityGalleryCallback const & cb)
{
if (osmId.empty())
{
GetPlatform().RunTask(Platform::Thread::Gui, [cb]() { cb({}); });
return;
}
CityGallery result;
std::string httpResult;
if (!WebApi::GetCityGalleryByOsmId(baseUrl, osmId, languages::GetCurrentNorm(), httpResult))
{
GetPlatform().RunTask(Platform::Thread::Gui, [cb]() { cb({}); });
return;
}
try
{
ParseCityGallery(httpResult, result);
}
catch (base::Json::Exception const & e)
{
LOG(LERROR, (e.Msg()));
result.clear();
}
GetPlatform().RunTask(Platform::Thread::Gui, [ cb, result = move(result) ]() { cb(result); });
}
} // namespace
namespace cross_reference
{
// static
bool WebApi::GetCityGalleryByOsmId(std::string const & baseUrl, std::string const & osmId,
std::string const & lang, std::string & result)
{
platform::HttpClient request(MakeCityGalleryUrl(baseUrl, osmId, lang));
return request.RunHttpRequest(result);
}
Api::Api(std::string const & baseUrl /* = "https://routes.maps.me/gallery/v1/city/" */)
: m_baseUrl(baseUrl)
{
}
void Api::SetDelegate(std::unique_ptr<Delegate> delegate)
{
m_delegate = std::move(delegate);
}
void Api::OnEnterForeground()
{
settings::TryGet("BookingCrossReferenceIsAwaiting", m_bookingCrossReferenceIsAwaiting);
if (!m_bookingCrossReferenceIsAwaiting)
return;
auto const eyeInfo = eye::Eye::Instance().GetInfo();
auto const timeSinceLastTransitionToBooking =
eye::Clock::now() - eyeInfo->m_crossReferences.m_transitionToBookingTime;
if (timeSinceLastTransitionToBooking < kMinMinutesCountAfterBooking ||
timeSinceLastTransitionToBooking > kMaxMinutesCountAfterBooking)
{
m_bookingCrossReferenceIsAwaiting = false;
settings::Set("BookingCrossReferenceIsAwaiting", false);
}
}
bool Api::NeedToShow() const
{
if (!m_bookingCrossReferenceIsAwaiting)
return false;
return NeedToShowImpl(eye::Eye::Instance().GetInfo());
}
void Api::GetCrossReferenceLinkAfterBooking(AfterBookingCallback const & cb) const
{
CHECK(m_delegate, ());
auto const eyeInfo = eye::Eye::Instance().GetInfo();
if (!m_bookingCrossReferenceIsAwaiting || !NeedToShowImpl(eyeInfo))
{
GetPlatform().RunTask(Platform::Thread::Gui, [cb]() { cb({}); });
return;
}
GetPlatform().RunTask(Platform::Thread::Background, [this, eyeInfo, cb]()
{
auto const targetTime = eyeInfo->m_crossReferences.m_transitionToBookingTime;
m2::PointD pos;
auto const found =
eyeInfo->m_mapObjects.FindNode([&pos, targetTime](eye::MapObject const & mapObject)
{
if (mapObject.GetEvents().empty())
return false;
auto const typeIt = std::find(kSupportedBookingTypes.cbegin(),
kSupportedBookingTypes.cend(), mapObject.GetBestType());
if (typeIt == kSupportedBookingTypes.cend())
return false;
for (auto const & event : mapObject.GetEvents())
{
switch (event.m_type)
{
case eye::MapObject::Event::Type::BookingBook:
case eye::MapObject::Event::Type::BookingMore:
case eye::MapObject::Event::Type::BookingReviews:
case eye::MapObject::Event::Type::BookingDetails:
{
if (event.m_eventTime == targetTime)
{
pos = mapObject.GetPos();
return true;
}
}
default: continue;
}
}
return false;
});
auto const osmId = found ? m_delegate->GetCityOsmId(pos) : "";
auto const resultUrl =
osmId.empty() ? "" : MakeCityGalleryUrl(m_baseUrl, osmId, languages::GetCurrentNorm());
GetPlatform().RunTask(Platform::Thread::Gui, [cb, resultUrl]() { cb(resultUrl); });
});
}
void Api::GetCrossReferenceCityGallery(std::string const & osmId,
CityGalleryCallback const & cb) const
{
GetCrossReferenceCityGalleryImpl(m_baseUrl, osmId, cb);
}
void Api::GetCrossReferenceCityGallery(m2::PointD const & point,
CityGalleryCallback const & cb) const
{
CHECK(m_delegate, ());
GetCrossReferenceCityGalleryImpl(m_baseUrl, m_delegate->GetCityOsmId(point), cb);
}
} // namespace cross_reference

View file

@ -0,0 +1,73 @@
#pragma once
#include "geometry/point2d.hpp"
#include <functional>
#include <memory>
#include <string>
namespace cross_reference
{
struct Author
{
std::string m_id;
std::string m_name;
};
struct LuxCategory
{
std::string m_name;
std::string m_color;
};
struct CityGalleryItem
{
std::string m_name;
std::string m_url;
std::string m_imageUrl;
std::string m_access;
std::string m_tier;
Author m_author;
LuxCategory m_luxCategory;
};
using CityGallery = std::vector<CityGalleryItem>;
class WebApi
{
public:
static bool GetCityGalleryByOsmId(std::string const & baseUrl, std::string const & osmId,
std::string const & lang, std::string & result);
};
using AfterBookingCallback = std::function<void(std::string const & url)>;
using CityGalleryCallback = std::function<void(CityGallery const & gallery)>;
class Api
{
public:
class Delegate
{
public:
virtual ~Delegate() = default;
virtual std::string GetCityOsmId(m2::PointD const & point) = 0;
};
explicit Api(std::string const & baseUrl = "https://routes.maps.me/gallery/v1/city/");
void SetDelegate(std::unique_ptr<Delegate> delegate);
void OnEnterForeground();
bool NeedToShow() const;
void GetCrossReferenceLinkAfterBooking(AfterBookingCallback const & cb) const;
void GetCrossReferenceCityGallery(std::string const & osmId,
CityGalleryCallback const & cb) const;
void GetCrossReferenceCityGallery(m2::PointD const & point,
CityGalleryCallback const & cb) const;
private:
std::unique_ptr<Delegate> m_delegate;
std::string m_baseUrl;
bool m_bookingCrossReferenceIsAwaiting = false;
};
} // namespace cross_reference

View file

@ -6,6 +6,7 @@ set(
SRC
ads_engine_tests.cpp
booking_tests.cpp
cross_reference_tests.cpp
facebook_tests.cpp
google_tests.cpp
maxim_tests.cpp
@ -23,7 +24,10 @@ omim_add_test(${PROJECT_NAME} ${SRC})
omim_link_libraries(
${PROJECT_NAME}
platform_tests_support
metrics_tests_support
partners_api
metrics
storage
indexer
platform

View file

@ -0,0 +1,291 @@
#include "testing/testing.hpp"
#include "partners_api/cross_reference_api.hpp"
#include "platform/settings.hpp"
#include "metrics/metrics_tests_support/eye_for_testing.hpp"
#include "platform/platform_tests_support/async_gui_thread.hpp"
#include <algorithm>
#include <memory>
using namespace eye;
using namespace platform::tests_support;
namespace
{
std::string const kTestOsmId = "TestOsmId";
class ScopedEyeWithAsyncGuiThread : public AsyncGuiThread
{
public:
ScopedEyeWithAsyncGuiThread()
{
EyeForTesting::ResetEye();
}
~ScopedEyeWithAsyncGuiThread() override
{
EyeForTesting::ResetEye();
}
};
class DelegateForTesting : public cross_reference::Api::Delegate
{
public:
std::string GetCityOsmId(m2::PointD const &) override { return kTestOsmId; }
};
} // namespace
UNIT_CLASS_TEST(ScopedEyeWithAsyncGuiThread, CrossReference_NeedToShow)
{
cross_reference::Api api;
Info info;
{
MapObject poi;
poi.SetBestType("tourism-hotel");
poi.SetPos({53.652007, 108.143443});
MapObject::Event eventInfo;
eventInfo.m_eventTime = Clock::now() - std::chrono::hours((24 * 30 * 3) + 1);
eventInfo.m_userPos = {72.045507, 81.408095};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime =
Clock::now() - (std::chrono::hours(24 * 30 * 3) + std::chrono::seconds(1));
eventInfo.m_userPos = {72.045400, 81.408200};
eventInfo.m_type = MapObject::Event::Type::AddToBookmark;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(24 * 30 * 3);
eventInfo.m_userPos = {72.045450, 81.408201};
eventInfo.m_type = MapObject::Event::Type::RouteToCreated;
poi.GetEditableEvents().emplace_back(eventInfo);
info.m_mapObjects.Add(poi);
}
EyeForTesting::SetInfo(info);
settings::Set("BookingCrossReferenceIsAwaiting", true);
api.OnEnterForeground();
TEST_EQUAL(api.NeedToShow(), false, ());
{
MapObject poi;
poi.SetBestType("tourism-hotel");
poi.SetPos({53.652005, 108.143448});
MapObject::Event eventInfo;
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(24 * 30 * 3);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(2);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::BookingBook;
poi.GetEditableEvents().emplace_back(eventInfo);
info.m_mapObjects.Add(poi);
}
info.m_crossReferences.m_transitionToBookingTime = Clock::now() - std::chrono::hours(2);
EyeForTesting::SetInfo(info);
settings::Set("BookingCrossReferenceIsAwaiting", true);
api.OnEnterForeground();
TEST_EQUAL(api.NeedToShow(), false, ());
{
MapObject poi;
poi.SetBestType("tourism-hotel");
poi.SetPos({53.653005, 108.143548});
MapObject::Event eventInfo;
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(24 * 20 * 3);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::minutes(6);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::BookingReviews;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::minutes(3);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::minutes(1);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::RouteToCreated;
poi.GetEditableEvents().emplace_back(eventInfo);
info.m_mapObjects.Add(poi);
}
info.m_crossReferences.m_transitionToBookingTime = Clock::now() - std::chrono::minutes(6);
EyeForTesting::SetInfo(info);
settings::Set("BookingCrossReferenceIsAwaiting", true);
api.OnEnterForeground();
TEST_EQUAL(api.NeedToShow(), true, ());
}
UNIT_CLASS_TEST(ScopedEyeWithAsyncGuiThread, CrossReference_GetCrossReferenceLinkAfterBooking)
{
cross_reference::Api api;
api.SetDelegate(std::make_unique<DelegateForTesting>());
Info info;
{
MapObject poi;
poi.SetBestType("tourism-hotel");
poi.SetPos({53.652007, 108.143443});
MapObject::Event eventInfo;
eventInfo.m_eventTime = Clock::now() - std::chrono::hours((24 * 30 * 3) + 1);
eventInfo.m_userPos = {72.045507, 81.408095};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime =
Clock::now() - (std::chrono::hours(24 * 30 * 3) + std::chrono::seconds(1));
eventInfo.m_userPos = {72.045400, 81.408200};
eventInfo.m_type = MapObject::Event::Type::AddToBookmark;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(24 * 30 * 3);
eventInfo.m_userPos = {72.045450, 81.408201};
eventInfo.m_type = MapObject::Event::Type::RouteToCreated;
poi.GetEditableEvents().emplace_back(eventInfo);
info.m_mapObjects.Add(poi);
}
EyeForTesting::SetInfo(info);
settings::Set("BookingCrossReferenceIsAwaiting", true);
api.OnEnterForeground();
std::string result{};
api.GetCrossReferenceLinkAfterBooking([&result](std::string const & url)
{
result = url;
testing::Notify();
});
testing::Wait();
TEST(result.empty(), ());
auto eventTime = Clock::now() - std::chrono::hours(2);
{
MapObject poi;
poi.SetBestType("tourism-hotel");
poi.SetPos({53.652005, 108.143448});
MapObject::Event eventInfo;
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(24 * 30 * 3);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = eventTime;
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::BookingBook;
poi.GetEditableEvents().emplace_back(eventInfo);
info.m_mapObjects.Add(poi);
}
info.m_crossReferences.m_transitionToBookingTime = Clock::now() - std::chrono::hours(2);
EyeForTesting::SetInfo(info);
settings::Set("BookingCrossReferenceIsAwaiting", true);
api.OnEnterForeground();
result = {};
api.GetCrossReferenceLinkAfterBooking([&result](std::string const & url)
{
result = url;
testing::Notify();
});
testing::Wait();
TEST(result.empty(), ());
eventTime = Clock::now() - std::chrono::minutes(6);
{
MapObject poi;
poi.SetBestType("tourism-hotel");
poi.SetPos({53.653005, 108.143548});
MapObject::Event eventInfo;
eventInfo.m_eventTime = Clock::now() - std::chrono::hours(24 * 20 * 3);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = eventTime;
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::BookingReviews;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::minutes(3);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::Open;
poi.GetEditableEvents().emplace_back(eventInfo);
eventInfo.m_eventTime = Clock::now() - std::chrono::minutes(1);
eventInfo.m_userPos = {53.016347, 158.683327};
eventInfo.m_type = MapObject::Event::Type::RouteToCreated;
poi.GetEditableEvents().emplace_back(eventInfo);
info.m_mapObjects.Add(poi);
}
info.m_crossReferences.m_transitionToBookingTime = eventTime;
EyeForTesting::SetInfo(info);
settings::Set("BookingCrossReferenceIsAwaiting", true);
api.OnEnterForeground();
result = {};
api.GetCrossReferenceLinkAfterBooking([&result](std::string const & url)
{
result = url;
testing::Notify();
});
testing::Wait();
TEST_NOT_EQUAL(result.find(kTestOsmId, 0), std::string::npos, ());
}
UNIT_CLASS_TEST(ScopedEyeWithAsyncGuiThread, CrossReference_GetCrossReferenceCityGallery)
{
{}
cross_reference::Api api("http://localhost:34568/gallery/city/");
api.SetDelegate(std::make_unique<DelegateForTesting>());
{
cross_reference::CityGallery result{};
api.GetCrossReferenceCityGallery(kTestOsmId, [&result](cross_reference::CityGallery const & gallery)
{
result = gallery;
testing::Notify();
});
testing::Wait();
TEST_EQUAL(result.size(), 2, ());
}
{
cross_reference::CityGallery result{};
m2::PointD pt;
api.GetCrossReferenceCityGallery(pt, [&result](cross_reference::CityGallery const & gallery)
{
result = gallery;
testing::Notify();
});
testing::Wait();
TEST_EQUAL(result.size(), 2, ());
}
}

View file

@ -147,6 +147,7 @@ class ResponseProvider:
"/partners/taxi_info": self.partners_yandex_taxi_info,
"/partners/get-offers-in-bbox/": self.partners_rent_nearby,
"/partners/CalculateByCoords": self.partners_calculate_by_coords,
"/gallery/city/TestOsmId/": self.cross_reference_gallery_city,
}[url]()
except:
return self.test_404()
@ -239,6 +240,9 @@ class ResponseProvider:
def partners_calculate_by_coords(self):
return Payload(jsons.PARTNERS_CALCULATE_BY_COORDS)
def cross_reference_gallery_city(self):
return Payload(jsons.CROSS_REFERENCE_GALLERY_CITY)
def kill(self):
logging.debug("Kill called in ResponseProvider")
self.delegate.kill()

View file

@ -513,3 +513,44 @@ PARTNERS_CALCULATE_BY_COORDS = """
"PriceString": "244.00 ₽"
}
"""
CROSS_REFERENCE_GALLERY_CITY = """
{
"data": [
{
"url": "bundle/73af3f02-b8e3-4f60-8ef0-1c3c5cff43ca",
"name": "По Виа Рипетта до мавзолея Августа и Алтаря мира",
"author": {
"key_id": "00000000-0000-0000-0000-000000000000",
"name": "The Village"
},
"image_url": "http://localhost:8000/images/73af3f02-b8e3-4f60-8ef0-1c3c5cff43ca.jpg",
"access": "public",
"lux_category": {
"name": "LUX",
"color": "666666"
},
"tier": "price.tier"
},
{
"url": "bundle/73af3f02-b8e3-4f60-8ef0-1c3c5cff43ca",
"name": "Полеты в метро",
"author": {
"key_id": "00000000-0000-0000-0000-000000000000",
"name": "Bmj"
},
"image_url": null,
"access": "public",
"lux_category": {
"name": null,
"color": "666666"
},
"tier": null
}
],
"errors": [],
"meta": {
"more": "search?city=666"
}
}
"""