/ServerlessWebApiWithCognito

An ASP.NET Core Web API project using AWS Cognito for authorization, for deployment to AWS Lambda as a demo of SPA with Cognito running serverless.

Primary LanguageC#

Custom Authorizer for Serverless ASP.NET Core Web APIs with Lambda and Cognito

Warning! This project is from September 2017, prior to .NET Core 2.0 support in Lambda

** If you use this, use the latest version of .NET, update the Nuget dependencies, and so on.**

** ==> UPDATE: for using this code with newer versions of .NET, you'll need to make a number of changes to the way that JWT tokens are validated, it has changed a bit <== **

In real-world web-applications, you nearly always want some way to authenticate users – either all of them, or some subset of them – in order to gate access to some features. In ASP.NET Web API, developers are used to adding the [Authorize] attribute to controller methods, and then relying on a membership database to store users and their roles. With Amazon Cognito User Pools, however, we can offload the storage, management and authentication of users and their roles, while still leveraging the [Authorize] attribute plus a custom AuthorizationHandler class, to control access to Web API methods.

In this walk-through, we’ll create an Amazon Cognito User Pool to store and authenticate our users, an ASP.NET Core Web API serverless project that will be hosted in AWS Lambda and fronted by API Gateway, and a simple Calendar SPA consisting of HTML and JavaScript, which will be part of the Web API project and served by the Lambda function. Our client-side script will pass a user's login credentials to Cognito, get back a JSON Web Token (JWT), and pass that in the HTTP Authorization header to our Web API methods that require authorization. We'll allow reading calendar events for everybody, but restrict creating and editing them to users in a group that we specify.

Create an Amazon Cognito User Pool

Our Cognito User Pool will contain our users, and the groups to which they belong. For this walk-through, we'll create and configure the User Pool using the AWS Management console, and then add a couple of users manually. For an application with lots of users, you would typically bulk import your users, or allow them to register via your web-application, both of which are supported in Cognito.

In the AWS Management Console, navigate to the AWS Cognito home page, and then click the "Manage your User Pools" button.

Next, create a user pool by clicking the blue, "Create a user pool" button at the top right. Give your user pool a name; for this walk-through, we'll use the name, "WebApi Calendar App Users", and then choose "Review Defaults" to see the default settings. You should see something like the following:

Before we create the pool, let's add an App Client that is allowed to authenticate against our pool. Click the, "Add app client…" link, which takes you to the App Clients list for this User Pool. There aren't any yet, so click the, 'Add an app client' link, and specify a name for the JavaScript web-app we'll be creating later. We'll use "CalendarWebClientApp" for this walk-thru. You need to uncheck the "Generate client secret" option, as the Cognito JavaScript SDK doesn't support using secrets (since anyone can view the source code in their browser). You should see something similar to the image below:

Create the app client, then the "Return to pool details" link. We can now see our App Client name next to "App clients". Click the blue "Create Pool" button to create the User Pool. After the pool is successfully created, copy the Pool Id from the top of the page and save it for later. Then click the "App client settings" link in the same navigation menu, and locate the App client ID for CalendarWebClientApp, and save that for later also.

Finally, let's create a Cognito group and a user for it. We'll log into our web application with this user later. Click the "Users and groups" link in the left-hand navigation menu, then on the Groups tab, create a new group called, "CalendarWriter". We don't need to select an IAM role. On the Users tab, create a new user. Pick a username, temporary password, and supply an email address. Uncheck "Mark phone number as verified" unless you want to provide a phone number. Then, add the user you just created to the "CalendarWriter" group. That's the group we'll require to create and edit events. New users created via the admin console this way will have to choose a new password on first login – we'll handle that in JavaScript.

Now we're ready to create the ASP.NET Core Web API project that will run in our Lambda function.

Create AWS Serverless Application (.NET Core)

Ensure you have already installed the latest version of the AWS Toolkit for Visual Studio, which installs the project templates for AWS projects. We're creating a new "AWS Serverless Application (.NET Core)" project. After selecting that project type, we'll choose the "ASP.NET Core Web API" blueprint from the blueprints list. This will generate skeleton code for our project, and also a CloudFormation template that will deploy the solution. This blueprint relies on the Amazon.Lambda.AspNetCoreServer NuGet package to translate calls between API Gateway and the ASP.NET Core framework.

Note that you can do all this on Mac OS X or Linux also. Once you have the .NET Core framework installed, you can install the AWS templates with the dotnet new command:

@:~$ dotnet new -i Amazon.Lambda.Templates::\*

The "Lambda ASP.NET Core Web API" template (short name lambda.AspNetCoreWebAPI) is then available to create new projects. You could then use Visual Studio Code, or any other code editor to write your code, and deploy using the CLI or AWS Management Console. For this walk-through, though, we'll use Visual Studio 2017 and the AWS Toolkit for Visual Studio.

The ASP.NET Core Web API blueprint project template creates two entry point classes, LocalEntryPoint.cs and LambdaEntryPoint.cs. The LocalEntryPoint class is used when running in your local dev environment, and leverages Kestrel, the ASP.NET Core webserver, while the LambdaEntryPoint relies on API Gateway. This makes it really easy to test out your project locally before deploying it. In fact, you should make sure everything is set up properly by building the project and then running it. Verify the endpoint http://localhost:5000/api/values returns the json array ["value1","value2"] by visiting it in a web browser. That will call the ValuesController’s Get method.

The blueprint project also creates an S3ProxyController, which we don't need for this project, so we can delete that. We can also remove the CloudFormation template parameters ShouldCreateBucket and BucketName, the conditions CreateS3Bucket and BucketNameGenerated, and all references to them from the serverless.template file.

Before we write any code, let’s add the NuGet package, “Microsoft.AspNetCore.Authentication.JwtBearer”, v1.0.1. Be careful not to install the latest version, since it requires .NET Standard 2.0, which isn’t yet supported in AWS Lambda. You’ll notice that some other NuGet packages are already installed as part of the project template. You can optionally remove the AWSSDK.S3 package, and then delete this line from Startup.ConfigureServices so that the project will still build: services.AddAWSService<Amazon.S3.IAmazonS3>();

Once we add the Microsoft.AspNetCore.Authentication.JwtBearer v1.0.1 package, we can add our implementation of IAuthorizationRequirement, and our customer AuthorizationHandler class as in the following code samples.

class CognitoGroupAuthorizationRequirement : IAuthorizationRequirement
{
    public string CognitoGroup { get; private set; }

    public CognitoGroupAuthorizationRequirement(string cognitoGroup)
    {
        CognitoGroup = cognitoGroup;
    }
}
class CognitoGroupAuthorizationHandler : AuthorizationHandler<CognitoGroupAuthorizationRequirement>
{
    protected override Task HandleRequirementAsync(AuthorizationHandlerContext context, CognitoGroupAuthorizationRequirement requirement)
    {
        if (!context.User.HasClaim(c => c.Type == "cognito:groups"))
        {
            context.Fail();
            return Task.CompletedTask;
        }

        var group = context.User.FindFirst(c => c.Type == "cognito:groups").Value;

        if (group == requirement.CognitoGroup)
        {
            context.Succeed(requirement);
        }
        else
        {
            context.Fail();
        }

        return Task.CompletedTask;
    }
}

Our custom handler just checks to see if the Cognito group is equal to whatever group we specify in the Authorization Policy we add. Let’s register our policy and add a singleton of the handler class by editing the Startup.ConfigureServices method to look like the below example.

public void ConfigureServices(IServiceCollection services)
{
    // add our Cognito group authorization requirement, specifying CalendarWriter as the group
    services.AddAuthorization(
        options => options.AddPolicy("InCalendarWriterGroup", policy => policy.Requirements.Add(new CognitoGroupAuthorizationRequirement("CalendarWriter")))
    );
    // add a singleton of our cognito authorization handler
    services.AddSingleton<IAuthorizationHandler, CognitoGroupAuthorizationHandler>();
            
    services.AddMvc();

    // Pull in any SDK configuration from Configuration object
    services.AddDefaultAWSOptions(Configuration.GetAWSOptions());
}

For our simple Calendar app, we'll create one controller, "EventsController", and one model, "CalendarEvent". The model will be serialized and deserialized automatically as it's passed to and from the controller methods. It's just a POCO class that maps to the required properties of a FullCalendar event. FullCalendar is an open-source JavaScript calendar built on JQuery, which we'll use in the client-side portion of the project.

public class CalendarEvent
{
    public string id { get; set; }
    public string title { get; set; }
    public bool allDay { get; set; }
    public DateTime start { get; set; }
    public DateTime end { get; set; }
}

In our API controller, we’ll implement three methods – a Get() method that returns all events in a date range, a Post() method to create new events, and a Put() method to edit existing events. We’ll decorate the Put() and Post() methods with our custom [Authorize] attribute and specify the InCalendarWriterGroup policy we registered in Startup.ConfigureServices. To keep things really simple, for this walk-thru we’ll just be storing events in a static collection in the controller – in a real application, you would want to persist them somewhere durable and shared between different Lambda invocations, like Amazon DynamoDB, a serverless NoSQL database. Lambda functions will "go away" after about 5 minutes of non-use, so any events we add will disappear when that happens. Also note the “test event” we create in the constructor – that’s just to demonstrate the ability to see events when not authenticated.

[Route("api/[controller]")]
public class EventsController : Controller
{
    private static List<CalendarEvent> events;

    public EventsController()
    {
        if (events == null)
        {
            var calEvent = new CalendarEvent()
            {
                id = Guid.NewGuid().ToString(),
                title = "Test Event Title",
                allDay = false,
                start = DateTime.Now,
                end = DateTime.Now.AddHours(2)
            };
            events = new List<CalendarEvent>( new[] { calEvent } );
        }
    }

    // GET api/events
    [HttpGet]
    public IEnumerable<CalendarEvent> Get(DateTime? start, DateTime? end)
    {   
        start = start ?? new DateTime(DateTime.Now.Year, DateTime.Now.Month, 1);
        end = end ?? new DateTime(DateTime.Now.Year, DateTime.Now.Month, DateTime.DaysInMonth(DateTime.Now.Year, DateTime.Now.Month));

        return events.FindAll(x => x.end >= start && x.start <= end);
    }

    // POST api/events
    [HttpPost]
    [Authorize(Policy = "InCalendarWriterGroup")]
    public void Post([FromForm]CalendarEvent calEvent)
    {
        calEvent.id = Guid.NewGuid().ToString();
        events.Add(calEvent);
    }

    // PUT api/events/5
    [HttpPut("{id}")]
    [Authorize(Policy = "InCalendarWriterGroup")]
    public void Put(string id, [FromForm]CalendarEvent calEvent)
    {
        var index = events.FindIndex(x => x.id == id);
        events[index] = calEvent;
    }

}

Finally, we’ll add code to the Startup.Configure method configure the JWT authentication options, and also to enable our app to serve static files (like html, css and JS) as well default files, so we don’t have to specify "index.html" in the URL of the main page. Static files support is enabled with the NuGet package, “Microsoft.AspNetCore.StaticFiles”, v1.0.0. Again, don't use the latest version since that requires .NET Standard 2.0.

We're just hard-coding the "Audience" (the app client ID from Cognito) and "Authority" (based on User Pool ID) values for simplicity. In a production app, you could pass those values as Environment Variables at deploy time, then read the values with Environment.GetEnvironmentVariable. Your Configure method should look like this.

public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
    loggerFactory.AddLambdaLogger(Configuration.GetLambdaLoggerOptions());
    app.UseDefaultFiles();  //needs to be before the app.UseStaticFiles() call below
    app.UseStaticFiles();
    app.UseJwtBearerAuthentication(newJwtBearerOptions
    {
        Audience = "<the app ID you copied from Cognito earlier>",
        Authority = "https://cognito-idp.<region>.amazonaws.com/<User Pool Id>",
        AutomaticAuthenticate = true,
        RequireHttpsMetadata = false  //false for dev only, for production the JWT should be sent via https
    });

    app.UseMvc();
}

To build the “Authority” value, just plug in your region and User Pool Id that you copied earlier. If your region was Oregon (us-west-2) and your User Pool Id was us-west-2_AbcDefGhi, for example, your Authority string would be “https://cognito-idp.us-west-2.amazonaws.com/us-west-2_AbcDefGhi”. For testing locally without worrying about certificates, set “RequireHttpsMetadata” to false. For any production app, you should set this to true so that the JWT token is only sent over HTTPS. The JwtBearer middleware validates the JWT’s digital signature, checks the expiration and does other checks, then creates a ClaimsPrincipal for the current user.

Now we're ready to add our client-side application!

Add HTML, CSS and JavaScript

ASP.NET Core uses the folder wwwroot as web root by default, so we'll create that folder in our project, and add our web files to it. You can either install the JavaScript files manually, or using a tool like npm. We'll need the following libraries for this demo app. I installed the Amazon Cognito, FullCalendar and Moment.js files locally, and referenced the JQuery files via URL, but you can change that to suit your needs. I recommend installing with npm to ensure you get all the dependencies, but links to downloads are included below.

Since this demo app is very simple, I embedded the JavaScript used to authenticate and create/edit events inline. You can see the full source code in the Git repo. We set the source for events in FullCalendar to "api/events" (FullCalendar passes the start and end arguments), and add handlers for dayClick and eventClick events. Reading events (the "Get()" method in EventsController) doesn't require authorization.

Creating or editing events will require a valid JWT showing the user in the CalendarWriter group. So we add a login button to our HTML, and wire it up to an authentication flow. When authenticating, we'll handle the onSuccess, onFailure, and newPasswordRequired events. On first login, the newPasswordRequired event is raised. We'll prompt the user for a new password, and then they'll need to log in with it. This will change the user's status in Cognito from FORCE_CHANGE_PASSWORD to CONFIRMED.

The JavaScript in the authenticate() function reads the username and password values from the form, configures the user pool (with the User Pool Id and App Client Id you copied earlier), then calls the CognitoUser.authenticateUser() function which makes the call out to Amazon Cognito. The response (if successful) includes the JWT token, which we put into local storage to use for subsequent Post or Put calls to our calendar API. To keep things simple for this walk-thru, we're not implementing any refresh strategy – when the JWT expires after 1 hour, subsequent PUT/POST calls will fail with a 401 Unauthorized response.

function authenticate() {

    var authenticationData = {
        Username: $('#username').val(),
        Password: $('#password').val(),
    };
    console.log('username = ' + authenticationData.Username);

    var CognitoIdentityServiceProvider = AWSCognito.CognitoIdentityServiceProvider;
    var authenticationDetails = new CognitoIdentityServiceProvider.AuthenticationDetails(authenticationData);

    var CognitoUserPool = AmazonCognitoIdentity.CognitoUserPool;

    var poolData = {
        UserPoolId: 'us-west-2_PoTkPSgPb', //user pool id
        ClientId: '5uibfabea1gvq8t8ivou1bge50' //app client id
    };
    var userPool = new CognitoUserPool(poolData);

    var userData = {
        Username: authenticationData.Username,
        Pool: userPool
    };
    var cognitoUser = new CognitoIdentityServiceProvider.CognitoUser(userData);

    cognitoUser.authenticateUser(authenticationDetails, {
        onSuccess: function (result) {
            localStorage.setItem('jwt', result.getIdToken().getJwtToken())
            console.log('jwt token + ' + localStorage.getItem('jwt'));
            $('#loginform').dialog('close');
            $('#login').prop("disabled", true);
        },

        onFailure: function (err) {
            alert(err);
        },

        newPasswordRequired: function (userAttributes, requiredAttributes) {
            // the api doesn't accept this field back
            delete userAttributes.email_verified;
            delete userAttributes.phone_number_verified;
            $('#loginform').dialog('close');
            $('#changepass').dialog({ modal: true, title: '1st Login - Change Password', width: 360 });
            $('#changebtn').click(function () {
                if ($('#newpass').val() == $('#confirmpass').val()) {
                    cognitoUser.completeNewPasswordChallenge($('#newpass').val(), userAttributes, this);
                    alert('Password changed, please log in with the new password');
                    $('#changepass').dialog('close');
                    Password: $('#password').val('');
                    showLogin();
                }
            });
        }
    });
}

One important note: we’re implementing the UI and authentication flow ourselves, in HTML and JavaScript. Cognito can actually host the login form itself, and handle the change password flow, and other scenarios as well. You can even customize the UI with your own logo and CSS.

The script above writes the JWT to the console, which makes it easy to test out the EventController API with Postman. Just remember that the Authorization header value is of the form, “bearer ”, and don’t forget the single space between them.

You can run the entire project locally from Visual Studio. Once the command-line window shows up, just go to http://localhost:5000/ in a browser (don’t forget the trailing slash). In order for API Gateway to handle requests to the root path, however, we need to add a resource corresponding to the “/” path to our serverless.template CloudFormation file, as a child of the “Events” node. Now GET requests to “/” will return our index.html file.

To deploy to AWS, first stop the application (if it’s running locally), then right-click the project node in Solution Explorer and select, “Publish to AWS Lambda”. The AWS Toolkit’s CloudFormation tab will open and track the status of the stack as it’s deployed. When it is finished, you can copy the AWS Serverless URL into a browser, and add a trailing slash (“/”). The initial load will take a few seconds, as it is a cold start for the Lambda function. Note that the URL includes the API Gateway stage, which in our case is “PROD”. If you use your own domain, you can configure it to point to that stage, so your website would be at www.yourdomain.com/ (for example) instead of the long API Gateway URL and stage name. ![](