Auth is the most critical part of the modern application stack. The abundance of data breaches signals that it is also challenging to get right. I find the idea of managing identities and secrets scary. I don’t want to be the cause of a worldwide data breach, and I doubt anybody does. Luckily for most developers, they can leverage the identity providers of popular web companies to bring ideas to market faster by focusing on the product’s core value rather than the plumbing of auth infrastructure.
This post will walk through the steps needed to configure an ASP.NET Core application to leverage the GitHub identity provider to add auth to a web application. As a bonus, you’ll see how to use your GitHub identity to make additional calls to the GitHub API.
OpenID All The Things
OAuth is the security mechanism for many web services, but it lacks the sufficient means to do both authentication and authorization. OpenID builds on OAuth to allow users to use existing accounts on their favorite websites to log into other websites without creating new logins. The OpenID technology is called OpenID Connect, which defines a specification for identity providers to follow. The specification allows interoperability between all vendors and clients in the space.
Since 2014, OpenID Connect has been widely adopted by most software companies as the approach to handle user identities, authentication, and authorization. The proliferation of OpenID Connect means it’s easier than ever to integrate the technology into any client.
If you’re interested in knowing more about OpenID Connect, I suggest visiting the IdentityServer documentation site. The folks behind the project are very knowledgeable and do an excellent job explaining the benefits of the technology.
ASP.NET Core with GitHub Auth
There will be a few steps in this process, but I hope they’ll be straightforward. I’ll also have a sample of the final solution available on GitHub at the end of this post.
Setting Application Secrets
The first step is to generate a clientId
and clientSecret
from your GitHub account. Next, visit your settings page and create a new OAuth App. Be sure to save these two values, as you’ll need to use them when creating the ASP.NET Core application.
Important: The values when setting up the OAuth app should be:
- Homepage URL: https://localhost:5001/
- Authorization callback URL: https://localhost:5001/signin-github
Let’s start by creating a brand new ASP.NET Core application. Now, create a new solution using the Web App
template from your favorite IDE or a terminal. Once created, you’ll want to store the clientId
and clientSecret
in the project. Using the user secrets feature of .NET will ensure you don’t accidentally leak sensitive values. From a terminal window, navigate to your solution directory. Next, you’ll initialize your project to retrieve user secrets.
The command adds a UserSecretsId
element to your project file. The key is used to retrieve your secrets at startup.
Be aware that these secrets will only be available during development. Accordingly, managing secrets in a production environment will depend on your setup, and the topic is beyond the scope of this post.
Now, let’s add your secrets to your user secrets store. You want to run the following commands from the same terminal used in the previous step, making sure to replace the <projectname>
with the name of your project.
Now you’re ready to start writing some code.
Configure GitHub Auth Provider
Before you start writing code, you’ll need to add a reference to a NuGet package to pull in GitHub auth handlers. While you’re here, also add the OctoKit
package.
You could also alter the project file.
In your Startup
file, you’ll want to add the Authentication
middleware and set up our Cookie
auth mechanism. While you’re using GitHub, you want to store our successful login information in a cookie for a set time. Writing to a cookie reduces the round trips to GitHub or any other identity provider.
In your ConfigureServices
method in the Startup
class, let’s add the necessary changes.
Let’s walk through this code from top to bottom and explain what each step accomplishes.
- The call to
AddAuthentication
adds the authentication services into the ASP.NET Core application service collection. The authentication services allows the request pipeline to challenge, sign in, and sign out users. - The call to
AddCookie
sets theLoginPath
andLogoutPath
for user actions. - The call to
AddGitHub
will allow you to set theclientId
andclientSecret
from the configuration object (remember the values come from user secrets). You also need to setup the callback path and the GitHub scope. A scope is a set of actions the app is allowed to perform on behalf of a user. Finally, when authenticating successfully, the code will store theaccess_token
claim on the auth cookie for later use.
Moving to the Configure
method in Startup
, you need to add two middleware.
I usually place these after the call to routing and before endpoints, but you need to determine where it makes sense for your application.
Login Pages and Signout Endpoints
While you’re still in the Startup
file, let’s handle the sign-out process. In the call to UseEndpoints
you’ll set up a new endpoint for signout
. I’ve included the entire call for UseEndpoints
for clarity. This endpoint will handle any requests to sign out of the current session.
Let’s move on to handling the logging-in process. First, in the web application, create a new Signin
page under the Pages directory. Then, within the view, add the following HTML.
The HTML will scan your application’s authentication providers and generate log-in buttons for your users to click. In the pages model, you’ll need to handle the form submission from the HTML page.
There you have it! Your application is now ready to log in your users. In the next section, you’ll see how to use the GitHub access token to make calls to the GitHub API.
Calling GitHub API using an AccessToken
In the previous section, you stored the GitHub access token as part of the identity claim. To retrieve the claim, you can use the User
property found in the context of a Razor Page. Let’s add some helper extension methods before continuing.
In the same project, create a new Razor page named Account
. For this blog post, let’s look exclusively at the code behind the page.
Wow! It couldn’t be any easier. First, you pull the access token from the User
property and then create a new GitHubClient
, which calls the GitHub API. Next, let’s update the Razor page to display our results.
Now, starting up the application, you can navigate to /account
and trigger the log-in process. The resulting page should look something like the following.
There is nothing left for you to do now but code your application.
Conclusion
Setting up OpenID connect authentication in ASP.NET Core is relatively straightforward. Much of the consideration needs to be put into storing secrets, saving access tokens, and utilizing it all. In addition, there are 100s of OpenID providers associated with the AspNet.Security.OAuth
library, which is fantastic for .NET developers. This post should help
To get a working sample of this post, head over to my GitHub repository and try it out for yourself. Remember to set up your OAuth application in GitHub first to get access to the necessary secrets.
As always, thanks for reading.