Streaks!
This commit is contained in:
@ -3,7 +3,8 @@ var _ = require('lodash'),
|
|||||||
Courseware = require('./../models/Courseware'),
|
Courseware = require('./../models/Courseware'),
|
||||||
User = require('./../models/User'),
|
User = require('./../models/User'),
|
||||||
resources = require('./resources'),
|
resources = require('./resources'),
|
||||||
R = require('ramda');
|
R = require('ramda'),
|
||||||
|
moment = require('moment');
|
||||||
|
|
||||||
/**
|
/**
|
||||||
* Courseware controller
|
* Courseware controller
|
||||||
@ -269,82 +270,86 @@ exports.completedCourseware = function (req, res, next) {
|
|||||||
};
|
};
|
||||||
|
|
||||||
exports.completedZiplineOrBasejump = function (req, res, next) {
|
exports.completedZiplineOrBasejump = function (req, res, next) {
|
||||||
var isCompletedWith = req.body.bonfireInfo.completedWith || undefined;
|
var isCompletedWith = req.body.bonfireInfo.completedWith || false;
|
||||||
var isCompletedDate = Math.round(+new Date());
|
var isCompletedDate = Math.round(+new Date());
|
||||||
var coursewareHash = req.body.coursewareInfo.coursewareHash;
|
var coursewareHash = req.body.coursewareInfo.coursewareHash;
|
||||||
var solutionLink = req.body.coursewareInfo.solutionLink;
|
var solutionLink = req.body.coursewareInfo.solutionLink;
|
||||||
if (!solutionLink) {
|
if (!solutionLink) {
|
||||||
// flash error and redirect
|
// flash error and redirect
|
||||||
|
return next(new Error('No solution provided'));
|
||||||
}
|
}
|
||||||
|
|
||||||
if (isCompletedWith) {
|
if (isCompletedWith) {
|
||||||
var paired = User.find({"profile.username": isCompletedWith.toLowerCase()}).limit(1);
|
var paired = User.find({'profile.username': isCompletedWith.toLowerCase()}).limit(1);
|
||||||
paired.exec(function (err, pairedWith) {
|
paired.exec(function (err, pairedWith) {
|
||||||
if (err) {
|
if (err) {
|
||||||
return err;
|
return next(err);
|
||||||
} else {
|
} else {
|
||||||
var index = req.user.uncompletedBonfires.indexOf(bonfireHash);
|
var index = req.user.uncompletedCoursewares.indexOf(coursewareHash);
|
||||||
if (index > -1) {
|
if (index > -1) {
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
req.user.progressTimestamps.push(Date.now() || 0);
|
||||||
req.user.uncompletedBonfires.splice(index, 1)
|
req.user.uncompletedCoursewares.splice(index, 1);
|
||||||
}
|
}
|
||||||
pairedWith = pairedWith.pop();
|
pairedWith = pairedWith.pop();
|
||||||
|
|
||||||
index = pairedWith.uncompletedBonfires.indexOf(bonfireHash);
|
index = pairedWith.uncompletedCoursewares.indexOf(coursewareHash);
|
||||||
if (index > -1) {
|
if (index > -1) {
|
||||||
pairedWith.progressTimestamps.push(Date.now() || 0);
|
pairedWith.progressTimestamps.push(Date.now() || 0);
|
||||||
pairedWith.uncompletedBonfires.splice(index, 1);
|
pairedWith.uncompletedCoursewares.splice(index, 1);
|
||||||
|
|
||||||
}
|
}
|
||||||
|
|
||||||
pairedWith.completedBonfires.push({
|
pairedWith.completedCoursewares.push({
|
||||||
_id: bonfireHash,
|
_id: coursewareHash,
|
||||||
completedWith: req.user._id,
|
completedWith: req.user._id,
|
||||||
completedDate: isCompletedDate,
|
completedDate: isCompletedDate,
|
||||||
solution: isSolution
|
solution: solutionLink
|
||||||
});
|
});
|
||||||
|
|
||||||
req.user.completedBonfires.push({
|
req.user.completedCoursewares.push({
|
||||||
_id: bonfireHash,
|
_id: coursewareHash,
|
||||||
completedWith: pairedWith._id,
|
completedWith: pairedWith._id,
|
||||||
completedDate: isCompletedDate,
|
completedDate: isCompletedDate,
|
||||||
solution: isSolution
|
solution: solutionLink
|
||||||
});
|
});
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
req.user.save(function (err, user) {
|
||||||
|
if (err) {
|
||||||
|
return next(err);
|
||||||
|
}
|
||||||
pairedWith.save(function (err, paired) {
|
pairedWith.save(function (err, paired) {
|
||||||
if (err) {
|
if (err) {
|
||||||
throw err;
|
return next(err);
|
||||||
}
|
}
|
||||||
if (user && paired) {
|
if (user && paired) {
|
||||||
res.send(true);
|
return res.send(true);
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
})
|
});
|
||||||
} else {
|
} else {
|
||||||
|
|
||||||
req.user.completedBonfires.push({
|
req.user.completedCoursewares.push({
|
||||||
_id: bonfireHash,
|
_id: coursewareHash,
|
||||||
completedWith: null,
|
completedWith: null,
|
||||||
completedDate: isCompletedDate,
|
completedDate: isCompletedDate,
|
||||||
solution: isSolution
|
solution: solutionLink
|
||||||
});
|
});
|
||||||
|
|
||||||
var index = req.user.uncompletedCourse.indexOf(bonfireHash);
|
var index = req.user.uncompletedCourse.indexOf(coursewareHash);
|
||||||
if (index > -1) {
|
if (index > -1) {
|
||||||
req.user.progressTimestamps.push(Date.now() || 0);
|
req.user.progressTimestamps.push(Date.now() || 0);
|
||||||
req.user.uncompletedBonfires.splice(index, 1)
|
req.user.uncompletedCoursewares.splice(index, 1);
|
||||||
}
|
}
|
||||||
|
|
||||||
req.user.save(function (err, user) {
|
req.user.save(function (err, user) {
|
||||||
if (err) {
|
if (err) {
|
||||||
throw err;
|
return next(err);
|
||||||
}
|
}
|
||||||
if (user) {
|
if (user) {
|
||||||
debug('Saving user');
|
debug('Saving user');
|
||||||
res.send(true)
|
return res.send(true);
|
||||||
}
|
}
|
||||||
});
|
});
|
||||||
}
|
}
|
||||||
|
@ -181,23 +181,33 @@ module.exports = {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
var timeStamps = [];
|
var oldChallengeKeys = R.keys(req.user.challengesHash);
|
||||||
R.keys(req.user.challengesHash).forEach(function(key) {
|
|
||||||
"use strict";
|
var updatedTimesFromOldChallenges = oldChallengeKeys.map(function(timeStamp) {
|
||||||
var timeStamp = parseInt(challengesHash[key], 10);
|
if (timeStamp.toString().length !== 13) {
|
||||||
timeStamps.push({timeStamp: timeStamp.length !== 13 ? (+timeStamp) : (+timeStamp * 1000)});
|
timeStamp *= 1000;
|
||||||
|
}
|
||||||
|
return timeStamp;
|
||||||
});
|
});
|
||||||
|
|
||||||
req.user.completedCoursewares = Array.zip(timeStamps, coursewares,
|
var newTimeStamps = R.map(function(timeStamp) {
|
||||||
|
if (timeStamp.toString().length !== 13) {
|
||||||
|
timeStamp *= 1000;
|
||||||
|
}
|
||||||
|
return timeStamp;
|
||||||
|
}, req.user.progressTimestamps);
|
||||||
|
|
||||||
|
req.user.progressTimestamps = newTimeStamps;
|
||||||
|
|
||||||
|
|
||||||
|
req.user.completedCoursewares = Array.zip(updatedTimesFromOldChallenges, coursewares,
|
||||||
function(left, right) {
|
function(left, right) {
|
||||||
"use strict";
|
|
||||||
return ({
|
return ({
|
||||||
completedDate: left.timeStamp,
|
completedDate: left.timeStamp,
|
||||||
_id: right._id,
|
_id: right._id,
|
||||||
name: right.name
|
name: right.name
|
||||||
});
|
});
|
||||||
}).filter(function(elem) {
|
}).filter(function(elem) {
|
||||||
"use strict";
|
|
||||||
return elem.completedDate !== 0;
|
return elem.completedDate !== 0;
|
||||||
});
|
});
|
||||||
req.user.pointsNeedMigration = false;
|
req.user.pointsNeedMigration = false;
|
||||||
|
@ -7,7 +7,8 @@ var _ = require('lodash'),
|
|||||||
secrets = require('../config/secrets'),
|
secrets = require('../config/secrets'),
|
||||||
moment = require('moment'),
|
moment = require('moment'),
|
||||||
debug = require('debug')('freecc:cntr:challenges'),
|
debug = require('debug')('freecc:cntr:challenges'),
|
||||||
resources = require('./resources');
|
resources = require('./resources'),
|
||||||
|
R = require('ramda');
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
@ -199,54 +200,44 @@ exports.postEmailSignup = function(req, res, next) {
|
|||||||
* For Calendar display
|
* For Calendar display
|
||||||
*/
|
*/
|
||||||
|
|
||||||
exports.getStreak = function(req, res) {
|
exports.getStreak = function(req, res, next) {
|
||||||
|
|
||||||
Array.prototype.timeReduce = function(combiner, initialValue) {
|
req.user.progressTimestamps = req.user.progressTimestamps.sort(function(a, b) {
|
||||||
var counter,
|
return a - b;
|
||||||
accumulatedValue;
|
});
|
||||||
|
|
||||||
// If the array is empty, do nothing
|
var timeObject = Object.create(null);
|
||||||
if (this.length === 0) {
|
R.forEach(function(time) {
|
||||||
return this;
|
timeObject[moment(time).format('YYYY-MM-DD')] = time;
|
||||||
} else {
|
}, req.user.progressTimestamps);
|
||||||
// If the user didn't pass an initial value, use the first item.
|
|
||||||
if (arguments.length === 1) {
|
var tmpLongest = 1;
|
||||||
counter = 1;
|
var timeKeys = R.keys(timeObject);
|
||||||
accumulatedValue = this[0];
|
for (var i = 1; i <= timeKeys.length; i++) {
|
||||||
|
if (moment(timeKeys[i - 1]).add(1, 'd').toString()
|
||||||
|
=== moment(timeKeys[i]).toString()) {
|
||||||
|
tmpLongest++;
|
||||||
|
if (tmpLongest > req.user.currentStreak) {
|
||||||
|
req.user.currentStreak = tmpLongest;
|
||||||
|
}
|
||||||
|
if ( req.user.currentStreak > req.user.longestStreak) {
|
||||||
|
req.user.longestStreak = req.user.currentStreak;
|
||||||
}
|
}
|
||||||
else if (arguments.length >= 2) {
|
|
||||||
counter = 0;
|
|
||||||
accumulatedValue = initialValue;
|
|
||||||
}
|
}
|
||||||
else {
|
|
||||||
throw "Invalid arguments.";
|
|
||||||
}
|
}
|
||||||
|
|
||||||
// Loop through the array, feeding the current value and the result of
|
req.user.save(function(err) {
|
||||||
// the previous computation back into the combiner function until
|
if (err) {
|
||||||
// we've exhausted the entire array and are left with only one function.
|
return next(err);
|
||||||
while (counter < this.length) {
|
|
||||||
accumulatedValue = combiner(accumulatedValue, this[counter]);
|
|
||||||
counter++;
|
|
||||||
}
|
|
||||||
|
|
||||||
return [accumulatedValue];
|
|
||||||
}
|
}
|
||||||
|
});
|
||||||
|
s
|
||||||
|
var payload = {
|
||||||
|
longest: req.user.longestStreak,
|
||||||
|
timeObject: timeObject
|
||||||
};
|
};
|
||||||
|
|
||||||
var timeObject = req.user.progressTimestamps.timeReduce(function(accumulatedTime, timeStamp) {
|
return res.send(payload);
|
||||||
|
|
||||||
var copyOfAccumulatedTime = Object.create(accumulatedTime);
|
|
||||||
|
|
||||||
copyOfAccumulatedTime[moment(timeStamp)
|
|
||||||
.format('MMMM Do YYYY')] = timeStamp;
|
|
||||||
|
|
||||||
return copyOfAccumulatedTime;
|
|
||||||
},
|
|
||||||
{});
|
|
||||||
|
|
||||||
debug('TimeObject is', timeObject);
|
|
||||||
return res.send(timeObject);
|
|
||||||
};
|
};
|
||||||
|
|
||||||
/**
|
/**
|
||||||
@ -326,7 +317,7 @@ exports.returnUser = function(req, res, next) {
|
|||||||
var data = {};
|
var data = {};
|
||||||
var progressTimestamps = user.progressTimestamps;
|
var progressTimestamps = user.progressTimestamps;
|
||||||
for (var i = 0; i < progressTimestamps.length; i++) {
|
for (var i = 0; i < progressTimestamps.length; i++) {
|
||||||
data[progressTimestamps[i].toString()] = 1;
|
data[(progressTimestamps[i] / 1000).toString()] = 1;
|
||||||
}
|
}
|
||||||
|
|
||||||
res.render('account/show', {
|
res.render('account/show', {
|
||||||
|
@ -1,7 +1,9 @@
|
|||||||
var bcrypt = require('bcrypt-nodejs');
|
var bcrypt = require('bcrypt-nodejs');
|
||||||
var crypto = require('crypto');
|
var crypto = require('crypto');
|
||||||
var mongoose = require('mongoose');
|
var mongoose = require('mongoose');
|
||||||
|
require('mongoose-long')(mongoose);
|
||||||
|
|
||||||
|
var Long = mongoose.Types.Long;
|
||||||
var userSchema = new mongoose.Schema({
|
var userSchema = new mongoose.Schema({
|
||||||
email: {
|
email: {
|
||||||
type: String,
|
type: String,
|
||||||
@ -21,7 +23,7 @@ var userSchema = new mongoose.Schema({
|
|||||||
type: Number,
|
type: Number,
|
||||||
default: 0
|
default: 0
|
||||||
},
|
},
|
||||||
progressTimestamps: { type: Array, default: [Date] },
|
progressTimestamps: [],
|
||||||
challengesCompleted: { type: Array, default: [] },
|
challengesCompleted: { type: Array, default: [] },
|
||||||
pointsNeedMigration: { type: Boolean, default: true },
|
pointsNeedMigration: { type: Boolean, default: true },
|
||||||
challengesHash: {
|
challengesHash: {
|
||||||
@ -332,9 +334,30 @@ var userSchema = new mongoose.Schema({
|
|||||||
resetPasswordToken: String,
|
resetPasswordToken: String,
|
||||||
resetPasswordExpires: Date,
|
resetPasswordExpires: Date,
|
||||||
uncompletedBonfires: Array,
|
uncompletedBonfires: Array,
|
||||||
completedBonfires: Array,
|
completedBonfires: [
|
||||||
|
{
|
||||||
|
_id: String,
|
||||||
|
completedWith: String,
|
||||||
|
completedDate: Long,
|
||||||
|
solution: String
|
||||||
|
}
|
||||||
|
],
|
||||||
uncompletedCoursewares: Array,
|
uncompletedCoursewares: Array,
|
||||||
completedCoursewares: Array
|
completedCoursewares: [
|
||||||
|
{
|
||||||
|
completedDate: Long,
|
||||||
|
_id: String,
|
||||||
|
name: String
|
||||||
|
}
|
||||||
|
],
|
||||||
|
currentStreak: {
|
||||||
|
type: Number,
|
||||||
|
default: 0
|
||||||
|
},
|
||||||
|
longestStreak: {
|
||||||
|
type: Number,
|
||||||
|
default: 0
|
||||||
|
}
|
||||||
});
|
});
|
||||||
|
|
||||||
/**
|
/**
|
||||||
|
Reference in New Issue
Block a user