Web application template for Petrock OpenID Connect (OIDC) service
Live example can be found at: https://unofficial-oidc-client.xvm.mit.edu
A short presentation summarizing our project can be found here.
- mit-oidc-client
We want to provide an easy-to-use template for MIT students looking to develop secure web services that supports MIT Touchstone authentication.
While supporting documentation exists to do user authentication using Petrock OpenID Connect service, we feel there is a knowledge gap and technical barrier that prevents its widespread adoption. In this project, we hope to provide a simple and secure client implementation that MIT student developers can adopt to quickly get authentication in their web services.
- Basic template for a secure web service containing front-end (React.js) and API back-end (Express.js)
- Features for integration with Touchstone authentication via Petrock OpenID Connect (OIDC) service
- Includes code for securely requesting, parsing, and validating OAuth tokens from MIT OIDC
- Provides logic for how to use those tokens to request information about users
- Support developer documentation for how to use our template and how OpenID Connect works
- Implementation of OpenPubKey, which is a client-side extension that extends the OIDC protcol and adds to its security by allowing for secure signing and verification of messages.
- As an example usage, we provide a simple chatroom service (in /frontend/src/chatroom and /backend/src/chatroom) that uses this feature. Users can verify the signature of messages to check if they actually came from the declared user.
OpenID Connect is an authentication protocal that builds on top of OAuth2.0. OIDC provides a secure and easy-to-use way for web applications to authenticate users via a Single-Sign-On (SSO) service like Google accounts or MIT Touchstone.
A helpful illustrative guide can be found here.
OIDC divides the authentication process to multiple steps, at which step you (or really the web service trying to authenticate on your behalf) will interact with a different OIDC API endpoint. The idea is that at each step, there will be certain keys or tokens that identifies who you are + what the web service is doing:
-
When you first successfully log in MIT Touchstone, the browser gets back an authorization code. The authorization code just says "a specific user logged in to our page".
-
The authorization code can then be exchanged for an access token, which grants it the ability to access information about you. To do this, it requires the web service to send a client secret, which uniquely identifies the browser/web service to the OIDC server (so OIDC can verify that it's a registered web service).
-
The access token is then used to fetch identifying information (like your email and name) at the OIDC
/oidc/userinfo
endpoint, after which point the web service is relayed information about your identity, and the login process completes.
Since OpenID Connect is a well-known protocol, open-source clients for it have been written, including:
- oidc-client-ts is a Typescript library that provides OIDC and OAuth2 support for browser-based client applications. The main issue we found with this template is that since all code is executed on the client side, it requires storing the
client_secret
in the frontend code. Because browsers are public clients (see Confidential vs. Public client), there is no way to guarantee the security of the secret. - passport-mitopenid is a project written by a MIT student in 2018 that implements a sample Node.js application that works with MIT OIDC service using Passport.js. This project is very similar to node-openid-client, which is the official package (certified by OpenID) for implementing OpenID Client and Relaying Party in Node.js applications. However, for both libraries, it is an incomplete solution, since they are intended for web applications running solely on Node.js, whereas for our solution we want to have an integrated authentication workflow between an Node.js backend and a frontend running a web framework like React.js.
To ensure the security of our overall client framework, we made the following design choices:
- HTTPS is required in development for both the front-end and back-end
- HTTPS/SSL encryption ensures all communication between the user and the web service is protected. This is a hard requirement for safe-handling of OAuth ID and access tokens
- Dependency on secure cryptographic libraries
- When generating randomness or relying on cryptographic primitives like hash functions, we use secure libraries like built in browser's
Crypto.getRandomValues()
andCrypto.subtle.digest()
functions, along with official JWT libraries. This ensures that we are generating secure randomness in our application and depends only on secure primitive implementation for signing and verification of data.
- When generating randomness or relying on cryptographic primitives like hash functions, we use secure libraries like built in browser's
- Utilization of state and nonce in authorization request
- As part of the OpenID Connect Basic Implementer's Guide 1.0, it is optional, but recommended to implement the state and nonce parameter in the authentication request flow. The state parameter is used to mitigate against Cross-Site Request Forgery (CSRF, XSRF) attacks, and the nonce parameter mitigate replay attacks.
- For our application, we decided to implement both of these variables, storing them in as a variable in localStorage and a cookie, respectively. Both contain high levels of entropy (2^128) for security. Note: In the optional OpenPubKey flow, the nonce is replaced by a different format than a random string. See the original paper for arguments about its security.
- Secure handling of cookies
- Whenever cookies are used in our implementation, namely the state parameter, we secure it by setting security parameters including:
path
: Restrict sending the cookie to a specific API endpoint on the backend onlysameSite
: Disallow sending cookie on cross-site requestssecure
: Force cookie to be sent over secure connections (HTTPS)
- Whenever cookies are used in our implementation, namely the state parameter, we secure it by setting security parameters including:
- Safe type checking using Typescript
- At MIT, especially in course 6.031, we promote the user of Typescript over Javascript because it allows for static typing, which helps code be more readable, bug-safe, and more easily maintainable. It's also the language of choice for most MIT students creating web services.
For this OIDC client framework, we assume you have access to the following:
- A public domain/IP address you own
- A server or VM instance which you will be hosting the code on
- Your own methodology for implementing sessions (and issuing session IDs)
- See the Sessions for a more in-depth explanation
In the framework, we have two primary services:
- React.js Frontend - what gets run in the browser
- Purpose: Display login page, redirects user to Petrock OIDC service for authentication, and talks to the backend (using the info it gets back from Petrock) to identify the user
- Express.js Backend - what gets run on your server
- Purpose: Handle requests from the frontend, uses the information it gets to talk to OIDC servers, eventually getting user profile information and sends it back to the frontend
- Runs on: port 4000 by default
You might be wondering why we need to separate between a frontend and a backend, instead of running everything in the browser. The reason is that for OpenID Connect code flow
(which is what Petrock implements), you need to store a client secret that is unique to every web application registered with Petrock. Because of this, we need to have a separate backend running on our own server (trusted environment) instead of having the entire code in the user's browser (untrusted environment).
Install:
- Node.js 16, at least Node 16.16.0
In both the backend
or frontend
folder, run:
npm install
to install dependenciesnpm run start
to run code locally (for development purposes)npm run build
to create build production
When deploying to production, you should build your code using npm run build
for both the frontend and backend so that it's optimized for performance. You should then use a web server like Nginx to serve your static frontend files and a process manager for Node like pm2
to run your backend server (don't run against Node directly!).
First thing you will need to do is to register your web app with the MIT Petrock service.
To register, you will need to follow the instructions on the Petrock Github repo (see this section). Main step is to fill out the Google Form with information about your service.
When it asks for a list of redirect URIs, you should provide: https://YOUR_DOMAIN_NAME/oidc-response
. For example, if the domain you own is dormdigest.mit.edu
, then you would input https://dormdigest.mit.edu/oidc-response
.
Within a few days, you should get an email back from the Petrock team with your client_id
and client_secret
. With this information, you should edit the following configuration files:
- Inside
frontend/src/auth/authConfig.ts
, update:DOMAIN_URI
: Domain name of your application (no trailing slash)client_id
: Client ID given to you from Petrock
- Inside
backend/src/auth/authConfig.ts
, update:DOMAIN_URI
: Domain name of your application (no trailing slash)
client_id
: Client ID given to you from Petrock
- Inside
cert/secrets.json
, update:client_secret
: Client secret given to you from Petrock- Note: This secret must never be publicly exposed! Make sure you don't commit updates to
secrets.json
to your Github repository.
And that's it for this step!
Whenever a user successfully logged into your application, ideally you need some artifact or value that uniquely identifies that user and can be use to verify (in future web activities) that they actually logged in at some point via Petrock OIDC service.
For most modern websites + applications, they (e.g, the backend) will generate a session_id
, which is a value that uniquely identifies a specific user's active session on a particular device, and returns it to the frontend/ the browser, where it is stored.
The session ID and is often used to authenticate all permission-sensitive actions a user performs on the website (normally included inside GET or POST requests to backend API endpoints). For more information about web sessions, see here.
Because session management is application-specific, and their implementation can widely differ depending on your needs (ex. using a SQL database on the backend or storing all information in a signed cookie, choice of cookie library, etc.).
Thus for the OIDC client framework, we leave the session ID management to you. By default, in the backend, every time a user is successfully logged in, we return a unique session ID via getSessionId()
(found inside backend/src/auth/authSession.ts
) to the frontend. This is inadequate for security since the session ID isn't saved anywhere (in the future the backend has no way of determining a given session ID string is actually valid).
Some ideas for how to proceed:
- If your application doesn't display any sensitive / personal data (ex. general wiki page), then you can leave it as-is
- Note: This is effectively the same as not having authentication on your website
- When you're first starting out with your web project, it is also acceptable to leave the default behavior, and come back to it once you have API endpoints that you want to protect with session IDs.
- Implement a SQL database on the backend
- Modify
getSessionId()
such that every time you store a generate a new sessionID, you store it in a SQL database (or some persistent table format). Later than, when you add API endpoints that require the session ID for authentication, you can check the string given by in the HTTP request and see if it's in the table. If so, proceed. Otherwise, deny the action
- Modify
- Implement a separate server that generate and verify session IDs
- Since not everyone wants to use an Express.js server as their backend, you can alternatively implement another server where your important API endpoints are served + is capable of generating and validating session IDs. Inside
backend/src/auth/authSession.ts
, we provide an alternative version ofgetSessionId()
that makes an HTTPS request to the session ID server (URL as defined bysession_id_uri
insidebackend/src/auth/authConfig.ts
).
- Since not everyone wants to use an Express.js server as their backend, you can alternatively implement another server where your important API endpoints are served + is capable of generating and validating session IDs. Inside
Whenever you want to use the return session ID in the frontend, you can call the getUserAuthInfo()
function (located inside frontend/src/api/localdata.tsx
). It returns an object with the user email and session ID by default.
With your web application, you will need a way of serving your static frontend files and also forward packets to your Express.js backend server. For this, we recommend using Nginx as a web server. If you have other methods of starting a web server, feel free to use them as well.
The idea is that you can point Nginx to serve your frontend build files (ex. index.html) at the web root path /
, and also forward all requests to URL paths starting with /api/
to the backend Express.js server (which runs on port 4000 by and only serves on localhost
by default). In addition, Nginx can also be used to enforce HTTPS for applications, which is useful for the next step, certificates.
To see how we configured our example website, see the following:
nginx.conf
- See here- Purpose: This file overrides the default Nginx config normally at
/etc/nginx/nginx.conf
. The only change it has is to specifyuser
tooidc
(e.g. the user on our server whose home directory contains both our frontend and backend code).
- Purpose: This file overrides the default Nginx config normally at
oidc_nginx.conf
- See here- Purpose: This file was added to
/etc/nginx/sites-available
and then symlinked to in/etc/nginx/sites-enabled
. It specifies the frontend file serving and backend request forwarding as described above.
- Purpose: This file was added to
To secure the frontend and backend, you will need to use SSL certificates. Notably, OIDC is not secure if your application only uses HTTP. For production, you should acquired certs from a trusted CA like Let's Encrypt.
For development work ONLY, you can generate self-signed certificates. See the following guide to use mkcert
utility. The certificates should be saved to the /cert folder, with SSL secret key file named key.pem
and public certificate file named cert.pem
.
We recommend using Let's Encrypt or other reputable certificate authority when deploying to production. If you're using platforms like Heroku or Render.com for hosting, they often will have their own SSL certificates management services. See this and this.
On our live example, we used Let's Encrypt Certbot tool configured for Nginx for the acquiring and the auto-renewal of TLS certificates.
Note: If you are using Lets Encrypt or some other SSL certificate service where these files are not being stored inside /cert/key.pem
and /cert/cert.pem
, you will need to modify the .env
SSL_CRT_FILE and SSL_KEY_FILE variables inside the frontend (frontend/.env
) and the backend (backend/.env
) files.
Once a user successfully logs in, the frontend will receive a successful response from the backend server inside the OidcResponseHandler()
function (located at frontend/src/auth/auth.tsx
).
You can see the logic in this code snippet:
if (data.success) {
//Login was successful! Expect id_token
setLoginMsg("Login successful!");
localStorage.setItem(AUTH_CONFIG.idtoken_localstorage_name, data.id_token); //Save id_token to local storage
localStorage.setItem(AUTH_CONFIG.useremail_localstoragge_name, data.email);
localStorage.setItem(AUTH_CONFIG.sessionid_localstorage_name, data.session_id); //Save session_id to local storage
//NOTE: If you want to do more with the additional user profile
// information (such as full name, family name, given name,
// MIT affiliation, you can do so here).
// Otherwise, they are not saved to localstorage by default
// out of privacy concerns.
By default, we save the user's email and session ID, which we establish as the minimum amount of info needed to identify a user. However, by default, the login response will have all of the following fields (defined inside in the same field), which if success
is True, will be populated accordingly.
/**
* Expected response for server to return to user's browser after querying /login endpoint
*/
interface loginResponse {
success: boolean, //Whether or not we were able to get user's info
error_msg: string, //If failed, provide error message. Else, empty string.
//All the values below will be populated if success,
//otherwise they will be empty strings.
//These are in accordance with: https://github.com/sipb/petrock#what-information-can-i-query
sub: string,
email: string,
affiliation: string,
name: string,
given_name: string,
family_name: string,
//For session management
session_id: string
//For identity management (useful for OpenPubKey extension)
id_token: string
}
Thus, if you want to do additional actions or save these profile information fields for later usage, you can modify the OidcResponseHandler()
under the comment lines we provided. You could, for example, also save the affiliation
and name
field also in LocalStorage, or send them along to some other API endpoints for profile creation on your website.
For more information about what the fields represent, see the Petrock docs: https://github.com/sipb/petrock#what-information-can-i-query
Our client implementation does not require a specific hosting solution, and indeed you can deploy it on platforms like Heroku and Render.com, or MIT-specific hosting services like XVM offered by the Student Information Processing Board (SIPB). Indeed, Heroku and Render.com offers fully managed TLS certificates to allow for HTTPS encryption.
For our purposes, we hosted our example website on an Ubuntu 18.02 VM running on SIPB's XVM service. We use Nginx as our web server and reverse proxy with TLS enabled, and pm2
as the process manager for the Express backend.
In this section, we'll break down how our frontend and backend work together to provide the OIDC client implementation. Note that this does not cover the OpenPubKey-specific details.
The follow diagram summarizes our system interactions between the frontend, backend, and the OIDC server:
---
title: OIDC Client Design
---
sequenceDiagram
autonumber
box lightyellow unofficial-oidc-client.xvm.mit.edu
participant Frontend as React.js Frontend
participant Backend as Express.js Backend
end
box lightblue oidc.mit.edu
participant OIDC as OIDC Server
end
Frontend ->> OIDC: redirects to /oidc/authorize
OIDC -->> Frontend: receives auth code
Frontend ->> Backend: sends auth code to /login
Backend ->> OIDC: sends auth code to /oidc/token with client_secret
OIDC -->> Backend: receives JWT with id_token and access token
Backend ->> OIDC: queries about user at /oidc/userinfo with access token
OIDC -->> Backend: receives user information
Backend -->> Frontend: receives id_token and user email
The primary structure of our frontend code is as follows:
<AuthProvider>
...
<Routes>
<Route element={<Layout />}>
...
<Route path="/login" element={<LoginPage />} />
<Route path="/oidc-response" element={<OidcResponseHandler />} />
<Route
path="/protected"
element={
<RequireAuth>
<ProtectedPage />
</RequireAuth>
}
/>
</Route>
</Routes>
</AuthProvider>
In App.tsx, we define an auth context manager <AuthProvider>
, which keeps track of the current logged in user. It provides a property auth.user
through useAuth()
, which stores the email of the current user as a string. Default value is an empty string.
To prevent displaying of certain component(s) until the user is logged in, we wrap them with <RequireAuth>
. In this case, we want to hide the <ProtectPage>
, which contains the chatroom. As noted in the security discussion, this only prevents browser rendering, but any actions done using these components need to be further authenticated in the backend (preferably using some form of user-identifiable session cookies).
In our frontend code, we provide two React router paths, namely /login
and /oidc-response
, to handle the OIDC logic workflow.
-
/login
handles the redirect from the current website to the OIDC's authentication endpoint. It supplies the client_id and redirect_uri to the OIDC server (in addition to other required fields) as query parameters. -
Once the user successfully authenticates to the OIDC server, they are redirected to the
/oidc-response
endpoint on the webpage. The browser takes in the authorization code in the URL, verifies the returned state parameter is correct, and forwards the auth code to the backend using the backend's POST/login
API endpoint. -
The backend returns a response of type
loginResponse
to the frontend. The frontend then checks thesuccess
parameter. If it is true, then it stores theid_token
to localStorage and signs the user in withemail
using theauth.signin()
function. Otherwise, it outputs the error message to the user via theOidcResponseHandler
React component. -
The webpage then reloads to show a
Welcome [email protected]!
message at the top (via<AuthStatus/>
component), along with a signout button. The user can then access the Protected page, which in this case allows them to interact with the OpenPubKey-enabled chatroom.
The Express.js backend requires only one API endpoint for OIDC, which we define as a POST request.
app.post("/api/login", handleLogin);
The /api/login
takes in a code
parameter in the request body, along with a nonce cookie named oidc-request-nonce
. All server responses will be objects of type loginResponse
(as defined above). In the normal case, it will return the user's ID token and their email. In the case that something fails to validate, instead of a 404 error we will return a descriptive error message that the frontend can then display to the user. Whether the login operation succeeds or fails is denoted by the "success" boolean parameter.
/api/login Workflow:
The server takes the code
variable and sends it to the /token
endpoint on the OIDC server to exchange for an ID token. It also includes the client_id
and client_secret
as HTTP Authorization Headers, along with a fixed grant_type
and redirect_uri
.
Note: This step is the reason why the ID token exchange is done on the backend (which is considered a confidential client), rather in the frontend (considered a public client), because we want to keep the application's client secret safe. See the Related Works section on why we view this to be necessary.
Following that step, the backend server receives back an JSON object containing the ID token from the OIDC server, which it then validates thoroughly.
Once we can be sure the ID token is valid, we proceed with querying the /userinfo
endpoint on the OIDC server for user information. Since we're solely interested in determining who the user that just logged in is, we decided to query for the email
field. This action is done by the getUserInfo()
function, which simply returns the user's email (or an error message if the lookup fails).
While we were able to achieve many of the original goals set out for this project, we recognize that there are aspects which could be further improved and developed on for our client framework.
- Advertisement to students:
- A big motivation for us in creating this project is that we want for MIT students to actually be able to use this code when creating their web services. As a result, we plan to advertise our project through dormspam and get in contact with professors + TA's teaching web development class like web.lab to see they can mention it to their students. We also want to get feedback on how we can best enable the adoption of our authentication framework into existing services (rather than just new ones).
As an extension to providing authentication via Petrock OpenID Connect (OIDC) service, we supply the client with a PK Token generated from the client's ID Token. The PK Token is a committment of a public/private key pair to the ID Token, which augments the method of authentication from Bearer's Authentication to Proof-of-Possession. This protocol is built upon and is fully compatible with the OpenID Connect service. We will show a possible use case of PK Tokens with an implementation of an authenticated chatroom.
Note: To access the chatroom, you must login with your MIT credentials through the Petrock OpenID Connect service.
In the authenticated chatroom, we demonstrate how PK Tokens can be use to verify whether data is coming from a trusted user or not, which fundamentally is what authentication is about. In our case, these pieces of data takes the form of a text message and the trusted user is an identity holding an MIT crediential.
Once logged in using an MIT crediential, the client may utilize methods from our pktoken
module to generate a PK Token opkService.getPKToken
, sign messages opkService.generateOSM
, and verify messages opkService.verifyOSM
. The user submits an OpenPubKey Signed Message (OSM) consisting of their message and a signture of their message using their PK Token. The authenticated chatroom is designed in a way to allow any users to verify any OSM at any time. All initial messages are unverified (grey check). To verify, click on the check and the verification may either accept (green check) or reject (red exclamation).
-
Committing bets: Two friends make a bet on something, and they want to prove they made it with their commitment. Say later one person tries to say they never made this bet, but because they signed their message (which is stored on the server), we have irrefutable proof that they did make the bet.
-
Code signing: Have a trusted verifier to have mapping of ID tokens to public keys, and then whenever a user signs a commit they made, they can send it to the verifier and have it put into an append-only log.
-
SSH: SSH keys are difficult to manage. Instead, have users sign in through Google (or some other OpenID provider), and then have a PK Token that can act as their public SSH key.
Contact us at [email protected].