We’ve all been there before, needing to bridge the gap between a legacy system and our modern ASP.NET Core MVC application. I recently had an office hour with a great developer named Ramakrishna that needed to support a legacy security format. A legacy application encrypted all the values in a single query string parameter then made an HTTP request to an ASP.NET Core MVC endpoint.
This post will see how we can leverage Action Filters in ASP.NET Core MVC to make the development experience in our action methods the same one we’ve been using all this time.
The Problem
Legacy systems may not have access to all the protocols and security patterns we have today, so they might have had to do some creative problem-solving. In our case, we have a query string parameter that is encrypted using TripleDES, and we expect the receiver to decrypt the value.
This technique is likely because secrets in URLs might end up in logs. Whatever the reason, we have a problem to solve.
Our receiving endpoint will have two parameters of number and name that we expect to have values by the time we’re in the context of our action.
We have a straightforward solution to this problem, and it involves Action Filters.
A Solution Using Action Filters and Action Arguments
Action Filters is a great way to intercept incoming requests and enhance the execution pipeline. In our case, we want to transform an existing query string parameter (secret),  decrypt its values, and set the parameters on the action method.
Our final usage will use an implemented EncryptedParameters attribute, which we’ll decorate on applicable methods.
Let’s look at the implementation of EncryptedParameters.
Using the ActionExecutingContext, we have access to the parameters of our target endpoint. Then, it’s only a matter of decrypting our secret, and then setting the ActionArguments using our known parameters.
The conversion of our types handles primitives and nullable types but does not handle complex models. Feel free to modify this code to meet your particular needs.
For folks interested in the CryptoEngine code, here it is, but I do not consider myself a cryptography expert.
In Practice
I created a simple ASP.NET Core MVC application to demonstrate this action filter in use. Starting with the view, we’ll encrypt some values and post them to our endpoint that uses the EncryptedParameters attribute.
@model IndexModel
@inject Microsoft.Extensions.Options.IOptions<CryptoEngine.Secrets> config
@{
    ViewData["Title"] = "Home Page";
    var values = "name=Khalid&number=57";
    var secret = CryptoEngine.Encrypt(values, config.Value.Key);
    var url = Url.Action("IndexPost", new {secret});
}
<div class="text-center">
    <h1 class="display-4">Welcome</h1>
    <p>Learn about <a href="https://docs.microsoft.com/aspnet/core">building Web apps with ASP.NET Core</a>.</p>
    <p>Click this link to transmit secret values via Querystring</p>
    <form action="@url" method="POST">
        <p>@url</p>
        <button type="submit">Submit</button>
    </form>
</div>
@if (Model != null)
{
    <section style="margin-top: 1em">
        <div class="text-center">
            <h2>Secrets</h2>
            <div>
                <label asp-for="Name"></label>
                @Model.Name
            </div>
            <div>
                <label asp-for="Number"></label>
                @Model.Number
            </div>
        </div>
    </section>
}
When we submit the form, we receive our secrets on the page.

Conclusion
There you have it! We can use this technique to transform any incoming query string into any other set of parameters. This technique is powerful and can be applied to any MVC action or can be applied globally.
As always, use this code as a starting point and change it to meet your particular needs.
If you want to play with a working sample, you can find the code at my GitHub repository.
As always, thanks for reading, and please follow me on Twitter @buhakmeh.
 
          
       Photo by
                
                    
                        Photo by 
 
                