-
Notifications
You must be signed in to change notification settings - Fork 6
New issue
Have a question about this project? Sign up for a free GitHub account to open an issue and contact its maintainers and the community.
By clicking “Sign up for GitHub”, you agree to our terms of service and privacy statement. We’ll occasionally send you account related emails.
Already on GitHub? Sign in to your account
Extract EngagementManager
and cover it with unit tests
#82
Changes from all commits
0327b22
a2aff95
7a2c289
becb67f
9d9e6b3
7767307
cf6366f
b0dd745
f66493c
97b35ad
7fbc013
3f6ead9
534f5b2
51a4674
1d2b94c
cb4d8a7
e4dd447
b9f8797
bf459f5
d752495
9855f16
e67e603
e6a556d
227f9a5
654e374
98dfeaa
File filter
Filter by extension
Conversations
Jump to
Diff view
Diff view
There are no files selected for viewing
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,120 @@ | ||
package com.parsely.parselyandroid; | ||
|
||
import java.util.Calendar; | ||
import java.util.HashMap; | ||
import java.util.Map; | ||
import java.util.TimeZone; | ||
import java.util.Timer; | ||
import java.util.TimerTask; | ||
|
||
/** | ||
* Engagement manager for article and video engagement. | ||
* <p> | ||
* Implemented to handle its own queuing of future executions to accomplish | ||
* two things: | ||
* <p> | ||
* 1. Flushing any engaged time before canceling. | ||
* 2. Progressive backoff for long engagements to save data. | ||
*/ | ||
class EngagementManager { | ||
|
||
private final ParselyTracker parselyTracker; | ||
public Map<String, Object> baseEvent; | ||
private boolean started; | ||
private final Timer parentTimer; | ||
private TimerTask waitingTimerTask; | ||
private long latestDelayMillis, totalTime; | ||
private Calendar startTime; | ||
private final UpdateEngagementIntervalCalculator intervalCalculator; | ||
|
||
public EngagementManager( | ||
ParselyTracker parselyTracker, | ||
Timer parentTimer, | ||
long intervalMillis, | ||
Map<String, Object> baseEvent, | ||
UpdateEngagementIntervalCalculator intervalCalculator | ||
) { | ||
this.parselyTracker = parselyTracker; | ||
this.baseEvent = baseEvent; | ||
this.parentTimer = parentTimer; | ||
this.intervalCalculator = intervalCalculator; | ||
latestDelayMillis = intervalMillis; | ||
totalTime = 0; | ||
startTime = Calendar.getInstance(TimeZone.getTimeZone("UTC")); | ||
} | ||
|
||
public boolean isRunning() { | ||
return started; | ||
} | ||
|
||
public void start() { | ||
scheduleNextExecution(latestDelayMillis); | ||
started = true; | ||
startTime = Calendar.getInstance(TimeZone.getTimeZone("UTC")); | ||
} | ||
|
||
public void stop() { | ||
waitingTimerTask.cancel(); | ||
started = false; | ||
} | ||
|
||
public boolean isSameVideo(String url, String urlRef, ParselyVideoMetadata metadata) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Suggestion (💡): Consider resolving/suppressing all the warnings related to this method:
FYI: For no.2 there is an easy fix, just flit the BEFORE: There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. I'd leave this to a different PR - I wanted this PR to be focused on extracting |
||
Map<String, Object> baseMetadata = (Map<String, Object>) baseEvent.get("metadata"); | ||
return (baseEvent.get("url").equals(url) && | ||
baseEvent.get("urlref").equals(urlRef) && | ||
baseMetadata.get("link").equals(metadata.link) && | ||
(int) (baseMetadata.get("duration")) == metadata.durationSeconds); | ||
} | ||
|
||
private void scheduleNextExecution(long delay) { | ||
TimerTask task = new TimerTask() { | ||
public void run() { | ||
doEnqueue(scheduledExecutionTime()); | ||
latestDelayMillis = intervalCalculator.updateLatestInterval(startTime); | ||
scheduleNextExecution(latestDelayMillis); | ||
} | ||
|
||
public boolean cancel() { | ||
boolean output = super.cancel(); | ||
// Only enqueue when we actually canceled something. If output is false then | ||
// this has already been canceled. | ||
if (output) { | ||
doEnqueue(scheduledExecutionTime()); | ||
} | ||
return output; | ||
} | ||
}; | ||
latestDelayMillis = delay; | ||
parentTimer.schedule(task, delay); | ||
waitingTimerTask = task; | ||
} | ||
|
||
private void doEnqueue(long scheduledExecutionTime) { | ||
// Create a copy of the base event to enqueue | ||
Map<String, Object> event = new HashMap<>(baseEvent); | ||
ParselyTracker.PLog(String.format("Enqueuing %s event.", event.get("action"))); | ||
|
||
// Update `ts` for the event since it's happening right now. | ||
Calendar now = Calendar.getInstance(TimeZone.getTimeZone("UTC")); | ||
@SuppressWarnings("unchecked") | ||
Map<String, Object> baseEventData = (Map<String, Object>) event.get("data"); | ||
assert baseEventData != null; | ||
Map<String, Object> data = new HashMap<>(baseEventData); | ||
data.put("ts", now.getTimeInMillis()); | ||
event.put("data", data); | ||
|
||
// Adjust inc by execution time in case we're late or early. | ||
long executionDiff = (System.currentTimeMillis() - scheduledExecutionTime); | ||
long inc = (latestDelayMillis + executionDiff); | ||
totalTime += inc; | ||
event.put("inc", inc / 1000); | ||
event.put("tt", totalTime); | ||
|
||
parselyTracker.enqueueEvent(event); | ||
} | ||
|
||
|
||
public double getIntervalMillis() { | ||
return latestDelayMillis; | ||
} | ||
} |
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,22 @@ | ||
package com.parsely.parselyandroid; | ||
|
||
import androidx.annotation.NonNull; | ||
|
||
import java.util.Calendar; | ||
import java.util.TimeZone; | ||
|
||
class UpdateEngagementIntervalCalculator { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Suggestion (💡): Ah, I forgot to mention this in my review, what about making this a There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. In next iterations, I'll inject some There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Nice to know, thanks for the heads-up on that. 💯 |
||
|
||
private static final long MAX_TIME_BETWEEN_HEARTBEATS = 60 * 60; | ||
private static final long OFFSET_MATCHING_BASE_INTERVAL = 35; | ||
private static final double BACKOFF_PROPORTION = 0.3; | ||
|
||
long updateLatestInterval(@NonNull final Calendar startTime) { | ||
There was a problem hiding this comment. Choose a reason for hiding this commentThe reason will be displayed to describe this comment to others. Learn more. Unit tests for this class are planned to be added in a separate PR. |
||
Calendar now = Calendar.getInstance(TimeZone.getTimeZone("UTC")); | ||
long totalTrackedTime = (now.getTime().getTime() - startTime.getTime().getTime()) / 1000; | ||
double totalWithOffset = totalTrackedTime + OFFSET_MATCHING_BASE_INTERVAL; | ||
double newInterval = totalWithOffset * BACKOFF_PROPORTION; | ||
long clampedNewInterval = (long) Math.min(MAX_TIME_BETWEEN_HEARTBEATS, newInterval); | ||
return clampedNewInterval * 1000; | ||
} | ||
} |
There was a problem hiding this comment.
Choose a reason for hiding this comment
The reason will be displayed to describe this comment to others. Learn more.
Suggestion (💡): I suggest making this
@Nullable
and then checking that duringstop()
:PS: Actually, I would even recommend for such refactor/test PRs to add any missing nullability annotation wherever possible going forward. 🤷