Skip to content

Commit dce9c95

Browse files
francois-rozetrickstaa
authored andcommitted
Ranking System v2 (anuraghazra#1186)
* Revise rank calculation * Replace contributions by commits * Lower average stats and S+ threshold * Fix calculateRank.test.js Missing key in dictionary constructor Co-authored-by: Rick Staa <[email protected]> * refactor: run prettier * feat: change star weight to 0.75 * Separate PRs and issues * Tweak weights * Add count_private back * fix: enable 'count_private' again * test: fix tests * refactor: improve code formatting * Higher targets --------- Co-authored-by: Rick Staa <[email protected]>
1 parent e1a4b4d commit dce9c95

7 files changed

+192
-243
lines changed

docs/readme_fr.md

+1-13
Original file line numberDiff line numberDiff line change
@@ -90,18 +90,6 @@ Pour masquer des statistiques spécifiques, vous pouvez passer un paramètre de
9090
![Les Stats GitHub de Anurag](https://github-readme-stats.vercel.app/api?username=anuraghazra&hide=contribs,prs)
9191
```
9292

93-
### Ajouter le compte des contributions privées au compte des commits totaux
94-
95-
Vous pouvez ajouter le compte de toutes vos contributions privées au compte total des engagements en utilisant le paramètre de requête `?count_private=true`.
96-
97-
_Note: Si vous déployez vous-même ce projet, les contributions privées seront comptées par défaut ; sinon, vous devez choisir de partager les comptes de vos contributions privées._
98-
99-
> Options: `&count_private=true`
100-
101-
```md
102-
![Les Stats GitHub de Anurag](https://github-readme-stats.vercel.app/api?username=anuraghazra&count_private=true)
103-
```
104-
10593
### Afficher les icônes
10694

10795
Pour activer les icônes, vous pouvez passer `show_icons=true` dans le paramètre de requête, comme ceci :
@@ -160,7 +148,7 @@ Vous pouvez fournir plusieurs valeurs (suivie d'une virgule) dans l'option bg_co
160148
- `hide_rank` - Masquer le rang _(boolean)_
161149
- `show_icons` - Afficher les icônes _(boolean)_
162150
- `include_all_commits` - Compter le total de commits au lieu de ne compter que les commits de l'année en cours _(boolean)_
163-
- `count_private` - Compter les commits privés _(boolean)_
151+
- `count_private` - Compter les contributions privées _(boolean)_
164152
- `line_height` - Fixer la hauteur de la ligne entre les textes _(number)_
165153

166154
#### Repo Card Exclusive Options:

readme.md

+1-14
Original file line numberDiff line numberDiff line change
@@ -120,19 +120,6 @@ You can pass a query parameter `&hide=` to hide any specific stats with comma-se
120120
![Anurag's GitHub stats](https://github-readme-stats.vercel.app/api?username=anuraghazra&hide=contribs,prs)
121121
```
122122

123-
### Adding private contributions count to total commits count
124-
125-
You can add the count of all your private contributions to the total commits count by using the query parameter `&count_private=true`.
126-
127-
> **Note**
128-
> If you are deploying this project yourself, the private contributions will be counted by default. If you are using the public Vercel instance, you need to choose to [share your private contributions](https://docs.github.com/en/account-and-profile/setting-up-and-managing-your-github-profile/managing-contribution-settings-on-your-profile/showing-your-private-contributions-and-achievements-on-your-profile).
129-
130-
> Options: `&count_private=true`
131-
132-
```md
133-
![Anurag's GitHub stats](https://github-readme-stats.vercel.app/api?username=anuraghazra&count_private=true)
134-
```
135-
136123
### Showing icons
137124

138125
To enable icons, you can pass `&show_icons=true` in the query param, like so:
@@ -283,7 +270,7 @@ You can provide multiple comma-separated values in the bg_color option to render
283270
- `rank_icon` - Shows alternative rank icon (i.e. `github` or `default`). Default: `default`.
284271
- `show_icons` - _(boolean)_. Default: `false`.
285272
- `include_all_commits` - Count total commits instead of just the current year commits _(boolean)_. Default: `false`.
286-
- `count_private` - Count private commits _(boolean)_. Default: `false`.
273+
- `count_private` - Count private contributions _(boolean)_. Default: `false`.
287274
- `line_height` - Sets the line height between text _(number)_. Default: `25`.
288275
- `exclude_repo` - Exclude stars from specified repositories _(Comma-separated values)_. Default: `[] (blank array)`.
289276
- `custom_title` - Sets a custom title for the card. Default: `<username> GitHub Stats`.

src/calculateRank.js

+51-84
Original file line numberDiff line numberDiff line change
@@ -1,105 +1,72 @@
1-
/**
2-
* Calculates the probability of x taking on x or a value less than x in a normal distribution
3-
* with mean and standard deviation.
4-
*
5-
* @see https://stackoverflow.com/a/5263759/10629172
6-
*
7-
* @param {string} mean The mean of the normal distribution.
8-
* @param {number} sigma The standard deviation of the normal distribution.
9-
* @param {number} to The value to calculate the probability for.
10-
* @returns {number} Probability.
11-
*/
12-
const normalcdf = (mean, sigma, to) => {
13-
var z = (to - mean) / Math.sqrt(2 * sigma * sigma);
14-
var t = 1 / (1 + 0.3275911 * Math.abs(z));
15-
var a1 = 0.254829592;
16-
var a2 = -0.284496736;
17-
var a3 = 1.421413741;
18-
var a4 = -1.453152027;
19-
var a5 = 1.061405429;
20-
var erf =
21-
1 - ((((a5 * t + a4) * t + a3) * t + a2) * t + a1) * t * Math.exp(-z * z);
22-
var sign = 1;
23-
if (z < 0) {
24-
sign = -1;
25-
}
26-
return (1 / 2) * (1 + sign * erf);
27-
};
1+
function expsf(x, lambda = 1) {
2+
return 2 ** (-lambda * x);
3+
}
284

295
/**
306
* Calculates the users rank.
317
*
328
* @param {object} params Parameters on which the user's rank depends.
33-
* @param {number} params.totalRepos Total number of repos.
34-
* @param {number} params.totalCommits Total number of commits.
35-
* @param {number} params.contributions The number of contributions.
36-
* @param {number} params.followers The number of followers.
9+
* @param {boolean} params.all_commits Whether `include_all_commits` was used.
10+
* @param {number} params.commits Number of commits.
3711
* @param {number} params.prs The number of pull requests.
3812
* @param {number} params.issues The number of issues.
39-
* @param {number} params.stargazers The number of stars.
13+
* @param {number} params.repos Total number of repos.
14+
* @param {number} params.stars The number of stars.
15+
* @param {number} params.followers The number of followers.
4016
* @returns {{level: string, score: number}}} The users rank.
4117
*/
42-
const calculateRank = ({
43-
totalRepos,
44-
totalCommits,
45-
contributions,
46-
followers,
18+
function calculateRank({
19+
all_commits,
20+
commits,
4721
prs,
4822
issues,
49-
stargazers,
50-
}) => {
51-
const COMMITS_OFFSET = 1.65;
52-
const CONTRIBS_OFFSET = 1.65;
53-
const ISSUES_OFFSET = 1;
54-
const STARS_OFFSET = 0.75;
55-
const PRS_OFFSET = 0.5;
56-
const FOLLOWERS_OFFSET = 0.45;
57-
const REPO_OFFSET = 1;
58-
59-
const ALL_OFFSETS =
60-
CONTRIBS_OFFSET +
61-
ISSUES_OFFSET +
62-
STARS_OFFSET +
63-
PRS_OFFSET +
64-
FOLLOWERS_OFFSET +
65-
REPO_OFFSET;
66-
67-
const RANK_S_VALUE = 1;
68-
const RANK_DOUBLE_A_VALUE = 25;
69-
const RANK_A2_VALUE = 45;
70-
const RANK_A3_VALUE = 60;
71-
const RANK_B_VALUE = 100;
23+
repos, // unused
24+
stars,
25+
followers,
26+
}) {
27+
const COMMITS_MEAN = all_commits ? 1000 : 250,
28+
COMMITS_WEIGHT = 2;
29+
const PRS_MEAN = 50,
30+
PRS_WEIGHT = 3;
31+
const ISSUES_MEAN = 25,
32+
ISSUES_WEIGHT = 1;
33+
const STARS_MEAN = 250,
34+
STARS_WEIGHT = 4;
35+
const FOLLOWERS_MEAN = 25,
36+
FOLLOWERS_WEIGHT = 1;
7237

73-
const TOTAL_VALUES =
74-
RANK_S_VALUE +
75-
RANK_DOUBLE_A_VALUE +
76-
RANK_A2_VALUE +
77-
RANK_A3_VALUE +
78-
RANK_B_VALUE;
38+
const TOTAL_WEIGHT =
39+
COMMITS_WEIGHT +
40+
PRS_WEIGHT +
41+
ISSUES_WEIGHT +
42+
STARS_WEIGHT +
43+
FOLLOWERS_WEIGHT;
7944

80-
// prettier-ignore
81-
const score = (
82-
totalCommits * COMMITS_OFFSET +
83-
contributions * CONTRIBS_OFFSET +
84-
issues * ISSUES_OFFSET +
85-
stargazers * STARS_OFFSET +
86-
prs * PRS_OFFSET +
87-
followers * FOLLOWERS_OFFSET +
88-
totalRepos * REPO_OFFSET
89-
) / 100;
45+
const rank =
46+
(COMMITS_WEIGHT * expsf(commits, 1 / COMMITS_MEAN) +
47+
PRS_WEIGHT * expsf(prs, 1 / PRS_MEAN) +
48+
ISSUES_WEIGHT * expsf(issues, 1 / ISSUES_MEAN) +
49+
STARS_WEIGHT * expsf(stars, 1 / STARS_MEAN) +
50+
FOLLOWERS_WEIGHT * expsf(followers, 1 / FOLLOWERS_MEAN)) /
51+
TOTAL_WEIGHT;
9052

91-
const normalizedScore = normalcdf(score, TOTAL_VALUES, ALL_OFFSETS) * 100;
53+
const RANK_S_PLUS = 0.025;
54+
const RANK_S = 0.1;
55+
const RANK_A_PLUS = 0.25;
56+
const RANK_A = 0.5;
57+
const RANK_B_PLUS = 0.75;
9258

9359
const level = (() => {
94-
if (normalizedScore < RANK_S_VALUE) return "S+";
95-
if (normalizedScore < RANK_DOUBLE_A_VALUE) return "S";
96-
if (normalizedScore < RANK_A2_VALUE) return "A++";
97-
if (normalizedScore < RANK_A3_VALUE) return "A+";
98-
return "B+";
60+
if (rank <= RANK_S_PLUS) return "S+";
61+
if (rank <= RANK_S) return "S";
62+
if (rank <= RANK_A_PLUS) return "A+";
63+
if (rank <= RANK_A) return "A";
64+
if (rank <= RANK_B_PLUS) return "B+";
65+
return "B";
9966
})();
10067

101-
return { level, score: normalizedScore };
102-
};
68+
return { level, score: rank * 100 };
69+
}
10370

10471
export { calculateRank };
10572
export default calculateRank;

src/fetchers/stats-fetcher.js

+15-24
Original file line numberDiff line numberDiff line change
@@ -192,7 +192,7 @@ const fetchStats = async (
192192
totalIssues: 0,
193193
totalStars: 0,
194194
contributedTo: 0,
195-
rank: { level: "C", score: 0 },
195+
rank: { level: "B", score: 0 },
196196
};
197197

198198
let res = await statsFetcher(username);
@@ -220,53 +220,44 @@ const fetchStats = async (
220220

221221
const user = res.data.data.user;
222222

223-
// populate repoToHide map for quick lookup
224-
// while filtering out
225-
let repoToHide = {};
226-
if (exclude_repo) {
227-
exclude_repo.forEach((repoName) => {
228-
repoToHide[repoName] = true;
229-
});
230-
}
231-
232223
stats.name = user.name || user.login;
233-
stats.totalIssues = user.openIssues.totalCount + user.closedIssues.totalCount;
234-
235-
// normal commits
236-
stats.totalCommits = user.contributionsCollection.totalCommitContributions;
237224

238-
// if include_all_commits then just get that,
239-
// since totalCommitsFetcher already sends totalCommits no need to +=
225+
// if include_all_commits, fetch all commits using the REST API.
240226
if (include_all_commits) {
241227
stats.totalCommits = await totalCommitsFetcher(username);
228+
} else {
229+
stats.totalCommits = user.contributionsCollection.totalCommitContributions;
242230
}
243231

244-
// if count_private then add private commits to totalCommits so far.
232+
// if count_private, add private contributions to totalCommits.
245233
if (count_private) {
246234
stats.totalCommits +=
247235
user.contributionsCollection.restrictedContributionsCount;
248236
}
249237

250238
stats.totalPRs = user.pullRequests.totalCount;
239+
stats.totalIssues = user.openIssues.totalCount + user.closedIssues.totalCount;
251240
stats.contributedTo = user.repositoriesContributedTo.totalCount;
252241

253-
// Retrieve stars while filtering out repositories to be hidden
242+
// Retrieve stars while filtering out repositories to be hidden.
243+
let repoToHide = new Set(exclude_repo);
244+
254245
stats.totalStars = user.repositories.nodes
255246
.filter((data) => {
256-
return !repoToHide[data.name];
247+
return !repoToHide.has(data.name);
257248
})
258249
.reduce((prev, curr) => {
259250
return prev + curr.stargazers.totalCount;
260251
}, 0);
261252

262253
stats.rank = calculateRank({
263-
totalCommits: stats.totalCommits,
264-
totalRepos: user.repositories.totalCount,
265-
followers: user.followers.totalCount,
266-
contributions: stats.contributedTo,
267-
stargazers: stats.totalStars,
254+
all_commits: include_all_commits,
255+
commits: stats.totalCommits,
268256
prs: stats.totalPRs,
269257
issues: stats.totalIssues,
258+
repos: user.repositories.totalCount,
259+
stars: stats.totalStars,
260+
followers: user.followers.totalCount,
270261
});
271262

272263
return stats;

tests/api.test.js

+7-38
Original file line numberDiff line numberDiff line change
@@ -12,17 +12,18 @@ const stats = {
1212
totalCommits: 200,
1313
totalIssues: 300,
1414
totalPRs: 400,
15-
contributedTo: 500,
15+
contributedTo: 50,
1616
rank: null,
1717
};
18+
1819
stats.rank = calculateRank({
19-
totalCommits: stats.totalCommits,
20-
totalRepos: 1,
21-
followers: 0,
22-
contributions: stats.contributedTo,
23-
stargazers: stats.totalStars,
20+
all_commits: false,
21+
commits: stats.totalCommits,
2422
prs: stats.totalPRs,
2523
issues: stats.totalIssues,
24+
repos: 1,
25+
stars: stats.totalStars,
26+
followers: 0,
2627
});
2728

2829
const data_stats = {
@@ -229,38 +230,6 @@ describe("Test /api/", () => {
229230
}
230231
});
231232

232-
it("should add private contributions", async () => {
233-
const { req, res } = faker(
234-
{
235-
username: "anuraghazra",
236-
count_private: true,
237-
},
238-
data_stats,
239-
);
240-
241-
await api(req, res);
242-
243-
expect(res.setHeader).toBeCalledWith("Content-Type", "image/svg+xml");
244-
expect(res.send).toBeCalledWith(
245-
renderStatsCard(
246-
{
247-
...stats,
248-
totalCommits: stats.totalCommits + 100,
249-
rank: calculateRank({
250-
totalCommits: stats.totalCommits + 100,
251-
totalRepos: 1,
252-
followers: 0,
253-
contributions: stats.contributedTo,
254-
stargazers: stats.totalStars,
255-
prs: stats.totalPRs,
256-
issues: stats.totalIssues,
257-
}),
258-
},
259-
{},
260-
),
261-
);
262-
});
263-
264233
it("should allow changing ring_color", async () => {
265234
const { req, res } = faker(
266235
{

0 commit comments

Comments
 (0)