База Asp.net TempData выдает ошибку 500 при добавлении списка и перенаправлении в другое представление

Я пытаюсь создать список предупреждений и добавить их в TempData. Но это работает, если я не делаю переадресацию. Когда я делаю перенаправление, это дает мне ошибку 500. Я также установил точку останова, но она не сработала, когда перенаправила, иначе она сработала правильно.

Метод действия

public IActionResult Create(CategoryCreateVM input)
{
    if (ModelState.IsValid)
    {
        var category = mapper.Map<Categories>(input);
        categoryBL.Add(category);

        List<Alert> alert = new List<Alert>();
        alert.Add(new Alert("alert-success", "success message"));
        alert.Add(new Alert("alert-danger", "danger message"));

        TempData["Alert"] = alert;

        return RedirectToAction("Index");
    }
    return View(input);
}

Как я получаю доступ к виду.

@{ 
    var alerts = TempData["Alert"] as List<Alert>;
}
@if (alerts != null && alerts.Count > 0)
{
    <div class="">
        @foreach (var alert in alerts)
        {
            <div class="alert @alert.AlertClass alert-dismissible">
                <button type="button" class="close" data-dismiss="alert" aria-hidden="true">&times;</button>
                @alert.Message
            </div>

        }
    </div>

}

Трассировки стека.

    Exception thrown: 'System.InvalidOperationException' in Microsoft.AspNetCore.Mvc.ViewFeatures.dll
Exception thrown: 'System.InvalidOperationException' in mscorlib.dll
Microsoft.AspNetCore.Mvc.Internal.ControllerActionInvoker: Information: Executed action MvcSandbox.Controllers.HomeController.Index (MvcSandbox) in 46.4619ms
Exception thrown: 'System.InvalidOperationException' in mscorlib.dll
Exception thrown: 'System.InvalidOperationException' in mscorlib.dll
Microsoft.AspNetCore.Diagnostics.DeveloperExceptionPageMiddleware: Error: An unhandled exception has occurred while executing the request

System.InvalidOperationException: The 'Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.TempDataSerializer' cannot serialize an object of type 'MvcSandbox.Controllers.Alert'.
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.TempDataSerializer.EnsureObjectCanBeSerialized(Object item) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\TempDataSerializer.cs:line 207
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.TempDataSerializer.Serialize(IDictionary`2 values) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\TempDataSerializer.cs:line 142
   at Microsoft.AspNetCore.Mvc.ViewFeatures.SessionStateTempDataProvider.SaveTempData(HttpContext context, IDictionary`2 values) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\ViewFeatures\SessionStateTempDataProvider.cs:line 62
   at Microsoft.AspNetCore.Mvc.ViewFeatures.TempDataDictionary.Save() in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\ViewFeatures\TempDataDictionary.cs:line 166
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.SaveTempDataFilter.SaveTempData(IActionResult result, ITempDataDictionaryFactory factory, HttpContext httpContext) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\SaveTempDataFilter.cs:line 91
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.SaveTempDataFilter.OnResultExecuted(ResultExecutedContext context) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\SaveTempDataFilter.cs:line 80
   at Microsoft.AspNetCore.Mvc.Internal.ControllerActionInvoker.Next(State& next, Scope& scope, Object& state, Boolean& isCompleted) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.Core\Internal\ControllerActionInvoker.cs:line 1023
   at Microsoft.AspNetCore.Mvc.Internal.ControllerActionInvoker.<InvokeNextResourceFilter>d__22.MoveNext() in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.Core\Internal\ControllerActionInvoker.cs:line 1105
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.ExceptionServices.ExceptionDispatchInfo.Throw()
   at Microsoft.AspNetCore.Mvc.Internal.ControllerActionInvoker.Rethrow(ResourceExecutedContext context) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.Core\Internal\ControllerActionInvoker.cs:line 1377
   at Microsoft.AspNetCore.Mvc.Internal.ControllerActionInvoker.Next(State& next, Scope& scope, Object& state, Boolean& isCompleted) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.Core\Internal\ControllerActionInvoker.cs:line 493
   at Microsoft.AspNetCore.Mvc.Internal.ControllerActionInvoker.<InvokeAsync>d__20.MoveNext() in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.Core\Internal\ControllerActionInvoker.cs:line 164
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at Microsoft.AspNetCore.Builder.RouterMiddleware.<Invoke>d__4.MoveNext()
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.GetResult()
   at Microsoft.AspNetCore.Session.SessionMiddleware.<Invoke>d__9.MoveNext() in D:\temp\Session-dev\Session-dev\src\Microsoft.AspNetCore.Session\SessionMiddleware.cs:line 106
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.ExceptionServices.ExceptionDispatchInfo.Throw()
   at Microsoft.AspNetCore.Session.SessionMiddleware.<Invoke>d__9.MoveNext() in D:\temp\Session-dev\Session-dev\src\Microsoft.AspNetCore.Session\SessionMiddleware.cs:line 123
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at Microsoft.AspNetCore.Diagnostics.DeveloperExceptionPageMiddleware.<Invoke>d__7.MoveNext()
Exception thrown: 'System.InvalidOperationException' in Microsoft.AspNetCore.Http.dll
Microsoft.AspNetCore.Server.Kestrel: Error: Connection id "0HL1I8OFS17SU": An unhandled exception was thrown by the application.

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, IDictionary`2 values) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\ViewFeatures\SessionStateTempDataProvider.cs:line 57
   at Microsoft.AspNetCore.Mvc.ViewFeatures.TempDataDictionary.Save() in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\ViewFeatures\TempDataDictionary.cs:line 166
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.SaveTempDataFilter.SaveTempData(IActionResult result, ITempDataDictionaryFactory factory, HttpContext httpContext) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\SaveTempDataFilter.cs:line 91
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.SaveTempDataFilter.<>c.<OnResultExecuting>b__5_0(Object state) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\SaveTempDataFilter.cs:line 57
   at Microsoft.AspNetCore.Server.Kestrel.Internal.Http.Frame.<FireOnStarting>d__178.MoveNext()
Microsoft.AspNetCore.Diagnostics.DeveloperExceptionPageMiddleware: Error: An exception was thrown attempting to display the error page.

System.ObjectDisposedException: The response has been aborted due to an unhandled application exception. ---> 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, IDictionary`2 values) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\ViewFeatures\SessionStateTempDataProvider.cs:line 57
   at Microsoft.AspNetCore.Mvc.ViewFeatures.TempDataDictionary.Save() in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\ViewFeatures\TempDataDictionary.cs:line 166
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.SaveTempDataFilter.SaveTempData(IActionResult result, ITempDataDictionaryFactory factory, HttpContext httpContext) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\SaveTempDataFilter.cs:line 91
   at Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.SaveTempDataFilter.<>c.<OnResultExecuting>b__5_0(Object state) in D:\temp\Mvc-dev\src\Microsoft.AspNetCore.Mvc.ViewFeatures\Internal\SaveTempDataFilter.cs:line 57
   at Microsoft.AspNetCore.Server.Kestrel.Internal.Http.Frame.<FireOnStarting>d__178.MoveNext()
   --- End of inner exception stack trace ---
   at Microsoft.AspNetCore.Server.Kestrel.Internal.Http.Frame.ThrowResponseAbortedException()
   at Microsoft.AspNetCore.Server.Kestrel.Internal.Http.Frame.<ProduceStartAndFireOnStartingAwaited>d__194.MoveNext()
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at Microsoft.AspNetCore.Server.Kestrel.Internal.Http.Frame.Write(ArraySegment`1 data)
   at Microsoft.AspNetCore.Server.Kestrel.Internal.Http.FrameResponseStream.Write(Byte[] buffer, Int32 offset, Int32 count)
   at System.IO.StreamWriter.Flush(Boolean flushStream, Boolean flushEncoder)
   at System.IO.StreamWriter.Write(String value)
   at Microsoft.Extensions.RazorViews.BaseView.Write(String value)
   at Microsoft.AspNetCore.Diagnostics.RazorViews.ErrorPage.<ExecuteAsync>d__6.MoveNext()
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at Microsoft.Extensions.RazorViews.BaseView.<ExecuteAsync>d__29.MoveNext()
--- End of stack trace from previous location where exception was thrown ---
   at System.Runtime.CompilerServices.TaskAwaiter.ThrowForNonSuccess(Task task)
   at System.Runtime.CompilerServices.TaskAwaiter.HandleNonSuccessAndDebuggerNotification(Task task)
   at Microsoft.AspNetCore.Diagnostics.DeveloperExceptionPageMiddleware.<Invoke>d__7.MoveNext()
Microsoft.AspNetCore.Server.Kestrel: Error: Connection id "0HL1I8OFS17SU": An unhandled exception was thrown by the application.

person Ahmar    schedule 01.01.2017    source источник
comment
И что показывает трассировка стека?   -  person Sami Kuhmonen    schedule 01.01.2017
comment
Трассировка Strack не отображается, локальный хост не работает, код ошибки 500   -  person Ahmar    schedule 01.01.2017
comment
Включите ведение журнала, чтобы увидеть трассировку стека   -  person Tratcher    schedule 01.01.2017
comment
Ни одно из приведенных ниже решений не сработало для меня. Я бы предпочел не заниматься сериализацией в JSON, но если мне придется...   -  person Daniel Jackson    schedule 05.04.2018


Ответы (3)


Наконец я понял, в чем проблема, копаясь в исходном коде. Asp.Net Core MVC в настоящее время не поддерживается сложный тип данных для TempData. Пока поддерживается только string. Именно через это исключение при сериализации данных, если мы передаем другое, чем string.

«Microsoft.AspNetCore.Mvc.ViewFeatures.Internal.TempDataSerializer» не может сериализовать объект типа «MvcSandbox.Controllers.Alert».

Я сериализую свой список в json, а затем сохраняю в TempData.

Вот как я это сделал. Я создаю метод расширения для добавления и извлечения данных из TempData.

const string Alerts = "Alerts";
public static List<Alert> GetAlert(this ITempDataDictionary tempData)
{
    CreateAlertTempData(tempData);
    return DeserializeAlerts(tempData[Alerts] as string);
}

public static void CreateAlertTempData(this ITempDataDictionary tempData)
{
    if (!tempData.ContainsKey(Alerts))
    {
        tempData[Alerts] = "";
    }
}
public static void AddAlert(this ITempDataDictionary tempData, Alert alert)
{
    if(alert == null)
    {
        throw new ArgumentNullException(nameof(alert));
    }
    var deserializeAlertList = tempData.GetAlert();
    deserializeAlertList.Add(alert);
    tempData[Alerts] = SerializeAlerts(deserializeAlertList);
}
public static string SerializeAlerts(List<Alert> tempData)
{
    return JsonConvert.SerializeObject(tempData);
}
public static List<Alert> DeserializeAlerts(string tempData)
{
    if(tempData.Length == 0)
    {
        return new List<Alert>();
    }
    return JsonConvert.DeserializeObject<List<Alert>>(tempData);
}

Добавить оповещение TempData

 tempData.AddAlert(new Alert(AlertClass, Message));

Наконец, получите «TempDatainView» и отобразите.

@if (TempData.GetAlert().Count > 0)
{
    <div class="alert-container">
        @foreach (var alert in TempData.GetAlert())
        {
            <div class="alert @alert.AlertClass alert-dismissible">
                <button type="button" class="close" data-dismiss="alert" aria-hidden="true">&times;</button>
                @alert.Message
            </div>
        }
    </div>
}
person Ahmar    schedule 02.01.2017
comment
Была такая же проблема, но это решение понравилось больше. stackoverflow.com/questions/34638823/ - person Stephen McDowell; 11.09.2018
comment
@StephenMcDowell Большое спасибо, ваше решение тоже помогло мне в моем проекте ASP.NET Core 2. очень ценю это - person PatsonLeaner; 11.11.2019

Вы настроили сессию? TempData использует сеанс за кулисами.

Project.json

"Microsoft.AspNetCore.Session": "1.1.0"

Вот файл Startup.cs. - ConfigureServices метод

public void ConfigureServices(IServiceCollection services)
{
    services.AddSession();
    services.AddMvc();
}

И Configure метод.

public void Configure(IApplicationBuilder app, IHostingEnvironment env, ILoggerFactory loggerFactory)
{
    app.UseSession();
        app.UseMvc(routes =>
        {
            routes.MapRoute(
                    name: "default",
                    template: "{controller=Home}/{action=Index}/{id?}");
    });
}

Теперь попробуйте с TempData, это сработает.

И вы можете установить среду с помощью переменной среды set ASPNETCORE_ENVIRONMENT=Development.

person Anuraj    schedule 02.01.2017
comment
Тем не менее, у вас есть проблема? Что такое трассировка стека? Если вы работаете из командной строки, какое исключение вы получаете? - person Anuraj; 02.01.2017
comment
Добавлена ​​трассировка стека @Sami-Kuhmonen. - person Ahmar; 02.01.2017
comment
@Ahmar Является ли трассировка стека перед включением сеанса? Я мог видеть исключение, связанное с сеансом - System.InvalidOperationException: Session has not been configured for this application or request. at Microsoft.AspNetCore.Http.DefaultHttpContext.get_Session() - person Anuraj; 02.01.2017
comment
Это странно, хотя я правильно настроил сеанс. - person Ahmar; 02.01.2017
comment
docs.microsoft.com/en-us/aspnet/core/ basics/app-state Помогли мне правильно настроить состояние сеанса, чтобы избавиться от моей ошибки. - person Matt Lengenfelder; 19.05.2017
comment
У меня это решение тоже не сработало :( но я проверю документы. - person Daniel Jackson; 05.04.2018

Это то, что я использую для включения данных сеанса и временных данных в Core Runtime 2.0.6.

Раздел ConfigureServices в Startup

// Add MVC service.
services.AddMvc(config =>
{
    // Configure global usage of antiforgery tokens.
    config.Filters.Add(new AutoValidateAntiforgeryTokenAttribute());
})
.AddSessionStateTempDataProvider();

// Adds the ability to use session variables.
services.AddSession(options =>
{
    options.Cookie.HttpOnly = true;
});
person Chris Go    schedule 14.12.2017
comment
Что ж, это было давно, и в Core 2.0.6 Runtime все изменилось. Хотя это близко. Я обновил комментарий тем, что использую сейчас, чтобы включить данные сеанса и временные данные. В наши дни для правильной работы сайта также требуется SSL, даже если это локальный временный сертификат для разработки в VS. TBH, не помните этот вопрос, и это немного не по теме, более того, просто заполнение информации о настройке для тех, кто в ней нуждается. - person Chris Go; 06.04.2018