乡下人产国偷v产偷v自拍,国产午夜片在线观看,婷婷成人亚洲综合国产麻豆,久久综合给合久久狠狠狠9

  • <output id="e9wm2"></output>
    <s id="e9wm2"><nobr id="e9wm2"><ins id="e9wm2"></ins></nobr></s>

    • 分享

      C#實(shí)現(xiàn)JWT無(wú)狀態(tài)驗(yàn)證的實(shí)戰(zhàn)應(yīng)用

       丹楓無(wú)跡 2021-04-07

      前言

      本文主要介紹JWT的實(shí)戰(zhàn)運(yùn)用。

      準(zhǔn)備工作

      首先我們創(chuàng)建一個(gè)Asp.Net的,包含MVC和WebApi的Web項(xiàng)目。

      然后使用Nuget搜索JWT,安裝JWT類(lèi)庫(kù),如下圖。

      設(shè)計(jì)思路

      這里我們簡(jiǎn)單的做了一個(gè)token驗(yàn)證的設(shè)計(jì),設(shè)計(jì)思路如下圖所示:

      代碼實(shí)現(xiàn)

      緩存

      首先,我們先開(kāi)發(fā)工具類(lèi),根據(jù)設(shè)計(jì)思路圖可得知,我們需要一個(gè)緩存類(lèi),用于在服務(wù)器端存儲(chǔ)token。

      編寫(xiě)緩存相關(guān)類(lèi)代碼如下:

       public class CacheHelper
       {
           public static object GetCache(string key)
           {
               return HttpRuntime.Cache[key];
           }
      
           public static T GetCache<T>(string key) where T : class
           {
               return (T)HttpRuntime.Cache[key];
           }
      
           public static bool ContainsKey(string key)
           {
               return GetCache(key) != null;
           }
      
           public static void RemoveCache(string key)
           {
               HttpRuntime.Cache.Remove(key);
           }
      
           public static void SetKeyExpire(string key, TimeSpan expire)
           {
               object value = GetCache(key);
               SetCache(key, value, expire);
           }
      
           public static void SetCache(string key, object value)
           {
               _SetCache(key, value, null, null);
           }
      
           public static void SetCache(string key, object value, TimeSpan timeout)
           {
               _SetCache(key, value, timeout, ExpireType.Absolute);
           }
      
           public static void SetCache(string key, object value, TimeSpan timeout, ExpireType expireType)
           {
               _SetCache(key, value, timeout, expireType);
           }
      
           private static void _SetCache(string key, object value, TimeSpan? timeout, ExpireType? expireType)
           {
               if (timeout == null)
                   HttpRuntime.Cache[key] = value;
               else
               {
                   if (expireType == ExpireType.Absolute)
                   {
                       DateTime endTime = DateTime.Now.AddTicks(timeout.Value.Ticks);
                       HttpRuntime.Cache.Insert(key, value, null, endTime, Cache.NoSlidingExpiration);
                   }
                   else
                   {
                       HttpRuntime.Cache.Insert(key, value, null, Cache.NoAbsoluteExpiration, timeout.Value);
                   }
               }
           }
       }
       /// <summary>
       /// 過(guò)期類(lèi)型
       /// </summary>
       public enum ExpireType
       {
           /// <summary>
           /// 絕對(duì)過(guò)期
           /// 注:即自創(chuàng)建一段時(shí)間后就過(guò)期
           /// </summary>
           Absolute,
      
           /// <summary>
           /// 相對(duì)過(guò)期
           /// 注:即該鍵未被訪問(wèn)后一段時(shí)間后過(guò)期,若此鍵一直被訪問(wèn)則過(guò)期時(shí)間自動(dòng)延長(zhǎng)
           /// </summary>
           Relative,
       }

      如上述代碼所示,我們編寫(xiě)了緩存幫助類(lèi)—CacheHelper類(lèi)。

      CacheHelper類(lèi):使用HttpRuntime的緩存,類(lèi)里實(shí)現(xiàn)緩存的增刪改,因?yàn)槭褂玫氖荋ttpRuntime,所以,如果沒(méi)有設(shè)置緩存的超時(shí)時(shí)間,則緩存的超時(shí)時(shí)間等于HttpRuntime.Cache配置的默認(rèn)超時(shí)時(shí)間。

      如果網(wǎng)站掛載在IIS里,那么,HttpRuntime.Cache配置超時(shí)時(shí)間的地方在該網(wǎng)站的應(yīng)用程序池中,如下圖:

      Jwt的幫助類(lèi)

      現(xiàn)在我們編寫(xiě)Jwt的幫助類(lèi),代碼如下:

      public class JwtHelper
      {
          //私鑰  
          public const string secret = "MIGfMA0GCSqGSIb3DQEBAQUAA4GNAmD7RTE2drj6hf3oZjJpMPZUQ1Qjb5H3K3PNwIDAQAB";
          
          /// <summary>
          /// <summary>
          /// 生成JwtToken
          /// </summary>
          /// <param name="payload">不敏感的用戶(hù)數(shù)據(jù)</param>
          /// <returns></returns>
          public static string SetJwtEncode(string username,int expiresMinutes)
          {
              //格式如下
              var payload = new Dictionary<string, object>
              {
                  { "username",username },
                  { "exp ", expiresMinutes },
                  { "domain", "" }
              };
      
              IJwtAlgorithm algorithm = new HMACSHA256Algorithm();
              IJsonSerializer serializer = new JsonNetSerializer();
              IBase64UrlEncoder urlEncoder = new JwtBase64UrlEncoder();
              IJwtEncoder encoder = new JwtEncoder(algorithm, serializer, urlEncoder);
      
              var token = encoder.Encode(payload, secret); 
              return token;
          }
         
          /// <summary>
          /// 根據(jù)jwtToken  獲取實(shí)體
          /// </summary>
          /// <param name="token">jwtToken</param>
          /// <returns></returns>
          public static IDictionary<string,object> GetJwtDecode(string token)
          {
              IJsonSerializer serializer = new JsonNetSerializer();
              IDateTimeProvider provider = new UtcDateTimeProvider();
              IJwtValidator validator = new JwtValidator(serializer, provider);
              IBase64UrlEncoder urlEncoder = new JwtBase64UrlEncoder();
              IJwtAlgorithm algorithm = new HMACSHA256Algorithm();
              IJwtDecoder decoder = new JwtDecoder(serializer, validator, urlEncoder, algorithm);
              var dicInfo = decoder.DecodeToObject(token, secret, verify: true);//token為之前生成的字符串
              return dicInfo;
          }
      }

      代碼很簡(jiǎn)單,實(shí)現(xiàn)了JWT的Code的創(chuàng)建和解析。

      注:JWT的Code雖然是密文,但它是可以被解析的,所以我們不要在Code里存儲(chǔ)重要信息,比如密碼。

      JWT的Code與解析后的內(nèi)容如下圖所示,左邊未Code,右邊未解析的內(nèi)容。

      AuthenticationHelper驗(yàn)證幫助類(lèi)

      現(xiàn)在,我們已經(jīng)可以編寫(xiě)驗(yàn)證類(lèi)了,利用剛剛已創(chuàng)建的緩存幫助類(lèi)和JWT幫助類(lèi)。

      AuthenticationHelper驗(yàn)證幫助類(lèi)代碼如下:

       public class AuthenticationHelper
       { 
           /// <summary>
           /// 默認(rèn)30分鐘
           /// </summary>
           /// <param name="username"></param>
           public static void AddUserAuth(string username)
           {
               var token = JwtHelper.SetJwtEncode(username, 30);
               CacheHelper.SetCache(username, token, new TimeSpan(TimeSpan.TicksPerHour / 2)); 
           } 
           public static void AddUserAuth(string username, TimeSpan ts)
           { 
               var token = JwtHelper.SetJwtEncode(username, ts.Minutes);
               CacheHelper.SetCache(username, token, ts);
      
           }
           public static string GetToken(string username)
           {
               var cachetoken = CacheHelper.GetCache(username);
               return cachetoken.ParseToString();  
           } 
           public static bool CheckAuth(string token)
           {
               var dicInfo = JwtHelper.GetJwtDecode(token);
               var username = dicInfo["username"];
      
               var cachetoken = CacheHelper.GetCache(username.ToString());
               if (!cachetoken.IsNullOrEmpty() && cachetoken.ToString() == token)
               {
                   return true;
               }
               else
               {
                   return false;
               } 
           }
       }

      如代碼所示,我們實(shí)現(xiàn)了驗(yàn)證token創(chuàng)建、驗(yàn)證token獲取、驗(yàn)證Token校驗(yàn)三個(gè)方法。


      到此,我們的基礎(chǔ)代碼已經(jīng)編寫(xiě)完了,下面進(jìn)入驗(yàn)證的應(yīng)用。

      Fliter

      首先,在Global.asax文件中,為我們WebApi添加一個(gè)過(guò)濾器,代碼如下:

       public class WebApiApplication : System.Web.HttpApplication
       {
           protected void Application_Start()
           {
               AreaRegistration.RegisterAllAreas();
               GlobalConfiguration.Configure(WebApiConfig.Register); 
               //webapiFilter
               System.Web.Http.GlobalConfiguration.Configuration.Filters.Add(new HttpPermissionFilter());
               System.Web.Http.GlobalConfiguration.Configuration.Filters.Add(new HttpExceptionFilter());
               //mvcFliter
               System.Web.Mvc.GlobalFilters.Filters.Add(new MvcExceptionFilter());
               System.Web.Mvc.GlobalFilters.Filters.Add(new MvcPermissionFilter()); 
               RouteConfig.RegisterRoutes(RouteTable.Routes);
               BundleConfig.RegisterBundles(BundleTable.Bundles);
           }
       }

      代碼中創(chuàng)建了四個(gè)過(guò)濾器,分別是MVC的請(qǐng)求和異常過(guò)濾器和WebApi的請(qǐng)求和異常過(guò)濾器。

      這里我們主要看WebApi的請(qǐng)求過(guò)濾器——HttpPermissionFilter。代碼如下:

      public class HttpPermissionFilter : System.Web.Http.Filters.ActionFilterAttribute
      {
          public override void OnActionExecuting(HttpActionContext actionContext)
          {
              string url ="請(qǐng)求Url" + actionContext.Request.RequestUri.ToString(); 
              var action = actionContext.ActionDescriptor.ActionName.ToLower();
              var controller = actionContext.ControllerContext.ControllerDescriptor.ControllerName.ToLower();
              if (controller != "login" && controller != "loginout")
              { 
                  //客戶(hù)端段token獲取
                  var token = actionContext.Request.Headers.Authorization != null ? actionContext.Request.Headers.Authorization.ToString() : "";
                  //服務(wù)端獲取token 與客戶(hù)端token進(jìn)行比較
                  if (!token.IsNullOrEmpty() && AuthenticationHelper.CheckAuth(token))
                  {
                      //認(rèn)證通過(guò),可進(jìn)行日志等處理 
                  }
                  else
                  { 
                      throw new Exception("Token無(wú)效");
                  } 
              }
          }
      }

      我們的HttpPermissionFilter類(lèi)繼承了System.Web.Http.Filters.ActionFilterAttribute,這樣他就可以截獲所有的WebApi請(qǐng)求了。

      然后我們重寫(xiě)了他的OnActionExecuting方法,在方法里,我們查詢(xún)到當(dāng)前請(qǐng)求的Controller的名稱(chēng),然后對(duì)其進(jìn)行了一個(gè)簡(jiǎn)單的判斷,如果是login(登錄)或loginout(登出),那我們就不對(duì)他的token進(jìn)行驗(yàn)證。如果是其他請(qǐng)求,則會(huì)從請(qǐng)求的Headers的Authorization屬性里讀取token,并使用AuthenticationHelper類(lèi)對(duì)這個(gè)token進(jìn)行正確性的驗(yàn)證。

      WebApi接口

      現(xiàn)在我們編寫(xiě)WebApi接口,編寫(xiě)一個(gè)登錄接口和一個(gè)普通請(qǐng)求接口。

      登錄接口:這里我們使用AuthenticationHelper類(lèi)創(chuàng)建一個(gè)token,并把他存儲(chǔ)到緩存中。

      然后再把token返回給調(diào)用者。

      普通接口:這里我們不做任何操作,就是簡(jiǎn)單的返回成功,因?yàn)槭欠窨梢栽L問(wèn)這個(gè)接口,已經(jīng)又Filter控制了。

      代碼如下:

      public class LoginController : ApiController
      {  
          public string Get(string username,string pwd)
          {
              AuthenticationHelper.AddUserAuth(username, new TimeSpan(TimeSpan.TicksPerMinute * 5));//5分鐘 
              string token = AuthenticationHelper.GetToken(username);
              return token;
          }  
      }
      public class RequestController : ApiController
      { 
          public string Get()
          {
              return "請(qǐng)求成功";
          }  
      }

      測(cè)試頁(yè)面

      現(xiàn)在我們編寫(xiě)測(cè)試頁(yè)面,這里我們實(shí)現(xiàn)三個(gè)按鈕,登錄、帶token訪問(wèn)Api、無(wú)token訪問(wèn)Api。

      代碼如下:

      <div> 
          <script>
              $(document).ready(function () {
                  $("#request").click(function () {
                      var token = window.localStorage.getItem('token');
                      if (token) {
      
                          $.ajax({
                              type: "GET",
                              url: "http://localhost:50525/api/Request",
                              success: function (data) {
                                  $('#con').append('<div> success:' + data + '</div>');
                                  console.log(data);
                              },
                              beforeSend: function (xhr) {
                                  //向Header頭中添加Authirization
                                  xhr.setRequestHeader("Authorization", token);
                              },
                              error: function (XMLHttpRequest, textStatus, errorThrown) { 
                                  $('#con').append('<div> error:' + errorThrown + '</div>');
                              }
                          });
                      }
                      else {
                          alert("token不存在");
                      }
                  });
                  $("#requestNotoken").click(function () {
                      var token = window.localStorage.getItem('token');
                      if (token) {
      
                          $.ajax({
                              type: "GET",
                              url: "http://localhost:50525/api/Request",
                              success: function (data) {
                                  $('#con').append('<div> success:' + data + '</div>');
                                  console.log(data);
                              },
                              error: function (XMLHttpRequest, textStatus, errorThrown) {
                                  $('#con').append('<div> error:' + errorThrown + '</div>');
                              }
                          });
                      }
                      else {
                          alert("token不存在");
                      }
                  });
                  $("#login").click(function () {
                      $.ajax({
                          type: "GET",
                          url: "http://localhost:50525/api/Login/?username=kiba&pwd=518",
                          success: function (data) {
                            
                              $('#con').append('<div> token:' + data + '</div>');
                              console.log(data);
                              window.localStorage.setItem('token', data)  
                          }
                      });
                  });
              }); 
          </script> 
          <h1>測(cè)試JWT</h1> 
          <button id="login">登錄</button>
          <button id="request">帶token訪問(wèn)Api</button>
          <button id="requestNotoken">無(wú)token訪問(wèn)Api</button>
          <div id="con"></div>
      </div>

      測(cè)試結(jié)果如下:

      如上圖所示,我們已經(jīng)成功實(shí)現(xiàn)簡(jiǎn)單的token驗(yàn)證。

      ----------------------------------------------------------------------------------------------------

      到此JWT的實(shí)戰(zhàn)應(yīng)用就已經(jīng)介紹完了。

      代碼已經(jīng)傳到Github上了,歡迎大家下載。

      Github地址: https://github.com/kiba518/JwtNet

      ----------------------------------------------------------------------------------------------------

      注:此文章為原創(chuàng),任何形式的轉(zhuǎn)載都請(qǐng)注明出處!

      https://www.cnblogs.com/kiba/p/14461836.html

        本站是提供個(gè)人知識(shí)管理的網(wǎng)絡(luò)存儲(chǔ)空間,所有內(nèi)容均由用戶(hù)發(fā)布,不代表本站觀點(diǎn)。請(qǐng)注意甄別內(nèi)容中的聯(lián)系方式、誘導(dǎo)購(gòu)買(mǎi)等信息,謹(jǐn)防詐騙。如發(fā)現(xiàn)有害或侵權(quán)內(nèi)容,請(qǐng)點(diǎn)擊一鍵舉報(bào)。
        轉(zhuǎn)藏 分享 獻(xiàn)花(0

        0條評(píng)論

        發(fā)表

        請(qǐng)遵守用戶(hù) 評(píng)論公約

        類(lèi)似文章 更多