This commit is contained in:
Nathan Leniz
2015-03-28 17:38:11 +09:00
parent 338d87424c
commit 0a07c1e397
4 changed files with 721 additions and 692 deletions

View File

@ -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);
} }
}); });
} }

View File

@ -26,7 +26,7 @@ Array.zip = function(left, right, combinerFunction) {
results = []; results = [];
for (counter = 0; counter < Math.min(left.length, right.length); counter++) { for (counter = 0; counter < Math.min(left.length, right.length); counter++) {
results.push(combinerFunction(left[counter],right[counter])); results.push(combinerFunction(left[counter], right[counter]));
} }
return results; return results;
@ -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;

View File

@ -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', {

View File

@ -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
}
}); });
/** /**