Подтвердить что ты не робот

Внедрение пользовательской идентификации и IPrincipal в MVC

У меня есть базовое бета-приложение MVC 2, в котором я пытаюсь реализовать пользовательские классы Identity и Principal.

Я создал свои классы, которые реализуют интерфейсы IIdentity и IPrincipal, создавал их, а затем назначал объект CustomPrincipal для моего Context.User в Application_AuthenticateRequest для Global.asax.

Это все успешно, и объекты выглядят хорошо. Когда я начинаю отображать Представления, страницы теперь терпят неудачу. Первый сбой находится в представлении LogoOnUserControl по умолчанию в следующей строке кода:

 [ <%= Html.ActionLink("Log Off", "LogOff", "Account") %> ]

Если я вытащил это, он завершится неудачей на другой строке кода "Html.ActionLink".

Получаемая ошибка:

Исключение типа 'System.Runtime.Serialization.SerializationException' произошел в WebDev.WebHost40.dll, но не обрабатывался в коде пользователя

Дополнительная информация: Тип не разрешено для члена "Model.Entities.UserIdentity, модель, Версия = 1.0.0.0, Культура = нейтральная, PublicKeyToken = NULL".

Есть ли какие-то дополнительные свойства, которые мне нужно реализовать в моей Identity, чтобы использовать пользовательский идентификатор в MVC? Я попытался реализовать [Serializable()] в классе Identity, но, похоже, это не повлияло.

UPDATE: Я пробовал 3-4 альтернативных способа реализовать это, но все еще не удается с той же ошибкой. Если я использую классы GenericIdentity/GenericPrincipal напрямую, это не ошибка.

GenericIdentity ident = new GenericIdentity("jzxcvcx");
GenericPrincipal princ = new GenericPrincipal(ident, null);
Context.User = princ;

Но это никуда не денется, так как я пытаюсь использовать CustomIdentity для хранения нескольких свойств. Если я реализую интерфейсы IIdentity/IPrincipal или наследую GenericIdentity/GenericPrincipal для моего CustomIdentity/CustomPrincipal, он терпит неудачу с исходной ошибкой выше.

4b9b3361

Ответ 1

Я понял это с небольшой помощью из Интернета. Трюк в том, что вы должны реализовать интерфейс ISerializable в своем классе, который реализует IIdentity. Надеюсь, это поможет сэкономить еще какое-то время:)

Объявление класса:

[Serializable]
    public class ForumUserIdentity : IIdentity, ISerializable

Реализация для ISerializable:

#region ISerializable Members

        public void GetObjectData(SerializationInfo info, StreamingContext context)
        {
            if (context.State == StreamingContextStates.CrossAppDomain)
            {
                GenericIdentity gIdent = new GenericIdentity(this.Name, this.AuthenticationType);
                info.SetType(gIdent.GetType());

                System.Reflection.MemberInfo[] serializableMembers;
                object[] serializableValues;

                serializableMembers = FormatterServices.GetSerializableMembers(gIdent.GetType());
                serializableValues = FormatterServices.GetObjectData(gIdent, serializableMembers);

                for (int i = 0; i < serializableMembers.Length; i++)
                {
                    info.AddValue(serializableMembers[i].Name, serializableValues[i]);
                }
            }
            else
            {
                throw new InvalidOperationException("Serialization not supported");
            }
        }

        #endregion

Вот ссылка к статье, в которой более подробно описывается "Feature"

Ответ 2

У меня была та же проблема. Я решил это, переместив мое основное создание из MvcApplication_AuthenticateRequest в MvcApplication_PostAuthenticateRequest. Я не знаю почему/как, но он решил проблему:)

        void MvcApplication_PostAuthenticateRequest(object sender, EventArgs e)
    {
        HttpCookie authCookie = HttpContext.Current.Request.Cookies[FormsAuthentication.FormsCookieName];
        if (authCookie != null)
        {
            string encTicket = authCookie.Value;
            if (!String.IsNullOrEmpty(encTicket))
            {
                FormsAuthenticationTicket ticket = FormsAuthentication.Decrypt(encTicket);
                BiamedIdentity id = new BiamedIdentity(ticket);
                GenericPrincipal prin = new GenericPrincipal(id, null);
                HttpContext.Current.User = prin;
            }
        }
    }

Ответ 3

Со мной это работает, когда я наследую свой класс Identity от MarshalByRefObject.

Также обратите внимание: при использовании Linq-to-Sql проблем не было. Я переключился на Entity-Framework и bang, я получил вышеприведенное сообщение.