mirror of
https://github.com/codeninjasllc/codecombat.git
synced 2024-11-28 10:06:08 -05:00
3fd5f49220
Add analytics per-day aggregation collection. Add analytics strings collection. Add per-day aggregation mongo insertion script. Update campaign editor to use aggregation collection. Update queries to use _id instead of created field.
252 lines
No EOL
8.6 KiB
JavaScript
252 lines
No EOL
8.6 KiB
JavaScript
// Print out level completion rates
|
|
|
|
// Usage:
|
|
// mongo <address>:<port>/<database> <script file> -u <username> -p <password>
|
|
|
|
// Bucketize start/finish events into days, then bucketize into levels
|
|
// Average playtime: level sessions created in timeframe, state.complete = true, then average 'playtime'
|
|
|
|
// TODO: Should this be total code problems / levels finished, or total / level session count instead?
|
|
// Average code problems: total code problems / levels staretd
|
|
|
|
// TODO: Why do a small number of 'Started level' not have properties.levelID set?
|
|
|
|
// TODO: spot check the data: NaN, only some 0.0 dates, etc.
|
|
// TODO: exclude levels with no interesting data?
|
|
|
|
var startTime = new Date();
|
|
|
|
var today = new Date();
|
|
today = today.toISOString().substr(0, 10);
|
|
print("Today is " + today);
|
|
|
|
var todayMinus6 = new Date();
|
|
todayMinus6.setUTCDate(todayMinus6.getUTCDate() - 6);
|
|
var startDate = todayMinus6.toISOString().substr(0, 10) + "T00:00:00.000Z";
|
|
print("Start date is " + startDate)
|
|
// startDate = "2015-01-02T00:00:00.000Z";
|
|
// var endDate = "2015-01-09T00:00:00.000Z";
|
|
|
|
function objectIdWithTimestamp(timestamp)
|
|
{
|
|
// Convert string date to Date object (otherwise assume timestamp is a date)
|
|
if (typeof(timestamp) == 'string') timestamp = new Date(timestamp);
|
|
// Convert date object to hex seconds since Unix epoch
|
|
var hexSeconds = Math.floor(timestamp/1000).toString(16);
|
|
// Create an ObjectId with that hex timestamp
|
|
var constructedObjectId = ObjectId(hexSeconds + "0000000000000000");
|
|
return constructedObjectId
|
|
}
|
|
|
|
function getCompletionRates() {
|
|
print("Getting completion rates...");
|
|
var queryParams = {
|
|
$and: [
|
|
{_id: {$gte: objectIdWithTimestamp(ISODate(startDate))}},
|
|
{$or: [ {"event" : 'Started Level'}, {"event" : 'Saw Victory'}]}
|
|
]
|
|
};
|
|
var cursor = db['analytics.log.events'].find(queryParams);
|
|
|
|
// <level><date><data>
|
|
var levelData = {};
|
|
while (cursor.hasNext()) {
|
|
var doc = cursor.next();
|
|
var created = doc.created.toISOString().substring(0, 10);
|
|
var event = doc.event;
|
|
var properties = doc.properties;
|
|
var level;
|
|
if (event === 'Saw Victory' && properties.level) level = properties.level.toLowerCase().replace(/ /g, '-');
|
|
else if (properties.levelID) level = properties.levelID
|
|
else continue
|
|
var user = doc.user;
|
|
|
|
if (level.length > longestLevelName) longestLevelName = level.length;
|
|
|
|
if (!levelData[level]) levelData[level] = {};
|
|
if (!levelData[level][created]) levelData[level][created] = {};
|
|
if (!levelData[level][created]['started']) levelData[level][created]['started'] = {};
|
|
if (!levelData[level][created]['finished']) levelData[level][created]['finished'] = {}
|
|
if (event === 'Started Level') levelData[level][created]['started'][user] = true;
|
|
else levelData[level][created]['finished'][user] = true;
|
|
}
|
|
longestLevelName += 2;
|
|
|
|
var levelRates = [];
|
|
for (level in levelData) {
|
|
var dateData = [];
|
|
var dateIndex = 0;
|
|
for (created in levelData[level]) {
|
|
var started =
|
|
dateData.push({
|
|
level: level,
|
|
created: created,
|
|
started: Object.keys(levelData[level][created]['started']).length,
|
|
finished: Object.keys(levelData[level][created]['finished']).length
|
|
});
|
|
if (dates.length === dateIndex) dates.push(created.substring(5));
|
|
dateIndex++;
|
|
}
|
|
levelRates.push(dateData);
|
|
}
|
|
// printjson(levelRates);
|
|
|
|
levelRates.sort(function(a,b) {return a[0].level < b[0].level ? -1 : 1});
|
|
for (levelKey in levelRates) levelRates[levelKey].sort(function(a,b) {return a.created < b.created ? 1 : -1});
|
|
|
|
return levelRates;
|
|
}
|
|
|
|
function addPlaytimeAverages(levelRates) {
|
|
print("Getting playtimes...");
|
|
// printjson(levelRates);
|
|
var match = {
|
|
"$match" : {
|
|
$and: [
|
|
{"created": { $gte: ISODate(startDate)}},
|
|
{"state.complete": true},
|
|
{"playtime": {$gt: 0}}
|
|
]
|
|
}};
|
|
|
|
var proj0 = {"$project": {
|
|
"_id" : 0,
|
|
"levelID" : 1,
|
|
"playtime": 1,
|
|
"created": {"$substr" : ["$created", 0, 10]}
|
|
}};
|
|
|
|
var group = {"$group" : {
|
|
"_id" : {
|
|
"created" : "$created",
|
|
"level": "$levelID"
|
|
},
|
|
"average" : {
|
|
"$avg" : "$playtime"
|
|
}
|
|
}};
|
|
|
|
var cursor = db['level.sessions'].aggregate(match, proj0, group);
|
|
|
|
var levelPlaytimeData = {};
|
|
while (cursor.hasNext()) {
|
|
var doc = cursor.next();
|
|
var created = doc._id.created;
|
|
var level = doc._id.level;
|
|
if (!levelPlaytimeData[level]) levelPlaytimeData[level] = {};
|
|
levelPlaytimeData[level][created] = doc.average;
|
|
}
|
|
|
|
for (levelIndex in levelRates) {
|
|
for (dateIndex in levelRates[levelIndex]) {
|
|
var level = levelRates[levelIndex][dateIndex].level;
|
|
var created = levelRates[levelIndex][dateIndex].created;
|
|
if (levelPlaytimeData[level] && levelPlaytimeData[level][created]) {
|
|
levelRates[levelIndex][dateIndex].averagePlaytime = levelPlaytimeData[level][created];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
function addUserCodeProblemCounts(levelRates) {
|
|
print("Getting user code problem counts...");
|
|
var match = {"$match" : {"created": { $gte: ISODate(startDate)}}};
|
|
|
|
var proj0 = {"$project": {
|
|
"_id" : 0,
|
|
"levelID" : 1,
|
|
"created": {"$substr" : ["$created", 0, 10]}
|
|
}};
|
|
|
|
var group = {"$group" : {
|
|
"_id" : {
|
|
"created" : "$created",
|
|
"level": "$levelID"
|
|
},
|
|
"count" : {
|
|
"$sum" : 1
|
|
}
|
|
}};
|
|
|
|
var cursor = db['level.sessions'].aggregate(match, proj0, group);
|
|
|
|
var levelPlaytimeData = {};
|
|
while (cursor.hasNext()) {
|
|
var doc = cursor.next();
|
|
var created = doc._id.created;
|
|
var level = doc._id.level;
|
|
if (!levelPlaytimeData[level]) levelPlaytimeData[level] = {};
|
|
levelPlaytimeData[level][created] = doc.count;
|
|
}
|
|
|
|
for (levelIndex in levelRates) {
|
|
for (dateIndex in levelRates[levelIndex]) {
|
|
var level = levelRates[levelIndex][dateIndex].level;
|
|
var created = levelRates[levelIndex][dateIndex].created;
|
|
if (levelPlaytimeData[level] && levelPlaytimeData[level][created]) {
|
|
levelRates[levelIndex][dateIndex].codeProblems = levelPlaytimeData[level][created];
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
var longestLevelName = -1;
|
|
var dates = [];
|
|
|
|
var levelRates = getCompletionRates();
|
|
// addPlaytimeAverages(levelRates);
|
|
// addUserCodeProblemCounts(levelRates);
|
|
|
|
// Print out all data
|
|
print("Columns: level, day, started, finished, completion rate, average finish playtime, average code problem count");
|
|
for (levelKey in levelRates) {
|
|
for (dateKey in levelRates[levelKey]) {
|
|
var created = levelRates[levelKey][dateKey].created;
|
|
var level = levelRates[levelKey][dateKey].level;
|
|
var started = levelRates[levelKey][dateKey].started;
|
|
var finished = levelRates[levelKey][dateKey].finished;
|
|
var completionRate = started > 0 ? finished / started : 0;
|
|
var averagePlaytime = levelRates[levelKey][dateKey].averagePlaytime;
|
|
averagePlaytime = averagePlaytime ? Math.round(averagePlaytime) : 0;
|
|
var averageCodeProblems = levelRates[levelKey][dateKey].codeProblems;
|
|
averageCodeProblems = averageCodeProblems ? (averageCodeProblems / started).toFixed(2) : 0.0;
|
|
var levelSpacer = new Array(longestLevelName - level.length).join(' ');
|
|
print(level + levelSpacer + created + "\t" + started + "\t" + finished + "\t" + (completionRate * 100).toFixed(2) + "% " + averagePlaytime + "s " + averageCodeProblems);
|
|
}
|
|
}
|
|
|
|
// Print out a nice grid of levels with 7 days of data
|
|
print("Columns: level, completion rate/average playtime/average code problems, completion rate/average playtime/average code problems ...");
|
|
print(new Array(longestLevelName).join(' ') + dates.join('\t\t'));
|
|
for (levelKey in levelRates) {
|
|
var hasStarted = false;
|
|
for (dateKey in levelRates[levelKey]) {
|
|
if (levelRates[levelKey][dateKey].started > 0) {
|
|
hasStarted = true;
|
|
break;
|
|
}
|
|
}
|
|
if (!hasStarted) continue;
|
|
|
|
if (levelRates[levelKey].length < 6) continue;
|
|
|
|
var level = levelRates[levelKey][0].level;
|
|
var levelSpacer = new Array(longestLevelName - level.length).join(' ');
|
|
var msg = level + levelSpacer;
|
|
|
|
for (dateKey in levelRates[levelKey]) {
|
|
var created = levelRates[levelKey][dateKey].created;
|
|
var started = levelRates[levelKey][dateKey].started;
|
|
var finished = levelRates[levelKey][dateKey].finished;
|
|
var averagePlaytime = levelRates[levelKey][dateKey].averagePlaytime;
|
|
averagePlaytime = averagePlaytime ? Math.round(averagePlaytime) : 0;
|
|
var averageCodeProblems = levelRates[levelKey][dateKey].codeProblems;
|
|
averageCodeProblems = averageCodeProblems ? averageCodeProblems / started : 0.0;
|
|
var completionRate = started > 0 ? finished / started : 0;
|
|
msg += (completionRate * 100).toFixed(2) + "/" + averagePlaytime + "/" + averageCodeProblems.toFixed(2) + "\t";
|
|
}
|
|
print(msg);
|
|
}
|
|
|
|
var endTime = new Date();
|
|
print("Runtime: " + (endTime - startTime)); |