TransactionScope с членством и ролями в одном блоке (способ использовать только одно соединение?) - PullRequest
3 голосов
/ 07 июня 2011

У меня есть вызовы к API Членства и API Ролей в одной и той же области транзакции.Я читал, что открытие более одного соединения вызывает эскалацию, требующую включения распределенных транзакций, поэтому я ищу способ открыть одно соединение и поделиться им с: Членство, роли, мои собственные вызовы.

Вот рабочий код, который вызывает нежелательную эскалацию:

public static void InsertUser(string userName, string email, string roleName, int contactId, string comment)
      {
         /*
          * Exceptions thrown here are caught in the DetailView's event handler and piped to the UI.
          */

         using(var transactionScope = new TransactionScope(TransactionScopeOption.RequiresNew))
         {
            string password = Membership.GeneratePassword(Membership.MinRequiredPasswordLength, Membership.MinRequiredNonAlphanumericCharacters);
            const string passwordQuestion = "Should you change your password question?";
            const string passwordAnswer = "yes";
            MembershipCreateStatus status;
            MembershipUser user = Membership.CreateUser(userName, password, email, passwordQuestion, passwordAnswer, true, out status);

            if(user == null)
            {
               throw new Exception(GetErrorMessage(status));
            }

            // Flesh out new user
            user.Comment = comment;
            Membership.UpdateUser(user);

            // Give user a role
            Roles.AddUserToRole(user.UserName, roleName);

            // Create bridge table record
            Guid userId = (Guid)ExceptionUtils.ThrowIfDefaultValue(user.ProviderUserKey, "ProviderUserkey is null!");
            insertIntoAspnet_Users_To_Contact(userId, contactId);

            // Send welcome email
            EmailUtils.SendWelcomeEmailFromAdmin(userName, email, password, passwordQuestion, passwordAnswer, roleName);

            transactionScope.Complete();
         }
      }

Спасибо

Ответы [ 2 ]

2 голосов
/ 07 июня 2011

Если у вас SQL2008 или выше, он может обрабатывать транзакции по нескольким соединениям, не переходя в MSDTC. Требуется, чтобы вы использовали одинаковую строку подключения для всех подключений.

Если вы используете более низкую версию SQL-сервера, я думаю, что вы проиграли. Я исследовал это несколько месяцев назад и не нашел способа справиться с этим, поэтому в итоге я пропустил транзакции и внедрил собственную обработку ошибок. У клиента был SQL2005, и обновление было невозможно.

0 голосов
/ 07 июня 2011

Вы указали TransactionScopeOption.RequiresNew, что означает, что вы хотите новую транзакцию каждый раз, когда вы передаете этот фрагмент кода, даже если уже есть подходящая окружающая.

Просто выполнитеэто:

using(var transactionScope = new TransactionScope())
...