Skip to content

Setting TempData on IExceptionFilter with SessionStateTempDataProvider #63845

@pedrobrasilborges

Description

@pedrobrasilborges

Is there an existing issue for this?

  • I have searched the existing issues

Describe the bug

I have implemented a IExceptionFilter that sets some data on TempData to be read on the redirect action, however since I have enabled SessionStateTempDataProvider when the pipeline reaches the response.OnStarting it will try to access Session that at that time has been closed on the SessionMiddleware. I know that I can pass the data as query string for example, but this is a legacy application migration that we are migrating slowly so the redirect endpoint is not to be migrated yet and it reads the data from the TempData.

Expected Behavior

To not give exception.

Steps To Reproduce

Example:

var builder = WebApplication.CreateBuilder();

builder.Services.AddSession();
builder.Services.AddScoped<BaseControllerExceptionFilter>();

builder.Services
			.AddControllersWithViews()
			.AddSessionStateTempDataProvider();
var app = builder.Build();

app.UseRouting();
app.UseSession();
app.MapControllers();
app.Run();
public sealed class BaseControllerExceptionFilter : IExceptionFilter
{
	private readonly ITempDataDictionaryFactory _tempDataFactory;

	public BaseControllerExceptionFilter(ITempDataDictionaryFactory tempDataFactory)
	{
		_tempDataFactory = tempDataFactory;
	}

	public void OnException(ExceptionContext context)
	{

		var tempData = _tempDataFactory.GetTempData(context.HttpContext);
		tempData["example"] = context.Exception.Message;

		context.Result = new RedirectToActionResult(null, "WeatherForecast", null);
		context.ExceptionHandled = true;
	}
}

[ApiController]
[Route("[controller]")]
[ServiceFilter(typeof(BaseControllerExceptionFilter))]
public class WeatherForecastController : Controller
{
	private static readonly string[] Summaries = new[]
	{
		"Freezing", "Bracing", "Chilly", "Cool", "Mild", "Warm", "Balmy", "Hot", "Sweltering", "Scorching"
	};

	private readonly ILogger<WeatherForecastController> _logger;

	public WeatherForecastController(ILogger<WeatherForecastController> logger)
	{
		_logger = logger;
	}

	[HttpGet(Name = "GetWeatherForecast")]
	public IEnumerable<WeatherForecast> Get()
	{
		var example = TempData["example"];
		if(example == null)
		{
			throw new Exception("123 = 123");
		}

		return Enumerable.Range(1, 5).Select(index => new WeatherForecast
		{
			Date = DateOnly.FromDateTime(DateTime.Now.AddDays(index)),
			TemperatureC = Random.Shared.Next(-20, 55),
			Summary = Summaries[Random.Shared.Next(Summaries.Length)]
		})
		.ToArray();
	}
}

Exceptions (if any)

System.InvalidOperationException: Session has not been configured for this application or request.
at Microsoft.AspNetCore.Http.DefaultHttpContext.get_Session()
at Microsoft.AspNetCore.Mvc.ViewFeatures.SessionStateTempDataProvider.SaveTempData(HttpContext context, IDictionary2 values) at Microsoft.AspNetCore.Mvc.ViewFeatures.TempDataDictionary.Save() at Microsoft.AspNetCore.Mvc.ViewFeatures.Filters.SaveTempDataFilter.OnStarting(HttpContext httpContext) at Microsoft.AspNetCore.Server.Kestrel.Core.Internal.Http.HttpProtocol.<FireOnStarting>g__ProcessEvents|241_0(HttpProtocol protocol, Stack1 events)

.NET Version

net8.0

Anything else?

No response

Metadata

Metadata

Assignees

No one assigned

    Labels

    area-mvcIncludes: MVC, Actions and Controllers, Localization, CORS, most templatesfeature-session

    Type

    No type

    Projects

    No projects

    Milestone

    No milestone

    Relationships

    None yet

    Development

    No branches or pull requests

    Issue actions