forked from organicmaps/organicmaps
dummy implementation and smoke test
This commit is contained in:
parent
ecc59be84f
commit
5e716d70d8
16 changed files with 423 additions and 11 deletions
|
@ -1,5 +1,7 @@
|
|||
#pragma once
|
||||
|
||||
#include "geometry/latlon.hpp"
|
||||
|
||||
#include "base/math.hpp"
|
||||
|
||||
#include "std/string.hpp"
|
||||
|
|
|
@ -1,7 +1,6 @@
|
|||
#pragma once
|
||||
|
||||
#include "map/api_mark_point.hpp"
|
||||
#include "map/booking_api.hpp"
|
||||
#include "map/bookmark.hpp"
|
||||
#include "map/bookmark_manager.hpp"
|
||||
#include "map/displacement_mode_manager.hpp"
|
||||
|
@ -34,6 +33,9 @@
|
|||
#include "storage/downloading_policy.hpp"
|
||||
#include "storage/storage.hpp"
|
||||
|
||||
#include "partners_api/booking_api.hpp"
|
||||
#include "partners_api/uber_api.hpp"
|
||||
|
||||
#include "platform/country_defines.hpp"
|
||||
#include "platform/location.hpp"
|
||||
#include "platform/platform.hpp"
|
||||
|
@ -156,6 +158,8 @@ protected:
|
|||
|
||||
BookingApi m_bookingApi;
|
||||
|
||||
uber::Api m_uberApi;
|
||||
|
||||
bool m_isRenderingEnabled;
|
||||
|
||||
/// This function will be called by m_storage when latest local files
|
||||
|
@ -185,6 +189,8 @@ public:
|
|||
BookingApi & GetBookingApi() { return m_bookingApi; }
|
||||
BookingApi const & GetBookingApi() const { return m_bookingApi; }
|
||||
|
||||
uber::Api & GetUberApi() { return m_uberApi;}
|
||||
|
||||
/// Migrate to new version of very different data.
|
||||
bool IsEnoughSpaceForMigrate() const;
|
||||
storage::TCountryId PreMigrate(ms::LatLon const & position, storage::Storage::TChangeCountryFunction const & change,
|
||||
|
|
|
@ -12,7 +12,6 @@ include($$ROOT_DIR/common.pri)
|
|||
|
||||
HEADERS += \
|
||||
api_mark_point.hpp \
|
||||
booking_api.hpp \
|
||||
bookmark.hpp \
|
||||
bookmark_manager.hpp \
|
||||
chart_generator.hpp \
|
||||
|
@ -36,7 +35,6 @@ SOURCES += \
|
|||
../api/src/c/api-client.c \
|
||||
address_finder.cpp \
|
||||
api_mark_point.cpp \
|
||||
booking_api.cpp \
|
||||
bookmark.cpp \
|
||||
bookmark_manager.cpp \
|
||||
chart_generator.cpp \
|
||||
|
|
|
@ -6,7 +6,7 @@ CONFIG -= app_bundle
|
|||
TEMPLATE = app
|
||||
|
||||
ROOT_DIR = ../..
|
||||
DEPENDENCIES = map drape_frontend routing search storage drape indexer platform editor geometry coding base \
|
||||
DEPENDENCIES = map drape_frontend routing search storage drape indexer partners_api platform editor geometry coding base \
|
||||
freetype fribidi expat protobuf tomcrypt jansson osrm stats_client minizip succinct pugixml stats_client
|
||||
|
||||
DEPENDENCIES *= opening_hours
|
||||
|
@ -36,7 +36,6 @@ macx-*: LIBS *= "-framework IOKit" "-framework SystemConfiguration"
|
|||
SOURCES += \
|
||||
../../testing/testingmain.cpp \
|
||||
address_tests.cpp \
|
||||
booking_tests.cpp \
|
||||
bookmarks_test.cpp \
|
||||
chart_generator_tests.cpp \
|
||||
feature_getters_tests.cpp \
|
||||
|
|
|
@ -1,7 +1,7 @@
|
|||
# mapshot binary
|
||||
|
||||
ROOT_DIR = ..
|
||||
DEPENDENCIES = map drape_frontend routing search storage indexer drape platform editor geometry coding base \
|
||||
DEPENDENCIES = map drape_frontend routing search storage indexer drape partners_api platform editor geometry coding base \
|
||||
freetype expat fribidi tomcrypt gflags jansson protobuf osrm stats_client minizip succinct \
|
||||
pugixml opening_hours
|
||||
|
||||
|
|
6
omim.pro
6
omim.pro
|
@ -23,7 +23,7 @@ HEADERS += defines.hpp
|
|||
CONFIG *= desktop
|
||||
}
|
||||
|
||||
SUBDIRS = 3party base coding geometry editor indexer routing search
|
||||
SUBDIRS = 3party base coding geometry editor indexer partners_api routing search
|
||||
|
||||
!CONFIG(osrm) {
|
||||
SUBDIRS *= platform stats storage
|
||||
|
@ -241,5 +241,9 @@ SUBDIRS = 3party base coding geometry editor indexer routing search
|
|||
drape_frontend_tests.subdir = drape_frontend/drape_frontend_tests
|
||||
drape_frontend_tests.depends = 3party base coding platform drape drape_frontend
|
||||
SUBDIRS *= drape_frontend_tests
|
||||
|
||||
partners_api_tests.subdir = partners_api/partners_api_tests
|
||||
partners_api_tests.depends = base platform partners_api
|
||||
SUBDIRS *= partners_api_tests
|
||||
} # !no-tests
|
||||
} # !gtool
|
||||
|
|
|
@ -1,4 +1,4 @@
|
|||
#include "map/booking_api.hpp"
|
||||
#include "partners_api/booking_api.hpp"
|
||||
|
||||
#include "base/gmtime.hpp"
|
||||
#include "base/logging.hpp"
|
18
partners_api/partners_api.pro
Normal file
18
partners_api/partners_api.pro
Normal file
|
@ -0,0 +1,18 @@
|
|||
TARGET = partners_api
|
||||
TEMPLATE = lib
|
||||
CONFIG += staticlib warn_on
|
||||
|
||||
ROOT_DIR = ..
|
||||
|
||||
INCLUDEPATH *= $$ROOT_DIR/3party/jansson/src
|
||||
|
||||
include($$ROOT_DIR/common.pri)
|
||||
|
||||
SOURCES += \
|
||||
booking_api.cpp \
|
||||
uber_api.cpp \
|
||||
|
||||
HEADERS += \
|
||||
booking_api.hpp \
|
||||
uber_api.hpp \
|
||||
|
|
@ -1,6 +1,6 @@
|
|||
#include "testing/testing.hpp"
|
||||
|
||||
#include "map/booking_api.hpp"
|
||||
#include "partners_api/booking_api.hpp"
|
||||
|
||||
UNIT_TEST(Booking_SmokeTest)
|
||||
{
|
16
partners_api/partners_api_tests/partners_api_tests.pro
Normal file
16
partners_api/partners_api_tests/partners_api_tests.pro
Normal file
|
@ -0,0 +1,16 @@
|
|||
TARGET = partners_api_tests
|
||||
CONFIG += console warn_on
|
||||
CONFIG -= app_bundle
|
||||
TEMPLATE = app
|
||||
|
||||
ROOT_DIR = ../..
|
||||
DEPENDENCIES = partners_api platform coding base tomcrypt jansson stats_client
|
||||
|
||||
include($$ROOT_DIR/common.pri)
|
||||
|
||||
QT *= core
|
||||
|
||||
SOURCES += \
|
||||
$$ROOT_DIR/testing/testingmain.cpp \
|
||||
booking_tests.cpp \
|
||||
uber_tests.cpp \
|
29
partners_api/partners_api_tests/uber_tests.cpp
Normal file
29
partners_api/partners_api_tests/uber_tests.cpp
Normal file
|
@ -0,0 +1,29 @@
|
|||
#include "testing/testing.hpp"
|
||||
|
||||
#include "geometry/latlon.hpp"
|
||||
|
||||
#include "partners_api/uber_api.hpp"
|
||||
|
||||
#include "base/logging.hpp"
|
||||
|
||||
UNIT_TEST(Uber_SmokeTest)
|
||||
{
|
||||
ms::LatLon from(59.856464, 30.371867);
|
||||
ms::LatLon to(59.856000, 30.371000);
|
||||
|
||||
{
|
||||
uber::Api uberApi;
|
||||
auto reqId = 0;
|
||||
reqId = uberApi.GetAvailableProducts(from, to, [&reqId](vector<uber::Product> const & products, size_t const requestId)
|
||||
{
|
||||
TEST(!products.empty(), ());
|
||||
TEST_EQUAL(requestId, reqId, ());
|
||||
|
||||
for(auto const & product : products)
|
||||
{
|
||||
LOG(LINFO, (product.m_productId, product.m_name, product.m_time, product.m_price));
|
||||
TEST(!product.m_productId.empty() && !product.m_name.empty() && !product.m_time.empty() && !product.m_price.empty(),());
|
||||
}
|
||||
});
|
||||
}
|
||||
}
|
266
partners_api/uber_api.cpp
Normal file
266
partners_api/uber_api.cpp
Normal file
|
@ -0,0 +1,266 @@
|
|||
#include "uber_api.hpp"
|
||||
|
||||
#include "platform/http_client.hpp"
|
||||
|
||||
#include "geometry/latlon.hpp"
|
||||
|
||||
#include "base/assert.hpp"
|
||||
#include "base/logging.hpp"
|
||||
|
||||
#include "std/future.hpp"
|
||||
|
||||
#include "3party/jansson/myjansson.hpp"
|
||||
|
||||
#include "private.h"
|
||||
|
||||
#define UBER_SERVER_TOKEN ""
|
||||
#define UBER_CLIENT_ID ""
|
||||
|
||||
using namespace platform;
|
||||
|
||||
namespace
|
||||
{
|
||||
string RunSimpleHttpRequest(string const & url)
|
||||
{
|
||||
HttpClient request(url);
|
||||
|
||||
if (request.RunHttpRequest() && !request.WasRedirected() && request.ErrorCode() == 200)
|
||||
{
|
||||
return request.ServerResponse();
|
||||
}
|
||||
|
||||
return {};
|
||||
}
|
||||
|
||||
bool CheckUberAnswer(json_t const * answer)
|
||||
{
|
||||
// Uber products are not available at this point.
|
||||
if (!json_is_array(answer))
|
||||
return false;
|
||||
|
||||
if (json_array_size(answer) <= 0)
|
||||
return false;
|
||||
|
||||
return true;
|
||||
}
|
||||
|
||||
void FillProducts(json_t const * time, json_t const * price, vector<uber::Product> & products)
|
||||
{
|
||||
auto const timeSize = json_array_size(time);
|
||||
for (size_t i = 0; i < timeSize; ++i)
|
||||
{
|
||||
uber::Product product;
|
||||
json_int_t estimatedTime = 0;
|
||||
auto const item = json_array_get(time, i);
|
||||
my::FromJSONObject(item, "display_name", product.m_name);
|
||||
my::FromJSONObject(item, "estimate", estimatedTime);
|
||||
product.m_time = strings::to_string(estimatedTime);
|
||||
products.push_back(product);
|
||||
}
|
||||
|
||||
auto const priceSize = json_array_size(price);
|
||||
for (size_t i = 0; i < priceSize; ++i)
|
||||
{
|
||||
string name;
|
||||
auto const item = json_array_get(price, i);
|
||||
|
||||
my::FromJSONObject(item, "display_name", name);
|
||||
auto const it = find_if(products.begin(), products.end(), [&name](uber::Product const & product){
|
||||
return product.m_name == name;
|
||||
});
|
||||
if (it == products.end())
|
||||
continue;
|
||||
|
||||
my::FromJSONObject(item, "product_id", it->m_productId);
|
||||
my::FromJSONObject(item, "estimate", it->m_price);
|
||||
}
|
||||
|
||||
products.erase(remove_if(products.begin(), products.end(), [](uber::Product const & p){
|
||||
return p.m_name.empty() || p.m_productId.empty() || p.m_time.empty() ||
|
||||
p.m_price.empty();
|
||||
}), products.end());
|
||||
}
|
||||
|
||||
void GetAvailableProductsAsync(ms::LatLon const & from, ms::LatLon const & to, size_t const requestId,
|
||||
uber::ProductsCallback const & fn)
|
||||
{
|
||||
auto time = async(launch::async, uber::RawApi::GetEstimatedTime, ref(from));
|
||||
auto price = async(launch::async, uber::RawApi::GetEstimatedPrice, ref(from), ref(to));
|
||||
|
||||
vector<uber::Product> products;
|
||||
|
||||
try
|
||||
{
|
||||
my::Json timeRoot(time.get().c_str());
|
||||
my::Json priceRoot(price.get().c_str());
|
||||
auto const timesArray = json_object_get(timeRoot.get(), "times");
|
||||
auto const pricesArray = json_object_get(priceRoot.get(), "prices");
|
||||
if (CheckUberAnswer(timesArray) && CheckUberAnswer(pricesArray))
|
||||
{
|
||||
FillProducts(timesArray, pricesArray, products);
|
||||
}
|
||||
}
|
||||
catch (my::Json::Exception const & e)
|
||||
{
|
||||
LOG(LERROR, (e.Msg()));
|
||||
products.clear();
|
||||
}
|
||||
|
||||
fn(products, requestId);
|
||||
}
|
||||
} // namespace
|
||||
|
||||
namespace uber
|
||||
{
|
||||
// static
|
||||
string RawApi::GetProducts(ms::LatLon const & pos)
|
||||
{
|
||||
stringstream url;
|
||||
url << "https://api.uber.com/v1/products?server_token=" << UBER_SERVER_TOKEN <<
|
||||
"&latitude=" << static_cast<float>(pos.lat) <<
|
||||
"&longitude=" << static_cast<float>(pos.lon);
|
||||
|
||||
return RunSimpleHttpRequest(url.str());
|
||||
}
|
||||
|
||||
// static
|
||||
string RawApi::GetEstimatedTime(ms::LatLon const & pos)
|
||||
{
|
||||
// stringstream url;
|
||||
// url << "https://api.uber.com/v1/products?server_token=" << UBER_SERVER_TOKEN <<
|
||||
// "&start_latitude=" << static_cast<float>(pos.lat) <<
|
||||
// "&start_longitude=" << static_cast<float>(pos.lon);
|
||||
|
||||
// return RunSimpleHttpRequest(url.str());
|
||||
return R"({
|
||||
"times":[
|
||||
{
|
||||
"localized_display_name":"uberPOOL",
|
||||
"estimate":180,
|
||||
"display_name":"uberPOOL",
|
||||
"product_id":"26546650-e557-4a7b-86e7-6a3942445247"
|
||||
},
|
||||
{
|
||||
"localized_display_name":"uberX",
|
||||
"estimate":180,
|
||||
"display_name":"uberX",
|
||||
"product_id":"a1111c8c-c720-46c3-8534-2fcdd730040d"
|
||||
},
|
||||
{
|
||||
"localized_display_name":"uberXL",
|
||||
"estimate":420,
|
||||
"display_name":"uberXL",
|
||||
"product_id":"821415d8-3bd5-4e27-9604-194e4359a449"
|
||||
},
|
||||
{
|
||||
"localized_display_name":"UberBLACK",
|
||||
"estimate":180,
|
||||
"display_name":"UberBLACK",
|
||||
"product_id":"d4abaae7-f4d6-4152-91cc-77523e8165a4"
|
||||
}
|
||||
]
|
||||
})";
|
||||
}
|
||||
|
||||
// static
|
||||
string RawApi::GetEstimatedPrice(ms::LatLon const & from, ms::LatLon const & to)
|
||||
{
|
||||
// stringstream url;
|
||||
// url << "https://api.uber.com/v1/products?server_token=" << UBER_SERVER_TOKEN <<
|
||||
// "&start_latitude=" << static_cast<float>(from.lat) <<
|
||||
// "&start_longitude=" << static_cast<float>(from.lon) <<
|
||||
// "&end_latitude=" << static_cast<float>(to.lat) <<
|
||||
// "&end_longitude=" << static_cast<float>(to.lon);
|
||||
|
||||
// return RunSimpleHttpRequest(url.str());
|
||||
return R"({
|
||||
"prices":[
|
||||
{
|
||||
"product_id": "26546650-e557-4a7b-86e7-6a3942445247",
|
||||
"currency_code": "USD",
|
||||
"display_name": "POOL",
|
||||
"estimate": "$7",
|
||||
"low_estimate": 7,
|
||||
"high_estimate": 7,
|
||||
"surge_multiplier": 1,
|
||||
"duration": 640,
|
||||
"distance": 5.34
|
||||
},
|
||||
{
|
||||
"product_id": "08f17084-23fd-4103-aa3e-9b660223934b",
|
||||
"currency_code": "USD",
|
||||
"display_name": "UberBLACK",
|
||||
"estimate": "$23-29",
|
||||
"low_estimate": 23,
|
||||
"high_estimate": 29,
|
||||
"surge_multiplier": 1,
|
||||
"duration": 640,
|
||||
"distance": 5.34
|
||||
},
|
||||
{
|
||||
"product_id": "9af0174c-8939-4ef6-8e91-1a43a0e7c6f6",
|
||||
"currency_code": "USD",
|
||||
"display_name": "UberSUV",
|
||||
"estimate": "$36-44",
|
||||
"low_estimate": 36,
|
||||
"high_estimate": 44,
|
||||
"surge_multiplier": 1.25,
|
||||
"duration": 640,
|
||||
"distance": 5.34
|
||||
},
|
||||
{
|
||||
"product_id": "aca52cea-9701-4903-9f34-9a2395253acb",
|
||||
"currency_code": null,
|
||||
"display_name": "uberTAXI",
|
||||
"estimate": "Metered",
|
||||
"low_estimate": null,
|
||||
"high_estimate": null,
|
||||
"surge_multiplier": 1,
|
||||
"duration": 640,
|
||||
"distance": 5.34
|
||||
},
|
||||
{
|
||||
"product_id": "a27a867a-35f4-4253-8d04-61ae80a40df5",
|
||||
"currency_code": "USD",
|
||||
"display_name": "uberX",
|
||||
"estimate": "$15",
|
||||
"low_estimate": 15,
|
||||
"high_estimate": 15,
|
||||
"surge_multiplier": 1,
|
||||
"duration": 640,
|
||||
"distance": 5.34
|
||||
}
|
||||
]
|
||||
})";
|
||||
}
|
||||
|
||||
size_t Api::GetAvailableProducts(ms::LatLon const & from, ms::LatLon const & to,
|
||||
ProductsCallback const & fn)
|
||||
{
|
||||
lock_guard<mutex> lock(m_mutex);
|
||||
|
||||
m_thread.reset();
|
||||
static size_t requestId = 0;
|
||||
++requestId;
|
||||
m_thread = unique_ptr<threads::SimpleThread, threadDeleter>(
|
||||
new threads::SimpleThread(GetAvailableProductsAsync, ref(from), ref(to), requestId, ref(fn)),
|
||||
[](threads::SimpleThread * ptr) { ptr->join(); delete ptr; });
|
||||
|
||||
return requestId;
|
||||
}
|
||||
|
||||
// static
|
||||
string Api::GetRideRequestLink(string const & m_productId, ms::LatLon const & from,
|
||||
ms::LatLon const & to)
|
||||
{
|
||||
stringstream url;
|
||||
url << "uber://?client_id=" << UBER_CLIENT_ID <<
|
||||
"&action=setPickup&product_id=" << m_productId <<
|
||||
"&pickup[latitude]=" << static_cast<float>(from.lat) <<
|
||||
"&pickup[longitude]=" << static_cast<float>(from.lon) <<
|
||||
"&dropoff[latitude]=" << static_cast<float>(to.lat)<<
|
||||
"&dropoff[longitude]=" << static_cast<float>(to.lon);
|
||||
|
||||
return url.str();
|
||||
}
|
||||
} // namespace uber
|
74
partners_api/uber_api.hpp
Normal file
74
partners_api/uber_api.hpp
Normal file
|
@ -0,0 +1,74 @@
|
|||
#pragma once
|
||||
|
||||
#include "base/thread.hpp"
|
||||
|
||||
#include "std/function.hpp"
|
||||
#include "std/mutex.hpp"
|
||||
#include "std/string.hpp"
|
||||
#include "std/unique_ptr.hpp"
|
||||
#include "std/vector.hpp"
|
||||
|
||||
namespace ms
|
||||
{
|
||||
class LatLon;
|
||||
} // namespace ms
|
||||
|
||||
namespace downloader
|
||||
{
|
||||
class HttpRequest;
|
||||
} // namespace downloader
|
||||
|
||||
namespace uber
|
||||
{
|
||||
// Uber api wrapper based on synchronous http requests.
|
||||
class RawApi
|
||||
{
|
||||
public:
|
||||
/// Returns information about the Uber products offered at a given location.
|
||||
/// The response includes the display name and other details about each product, and lists the
|
||||
/// products in the proper display order. This endpoint does not reflect real-time availability
|
||||
/// of the products.
|
||||
static string GetProducts(ms::LatLon const & pos);
|
||||
/// Returns ETAs for all products currently available at a given location, with the ETA for each
|
||||
/// product expressed as integers in seconds. If a product returned from GetProducts is not
|
||||
/// returned from this endpoint for a given latitude/longitude pair then there are currently none
|
||||
/// of that product available to request. Call this endpoint every minute to provide the most
|
||||
/// accurate, up-to-date ETAs.
|
||||
static string GetEstimatedTime(ms::LatLon const & pos);
|
||||
/// Returns an estimated price range for each product offered at a given location. The price
|
||||
/// estimate is provided as a formatted string with the full price range and the localized
|
||||
/// currency symbol.
|
||||
static string GetEstimatedPrice(ms::LatLon const & from, ms::LatLon const & to);
|
||||
};
|
||||
|
||||
struct Product
|
||||
{
|
||||
string m_productId;
|
||||
string m_name;
|
||||
string m_time;
|
||||
string m_price;
|
||||
};
|
||||
/// @products - vector of available products for requested route.
|
||||
/// @requestId - identificator which was provided to GetAvailableProducts to identify request.
|
||||
using ProductsCallback = function<void(vector<Product> const & products, size_t const requestId)>;
|
||||
|
||||
class Api
|
||||
{
|
||||
public:
|
||||
/// Requests list of available products from Uber. Returns request identificator immediately.
|
||||
size_t GetAvailableProducts(ms::LatLon const & from, ms::LatLon const & to,
|
||||
ProductsCallback const & fn);
|
||||
|
||||
/// Returns link which allows you to launch the Uber app.
|
||||
static string GetRideRequestLink(string const & m_productId, ms::LatLon const & from, ms::LatLon const & to);
|
||||
|
||||
private:
|
||||
using threadDeleter = function<void(threads::SimpleThread *)>;
|
||||
unique_ptr<threads::SimpleThread, threadDeleter> m_thread;
|
||||
|
||||
mutex m_mutex;
|
||||
};
|
||||
} // namespace uber
|
||||
|
||||
|
||||
|
|
@ -1,6 +1,6 @@
|
|||
# Main application in qt.
|
||||
ROOT_DIR = ..
|
||||
DEPENDENCIES = map drape_frontend routing search storage indexer drape platform editor geometry \
|
||||
DEPENDENCIES = map drape_frontend routing search storage indexer drape partners_api platform editor geometry \
|
||||
coding base freetype expat fribidi tomcrypt jansson protobuf osrm stats_client \
|
||||
minizip succinct pugixml oauthcpp
|
||||
|
||||
|
|
|
@ -6,7 +6,7 @@ CONFIG -= app_bundle
|
|||
TEMPLATE = app
|
||||
|
||||
ROOT_DIR = ../..
|
||||
DEPENDENCIES = map drape_frontend routing search storage indexer drape platform_tests_support platform editor opening_hours geometry \
|
||||
DEPENDENCIES = map drape_frontend routing search storage indexer drape partners_api platform_tests_support platform editor opening_hours geometry \
|
||||
coding base freetype expat fribidi tomcrypt jansson protobuf osrm stats_client \
|
||||
minizip succinct pugixml oauthcpp
|
||||
|
||||
|
|
Loading…
Add table
Reference in a new issue