Skip to content

Commit 88322ad

Browse files
committed
Merge branch 'firebase'
Add firebase version to GitHub Invite
2 parents aa9c034 + 237d8c7 commit 88322ad

14 files changed

+3643
-4
lines changed

README.md

+14-3
Original file line numberDiff line numberDiff line change
@@ -7,7 +7,7 @@ To send an invitation three things must be provided:
77
2. Personal access token of an admin in that organization (obtainable in GitHub account settings) that has at least, admin:org (full control of orgs and teams, read and write org projects) permission, when the token was created.
88
3. Username of the GitHub user to be invited.
99

10-
The way this app requires the above mentioned three items, depends on your choice of version to use. Three versions are available and they require different ways of providing the above three items.
10+
The way this app requires the above mentioned three items, depends on your choice of version to use. Four versions are available and they require different ways of providing the above three items.
1111

1212

1313

@@ -19,8 +19,16 @@ In this version, provide an `org.js` file in the root folder that exports a JSON
1919

2020
![Demo of Node Monolith Version of GitHub Organization Invitation](/demos/node-monolith.gif)
2121

22+
2. ## [Firebase Version](/firebase)
2223

23-
2. ## [Static Web App](/docs)
24+
In this version, create a firebase project at https://console.firebase.google.com and set it up and deploy as in the [Firebase Version README](/firebase/README.md#setting-up). Your Github-Invite is at your `<project_id>`.web.app. Share this link with your invitees. It contains a frontend similar to that of the Node Monolith above, where the GitHub user can enter their username and get invited to your organization.
25+
26+
#### Demo
27+
28+
![Demo of Node Monolith Version of GitHub Organization Invitation](/demos/node-monolith.gif)
29+
30+
31+
3. ## [Static Web App](/docs)
2432

2533
In this version, you enter your organization's name on GitHub, an admin's personal access token and the username of the GitHub user to be invited, directly in the static frontend of the website and JavaScript will do the invitation. Proper feedback is equally shown directly in the website whether the invitation was successful or not and possible reasons for that.
2634

@@ -32,7 +40,7 @@ So, you can click on that link and perform as many invitations as you wish.
3240
![Demo of Static Frontend Version of GitHub Organization Invitation](/demos/static-frontend.gif)
3341

3442

35-
3. ## [NPM Package](/npm-package)
43+
4. ## [NPM Package](/npm-package)
3644

3745
[![npm version](https://badge.fury.io/js/github-invite.svg)](https://badge.fury.io/js/github-invite)
3846

@@ -59,6 +67,9 @@ The `github-invite` CLI also outputs feedback properly as invitations are done,
5967
<br/>
6068
<br/>
6169

70+
## Customizing
71+
Feel free to customize the frontends of the ![Node Monolith](/node-monolith) and ![Firebase](/firebase) versions to suit your brand's guidelines. However, avoid tampering with the parts of the code that give feedback and make the app work.
72+
6273
## Bugs? Features?
6374
If you have find any problems while using this software, or you come accross any bugs, or you want to add a feature, please feel free to create an issue or pull request.
6475
Thank you!

firebase/.gitignore

+68
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,68 @@
1+
# Logs
2+
logs
3+
*.log
4+
npm-debug.log*
5+
yarn-debug.log*
6+
yarn-error.log*
7+
firebase-debug.log*
8+
9+
# Firebase cache
10+
.firebase/
11+
12+
# Firebase config
13+
14+
# Uncomment this if you'd like others to create their own Firebase project.
15+
# For a team working on the same Firebase project(s), it is recommended to leave
16+
# it commented so all members can deploy to the same project(s) in .firebaserc.
17+
# .firebaserc
18+
19+
# Runtime data
20+
pids
21+
*.pid
22+
*.seed
23+
*.pid.lock
24+
25+
# Directory for instrumented libs generated by jscoverage/JSCover
26+
lib-cov
27+
28+
# Coverage directory used by tools like istanbul
29+
coverage
30+
31+
# nyc test coverage
32+
.nyc_output
33+
34+
# Grunt intermediate storage (http://gruntjs.com/creating-plugins#storing-task-files)
35+
.grunt
36+
37+
# Bower dependency directory (https://bower.io/)
38+
bower_components
39+
40+
# node-waf configuration
41+
.lock-wscript
42+
43+
# Compiled binary addons (http://nodejs.org/api/addons.html)
44+
build/Release
45+
46+
# Dependency directories
47+
node_modules/
48+
49+
# Optional npm cache directory
50+
.npm
51+
52+
# Optional eslint cache
53+
.eslintcache
54+
55+
# Optional REPL history
56+
.node_repl_history
57+
58+
# Output of 'npm pack'
59+
*.tgz
60+
61+
# Yarn Integrity file
62+
.yarn-integrity
63+
64+
# dotenv environment variables file
65+
.env
66+
67+
# add .firebaserc to .gitignore so deployment can be from any firebase project
68+
.firebaserc

firebase/README.md

+35
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,35 @@
1+
# GitHub Organisation Invitation - Firebase Version
2+
3+
A web app that can accept a GitHub username and send them an invitation to join a GitHub Organisation.
4+
5+
You can deploy it with firebase and provide users with the frontend, in which they can enter their GitHub username and get invited to your GitHub organization.
6+
7+
## On Firebase
8+
Firebase is a backend as a service from Google that combines authentication, database, file storage, machine learning, analytics and cloud functions in one place, so that you concentrate on building the app. Checkout https://firebase.google.com for more information.
9+
10+
This version of GitHub Invite uses Firebase to work. Fundamentally, the static files hosted from the [public folder](./public) makeup the frontend where users enter their `username`s get invited. The frontend submits to a cloud function that makes the invitation and responds with the proper feedback if invitation was successful or not via the URL parameters.
11+
12+
## Get an admin token
13+
Only admin members of a GitHub organisation can send an invitation to other GitHub users, to join that organisation. The admin member must be authenticated before invitation can be sent. Go to settings in admin's GitHub account and get a new personal access token or use an existing token, if you already have one. Ensure that the token has at least, the admin:org (full control of orgs and teams, read and write org projects) permission (select that permission while creating or edit existing token permissions to include). Keep the token, it will be useful when setting up the Firebase Environment
14+
15+
## Setting Up
16+
* Create a Firebase project at https://console.firebase.google.com. Take note of your project_id, it is globally unique across all firebase projects and can't be changed after your project is created. You can also enable Google Analytics (recommended).
17+
18+
* After creating your project, create a new web app, enable firebase hosting and follow the terminal/command line steps.
19+
20+
* Clone or Download this repo, and in the terminal/command line, be sure you are in this directory
21+
22+
* Run `firebase use --add` and select your project
23+
24+
* Run `firebase functions:config:set org.name="ORG NAME ON GITHUB" org.token="GENERATED ADMIN TOKEN"`
25+
26+
* Enable billing for your firebase project, in the firebase console, inorder to use cloud functions at https://console.firebase.google.com/project/_/usage/details. In other words, switch from the free (spark) plan to the pay as you go (blaze) plan. Checkout the cloud functions pricing at https://firebase.google.com/pricing#cloud-functions
27+
28+
* Run `firebase deploy`, your Github-Invite is at your `<project_id>`.web.app
29+
30+
## Customizing
31+
Feel free to customize the frontend to suit your brand's guidelines. However, avoid tampering with the parts of the code that give feedback and make the app work.
32+
33+
## Demo
34+
35+
![Demo of Firebase Version of GitHub Organization Invitation](../demos/node-monolith.gif)

firebase/firebase.json

+29
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,29 @@
1+
{
2+
"functions": {
3+
"predeploy": [
4+
"npm --prefix \"$RESOURCE_DIR\" run lint"
5+
]
6+
},
7+
"hosting": {
8+
"public": "public",
9+
"ignore": [
10+
"firebase.json",
11+
"**/.*",
12+
"**/node_modules/**"
13+
],
14+
"rewrites": [
15+
{
16+
"source": "/org",
17+
"function": "app"
18+
},
19+
{
20+
"source": "/invite",
21+
"function": "app"
22+
},
23+
{
24+
"source": "**",
25+
"destination": "/index.html"
26+
}
27+
]
28+
}
29+
}

firebase/functions/.eslintrc.json

+123
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,123 @@
1+
{
2+
"parserOptions": {
3+
// Required for certain syntax usages
4+
"ecmaVersion": 2017
5+
},
6+
"plugins": [
7+
"promise"
8+
],
9+
"extends": "eslint:recommended",
10+
"rules": {
11+
// Removed rule "disallow the use of console" from recommended eslint rules
12+
"no-console": "off",
13+
14+
// Removed rule "disallow multiple spaces in regular expressions" from recommended eslint rules
15+
"no-regex-spaces": "off",
16+
17+
// Removed rule "disallow the use of debugger" from recommended eslint rules
18+
"no-debugger": "off",
19+
20+
// Removed rule "disallow unused variables" from recommended eslint rules
21+
"no-unused-vars": "off",
22+
23+
// Removed rule "disallow mixed spaces and tabs for indentation" from recommended eslint rules
24+
"no-mixed-spaces-and-tabs": "off",
25+
26+
// Removed rule "disallow the use of undeclared variables unless mentioned in /*global */ comments" from recommended eslint rules
27+
"no-undef": "off",
28+
29+
// Warn against template literal placeholder syntax in regular strings
30+
"no-template-curly-in-string": 1,
31+
32+
// Warn if return statements do not either always or never specify values
33+
"consistent-return": 1,
34+
35+
// Warn if no return statements in callbacks of array methods
36+
"array-callback-return": 1,
37+
38+
// Require the use of === and !==
39+
"eqeqeq": 2,
40+
41+
// Disallow the use of alert, confirm, and prompt
42+
"no-alert": 2,
43+
44+
// Disallow the use of arguments.caller or arguments.callee
45+
"no-caller": 2,
46+
47+
// Disallow null comparisons without type-checking operators
48+
"no-eq-null": 2,
49+
50+
// Disallow the use of eval()
51+
"no-eval": 2,
52+
53+
// Warn against extending native types
54+
"no-extend-native": 1,
55+
56+
// Warn against unnecessary calls to .bind()
57+
"no-extra-bind": 1,
58+
59+
// Warn against unnecessary labels
60+
"no-extra-label": 1,
61+
62+
// Disallow leading or trailing decimal points in numeric literals
63+
"no-floating-decimal": 2,
64+
65+
// Warn against shorthand type conversions
66+
"no-implicit-coercion": 1,
67+
68+
// Warn against function declarations and expressions inside loop statements
69+
"no-loop-func": 1,
70+
71+
// Disallow new operators with the Function object
72+
"no-new-func": 2,
73+
74+
// Warn against new operators with the String, Number, and Boolean objects
75+
"no-new-wrappers": 1,
76+
77+
// Disallow throwing literals as exceptions
78+
"no-throw-literal": 2,
79+
80+
// Require using Error objects as Promise rejection reasons
81+
"prefer-promise-reject-errors": 2,
82+
83+
// Enforce “for” loop update clause moving the counter in the right direction
84+
"for-direction": 2,
85+
86+
// Enforce return statements in getters
87+
"getter-return": 2,
88+
89+
// Disallow await inside of loops
90+
"no-await-in-loop": 2,
91+
92+
// Disallow comparing against -0
93+
"no-compare-neg-zero": 2,
94+
95+
// Warn against catch clause parameters from shadowing variables in the outer scope
96+
"no-catch-shadow": 1,
97+
98+
// Disallow identifiers from shadowing restricted names
99+
"no-shadow-restricted-names": 2,
100+
101+
// Enforce return statements in callbacks of array methods
102+
"callback-return": 2,
103+
104+
// Require error handling in callbacks
105+
"handle-callback-err": 2,
106+
107+
// Warn against string concatenation with __dirname and __filename
108+
"no-path-concat": 1,
109+
110+
// Prefer using arrow functions for callbacks
111+
"prefer-arrow-callback": 1,
112+
113+
// Return inside each then() to create readable and reusable Promise chains.
114+
// Forces developers to return console logs and http calls in promises.
115+
"promise/always-return": 2,
116+
117+
//Enforces the use of catch() on un-returned promises
118+
"promise/catch-or-return": 2,
119+
120+
// Warn against nested then() or catch() statements
121+
"promise/no-nesting": 1
122+
}
123+
}

firebase/functions/.gitignore

+2
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,2 @@
1+
node_modules/
2+
.runtimeconfig.json

firebase/functions/index.js

+87
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,87 @@
1+
const functions = require('firebase-functions');
2+
const express = require('express');
3+
const app = express();
4+
const fetch = require('node-fetch');
5+
const host = 'https://api.github.com';
6+
7+
app.get('/org', (req, res) => {
8+
try {
9+
res.status(200).send(functions.config().org.name);
10+
} catch (error) {
11+
res.status(500).end();
12+
}
13+
});
14+
15+
app.post('/invite', async (req, res) => {
16+
try {
17+
// check if github account exists with the given username, if no return that the username is invalid
18+
const user = await fetch(
19+
`${host}/users/${req.body.username}`
20+
).then((response) => response.json());
21+
22+
if (!user.id)
23+
return res.status(401).json({
24+
status: false,
25+
message: 'Invalid Username',
26+
body: `GitHub user with username: ${req.body.username} not found. Please check username's spelling or create GitHub account with that username. Thank you.`,
27+
});
28+
29+
// ensure that GitHub organization exists else return
30+
const organization = await fetch(
31+
`${host}/orgs/${functions.config().org.name}`
32+
).then((response) => response.json());
33+
34+
if (!organization.id)
35+
return res.status(401).json({
36+
status: false,
37+
message: 'Server Error',
38+
body: `GitHub organization with name: ${functions.config().org.name} not found.`,
39+
});
40+
41+
// invite user
42+
return fetch(`${host}/orgs/${functions.config().org.name}/invitations`, {
43+
method: 'POST',
44+
headers: {
45+
'Content-Type': 'application/json',
46+
Accept: 'application/json',
47+
Authorization: `token ${functions.config().org.token}`,
48+
},
49+
body: `{"invitee_id":${user.id}}`,
50+
}).then((response) => {
51+
// respond apprioprately
52+
if (response.status === 201) {
53+
return res.status(201).json({
54+
status: true,
55+
message: 'Successfully Invited',
56+
body: `Dear ${req.body.username},<br>Kindly check your inbox and accept the invitation that has been sent to you.<br>Thank you!`,
57+
});
58+
} else {
59+
return response.json().then((data) => {
60+
let messages = [data.message];
61+
if (data.errors) {
62+
for (let error of data.errors) {
63+
messages.push(error.message);
64+
}
65+
}
66+
return res.status(401).json({
67+
status: false,
68+
message: response.statusText,
69+
body: messages.join('<br>'),
70+
});
71+
});
72+
}
73+
});
74+
} catch (error) {
75+
return res.status(500).json({
76+
status: false,
77+
message: 'An Error Occured',
78+
body: error.toString(),
79+
});
80+
}
81+
});
82+
83+
app.use('*', (req, res) => {
84+
res.redirect('/');
85+
});
86+
87+
exports.app = functions.https.onRequest(app);

0 commit comments

Comments
 (0)