[android] Added realtime statistics for guides

This commit is contained in:
r.kuznetsov 2019-09-03 18:44:56 +03:00 committed by Aleksandr Zatsepin
parent 59e74dae4b
commit 3fc8e2ecd5
16 changed files with 206 additions and 39 deletions

@ -1 +1 @@
Subproject commit 98e8f7dfbc9103ed8371a9c6bd180b4a49eed333
Subproject commit 7dfbee9ba94f918836c276b4eee920883a6c1fb6

View file

@ -2,6 +2,8 @@
#include "android/jni/com/mapswithme/core/jni_helper.hpp"
#include "base/url_helpers.hpp"
#include <jni.h>
#include <string>
@ -14,6 +16,12 @@ extern "C"
return env->NewStringUTF(ALOHALYTICS_URL);
}
JNIEXPORT jstring JNICALL
Java_com_mapswithme_maps_PrivateVariables_alohalyticsRealtimeUrl(JNIEnv * env, jclass clazz)
{
return env->NewStringUTF(base::url::Join(ALOHALYTICS_URL, "realtime").c_str());
}
JNIEXPORT jstring JNICALL
Java_com_mapswithme_maps_PrivateVariables_flurryKey(JNIEnv * env, jclass clazz)
{

View file

@ -11,6 +11,8 @@
#include "platform/preferred_languages.hpp"
#include "base/string_utils.hpp"
#include <utility>
using namespace std::placeholders;
@ -1078,6 +1080,22 @@ Java_com_mapswithme_maps_bookmarks_data_BookmarkManager_nativeResetInvalidCatego
frm()->GetBookmarkManager().ResetInvalidCategories();
}
JNIEXPORT jstring JNICALL
Java_com_mapswithme_maps_bookmarks_data_BookmarkManager_nativeGuidesIds(JNIEnv * env, jobject)
{
auto & bm = frm()->GetBookmarkManager();
auto const guides = bm.GetCategoriesFromCatalog(
std::bind(&BookmarkManager::IsGuide, std::placeholders::_1));
return ToJavaString(env, strings::JoinStrings(guides.begin(), guides.end(), ','));
}
JNIEXPORT jboolean JNICALL
Java_com_mapswithme_maps_bookmarks_data_BookmarkManager_nativeIsGuide(JNIEnv * env, jobject,
jint accessRulesIndex)
{
return static_cast<jboolean>(BookmarkManager::IsGuide(static_cast<kml::AccessRules>(accessRulesIndex)));
}
JNIEXPORT jobjectArray JNICALL
Java_com_mapswithme_maps_bookmarks_data_BookmarkManager_nativeGetBookmarkCategories(JNIEnv *env,
jobject thiz)

View file

@ -10,6 +10,8 @@ public class PrivateVariables
@NonNull
public static native String alohalyticsUrl();
@NonNull
public static native String alohalyticsRealtimeUrl();
@NonNull
public static native String flurryKey();
@NonNull
public static native String appsFlyerKey();

View file

@ -512,12 +512,16 @@ public class BookmarksListFragment extends BaseMwmRecyclerFragment<BookmarkListA
final Bookmark bookmark = (Bookmark) adapter.getItem(position);
i.putExtra(MwmActivity.EXTRA_TASK,
new Factory.ShowBookmarkTask(bookmark.getCategoryId(), bookmark.getBookmarkId()));
if (BookmarkManager.INSTANCE.isGuide(mCategoryDataSource.getData()))
Statistics.INSTANCE.trackGuideBookmarkSelect(mCategoryDataSource.getData().getServerId());
break;
case BookmarkListAdapter.TYPE_TRACK:
final Track track = (Track) adapter.getItem(position);
i.putExtra(MwmActivity.EXTRA_TASK,
new Factory.ShowTrackTask(track.getCategoryId(), track.getTrackId()));
if (BookmarkManager.INSTANCE.isGuide(mCategoryDataSource.getData()))
Statistics.INSTANCE.trackGuideTrackSelect(mCategoryDataSource.getData().getServerId());
break;
}

View file

@ -82,6 +82,14 @@ public class CachedBookmarkCategoriesFragment extends BaseBookmarkCategoriesFrag
return new CatalogBookmarkCategoriesAdapter(requireContext());
}
@Override
public void onItemClick(@NonNull View v, @NonNull BookmarkCategory category)
{
if (BookmarkManager.INSTANCE.isGuide(category))
Statistics.INSTANCE.trackGuideOpen(category.getServerId());
super.onItemClick(v, category);
}
@Override
public void onFooterClick()
{
@ -122,6 +130,8 @@ public class CachedBookmarkCategoriesFragment extends BaseBookmarkCategoriesFrag
UiUtils.showIf(isEmptyAdapter, mEmptyViewContainer);
mPayloadContainer.setVisibility(isEmptyAdapter ? View.GONE : View.VISIBLE);
mProgressContainer.setVisibility(View.GONE);
if (!isEmptyAdapter)
Statistics.INSTANCE.trackGuidesShown(BookmarkManager.INSTANCE.getGuidesIds());
}
}
@ -207,6 +217,7 @@ public class CachedBookmarkCategoriesFragment extends BaseBookmarkCategoriesFrag
UiUtils.show(mPayloadContainer);
UiUtils.hide(mProgressContainer, mEmptyViewContainer);
getAdapter().notifyDataSetChanged();
Statistics.INSTANCE.trackGuidesShown(BookmarkManager.INSTANCE.getGuidesIds());
}
else
{

View file

@ -711,6 +711,17 @@ public enum BookmarkManager
return nativeGetCatalogFrontendUrl(utm);
}
@NonNull
public String getGuidesIds()
{
return nativeGuidesIds();
}
public boolean isGuide(@NonNull BookmarkCategory category)
{
return category.isFromCatalog() && nativeIsGuide(category.getAccessRules().ordinal());
}
public void requestRouteTags()
{
nativeRequestCatalogTags();
@ -938,6 +949,10 @@ public enum BookmarkManager
boolean hasMyPosition, double lat, double lon,
long timestamp);
@NonNull
private static native String nativeGuidesIds();
private static native boolean nativeIsGuide(int accessRulesIndex);
public interface BookmarksLoadingListener
{
void onBookmarksLoadingStarted();

View file

@ -207,7 +207,7 @@ public class AdsRemovalPurchaseDialog extends BaseMwmDialogFragment
Statistics.INSTANCE.trackPurchasePreviewSelect(PrivateVariables.adsRemovalServerId(),
details.getProductId());
Statistics.INSTANCE.trackPurchaseEvent(Statistics.EventName.INAPP_PURCHASE_PREVIEW_PAY,
purchaseId);
purchaseId, Statistics.STATISTICS_CHANNEL_REALTIME);
}
void onExplanationClick()

View file

@ -134,7 +134,7 @@ public class BookmarkPaymentFragment extends BaseMwmFragment
Statistics.INSTANCE.trackPurchasePreviewSelect(mPaymentData.getServerId(),
mPaymentData.getProductId());
Statistics.INSTANCE.trackPurchaseEvent(Statistics.EventName.INAPP_PURCHASE_PREVIEW_PAY,
mPaymentData.getServerId());
mPaymentData.getServerId(), Statistics.STATISTICS_CHANNEL_REALTIME);
startPurchaseTransaction();
}

View file

@ -111,7 +111,8 @@ public class BookmarkSubscriptionFragment extends BaseAuthFragment
activateState(BookmarkSubscriptionPaymentState.PINGING);
Statistics.INSTANCE.trackPurchaseEvent(Statistics.EventName.INAPP_PURCHASE_PREVIEW_PAY,
PrivateVariables.bookmarksSubscriptionServerId());
PrivateVariables.bookmarksSubscriptionServerId(),
Statistics.STATISTICS_CHANNEL_REALTIME);
}
@Override

View file

@ -809,7 +809,8 @@ public class PlacePageView extends NestedScrollView
private void showDescriptionScreen()
{
Statistics.INSTANCE.trackEvent(Statistics.EventName.PLACEPAGE_DESCRIPTION_MORE);
Statistics.INSTANCE.trackEvent(Statistics.EventName.PLACEPAGE_DESCRIPTION_MORE,
Statistics.STATISTICS_CHANNEL_REALTIME);
Context context = mPlaceDescriptionContainer.getContext();
String description = Objects.requireNonNull(mMapObject).getDescription();
Intent intent = new Intent(context, PlaceDescriptionActivity.class)

View file

@ -69,6 +69,10 @@ import static com.mapswithme.util.statistics.Statistics.EventName.BM_SYNC_PROPOS
import static com.mapswithme.util.statistics.Statistics.EventName.BM_SYNC_PROPOSAL_TOGGLE;
import static com.mapswithme.util.statistics.Statistics.EventName.BM_SYNC_SUCCESS;
import static com.mapswithme.util.statistics.Statistics.EventName.DOWNLOADER_DIALOG_ERROR;
import static com.mapswithme.util.statistics.Statistics.EventName.GUIDES_BOOKMARK_SELECT;
import static com.mapswithme.util.statistics.Statistics.EventName.GUIDES_SHOWN;
import static com.mapswithme.util.statistics.Statistics.EventName.GUIDES_OPEN;
import static com.mapswithme.util.statistics.Statistics.EventName.GUIDES_TRACK_SELECT;
import static com.mapswithme.util.statistics.Statistics.EventName.INAPP_PURCHASE_PREVIEW_SELECT;
import static com.mapswithme.util.statistics.Statistics.EventName.INAPP_PURCHASE_PREVIEW_SHOW;
import static com.mapswithme.util.statistics.Statistics.EventName.INAPP_PURCHASE_PRODUCT_DELIVERED;
@ -128,6 +132,8 @@ import static com.mapswithme.util.statistics.Statistics.EventParam.PURCHASE;
import static com.mapswithme.util.statistics.Statistics.EventParam.RESTAURANT;
import static com.mapswithme.util.statistics.Statistics.EventParam.RESTAURANT_LAT;
import static com.mapswithme.util.statistics.Statistics.EventParam.RESTAURANT_LON;
import static com.mapswithme.util.statistics.Statistics.EventParam.SERVER_ID;
import static com.mapswithme.util.statistics.Statistics.EventParam.SERVER_IDS;
import static com.mapswithme.util.statistics.Statistics.EventParam.STATE;
import static com.mapswithme.util.statistics.Statistics.EventParam.TYPE;
import static com.mapswithme.util.statistics.Statistics.EventParam.VALUE;
@ -316,6 +322,14 @@ public enum Statistics
public static final int PP_BANNER_STATE_PREVIEW = 0;
public static final int PP_BANNER_STATE_DETAILS = 1;
@Retention(RetentionPolicy.SOURCE)
@IntDef({ STATISTICS_CHANNEL_DEFAULT, STATISTICS_CHANNEL_REALTIME })
public @interface StatisticsChannel {}
public static final int STATISTICS_CHANNEL_DEFAULT = org.alohalytics.Statistics.ONLY_CHANNEL;
private static final int REALTIME_CHANNEL_INDEX = 1;
public static final int STATISTICS_CHANNEL_REALTIME = STATISTICS_CHANNEL_DEFAULT | (1 << REALTIME_CHANNEL_INDEX);
// Statistics counters
private int mBookmarksCreated;
private int mSharedTimes;
@ -534,7 +548,12 @@ public enum Statistics
public static final String INAPP_SUGGESTION_SHOWN = "MapsMe_InAppSuggestion_shown";
public static final String INAPP_SUGGESTION_CLICKED = "MapsMe_InAppSuggestion_clicked";
public static final String INAPP_SUGGESTION_CLOSED = "MapsMe_InAppSuggestion_closed";
public static final String GUIDES_SHOWN = "Bookmarks_Downloaded_Guides_list";
public static final String GUIDES_OPEN = "Bookmarks_Downloaded_Guide_open";
public static final String GUIDES_BOOKMARK_SELECT = "Bookmarks_BookmarksList_Bookmark_select";
public static final String GUIDES_TRACK_SELECT = "Bookmarks_BookmarksList_Track_select";
public static class Settings
{
public static final String WEB_SITE = "Setings. Go to website";
@ -638,6 +657,8 @@ public enum Statistics
static final String VENDOR = "vendor";
static final String PRODUCT = "product";
static final String PURCHASE = "purchase";
static final String SERVER_ID = "server_id";
static final String SERVER_IDS = "server_ids";
private EventParam() {}
}
@ -755,25 +776,44 @@ public enum Statistics
// Statistics is still enabled/disabled separately and never sent anywhere if turned off.
// TODO (AlexZ): Remove this initialization dependency from JNI part.
org.alohalytics.Statistics.setDebugMode(BuildConfig.DEBUG);
org.alohalytics.Statistics.setup(PrivateVariables.alohalyticsUrl(), context);
org.alohalytics.Statistics.setup(new String[] { PrivateVariables.alohalyticsUrl(),
PrivateVariables.alohalyticsRealtimeUrl()}, context);
}
public void trackEvent(@NonNull String name)
{
trackEvent(name, STATISTICS_CHANNEL_DEFAULT);
}
public void trackEvent(@NonNull String name, @StatisticsChannel int channel)
{
if (mEnabled)
org.alohalytics.Statistics.logEvent(name);
org.alohalytics.Statistics.logEvent(name, channel);
mMediator.getEventLogger().logEvent(name, Collections.emptyMap());
}
public void trackEvent(@NonNull String name, @NonNull Map<String, String> params)
{
trackEvent(name, params, STATISTICS_CHANNEL_DEFAULT);
}
public void trackEvent(@NonNull String name, @NonNull Map<String, String> params,
@StatisticsChannel int channel)
{
if (mEnabled)
org.alohalytics.Statistics.logEvent(name, params);
org.alohalytics.Statistics.logEvent(name, params, channel);
mMediator.getEventLogger().logEvent(name, params);
}
public void trackEvent(@NonNull String name, @Nullable Location location, @NonNull Map<String, String> params)
public void trackEvent(@NonNull String name, @Nullable Location location,
@NonNull Map<String, String> params)
{
trackEvent(name, location, params, STATISTICS_CHANNEL_DEFAULT);
}
public void trackEvent(@NonNull String name, @Nullable Location location,
@NonNull Map<String, String> params, @StatisticsChannel int channel)
{
List<String> eventDictionary = new ArrayList<String>();
for (Map.Entry<String, String> entry : params.entrySet())
@ -785,14 +825,20 @@ public enum Statistics
params.put("lon", (location == null ? "N/A" : String.valueOf(location.getLongitude())));
if (mEnabled)
org.alohalytics.Statistics.logEvent(name, eventDictionary.toArray(new String[0]), location);
org.alohalytics.Statistics.logEvent(name, eventDictionary.toArray(new String[0]), location, channel);
mMediator.getEventLogger().logEvent(name, params);
}
public void trackEvent(@NonNull String name, @NonNull ParameterBuilder builder)
{
trackEvent(name, builder.get());
trackEvent(name, builder.get(), STATISTICS_CHANNEL_DEFAULT);
}
public void trackEvent(@NonNull String name, @NonNull ParameterBuilder builder,
@StatisticsChannel int channel)
{
trackEvent(name, builder.get(), channel);
}
public void startActivity(Activity activity)
@ -995,22 +1041,22 @@ public enum Statistics
@NonNull MapObject mapObject)
{
String provider = restaurant.getType() == Sponsored.TYPE_OPENTABLE ? OPENTABLE : "Unknown restaurant";
Statistics.INSTANCE.trackEvent(eventName, LocationHelper.INSTANCE.getLastKnownLocation(),
Statistics.params().add(PROVIDER, provider)
.add(RESTAURANT, restaurant.getId())
.add(RESTAURANT_LAT, mapObject.getLat())
.add(RESTAURANT_LON, mapObject.getLon()).get());
trackEvent(eventName, LocationHelper.INSTANCE.getLastKnownLocation(),
Statistics.params().add(PROVIDER, provider)
.add(RESTAURANT, restaurant.getId())
.add(RESTAURANT_LAT, mapObject.getLat())
.add(RESTAURANT_LON, mapObject.getLon()).get());
}
public void trackHotelEvent(@NonNull String eventName, @NonNull Sponsored hotel,
@NonNull MapObject mapObject)
{
String provider = hotel.getType() == Sponsored.TYPE_BOOKING ? BOOKING_COM : "Unknown hotel";
Statistics.INSTANCE.trackEvent(eventName, LocationHelper.INSTANCE.getLastKnownLocation(),
Statistics.params().add(PROVIDER, provider)
.add(HOTEL, hotel.getId())
.add(HOTEL_LAT, mapObject.getLat())
.add(HOTEL_LON, mapObject.getLon()).get());
trackEvent(eventName, LocationHelper.INSTANCE.getLastKnownLocation(),
Statistics.params().add(PROVIDER, provider)
.add(HOTEL, hotel.getId())
.add(HOTEL_LAT, mapObject.getLat())
.add(HOTEL_LON, mapObject.getLon()).get());
}
public void trackBookHotelEvent(@NonNull Sponsored hotel, @NonNull MapObject mapObject)
@ -1342,7 +1388,7 @@ public enum Statistics
@NonNull MapObject mapObject)
{
// Here we code category by means of rating.
Statistics.INSTANCE.trackEvent(eventName, LocationHelper.INSTANCE.getLastKnownLocation(),
trackEvent(eventName, LocationHelper.INSTANCE.getLastKnownLocation(),
Statistics.params().add(PROVIDER, convertToSponsor(sponsoredObj))
.add(CATEGORY, sponsoredObj.getRating())
.add(OBJECT_LAT, mapObject.getLat())
@ -1567,12 +1613,19 @@ public enum Statistics
{
trackEvent(INAPP_PURCHASE_PREVIEW_SHOW, params().add(VENDOR, vendor)
.add(PRODUCT, productId)
.add(PURCHASE, purchaseId));
.add(PURCHASE, purchaseId),
STATISTICS_CHANNEL_REALTIME);
}
public void trackPurchaseEvent(@NonNull String event, @NonNull String purchaseId)
{
trackEvent(event, params().add(PURCHASE, purchaseId));
trackPurchaseEvent(event, purchaseId, STATISTICS_CHANNEL_DEFAULT);
}
public void trackPurchaseEvent(@NonNull String event, @NonNull String purchaseId,
@StatisticsChannel int channel)
{
trackEvent(event, params().add(PURCHASE, purchaseId), channel);
}
public void trackPurchasePreviewSelect(@NonNull String purchaseId, @NonNull String productId)
@ -1632,18 +1685,39 @@ public enum Statistics
public void trackPurchaseProductDelivered(@NonNull String purchaseId, @NonNull String vendor)
{
trackEvent(INAPP_PURCHASE_PRODUCT_DELIVERED, params().add(VENDOR, vendor)
.add(PURCHASE, purchaseId));
.add(PURCHASE, purchaseId),
STATISTICS_CHANNEL_REALTIME);
}
public void trackTipsEvent(@NonNull String eventName, int type)
{
Statistics.INSTANCE.trackEvent(eventName, params().add(TYPE, type));
trackEvent(eventName, params().add(TYPE, type));
}
public void trackTipsClose(int type)
{
Statistics.INSTANCE.trackEvent(TIPS_TRICKS_CLOSE, params().add(TYPE, type)
.add(OPTION, OFFSCREEEN));
trackEvent(TIPS_TRICKS_CLOSE, params().add(TYPE, type).add(OPTION, OFFSCREEEN));
}
public void trackGuidesShown(@NonNull String serverIds)
{
if (!serverIds.isEmpty())
trackEvent(GUIDES_SHOWN, params().add(SERVER_IDS, serverIds), STATISTICS_CHANNEL_REALTIME);
}
public void trackGuideOpen(@NonNull String serverId)
{
trackEvent(GUIDES_OPEN, params().add(SERVER_ID, serverId), STATISTICS_CHANNEL_REALTIME);
}
public void trackGuideBookmarkSelect(@NonNull String serverId)
{
trackEvent(GUIDES_BOOKMARK_SELECT, params().add(SERVER_ID, serverId), STATISTICS_CHANNEL_REALTIME);
}
public void trackGuideTrackSelect(@NonNull String serverId)
{
trackEvent(GUIDES_TRACK_SELECT, params().add(SERVER_ID, serverId), STATISTICS_CHANNEL_REALTIME);
}
public static ParameterBuilder params()

View file

@ -55,7 +55,8 @@ void checkFlurryLogStatus(FlurryEventRecordStatus status)
[MRMyTracker setupTracker];
}
[Alohalytics setup:@(ALOHALYTICS_URL) withLaunchOptions:launchOptions];
[Alohalytics setup:@[@(ALOHALYTICS_URL), [NSString stringWithFormat:@"%@/%@", @(ALOHALYTICS_URL), @"realtime"]]
withLaunchOptions:launchOptions];
}
// Always call Facebook method, looks like it is required to handle some url schemes and sign on scenarios.
return [[FBSDKApplicationDelegate sharedInstance] application:application didFinishLaunchingWithOptions:launchOptions];

View file

@ -995,6 +995,25 @@ std::string BookmarkManager::GetLocalizedRegionAddress(m2::PointD const & pt)
return m_regionAddressGetter->GetLocalizedRegionAddress(pt);
}
std::vector<std::string> BookmarkManager::GetCategoriesFromCatalog(AccessRulesFilter && filter) const
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
std::vector<std::string> result;
for (auto const & c : m_categories)
{
if (!filter || filter(c.second->GetCategoryData().m_accessRules))
result.emplace_back(c.second->GetServerId());
}
return result;
}
// static
bool BookmarkManager::IsGuide(kml::AccessRules accessRules)
{
return accessRules == kml::AccessRules::Public || accessRules == kml::AccessRules::Paid ||
accessRules == kml::AccessRules::P2P;
}
void BookmarkManager::PrepareBookmarksAddresses(std::vector<SortBookmarkData> & bookmarksForSort,
AddressesCollection & newAddresses)
{
@ -2720,15 +2739,9 @@ void BookmarkManager::SetAllCategoriesVisibility(CategoryFilterType const filter
std::vector<std::string> BookmarkManager::GetAllPaidCategoriesIds() const
{
CHECK_THREAD_CHECKER(m_threadChecker, ());
std::vector<std::string> ids;
for (auto const & category : m_categories)
{
if (category.second->GetCategoryData().m_accessRules != kml::AccessRules::Paid)
continue;
ids.emplace_back(category.second->GetServerId());
}
return ids;
return GetCategoriesFromCatalog([](kml::AccessRules accessRules) {
return accessRules == kml::AccessRules::Paid;
});
}
bool BookmarkManager::CanConvert() const

View file

@ -439,6 +439,10 @@ public:
static std::string GetSortedByTimeBlockName(SortedByTimeBlockType blockType);
std::string GetLocalizedRegionAddress(m2::PointD const & pt);
using AccessRulesFilter = std::function<bool(kml::AccessRules)>;
std::vector<std::string> GetCategoriesFromCatalog(AccessRulesFilter && filter) const;
static bool IsGuide(kml::AccessRules accessRules);
private:
class MarksChangesTracker : public df::UserMarksProvider
{

View file

@ -2520,6 +2520,21 @@ void Framework::OnTapEvent(TapEvent const & tapEvent)
alohalytics::Stats::Instance().LogEvent("$SelectMapObject", kv,
alohalytics::Location::FromLatLon(ll.m_lat, ll.m_lon));
// Send realtime statistics about bookmark selection.
if (info.IsBookmark())
{
auto const serverId = m_bmManager->GetCategoryServerId(info.GetBookmarkCategoryId());
if (!serverId.empty())
{
auto const accessRules = m_bmManager->GetCategoryData(info.GetBookmarkCategoryId()).m_accessRules;
if (BookmarkManager::IsGuide(accessRules))
{
alohalytics::TStringMap params = {{"server_id", serverId}};
alohalytics::Stats::Instance().LogEvent("Map_GuideBookmark_open", params, alohalytics::kAllChannels);
}
}
}
if (info.GetSponsoredType() == SponsoredType::Booking)
{
GetPlatform().GetMarketingService().SendPushWooshTag(marketing::kBookHotelOnBookingComDiscovered);