構建NetCore應用框架之實戰篇(七):BitAdminCore框架登錄功能源碼解讀

来源:https://www.cnblogs.com/chenyinxin/archive/2018/05/10/9018689.html
-Advertisement-
Play Games

本篇承接上篇內容,如果你不小心點擊進來,建議從第一篇開始完整閱讀,文章內容繼承性連貫性。 構建NetCore應用框架之實戰篇系列 一、簡介 1、登錄功能完成後,框架的雛形已經形成,有必要進行複習。 2、本篇簡單對框架代碼進行一些解釋。同時可以簡單理解框架的規範。 二、目錄結構規範 1、直接上圖,目錄 ...


 

 

本篇承接上篇內容,如果你不小心點擊進來,建議從第一篇開始完整閱讀,文章內容繼承性連貫性。

構建NetCore應用框架之實戰篇系列

 

一、簡介


1、登錄功能完成後,框架的雛形已經形成,有必要進行複習。

2、本篇簡單對框架代碼進行一些解釋。同時可以簡單理解框架的規範。

 

二、目錄結構規範


1、直接上圖,目錄結構已經包含規範,哪類文件該放哪裡。

 

 

 

 三、Startup中的代碼功能解釋


1、想了很多辦法,最終還是覺得用註釋的方式,請細讀文中註釋。

2、見代碼

 

namespace BitAdminCoreLearn
{
    public class Startup
    {
        public Startup(IConfiguration configuration)
        {
            Configuration = configuration;
        }

        public IConfiguration Configuration { get; }

        // This method gets called by the runtime. Use this method to add services to the container.
        public void ConfigureServices(IServiceCollection services)
        {
            //註冊Session服務
            services.AddSession();

            //註冊HttpContext單例,這個HttpContextCore.Current要用到,不註冊取出來是null。
            services.TryAddSingleton<IHttpContextAccessor, HttpContextAccessor>();

            //使用登錄認證
            services.AddAuthentication(CookieAuthenticationDefaults.AuthenticationScheme)
                .AddCookie(options => options.TicketDataFormat = new TicketDataFormat<AuthenticationTicket>());
            services.AddMvc();
        }

        // This method gets called by the runtime. Use this method to configure the HTTP request pipeline.
        public void Configure(IApplicationBuilder app, IHostingEnvironment env, IServiceProvider svc)
        {
            if (env.IsDevelopment())
            {
                app.UseDeveloperExceptionPage();
            }

            //使用配置信息,就是一個靜態變數付值。
            HttpContextCore.Configuration = this.Configuration;
            HttpContextCore.ServiceProvider = svc;
            HttpContextCore.HostingEnvironment = env;

            //啟用靜態文件
            app.UseStaticFiles();

            //啟用Session緩存
            app.UseSession();


            //啟用登錄認證服務
            app.UseAuthentication();

            //使用自定義路由模版
            app.UseMvc(routes => routes.MapRoute(name: "default", template: "{controller=Account}/{action=Index}/{id?}"));
        }
    }

    /// <summary>
    /// 這個是自定義Ticket加密解密類。
    /// 功能:實現負責均衡支持,雖然Cookie模式,預設實現會與伺服器綁定,伺服器A產生的Cookies無法在伺服器B解釋。
    /// 應用:登錄組件在加解密Cookies時會調用本方法,具體見services.AddAuthentication
    /// </summary>
    /// <typeparam name="T"></typeparam>
    public class TicketDataFormat<T> : ISecureDataFormat<T> where T : AuthenticationTicket
    {
        public string Protect(T data, string purpose)
        {
            TicketSerializer _serializer = new TicketSerializer();
            byte[] userData = _serializer.Serialize(data);
            return Convert.ToBase64String(userData);
        }

        public T Unprotect(string protectedText, string purpose)
        {
            TicketSerializer _serializer = new TicketSerializer();
            byte[] bytes = Convert.FromBase64String(protectedText);
            return _serializer.Deserialize(bytes) as T;
        }

        string ISecureDataFormat<T>.Protect(T data)
        {
            TicketSerializer _serializer = new TicketSerializer();
            byte[] userData = _serializer.Serialize(data);
            return Convert.ToBase64String(userData);
        }

        T ISecureDataFormat<T>.Unprotect(string protectedText)
        {
            TicketSerializer _serializer = new TicketSerializer();
            byte[] bytes = Convert.FromBase64String(protectedText);
            return _serializer.Deserialize(bytes) as T;
        }
    }

    /// <summary>
    /// 這裡是自定義許可權過濾器。
    /// 功能:如果未登錄,返回405
    /// 應用:如果api定義了過濾,會運行以下代碼,需要登錄才能訪問
    /// 前端:前端ajax有全局設置,跳轉到登錄頁。
    /// </summary>
    [AttributeUsage(AttributeTargets.Class | AttributeTargets.Method, AllowMultiple = false)]
    public class BitAuthorizeAttribute : AuthorizeAttribute, IAuthorizationFilter
    {
        public void OnAuthorization(AuthorizationFilterContext context)
        {
            if (!SSOClient.IsLogin)
            {
                context.Result = new StatusCodeResult((int)HttpStatusCode.MethodNotAllowed);
            }
        }
    }
}

 

 

四、AccountController中的代碼功能解釋


 

 1、上代碼、見註釋。

 

namespace BitAdminCore.Controllers
{
    public class AccountController : Controller
    {
        DataContext dbContext = new DataContext();
        /// <summary>
        /// 首頁跳轉,當用戶輸入功能變數名稱時,可以路轉到登錄頁,而不會出現錯誤。
        /// </summary>
        /// <returns></returns>
        public ActionResult Index()
        {
            return Redirect("/pages/account/login.html");
        }

        /// <summary>
        /// 判斷是否登錄,前端每個頁面載入時,都會進行判斷。
        /// 登錄頁:如果已登錄,跳轉到首頁。
        /// 其它頁:如果未登錄,跳轉到登錄頁。
        /// </summary>
        /// <returns></returns>
        public JsonResult IsLogin()
        {
            return Json(Convert.ToString(SSOClient.IsLogin).ToLower());
        }

        /// <summary>
        /// 驗證碼,直接返回image類型,把路徑寫在src上即可
        /// </summary>
        /// <returns></returns>
        public ActionResult VerifyCode()
        {
            try
            {
                string code = VerificationCode.CreateCode(4);
                Bitmap image = VerificationCode.CreateImage(code);
                MemoryStream ms = new MemoryStream();
                image.Save(ms, ImageFormat.Png);
                byte[] bytes = ms.GetBuffer();
                ms.Close();

                HttpContextCore.Current.Session.Set("VerificationCode", code);
                return File(bytes, "image/jpeg");
            }
            catch (Exception ex)
            {
                LogHelper.SaveLog(ex);
                return Json(new { Code = 1, Msg = "伺服器異常,請聯繫管理員!" });
            }
        }

        /// <summary>
        /// 登錄驗證方法,你懂的
        /// </summary>
        /// <param name="account"></param>
        /// <param name="password"></param>
        /// <param name="verifyCode"></param>
        /// <returns></returns>
        public JsonResult Login(string account, string password,string verifyCode)
        {
            try
            {
                string vcode = HttpContextCore.Current.Session.Get<string>("VerificationCode");
                if (Convert.ToString(verifyCode).ToLower() != Convert.ToString(vcode).ToLower())
                    return Json(new { Code = 1, Msg = "驗證碼不正確,請重新輸入!" });

                if (!SSOClient.Validate(account, password, out Guid userId))
                    return Json(new { Code = 1, Msg = "帳號或密碼不正確,請重新輸入!" });

                HttpContextCore.Current.Session.Set("VerificationCode", string.Empty);

                SSOClient.SignIn(userId);
                return Json(new { Code = 0 });
            }
            catch (Exception ex)
            {
                LogHelper.SaveLog(ex);
                return Json(new { Code = 1, Msg = "伺服器異常,請聯繫管理員!" });
            }
        }
      
        /// <summary>
        /// 登錄後獲取當前用戶信息,首頁會用到。
        /// 這裡需要登錄後才能調用,所以加了過濾。
        /// </summary>
        /// <returns></returns>
        [BitAuthorize]
        public JsonResult GetUser()
        {
            try
            {
                SysUser user = SSOClient.User;
                //SysDepartment department = SSOClient.Department;
                return Json(new
                {
                    userCode = Convert.ToString(user.UserCode),
                    userName = Convert.ToString(user.UserName),
                    idCard = Convert.ToString(user.IdCard),
                    mobile = Convert.ToString(user.Mobile),
                    email = Convert.ToString(user.Email),
                    //departmentName = Convert.ToString(department.DepartmentName)
                });
            }
            catch (Exception ex)
            {
                LogHelper.SaveLog(ex);
                return Json(new { Code = 1, Msg = "伺服器異常,請聯繫管理員!" });
            }
        }

        /// <summary>
        /// 登出
        /// </summary>
        /// <returns></returns>
        public ActionResult SignOut()
        {
            SSOClient.SignOut();
            return Json(new { Code = 0 });
        }         
    }
}

 

 

 

五、SSOClient中的代碼功能解釋


 

1、上代碼、見註釋。

 

namespace BitAdminCore.Helpers
{
    public partial class SSOClient
    {
        public static bool IsLogin
        {
            get
            {
                if (HttpContextCore.Current.User == null || HttpContextCore.Current.User.Identity == null)
                    return false;
                return HttpContextCore.Current.User.Identity.IsAuthenticated;
            }
        }
        /// <summary>
        /// 通過用戶名密碼驗證用戶信息,如果成功,返回用戶ID,供系統預設登錄使用。
        /// </summary>
        /// <param name="sign"></param>
        /// <param name="password"></param>
        /// <param name="userid"></param>
        /// <returns></returns>
        public static bool Validate(string sign, string password,out Guid userid)
        {
            userid = Guid.Empty;
            DataContext dbContext = new DataContext();
            password = EncryptHelper.MD5(password);
            var userModel = dbContext.SysUser.FirstOrDefault(t => (t.Mobile == sign || t.Email == sign || t.UserCode == sign) && t.UserPassword == password);
            if (userModel == null)
                return false;

            userid = userModel.UserId;
            return true;
        }
        /// <summary>
        /// 通過用戶標識驗證用戶。不需要密碼,通常是第三方登錄之後,返回標識,再通過標識驗證。
        /// </summary>
        /// <param name="sign"></param>
        /// <param name="user"></param>
        /// <returns></returns>
        public static bool Validate(string sign, out SysUser user)
        {
            DataContext dbContext = new DataContext();
            user = dbContext.SysUser.FirstOrDefault(t => (t.Mobile == sign || t.Email == sign || t.UserCode == sign) );
            if (user == null)
                return false;
            
            return true;
        }
        /// <summary>
        /// 登錄函數,寫入登錄狀態和登錄信息。
        /// 這裡需要在Startup中的那些配置項(具體機制後續寫一篇文章介紹)。
        /// </summary>
        /// <param name="userid"></param>
        public static void SignIn(Guid userid)
        {
            DataContext dbContext = new DataContext();
            SysUser user = dbContext.SysUser.FirstOrDefault(x => x.UserId == userid);
            //var roles = dbContext.SysRoleUser.Where(x => x.UserId == user.UserId).ToList();

            ClaimsIdentity identity = new ClaimsIdentity(CookieAuthenticationDefaults.AuthenticationScheme);
            identity.AddClaim(new Claim(ClaimTypes.Sid, user.UserId.ToString()));
            identity.AddClaim(new Claim(ClaimTypes.Name, user.UserCode));

            //foreach (var role in roles)
            //{
            //    identity.AddClaim(new Claim(ClaimTypes.Role, role.RoleId.ToString()));
            //}
            SignOut();
            HttpContextCore.Current.SignInAsync(CookieAuthenticationDefaults.AuthenticationScheme, new ClaimsPrincipal(identity));
        }
        /// <summary>
        /// 登出,也就是清除登錄Cookies和Session
        /// </summary>
        public static void SignOut()
        {
            HttpContextCore.Current.SignOutAsync(CookieAuthenticationDefaults.AuthenticationScheme);
            HttpContextCore.Current.Session.Clear();
        }
    }
}

 

本篇代碼解讀到此,接下來怎麼接著寫,需要思考一段時間了,連續寫太多,有些思路不清晰。

 


您的分享是我們最大的動力!

-Advertisement-
Play Games
更多相關文章
  • 1.C#連接連接Access 2.C#連接SQL Server 3.C#連接Oracle 4.C#連接MySQL 5.C#連接IBM DB2 6.C#連接SyBase 7.C#連接FireBird 8.C#連接Interbase ...
  • .net自定義錯誤頁面實現:其一、通過節點system.web新增customErrors配置節點實現、其二、通過節點system.webServer新增httpErrors配置節點實現 ...
  • 近期接手一個winform 項目,雖然之前有.net 的經驗,但是對一些控制項的用法還不是很熟悉。 這段時間將會記錄一些在工作中遇到的坎坷以及對應的解決辦法,寫出來與大家分享並希望大神提出更好解決方法來促進進步。 我也會儘可能把我查找到資料的出處引出來,以此來感恩對我提供幫助的人們。 正題如下 一、關 ...
  • .net捕捉全局未處理異常的3種方式:方式一、Page_Error處理頁面級未處理異常、方式二、通過HttpModule來捕獲未處理的異常、方式三、通過Global中捕獲未處理的異常 ...
  • 官方文檔:https://cloud.tencent.com/document/product/641/12422 請求官方API及簽名的生成代碼如下: ...
  • 有些項目尤其是WinForm或者是WPF項目,針對一些工具形式的小項目,不想軟體流出去之後,懂程式的的拿到手之後一看配置文件就知道了我們資料庫的用戶名和密碼,如果外網能訪問的話,那就麻煩大了。所以這裡為了防止項目外泄之後這些信息不被別人看到,我們就需要對鏈接字元串或者其他重要信息進行加密,用的時候在 ...
  • 比如Json.NET的JObject明明實現了IEnumerable<T>,具體來說是IEnumerable<KeyValuePair<string, JToken>>,按說JObject類型的對象是可以直接調用Select、Where等linq擴展方法的,但偏偏就是不行,代碼如下: 究竟是人性的扭 ...
  • 經緯度轉換為詳細地址信息 參考文檔:http://lbs.qq.com/webservice_v1/guide-gcoder.html 首先申請key,如果使用的是服務端請求webservice API ,申請密鑰的時候要選擇“服務端”,創建成功之後設置ip白名單,否則預設全部ip都可以使用的你的k ...
一周排行
    -Advertisement-
    Play Games
  • 移動開發(一):使用.NET MAUI開發第一個安卓APP 對於工作多年的C#程式員來說,近來想嘗試開發一款安卓APP,考慮了很久最終選擇使用.NET MAUI這個微軟官方的框架來嘗試體驗開發安卓APP,畢竟是使用Visual Studio開發工具,使用起來也比較的順手,結合微軟官方的教程進行了安卓 ...
  • 前言 QuestPDF 是一個開源 .NET 庫,用於生成 PDF 文檔。使用了C# Fluent API方式可簡化開發、減少錯誤並提高工作效率。利用它可以輕鬆生成 PDF 報告、發票、導出文件等。 項目介紹 QuestPDF 是一個革命性的開源 .NET 庫,它徹底改變了我們生成 PDF 文檔的方 ...
  • 項目地址 項目後端地址: https://github.com/ZyPLJ/ZYTteeHole 項目前端頁面地址: ZyPLJ/TreeHoleVue (github.com) https://github.com/ZyPLJ/TreeHoleVue 目前項目測試訪問地址: http://tree ...
  • 話不多說,直接開乾 一.下載 1.官方鏈接下載: https://www.microsoft.com/zh-cn/sql-server/sql-server-downloads 2.在下載目錄中找到下麵這個小的安裝包 SQL2022-SSEI-Dev.exe,運行開始下載SQL server; 二. ...
  • 前言 隨著物聯網(IoT)技術的迅猛發展,MQTT(消息隊列遙測傳輸)協議憑藉其輕量級和高效性,已成為眾多物聯網應用的首選通信標準。 MQTTnet 作為一個高性能的 .NET 開源庫,為 .NET 平臺上的 MQTT 客戶端與伺服器開發提供了強大的支持。 本文將全面介紹 MQTTnet 的核心功能 ...
  • Serilog支持多種接收器用於日誌存儲,增強器用於添加屬性,LogContext管理動態屬性,支持多種輸出格式包括純文本、JSON及ExpressionTemplate。還提供了自定義格式化選項,適用於不同需求。 ...
  • 目錄簡介獲取 HTML 文檔解析 HTML 文檔測試參考文章 簡介 動態內容網站使用 JavaScript 腳本動態檢索和渲染數據,爬取信息時需要模擬瀏覽器行為,否則獲取到的源碼基本是空的。 本文使用的爬取步驟如下: 使用 Selenium 獲取渲染後的 HTML 文檔 使用 HtmlAgility ...
  • 1.前言 什麼是熱更新 游戲或者軟體更新時,無需重新下載客戶端進行安裝,而是在應用程式啟動的情況下,在內部進行資源或者代碼更新 Unity目前常用熱更新解決方案 HybridCLR,Xlua,ILRuntime等 Unity目前常用資源管理解決方案 AssetBundles,Addressable, ...
  • 本文章主要是在C# ASP.NET Core Web API框架實現向手機發送驗證碼簡訊功能。這裡我選擇是一個互億無線簡訊驗證碼平臺,其實像阿裡雲,騰訊雲上面也可以。 首先我們先去 互億無線 https://www.ihuyi.com/api/sms.html 去註冊一個賬號 註冊完成賬號後,它會送 ...
  • 通過以下方式可以高效,並保證數據同步的可靠性 1.API設計 使用RESTful設計,確保API端點明確,並使用適當的HTTP方法(如POST用於創建,PUT用於更新)。 設計清晰的請求和響應模型,以確保客戶端能夠理解預期格式。 2.數據驗證 在伺服器端進行嚴格的數據驗證,確保接收到的數據符合預期格 ...