forked from arcanis/secretsanta
-
Notifications
You must be signed in to change notification settings - Fork 0
/
Copy pathSecretSanta.js
111 lines (63 loc) · 3.04 KB
/
SecretSanta.js
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
var SecretSanta = function () {
this.names = [];
this.enforced = Object.create( null );
this.blacklists = Object.create( null );
};
SecretSanta.prototype.add = function ( name ) {
if ( this.names.indexOf( name ) !== -1 )
throw new Error( 'Cannot redefine ' + name );
this.names.push( name );
var subapi = { };
subapi.enforce = function ( other ) {
this.enforced[ name ] = other;
return subapi;
}.bind( this );
subapi.blacklist = function ( other ) {
if ( ! Object.prototype.hasOwnProperty.call( this.blacklists, name ) )
this.blacklists[ name ] = [];
if ( this.blacklists[ name ].indexOf( other ) === -1 )
this.blacklists[ name ].push( other );
return subapi;
}.bind( this );
return subapi;
};
SecretSanta.prototype.generate = function () {
var pairings = Object.create( null );
var candidatePairings = Object.create( null );
this.names.forEach( function ( name ) {
if ( Object.prototype.hasOwnProperty.call( this.enforced, name ) ) {
var enforced = this.enforced[ name ];
if ( this.names.indexOf( enforced ) === -1 )
throw new Error( name + ' is paired with ' + enforced + ', which hasn\'t been declared as a possible pairing' );
Object.keys( pairings ).forEach( function ( name ) {
if ( pairings[ name ] === enforced ) {
throw new Error( 'Per your rules, multiple persons are paired with ' + enforced );
}
} );
candidatePairings[ name ] = [ this.enforced[ name ] ];
} else {
var candidates = _.difference( this.names, [ name ] );
if ( Object.prototype.hasOwnProperty.call( this.blacklists, name ) )
candidates = _.difference( candidates, this.blacklists[ name ] );
candidatePairings[ name ] = candidates;
}
}, this );
var findNextGifter = function () {
var names = Object.keys( candidatePairings );
var minCandidateCount = _.min( names.map( function ( name ) { return candidatePairings[ name ].length; } ) );
var potentialGifters = names.filter( function ( name ) { return candidatePairings[ name ].length === minCandidateCount; } );
return _.sample( potentialGifters );
};
while ( Object.keys( candidatePairings ).length > 0 ) {
var name = findNextGifter();
if ( candidatePairings[ name ].length === 0 )
throw new Error('We haven\'t been able to find a match for ' + name + '! Press "Generate" to try again and, if it still doesn\'t work, try removing some exclusions from your rules. Sorry for the inconvenience!');
var pairing = _.sample( candidatePairings[ name ] );
delete candidatePairings[ name ];
Object.keys( candidatePairings ).forEach( function ( name ) {
candidatePairings[ name ] = _.without( candidatePairings[ name ], pairing );
} );
pairings[ name ] = pairing;
}
return pairings;
};