-
Notifications
You must be signed in to change notification settings - Fork 83
/
Copy pathladders.js
330 lines (290 loc) · 9.9 KB
/
ladders.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
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
161
162
163
164
165
166
167
168
169
170
171
172
173
174
175
176
177
178
179
180
181
182
183
184
185
186
187
188
189
190
191
192
193
194
195
196
197
198
199
200
201
202
203
204
205
206
207
208
209
210
211
212
213
214
215
216
217
218
219
220
221
222
223
224
225
226
227
228
229
230
231
232
233
234
235
236
237
238
239
240
241
242
243
244
245
246
247
248
249
250
251
252
253
254
255
256
257
258
259
260
261
262
263
264
265
266
267
268
269
270
271
272
273
274
275
276
277
278
279
280
281
282
283
284
285
286
287
288
289
290
291
292
293
294
295
296
297
298
299
300
301
302
303
304
305
306
307
308
309
310
311
312
313
314
315
316
317
318
319
320
321
322
323
324
325
326
327
328
329
330
/**
* Ladder library
* Pokemon Showdown - http://pokemonshowdown.com/
*
* This file handles ladders for all servers other than
* play.pokemonshowdown.com.
*
* Specifically, this is the file that handles calculating and keeping
* track of players' Elo ratings for all formats.
*
* Matchmaking is currently still implemented in rooms.js.
*
* @license MIT license
*/
'use strict';
const fs = require('fs');
let Ladders = module.exports = getLadder;
Ladders.get = Ladders;
// tells the client to ask the server for format information
Ladders.formatsListPrefix = '|,LL';
// ladderCaches = {formatid: ladder OR Promise(ladder)}
// Use Ladders(formatid).ladder to guarantee a Promise(ladder).
// ladder is basically a 2D array representing the corresponding ladder.tsv
// with userid in front
// ladder = [ladderRow]
// ladderRow = [userid, elo, username, w, l, t]
let ladderCaches = Ladders.ladderCaches = Object.create(null);
class Ladder {
constructor(formatid) {
this.formatid = toId(formatid);
this.loadedLadder = null;
this.ladder = this.load();
}
/**
* Internal function, returns a Promise for a ladder
*/
load() {
// ladderCaches[formatid]
if (this.formatid in ladderCaches) {
let cachedLadder = ladderCaches[this.formatid];
if (cachedLadder.then) {
return cachedLadder.then(ladder => {
this.loadedLadder = ladder;
return ladder;
});
} else {
return Promise.resolve(this.loadedLadder = cachedLadder);
}
}
return (ladderCaches[this.formatid] = new Promise((resolve, reject) => {
fs.readFile('config/ladders/' + this.formatid + '.tsv', (err, data) => {
if (err) {
this.loadedLadder = ladderCaches[this.formatid] = [];
// console.log('Ladders(' + this.formatid + ') err loading tsv: ' + JSON.stringify(this.loadedLadder));
resolve(this.loadedLadder);
return;
}
let ladder = [];
let dataLines = ('' + data).split('\n');
for (let i = 1; i < dataLines.length; i++) {
let line = dataLines[i].trim();
if (!line) continue;
let row = line.split('\t');
ladder.push([toId(row[1]), Number(row[0]), row[1], Number(row[2]), Number(row[3]), Number(row[4]), row[5]]);
}
this.loadedLadder = ladderCaches[this.formatid] = ladder;
// console.log('Ladders(' + this.formatid + ') loaded tsv: ' + JSON.stringify(this.loadedLadder));
resolve(this.loadedLadder);
});
}));
}
/**
* Saves the ladder in config/ladders/[formatid].tsv
*
* Called automatically by updateRating, so you don't need to manually
* call this.
*/
save() {
if (this.saving) return;
this.saving = true;
if (!this.loadedLadder) {
this.ladder.then(() => {
this.save();
});
return;
}
if (!this.loadedLadder.length) {
this.saving = false;
return;
}
let stream = fs.createWriteStream('config/ladders/' + this.formatid + '.tsv');
stream.write('Elo\tUsername\tW\tL\tT\tLast update\r\n');
for (let i = 0; i < this.loadedLadder.length; i++) {
let row = this.loadedLadder[i];
stream.write(row.slice(1).join('\t') + '\r\n');
}
stream.end();
this.saving = false;
}
/**
* Gets the index of a user in the ladder array.
*
* If createIfNeeded is true, the user will be created and added to
* the ladder array if it doesn't already exist.
*/
indexOfUser(username, createIfNeeded) {
let userid = toId(username);
for (let i = 0; i < this.loadedLadder.length; i++) {
if (this.loadedLadder[i][0] === userid) return i;
}
if (createIfNeeded) {
let index = this.loadedLadder.length;
this.loadedLadder.push([userid, 1000, username, 0, 0, 0]);
return index;
}
return -1;
}
/**
* Returns [formatid, html], where html is an the HTML source of a
* ladder toplist, to be displayed directly in the ladder tab of the
* client.
*/
getTop() {
let formatid = this.formatid;
let name = Tools.getFormat(formatid).name;
return this.ladder.then(ladder => {
let buf = '<h3>' + name + ' Top 100</h3>';
buf += '<table>';
buf += '<tr><th>' + ['', 'Username', '<abbr title="Elo rating">Elo</abbr>', 'W', 'L', 'T'].join('</th><th>') + '</th></tr>';
for (let i = 0; i < ladder.length; i++) {
let row = ladder[i];
buf += '<tr><td>' + [
i + 1, row[2], '<strong>' + Math.round(row[1]) + '</strong>', row[3], row[4], row[5],
].join('</td><td>') + '</td></tr>';
}
return [formatid, buf];
});
}
/**
* Returns a Promise for the Elo rating of a user
*/
getRating(userid) {
let formatid = this.formatid;
let user = Users.getExact(userid);
if (user && user.mmrCache[formatid]) {
return Promise.resolve(user.mmrCache[formatid]);
}
return this.ladder.then(() => {
if (user.userid !== userid) return;
let index = this.indexOfUser(userid);
if (index < 0) return (user.mmrCache[formatid] = 1000);
return (user.mmrCache[formatid] = this.loadedLadder[index][1]);
});
}
/**
* Internal method. Update the Elo rating of a user.
*/
updateRow(row, score, foeElo) {
let elo = row[1];
// The K factor determines how much your Elo changes when you win or
// lose games. Larger K means more change.
// In the "original" Elo, K is constant, but it's common for K to
// get smaller as your rating goes up
let K = 50;
// dynamic K-scaling (optional)
if (elo < 1200) {
if (score < 0.5) {
K = 10 + (elo - 1000) * 40 / 200;
} else if (score > 0.5) {
K = 90 - (elo - 1000) * 40 / 200;
}
} else if (elo > 1350) {
K = 40;
} else if (elo > 1600) {
K = 32;
}
// main Elo formula
let E = 1 / (1 + Math.pow(10, (foeElo - elo) / 400));
elo += K * (score - E);
if (elo < 1000) elo = 1000;
row[1] = elo;
if (score > 0.6) {
row[3]++; // win
} else if (score < 0.4) {
row[4]++; // loss
} else {
row[5]++; // tie
}
row[6] = '' + new Date();
}
/**
* Update the Elo rating for two players after a battle, and display
* the results in the passed room.
*/
updateRating(p1name, p2name, p1score, room) {
let formatid = this.formatid;
this.ladder.then(() => {
let p1newElo, p2newElo;
try {
let p1index = this.indexOfUser(p1name, true);
let p1elo = this.loadedLadder[p1index][1];
let p2index = this.indexOfUser(p2name, true);
let p2elo = this.loadedLadder[p2index][1];
this.updateRow(this.loadedLadder[p1index], p1score, p2elo);
this.updateRow(this.loadedLadder[p2index], 1 - p1score, p1elo);
p1newElo = this.loadedLadder[p1index][1];
p2newElo = this.loadedLadder[p2index][1];
// console.log('L: ' + this.loadedLadder.map(r => ''+Math.round(r[1])+' '+r[2]).join('\n'));
// move p1 to its new location
let newIndex = p1index;
while (newIndex > 0 && this.loadedLadder[newIndex - 1][1] <= p1newElo) newIndex--;
while (newIndex === p1index || (this.loadedLadder[newIndex] && this.loadedLadder[newIndex][1] > p1newElo)) newIndex++;
// console.log('ni='+newIndex+', p1i='+p1index);
if (newIndex !== p1index && newIndex !== p1index + 1) {
let row = this.loadedLadder.splice(p1index, 1)[0];
// adjust for removed row
if (newIndex > p1index) newIndex--;
if (p2index > p1index) p2index--;
this.loadedLadder.splice(newIndex, 0, row);
// adjust for inserted row
if (p2index >= newIndex) p2index++;
}
// move p2
newIndex = p2index;
while (newIndex > 0 && this.loadedLadder[newIndex - 1][1] <= p2newElo) newIndex--;
while (newIndex === p2index || (this.loadedLadder[newIndex] && this.loadedLadder[newIndex][1] > p2newElo)) newIndex++;
// console.log('ni='+newIndex+', p2i='+p2index);
if (newIndex !== p2index && newIndex !== p2index + 1) {
let row = this.loadedLadder.splice(p2index, 1)[0];
// adjust for removed row
if (newIndex > p2index) newIndex--;
this.loadedLadder.splice(newIndex, 0, row);
}
let p1 = Users.getExact(p1name);
if (p1) p1.mmrCache[formatid] = +p1newElo;
let p2 = Users.getExact(p2name);
if (p2) p2.mmrCache[formatid] = +p2newElo;
this.save();
if (!room.battle) {
console.log('room expired before ladder update was received');
return;
}
let reasons = '' + (Math.round(p1newElo) - Math.round(p1elo)) + ' for ' + (p1score > 0.9 ? 'winning' : (p1score < 0.1 ? 'losing' : 'tying'));
if (reasons.charAt(0) !== '-') reasons = '+' + reasons;
room.addRaw(Tools.escapeHTML(p1name) + '\'s rating: ' + Math.round(p1elo) + ' → <strong>' + Math.round(p1newElo) + '</strong><br />(' + reasons + ')');
reasons = '' + (Math.round(p2newElo) - Math.round(p2elo)) + ' for ' + (p1score > 0.9 ? 'losing' : (p1score < 0.1 ? 'winning' : 'tying'));
if (reasons.charAt(0) !== '-') reasons = '+' + reasons;
room.addRaw(Tools.escapeHTML(p2name) + '\'s rating: ' + Math.round(p2elo) + ' → <strong>' + Math.round(p2newElo) + '</strong><br />(' + reasons + ')');
room.update();
} catch (e) {
if (!room.battle) return;
room.addRaw('There was an error calculating rating changes:');
room.add(e.stack);
room.update();
}
if (!Tools.getFormat(formatid).noLog) {
room.logBattle(p1score, p1newElo, p2newElo);
}
});
}
/**
* Returns a promise for a <tr> with all ratings for the current format.
*/
visualize(username) {
return this.ladder.then(() => {
let index = this.indexOfUser(username, false);
if (index < 0) return '';
let ratings = this.loadedLadder[index];
let output = '<tr><td>' + this.formatid + '</td><td><strong>' + Math.round(ratings[1]) + '</strong></td>';
return output + '<td>' + ratings[3] + '</td><td>' + ratings[4] + '</td><td>' + (ratings[3] + ratings[4]) + '</td></tr>';
});
}
/**
* Returns a Promise for an array of strings of <tr>s for ladder ratings of the user
*/
static visualizeAll(username) {
let ratings = [];
for (let i in Tools.data.Formats) {
if (Tools.data.Formats[i].searchShow) {
ratings.push(Ladders(i).visualize(username));
}
}
return Promise.all(ratings);
}
}
function getLadder(formatid) {
return new Ladder(formatid);
}
Ladders.Ladder = Ladder;
Ladders.visualizeAll = Ladder.visualizeAll;