myesn

myEsn2E9

hi
github

ABP: 携帯電話番号でログイン

まず、abp modules identity のソースコードを参考にして、自分の Web プロジェクトに以下を作成します:

  • Pages/Account/Login.cshtml
  • Pages/Account/Login.cshtml.cs

最初にLogin.cshtmlのソースコードをコピーし、次にLogin.cshtml.csファイルのコードを以下のように修正します:

using Microsoft.AspNetCore.Authentication;
using Microsoft.AspNetCore.Identity;
using Microsoft.AspNetCore.Mvc;
using Microsoft.EntityFrameworkCore;
using Microsoft.Extensions.Options;
using System.ComponentModel;
using System.ComponentModel.DataAnnotations;
using System.Diagnostics;
using System.Text.RegularExpressions;
using System.Threading.Tasks;
using Volo.Abp.Account.Settings;
using Volo.Abp.Account.Web;
using Volo.Abp.Auditing;
using Volo.Abp.Domain.Repositories;
using Volo.Abp.Identity;
using Volo.Abp.Identity.AspNetCore;
using Volo.Abp.Settings;
using Volo.Abp.Validation;
using IdentityUser = Volo.Abp.Identity.IdentityUser;

namespace MiaoXin.Web.Pages.Account
{
    /// <summary>
    /// Volo.Abp.Account.Web.Pages.Account.LoginModel
    /// </summary>
    public class LoginModel : Volo.Abp.Account.Web.Pages.Account.LoginModel
    {
        // https://github.com/abpframework/abp/issues/8068#issuecomment-799981650
        private readonly IIdentityUserRepository _identityUserRepository;

        public LoginModel(
            IIdentityUserRepository identityUserRepository,
            IAuthenticationSchemeProvider schemeProvider,
            IOptions<AbpAccountOptions> accountOptions,
            IOptions<IdentityOptions> identityOptions,
            IdentityDynamicClaimsPrincipalContributorCache identityDynamicClaimsPrincipalContributorCache
            )
            : base(schemeProvider, accountOptions, identityOptions, identityDynamicClaimsPrincipalContributorCache)
        {
            _identityUserRepository = identityUserRepository;
        }

        [BindProperty]
        public new LoginInputModel LoginInput { get; set; }

        // https://github.com/abpframework/abp/issues/7978#issuecomment-830715330
        public override async Task<IActionResult> OnPostAsync(string action)
        {
            //return base.OnPostAsync(action);
            await CheckLocalLoginAsync();

            ValidateModel();

            ExternalProviders = await GetExternalProviders();

            EnableLocalLogin = await SettingProvider.IsTrueAsync(AccountSettingNames.EnableLocalLogin);

            await ReplacePhoneNumberToUsernameOfInputIfNeeds();
            await ReplaceEmailToUsernameOfInputIfNeeds();

            await IdentityOptions.SetAsync();

            var result = await SignInManager.PasswordSignInAsync(
                LoginInput.UserNameOrEmailAddressOrPhoneNumber,
                LoginInput.Password,
                LoginInput.RememberMe,
                true
            );

            await IdentitySecurityLogManager.SaveAsync(new IdentitySecurityLogContext()
            {
                Identity = IdentitySecurityLogIdentityConsts.Identity,
                Action = result.ToIdentitySecurityLogAction(),
                UserName = LoginInput.UserNameOrEmailAddressOrPhoneNumber
            });

            if (result.RequiresTwoFactor)
            {
                return await TwoFactorLoginResultAsync();
            }

            if (result.IsLockedOut)
            {
                Alerts.Warning(L["UserLockedOutMessage"]);
                return Page();
            }

            if (result.IsNotAllowed)
            {
                Alerts.Warning(L["LoginIsNotAllowed"]);
                return Page();
            }

            if (!result.Succeeded)
            {
                Alerts.Danger(L["InvalidUserNameOrPassword"]);
                return Page();
            }

            //TODO: ログインユーザーからユーザーIDを取得する方法を見つけ、再度クエリを実行しないようにする!
            var user = await FindUserByPhoneNumberAsync(LoginInput.UserNameOrEmailAddressOrPhoneNumber) ??
                       await UserManager.FindByNameAsync(LoginInput.UserNameOrEmailAddressOrPhoneNumber) ??
                       await UserManager.FindByEmailAsync(LoginInput.UserNameOrEmailAddressOrPhoneNumber);

            Debug.Assert(user != null, nameof(user) + " != null");

            // 動的クレームキャッシュをクリアします。
            await IdentityDynamicClaimsPrincipalContributorCache.ClearAsync(user.Id, user.TenantId);

            return await RedirectSafelyAsync(ReturnUrl, ReturnUrlHash);
        }

        protected override async Task ReplaceEmailToUsernameOfInputIfNeeds()
        {
            //return base.ReplaceEmailToUsernameOfInputIfNeeds();
            if (!ValidationHelper.IsValidEmailAddress(LoginInput.UserNameOrEmailAddressOrPhoneNumber))
            {
                return;
            }

            var userByUsername = await UserManager.FindByNameAsync(LoginInput.UserNameOrEmailAddressOrPhoneNumber);
            if (userByUsername != null)
            {
                return;
            }

            var userByEmail = await UserManager.FindByEmailAsync(LoginInput.UserNameOrEmailAddressOrPhoneNumber);
            if (userByEmail == null)
            {
                return;
            }

            LoginInput.UserNameOrEmailAddressOrPhoneNumber = userByEmail.UserName;
        }
        
        protected async Task ReplacePhoneNumberToUsernameOfInputIfNeeds()
        {
            // 電話番号は11桁の数字でなければなりません
            if (!IsPhoneNumber(LoginInput.UserNameOrEmailAddressOrPhoneNumber))
            {
                return;
            }

            var userByPhoneNumber = await FindUserByPhoneNumberAsync(LoginInput.UserNameOrEmailAddressOrPhoneNumber);
            if (userByPhoneNumber == null)
            {
                return;
            }

            LoginInput.UserNameOrEmailAddressOrPhoneNumber = userByPhoneNumber.UserName;
        }

        /// <summary>
        /// 電話番号でユーザーを検索します
        /// https://stackoverflow.com/a/70344937
        /// </summary>
        /// <param name="phoneNumber"></param>
        /// <returns></returns>
        private async Task<IdentityUser?> FindUserByPhoneNumberAsync(string phoneNumber)
        {
            // https://github.com/abpframework/abp/issues/8068
            // UserManager.Users.FirstOrDefaultAsync(x => x.PhoneNumber == LoginInput.UserNameOrEmailAddressOrPhoneNumber);
            var identityUserDbSet = await _identityUserRepository.GetDbSetAsync();

            return await identityUserDbSet.FirstOrDefaultAsync(x => x.PhoneNumber == LoginInput.UserNameOrEmailAddressOrPhoneNumber);
        }


        /// <summary>
        /// 電話番号かどうかを判断します
        /// https://stackoverflow.com/a/29970789
        /// </summary>
        /// <param name="number"></param>
        /// <returns></returns>
        private static bool IsPhoneNumber(string number)
        {
            return Regex.Match(number, @"^1[0-9]{10}$").Success;
        }


        public new class LoginInputModel
        {
            [Required]
            [DynamicStringLength(typeof(IdentityUserConsts), nameof(IdentityUserConsts.MaxEmailLength))]
            [DisplayName("ユーザー名または電話番号")]
            public string UserNameOrEmailAddressOrPhoneNumber { get; set; }

            [Required]
            [DynamicStringLength(typeof(IdentityUserConsts), nameof(IdentityUserConsts.MaxPasswordLength))]
            [DataType(DataType.Password)]
            [DisableAuditing]
            [DisplayName("パスワード")]
            public string Password { get; set; }

            public bool RememberMe { get; set; }

            /// <summary>
            /// 画像認証コード
            /// </summary>
            [Required]
            [StringLength(4)]
            [DisplayName("画像認証コード")]
            public string Captcha { get; set; }
        }
    }
}
読み込み中...
文章は、創作者によって署名され、ブロックチェーンに安全に保存されています。