630 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
			
		
		
	
	
			630 lines
		
	
	
		
			19 KiB
		
	
	
	
		
			JavaScript
		
	
	
	
	
	
| /* global describe it expect afterAll  */
 | |
| import moment from 'moment-timezone';
 | |
| import sinon from 'sinon';
 | |
| 
 | |
| import {
 | |
|   prepUniqueDaysByHours,
 | |
|   calcCurrentStreak,
 | |
|   calcLongestStreak,
 | |
|   getUserById
 | |
| } from './user-stats';
 | |
| import { mockUserID, mockApp, mockUser } from '../boot_tests/fixtures';
 | |
| 
 | |
| // setting now to 2016-02-03T11:00:00 (PST)
 | |
| const clock = sinon.useFakeTimers(1454526000000);
 | |
| const PST = 'America/Los_Angeles';
 | |
| 
 | |
| describe('user stats', () => {
 | |
|   afterAll(() => {
 | |
|     clock.restore();
 | |
|   });
 | |
| 
 | |
|   describe('prepUniqueDaysByHours', () => {
 | |
|     it(
 | |
|       'should return correct epoch when all entries fall into ' +
 | |
|         'one day in UTC',
 | |
|       () => {
 | |
|         expect(
 | |
|           prepUniqueDaysByHours([
 | |
|             moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('8/3/2015 14:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('8/3/2015 20:00', 'M/D/YYYY H:mm').valueOf()
 | |
|           ])
 | |
|         ).toEqual([1438567200000]);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it('should return correct epoch when given two identical dates', () => {
 | |
|       expect(
 | |
|         prepUniqueDaysByHours([
 | |
|           moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|           moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf()
 | |
|         ])
 | |
|       ).toEqual([1438567200000]);
 | |
|     });
 | |
| 
 | |
|     it('should return 2 epochs when dates fall into two days in PST', () => {
 | |
|       expect(
 | |
|         prepUniqueDaysByHours(
 | |
|           [
 | |
|             // 8/2/2015 in America/Los_Angeles
 | |
|             moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('8/3/2015 14:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('8/3/2015 20:00', 'M/D/YYYY H:mm').valueOf()
 | |
|           ],
 | |
|           PST
 | |
|         )
 | |
|       ).toEqual([1438567200000, 1438610400000]);
 | |
|     });
 | |
| 
 | |
|     it('should return 3 epochs when dates fall into three days', () => {
 | |
|       expect(
 | |
|         prepUniqueDaysByHours([
 | |
|           moment.utc('8/1/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|           moment.utc('3/3/2015 14:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|           moment.utc('9/30/2014 20:00', 'M/D/YYYY H:mm').valueOf()
 | |
|         ])
 | |
|       ).toEqual([1412107200000, 1425391200000, 1438394400000]);
 | |
|     });
 | |
| 
 | |
|     it(
 | |
|       'should return same but sorted array if all input dates are ' +
 | |
|         'start of day',
 | |
|       () => {
 | |
|         expect(
 | |
|           prepUniqueDaysByHours([1438387200000, 1425340800000, 1412035200000])
 | |
|         ).toEqual([1412035200000, 1425340800000, 1438387200000]);
 | |
|       }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   describe('calcCurrentStreak', function() {
 | |
|     it('should return 1 day when today one challenge was completed', () => {
 | |
|       expect(
 | |
|         calcCurrentStreak(
 | |
|           prepUniqueDaysByHours([
 | |
|             moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|           ])
 | |
|         )
 | |
|       ).toEqual(1);
 | |
|     });
 | |
| 
 | |
|     it(
 | |
|       'should return 1 day when today more than one challenge ' +
 | |
|         'was completed',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(1);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it('should return 0 days when today 0 challenges were completed', () => {
 | |
|       expect(
 | |
|         calcCurrentStreak(
 | |
|           prepUniqueDaysByHours([
 | |
|             moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf()
 | |
|           ])
 | |
|         )
 | |
|       ).toEqual(0);
 | |
|     });
 | |
| 
 | |
|     it(
 | |
|       'should return 2 days when today and yesterday challenges were ' +
 | |
|         'completed',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(1, 'days')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(2);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 3 when today and for two days before user was ' + 'active',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 3:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/13/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/14/2015 5:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc(moment.utc().subtract(2, 'days')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'days')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(3);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 1 when there is a 1.5 day long break and ' +
 | |
|         'dates fall into two days separated by third',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(47, 'hours')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(11, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(1);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 2 when the break is more than 1.5 days ' +
 | |
|         'but dates fall into two consecutive days',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(40, 'hours')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(2);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return correct count in default timezone UTC ' +
 | |
|         'given `undefined` timezone',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(1);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 2 days when today and yesterday ' +
 | |
|         'challenges were completed given PST',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours(
 | |
|               [
 | |
|                 moment.utc(moment.utc().subtract(1, 'days')).valueOf(),
 | |
|                 moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|               ],
 | |
|               PST
 | |
|             ),
 | |
|             PST
 | |
|           )
 | |
|         ).toEqual(2);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 17 when there is no break in given timezone ' +
 | |
|         '(but would be the break if in UTC)',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours(
 | |
|               [
 | |
|                 1453174506164,
 | |
|                 1453175436725,
 | |
|                 1453252466853,
 | |
|                 1453294968225,
 | |
|                 1453383782844,
 | |
|                 1453431903117,
 | |
|                 1453471373080,
 | |
|                 1453594733026,
 | |
|                 1453645014058,
 | |
|                 1453746762747,
 | |
|                 1453747659197,
 | |
|                 1453748029416,
 | |
|                 1453818029213,
 | |
|                 1453951796007,
 | |
|                 1453988570615,
 | |
|                 1454069704441,
 | |
|                 1454203673979,
 | |
|                 1454294055498,
 | |
|                 1454333545125,
 | |
|                 1454415163903,
 | |
|                 1454519128123,
 | |
|                 moment.tz(PST).valueOf()
 | |
|               ],
 | |
|               PST
 | |
|             ),
 | |
|             PST
 | |
|           )
 | |
|         ).toEqual(17);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 4 when there is a break in UTC ' +
 | |
|         '(but would be no break in PST)',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcCurrentStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               1453174506164,
 | |
|               1453175436725,
 | |
|               1453252466853,
 | |
|               1453294968225,
 | |
|               1453383782844,
 | |
|               1453431903117,
 | |
|               1453471373080,
 | |
|               1453594733026,
 | |
|               1453645014058,
 | |
|               1453746762747,
 | |
|               1453747659197,
 | |
|               1453748029416,
 | |
|               1453818029213,
 | |
|               1453951796007,
 | |
|               1453988570615,
 | |
|               1454069704441,
 | |
|               1454203673979,
 | |
|               1454294055498,
 | |
|               1454333545125,
 | |
|               1454415163903,
 | |
|               1454519128123,
 | |
|               moment.utc().valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(4);
 | |
|       }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   describe('calcLongestStreak', function() {
 | |
|     it(
 | |
|       'should return 1 when there is the only one one-day-long ' +
 | |
|         'streak available',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('9/12/2015 4:00', 'M/D/YYYY H:mm').valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(1);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 4 when there is the only one ' +
 | |
|         'more-than-one-days-long streak available',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/13/2015 3:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/14/2015 1:00', 'M/D/YYYY H:mm').valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(4);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 1 when there is only one one-day-long streak ' +
 | |
|         'and it is today',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(1);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it('should return 2 when yesterday and today makes longest streak', () => {
 | |
|       expect(
 | |
|         calcLongestStreak(
 | |
|           prepUniqueDaysByHours([
 | |
|             moment.utc(moment.utc().subtract(1, 'days')).valueOf(),
 | |
|             moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|           ])
 | |
|         )
 | |
|       ).toEqual(2);
 | |
|     });
 | |
| 
 | |
|     it('should return 4 when there is a month long break', () => {
 | |
|       expect(
 | |
|         calcLongestStreak(
 | |
|           prepUniqueDaysByHours([
 | |
|             moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('9/12/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('10/4/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('10/5/2015 5:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('10/6/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('10/7/2015 5:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|             moment.utc('11/3/2015 2:00', 'M/D/YYYY H:mm').valueOf()
 | |
|           ])
 | |
|         )
 | |
|       ).toEqual(4);
 | |
|     });
 | |
| 
 | |
|     it(
 | |
|       'should return 2 when there is a more than 1.5 days ' +
 | |
|         'long break of (36 hours)',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 15:30', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment
 | |
|                 .utc(
 | |
|                   moment
 | |
|                     .utc('9/12/2015 15:30', 'M/D/YYYY H:mm')
 | |
|                     .add(37, 'hours')
 | |
|                 )
 | |
|                 .valueOf(),
 | |
|               moment.utc('9/14/2015 22:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/15/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('10/3/2015 2:00', 'M/D/YYYY H:mm').valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(2);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 4 when the longest streak consist of ' +
 | |
|         'several same day timestamps',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 3:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/13/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/14/2015 5:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc(moment.utc().subtract(2, 'days')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'days')).valueOf(),
 | |
|               moment.utc().valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(4);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 4 when there are several longest streaks ' +
 | |
|         '(same length)',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('8/3/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/13/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/14/2015 5:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('10/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('10/12/2015 1:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('10/13/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('10/14/2015 5:00', 'M/D/YYYY H:mm').valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(4);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return correct longest streak when there is a very ' +
 | |
|         'long period',
 | |
|       () => {
 | |
|         let cals = [];
 | |
|         const n = 100;
 | |
|         for (let i = 0; i < n; i++) {
 | |
|           cals.push(moment.utc(moment.utc().subtract(i, 'days')).valueOf());
 | |
|         }
 | |
| 
 | |
|         expect(calcLongestStreak(prepUniqueDaysByHours(cals))).toEqual(n);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return correct longest streak in default timezone ' +
 | |
|         'UTC given `undefined` timezone',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc(moment.utc().subtract(1, 'days')).valueOf(),
 | |
|               moment.utc(moment.utc().subtract(1, 'hours')).valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(2);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 4 when there is the only one more-than-one-days-long ' +
 | |
|         'streak available given PST',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               moment.utc('9/11/2015 4:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/12/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/13/2015 3:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|               moment.utc('9/14/2015 1:00', 'M/D/YYYY H:mm').valueOf()
 | |
|             ]),
 | |
|             PST
 | |
|           )
 | |
|         ).toEqual(4);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 3 when longest streak is 3 in PST ' +
 | |
|         '(but would be different in default timezone UTC)',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours(
 | |
|               [
 | |
|                 moment.utc('9/11/2015 23:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|                 moment.utc('9/12/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|                 moment.utc('9/13/2015 2:00', 'M/D/YYYY H:mm').valueOf(),
 | |
|                 moment.utc('9/14/2015 6:00', 'M/D/YYYY H:mm').valueOf()
 | |
|               ],
 | |
|               PST
 | |
|             ),
 | |
|             PST
 | |
|           )
 | |
|         ).toEqual(3);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return 17 when there is no break in PST ' +
 | |
|         '(but would be break in UTC) and it is current',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours(
 | |
|               [
 | |
|                 1453174506164,
 | |
|                 1453175436725,
 | |
|                 1453252466853,
 | |
|                 1453294968225,
 | |
|                 1453383782844,
 | |
|                 1453431903117,
 | |
|                 1453471373080,
 | |
|                 1453594733026,
 | |
|                 1453645014058,
 | |
|                 1453746762747,
 | |
|                 1453747659197,
 | |
|                 1453748029416,
 | |
|                 1453818029213,
 | |
|                 1453951796007,
 | |
|                 1453988570615,
 | |
|                 1454069704441,
 | |
|                 1454203673979,
 | |
|                 1454294055498,
 | |
|                 1454333545125,
 | |
|                 1454415163903,
 | |
|                 1454519128123,
 | |
|                 moment.tz(PST).valueOf()
 | |
|               ],
 | |
|               PST
 | |
|             ),
 | |
|             PST
 | |
|           )
 | |
|         ).toEqual(17);
 | |
|       }
 | |
|     );
 | |
| 
 | |
|     it(
 | |
|       'should return a streak of 4 when there is a break in UTC ' +
 | |
|         '(but no break in PST)',
 | |
|       () => {
 | |
|         expect(
 | |
|           calcLongestStreak(
 | |
|             prepUniqueDaysByHours([
 | |
|               1453174506164,
 | |
|               1453175436725,
 | |
|               1453252466853,
 | |
|               1453294968225,
 | |
|               1453383782844,
 | |
|               1453431903117,
 | |
|               1453471373080,
 | |
|               1453594733026,
 | |
|               1453645014058,
 | |
|               1453746762747,
 | |
|               1453747659197,
 | |
|               1453748029416,
 | |
|               1453818029213,
 | |
|               1453951796007,
 | |
|               1453988570615,
 | |
|               1454069704441,
 | |
|               1454203673979,
 | |
|               1454294055498,
 | |
|               1454333545125,
 | |
|               1454415163903,
 | |
|               1454519128123,
 | |
|               moment.utc().valueOf()
 | |
|             ])
 | |
|           )
 | |
|         ).toEqual(4);
 | |
|       }
 | |
|     );
 | |
|   });
 | |
| 
 | |
|   describe('getUserById', () => {
 | |
|     const stubUser = {
 | |
|       findById(id, cb) {
 | |
|         cb(null, { id: 123 });
 | |
|       }
 | |
|     };
 | |
|     it('returns a promise', () => {
 | |
|       expect.assertions(3);
 | |
|       expect(typeof getUserById('123', stubUser).then).toEqual('function');
 | |
|       expect(typeof getUserById('123', stubUser).catch).toEqual('function');
 | |
|       expect(typeof getUserById('123', stubUser).finally).toEqual('function');
 | |
|     });
 | |
| 
 | |
|     it('resolves a user for a given id', done => {
 | |
|       expect.assertions(7);
 | |
|       return getUserById(mockUserID, mockApp.models.User)
 | |
|         .then(user => {
 | |
|           expect(user).toEqual(mockUser);
 | |
| 
 | |
|           // fields added or removed by getUserById
 | |
|           expect(user).not.toHaveProperty('progressTimestamps');
 | |
|           expect(user).toHaveProperty('completedChallengeCount');
 | |
|           expect(user).toHaveProperty('completedProjectCount');
 | |
|           expect(user).toHaveProperty('completedCertCount');
 | |
|           expect(user).toHaveProperty('completedLegacyCertCount');
 | |
|           expect(user.completedChallenges).toEqual([]);
 | |
|         })
 | |
|         .then(done)
 | |
|         .catch(done);
 | |
|     });
 | |
| 
 | |
|     it('throws when no user is found', done => {
 | |
|       const noUserError = new Error('No user found');
 | |
|       const throwyUserModel = {
 | |
|         findById(_, cb) {
 | |
|           cb(noUserError);
 | |
|         }
 | |
|       };
 | |
|       expect(
 | |
|         getUserById('not-a-real-id', throwyUserModel).catch(error => {
 | |
|           expect(error).toEqual(noUserError);
 | |
|           done();
 | |
|         })
 | |
|       );
 | |
|     });
 | |
|   });
 | |
| });
 |