Skip to content

Commit 4ae95c9

Browse files
committed
feat: CLI for contributors archiving
1 parent ba4f1b1 commit 4ae95c9

File tree

5 files changed

+108
-23
lines changed

5 files changed

+108
-23
lines changed

README.md

+42-3
Original file line numberDiff line numberDiff line change
@@ -1,15 +1,54 @@
11
# open-community-kit
22
Tools and stats for open-source communities
33

4+
# Installation
5+
6+
```
7+
npm install -g open-community-kit
8+
```
9+
10+
You can use npx as well if you just want to test a CLI command
11+
412
# Usage
513

614
## Create a leaderboard of github contributors for all the repos of a user/org
715
(Contributor with highest # of contributions at the top)
816

9-
Run `node contributors.js` from your terminal
17+
### Using CLI
18+
19+
Run `open-community-kit yourGithubOrgName` from your terminal
20+
21+
Note: You can also use the shorthand `ock` in place of `open-commmunity-kit` i.e.
22+
23+
```
24+
ock yourGitHubOrgName
25+
```
1026

1127
This will
12-
* Fetch data from Github APIs and prepare a leaderboard of all the contributors to (default "Git-Commit-Show" org) the user or org you mention in REPO_OWNER variable inside `contributors.js`
28+
* Fetch data from Github APIs and prepare a leaderboard of all the contributors to public repositories of your GitHub organization/user accout
1329
* Save the leaderboard in a csv file in the same folder
1430

15-
You will hit the API limits soon. To increase API limits, add GITHUB_PERSONAL_TOKEN in `contributors.js`
31+
You will hit the API limits soon. **To increase API limits**, add [`GITHUB_PERSONAL_TOKEN`](https://github.com/settings/tokens) as well in the arguments i.e.
32+
33+
```
34+
ock yourGitHubOrgName yourGitHubPersonalToken
35+
```
36+
37+
### Using code
38+
39+
```javascript
40+
const OCK = require('open-community-kit').OCK;
41+
OCK.contributors.github.archive('your_github_org_or_username',
42+
{ GITHUB_PERSONAL_TOKEN: 'your_gh_personal_token_optional'
43+
});
44+
```
45+
46+
## Settings for repeated usage
47+
48+
If you are going to use this command frequently, you might not want to set organization name and personal token again and again. Instead, you can set following environment variables and then you don't need to pass those variables as CLI arguments or function parameters
49+
50+
```
51+
# Set these variables in the environment to avoid repeatedly specifying these variables
52+
1. REPO_OWNER
53+
2. GITHUB_PERSONAL_TOKEN
54+
```

cli.js

+14
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,14 @@
1+
#!/usr/bin/env node
2+
3+
const OCK = require('./index').OCK;
4+
5+
let mainCommand = process.argv[1];
6+
console.log("Running "+mainCommand+"...");
7+
let REPO_OWNER = process.argv[2];
8+
let GITHUB_PERSONAL_TOKEN = process.argv[3];
9+
10+
let options = {};
11+
if(GITHUB_PERSONAL_TOKEN){
12+
options.GITHUB_PERSONAL_TOKEN = GITHUB_PERSONAL_TOKEN;
13+
}
14+
OCK.contributors.github.archive(REPO_OWNER, options);

contributors.js

+29-19
Original file line numberDiff line numberDiff line change
@@ -3,12 +3,14 @@
33
* @example To archive contributors leaderboard data in csv file, run `node contributors.js`
44
*/
55

6+
exports.archiveContributorsLeaderboard = archiveContributorsLeaderboard
7+
68
const https = require('https');
79

8-
// INPUTS
9-
// Mandatory: Repo owner that you want to analyze
10+
// Configurations (Optional)
11+
// Repo owner that you want to analyze
1012
const REPO_OWNER = process.env.REPO_OWNER;
11-
// Optional: Authentication using github token. When used, it will increase the API limits from 60 to 5000/hr
13+
// Authentication using github token. When used, it will increase the API limits from 60 to 5000/hr
1214
const GITHUB_PERSONAL_TOKEN = process.env.GITHUB_PERSONAL_TOKEN;
1315
// END OF INPUTS
1416

@@ -25,12 +27,16 @@ if(GITHUB_PERSONAL_TOKEN){
2527

2628
/**
2729
* Get all github repos of an owner(user/org)
28-
* @param {String} owner
29-
* @param {Number} pageNo
30+
* @param {string} owner The organization or user name on GitHub
31+
* @param {Object} options Additional options e.g. { pageNo: 1 }
3032
* @returns Promise<Array<Object> | String> JSON array of data on success, error on failure
3133
* @example getAllRepos('myorghandle').then((repos) => console.log(repos)).catch((err) => console.log(err))
3234
*/
33-
async function getAllRepos(owner, pageNo = 1) {
35+
async function getAllRepos(owner=REPO_OWNER, options) {
36+
let pageNo = (options && options.pageNo) ? options.pageNo : 1;
37+
if(options && options.GITHUB_PERSONAL_TOKEN){
38+
GITHUB_REQUEST_OPTIONS.headers["Authorization"] = "token "+options.GITHUB_PERSONAL_TOKEN;
39+
}
3440
return new Promise((resolve, reject) => {
3541
let url = `https://api.github.com/orgs/${owner}/repos?per_page=100&page=${pageNo}`;
3642
console.log(url);
@@ -49,7 +55,7 @@ async function getAllRepos(owner, pageNo = 1) {
4955
//It might have more data on the next page
5056
pageNo++;
5157
try {
52-
let dataFromNextPage = await getAllRepos(owner, pageNo);
58+
let dataFromNextPage = await getAllRepos(owner, { pageNo: pageNo } );
5359
dataJsonArray.push(...dataFromNextPage);
5460
} catch (err) {
5561
console.log("No more pagination needed")
@@ -66,8 +72,8 @@ async function getAllRepos(owner, pageNo = 1) {
6672

6773
/**
6874
* Get contributors for a Github repo
69-
* @param {*} fullRepoName e.g. myorghandle/myreponame
70-
* @param {*} pageNo
75+
* @param {string} fullRepoName e.g. myorghandle/myreponame
76+
* @param {number} pageNo
7177
* @returns Promise<Array<Object> | String>
7278
* @example getRepoContributors('myorghandle/myreponame').then((contributors) => console.log(contributors)).catch((err) => console.log(err))
7379
*/
@@ -107,10 +113,11 @@ async function getRepoContributors(fullRepoName, pageNo = 1) {
107113

108114
/**
109115
* Get all contributors across all the repos of an owner
110-
* @param {*} owner github user or org handle
116+
* @param {string} owner github user or org handle
117+
* @param {Object} options Additional options
111118
*/
112-
async function getAllContributors(owner) {
113-
let repos = await getAllRepos(owner);
119+
async function getAllContributors(owner=REPO_OWNER, options) {
120+
let repos = await getAllRepos(owner, options);
114121
if (!repos || repos.length < 1) {
115122
console.log("Error in getting repos for " + owner)
116123
throw ("Error in getting repos for " + owner)
@@ -149,7 +156,7 @@ async function getAllContributors(owner) {
149156

150157
/**
151158
* Adds up all the contributions by a contributor to different repos
152-
* @param {*} contributors
159+
* @param {Array} contributors
153160
*/
154161
function aggregateAllContributors(contributors) {
155162
return contributors.reduce(function (grouped, currentItem) {
@@ -185,6 +192,10 @@ function sortReposByContributionsCount(repoContributionMappingArray){
185192
})
186193
}
187194

195+
/**
196+
* Writes all contributors data to a file
197+
* @param {Array} contributors
198+
*/
188199
function writeContributorLeaderboardToFile(contributors) {
189200
const fs = require('fs');
190201
let ghContributorLeaderboard = contributors.map((contributor) => {
@@ -201,10 +212,11 @@ function writeContributorLeaderboardToFile(contributors) {
201212

202213
/**
203214
* Archives contributors leaderboard data sorted by contrbutions in a file
204-
* @param {*} owner
215+
* @param {string} owner The organization or user name on GitHub
216+
* @param {Object} options Additional options
205217
*/
206-
async function archiveContributorsLeaderboard(owner) {
207-
let contributors = await getAllContributors();
218+
async function archiveContributorsLeaderboard(owner=REPO_OWNER, options) {
219+
let contributors = await getAllContributors(owner, options);
208220
if (!contributors || contributors.length < 1) {
209221
console.log("Failed to get contributors for "+owner);
210222
return;
@@ -218,6 +230,4 @@ async function archiveContributorsLeaderboard(owner) {
218230
writeContributorLeaderboardToFile(contributors);
219231

220232
return ghHandles;
221-
}
222-
223-
archiveContributorsLeaderboard(REPO_OWNER)
233+
}

index.js

+18
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,18 @@
1+
const contributorsLib = require('./contributors');
2+
3+
/**
4+
* Bundling all APIs together
5+
* @param {*} options
6+
*/
7+
8+
const OCK = {
9+
contributors: {
10+
github: {
11+
archive: async function(owner, options){
12+
contributorsLib.archiveContributorsLeaderboard(owner, options)
13+
}
14+
}
15+
}
16+
}
17+
18+
exports.OCK = OCK;

package.json

+5-1
Original file line numberDiff line numberDiff line change
@@ -1,8 +1,12 @@
11
{
22
"name": "open-community-kit",
3-
"version": "1.0.0",
3+
"version": "1.1.0",
44
"description": "Tools and stats for open-source communities",
55
"main": "contributors.js",
6+
"bin": {
7+
"ock": "cli.js",
8+
"open-community-kit": "cli.js"
9+
},
610
"scripts": {
711
"test": "echo \"Error: no test specified\" && exit 1"
812
},

0 commit comments

Comments
 (0)