TOTP 介绍及基于 C# 的简单实现

TOTP 介绍及基于 C# 的简单实现

640?wx_fmt=png

Intro

TOTP 是基于时间的一次性密码生成算法,它由 RFC 6238 定义。和基于事件的一次性密码生成算法不同 HOTP,TOTP 是基于时间的,它和 HOTP 具有如下关系:

  1. TOTP = HOTP(K, T)

  2. HOTP(K,C) = Truncate(HMAC-SHA-1(K,C))

其中:

  • T:T = (Current Unix time - T0) / X, T0 = 0,X = 30

  • K:客户端和服务端的共享密钥,不同的客户端的密钥各不相同。

  • HOTP:该算法请参考 RFC,也可参考 理解 HMAC-Based One-Time Password Algorithm

TOTP 算法是基于 HOTP 的,对于 HOTP 算法来说,HOTP 的输入一致时始终输出相同的值,而 TOTP 是基于时间来算出来的一个值,可以在一段时间内(官方推荐是30s)保证这个值是固定以实现,在一段时间内始终是同一个值,以此来达到基于时间的一次性密码生成算法,使用下来整体还不错,有个小问题,如果需要实现一个密码只能验证一次需要自己在业务逻辑里实现,只能自己实现,TOTP 只负责生成和验证。

C# 实现 TOTP

实现代码

  1. using System;

  2. using System.Security.Cryptography;

  3. using System.Text;


  4. namespace WeihanLi.Totp

  5. {

  6. public class Totp

  7. {

  8. private readonly OtpHashAlgorithm _hashAlgorithm;

  9. private readonly int _codeSize;


  10. public Totp() : this(OtpHashAlgorithm.SHA1, 6)

  11. {

  12. }


  13. public Totp(OtpHashAlgorithm otpHashAlgorithm, int codeSize)

  14. {

  15. _hashAlgorithm = otpHashAlgorithm;


  16. // valid input parameter

  17. if (codeSize <= 0 || codeSize > 10)

  18. {

  19. throw new ArgumentOutOfRangeException(nameof(codeSize), codeSize, "length must between 1 and 9");

  20. }

  21. _codeSize = codeSize;

  22. }


  23. private static readonly Encoding Encoding = new UTF8Encoding(false, true);


  24. public virtual string Compute(string securityToken) => Compute(Encoding.GetBytes(securityToken));


  25. public virtual string Compute(byte[] securityToken) => Compute(securityToken, GetCurrentTimeStepNumber());


  26. private string Compute(byte[] securityToken, long counter)

  27. {

  28. HMAC hmac;

  29. switch (_hashAlgorithm)

  30. {

  31. case OtpHashAlgorithm.SHA1:

  32. hmac = new HMACSHA1(securityToken);

  33. break;


  34. case OtpHashAlgorithm.SHA256:

  35. hmac = new HMACSHA256(securityToken);

  36. break;


  37. case OtpHashAlgorithm.SHA512:

  38. hmac = new HMACSHA512(securityToken);

  39. break;


  40. default:

  41. throw new ArgumentOutOfRangeException(nameof(_hashAlgorithm), _hashAlgorithm, null);

  42. }


  43. using (hmac)

  44. {

  45. var stepBytes = BitConverter.GetBytes(counter);

  46. if (BitConverter.IsLittleEndian)

  47. {

  48. Array.Reverse(stepBytes); // need BigEndian

  49. }

  50. // See https://tools.ietf.org/html/rfc4226

  51. var hashResult = hmac.ComputeHash(stepBytes);


  52. var offset = hashResult[hashResult.Length - 1] & 0xf;

  53. var p = "";

  54. for (var i = 0; i < 4; i++)

  55. {

  56. p += hashResult[offset + i].ToString("X2");

  57. }

  58. var num = Convert.ToInt64(p, 16) & 0x7FFFFFFF;


  59. //var binaryCode = (hashResult[offset] & 0x7f) << 24

  60. // | (hashResult[offset + 1] & 0xff) << 16

  61. // | (hashResult[offset + 2] & 0xff) << 8

  62. // | (hashResult[offset + 3] & 0xff);


  63. return (num % (int)Math.Pow(10, _codeSize)).ToString();

  64. }

  65. }


  66. public virtual bool Verify(string securityToken, string code) => Verify(Encoding.GetBytes(securityToken), code);


  67. public virtual bool Verify(string securityToken, string code, TimeSpan timeToleration) => Verify(Encoding.GetBytes(securityToken), code, timeToleration);


  68. public virtual bool Verify(byte[] securityToken, string code) => Verify(securityToken, code, TimeSpan.Zero);


  69. public virtual bool Verify(byte[] securityToken, string code, TimeSpan timeToleration)

  70. {

  71. var futureStep = (int)(timeToleration.TotalSeconds / 30);

  72. var step = GetCurrentTimeStepNumber();

  73. for (int i = -futureStep; i <= futureStep; i++)

  74. {

  75. if (step + i < 0)

  76. {

  77. continue;

  78. }

  79. var totp = Compute(securityToken, step + i);

  80. if (totp == code)

  81. {

  82. return true;

  83. }

  84. }

  85. return false;

  86. }


  87. private static readonly DateTime _unixEpoch = new DateTime(1970, 1, 1, 0, 0, 0, DateTimeKind.Utc);


  88. /// <summary>

  89. /// timestep

  90. /// 30s(Recommend)

  91. /// </summary>

  92. private static readonly long _timeStepTicks = TimeSpan.TicksPerSecond * 30;


  93. // More info: https://tools.ietf.org/html/rfc6238#section-4

  94. private static long GetCurrentTimeStepNumber()

  95. {

  96. var delta = DateTime.UtcNow - _unixEpoch;

  97. return delta.Ticks / _timeStepTicks;

  98. }

  99. }

  100. }

使用方式:

  1. var otp = new Totp(OtpHashAlgorithm.SHA1, 4); // 使用 SHA1算法,输出4位

  2. var secretKey = "12345678901234567890";

  3. var output = otp.Compute(secretKey);

  4. Console.WriteLine($"output: {output}");

  5. Thread.Sleep(1000 * 30);

  6. var verifyResult = otp.Verify(secretKey, output); // 使用默认的验证方式,30s内有效

  7. Console.WriteLine($"Verify result: {verifyResult}");

  8. verifyResult = otp.Verify(secretKey, output, TimeSpan.FromSeconds(60)); // 指定可容忍的时间差,60s内有效

  9. Console.WriteLine($"Verify result: {verifyResult}");

输出示例:

640?wx_fmt=png

Reference

  • https://tools.ietf.org/html/rfc4226

  • https://tools.ietf.org/html/rfc6238

  • http://wsfdl.com/algorithm/2016/04/05/%E7%90%86%E8%A7%A3HOTP.html

  • http://wsfdl.com/algorithm/2016/04/14/%E7%90%86%E8%A7%A3TOTP.html

  • https://www.cnblogs.com/voipman/p/6216328.html


本文来自互联网用户投稿,该文观点仅代表作者本人,不代表本站立场。本站仅提供信息存储空间服务,不拥有所有权,不承担相关法律责任。如若转载,请注明出处:http://www.mzph.cn/news/315944.shtml

如若内容造成侵权/违法违规/事实不符,请联系多彩编程网进行投诉反馈email:809451989@qq.com,一经查实,立即删除!

相关文章

微软携手红帽,共筑开源新未来

官宣了&#xff0c;官宣了&#xff01;微软携手红帽正式推出重要力作Global Azure 公有云第一款联合管理的 OpenShift 产品&#xff0c;业界欢腾&#xff01;自此&#xff0c;微软在拥抱云计算开源之路上&#xff0c;多了一股强劲的推动力。5月初刚刚举办的全球 Red Hat Summit…

.NET Core 3.0之深入源码理解Startup的注册及运行

开发.NET Core应用&#xff0c;直接映入眼帘的就是Startup类和Program类&#xff0c;它们是.NET Core应用程序的起点。通过使用Startup&#xff0c;可以配置化处理所有向应用程序所做的请求的管道&#xff0c;同时也可以减少.NET应用程序对单一服务器的依赖性&#xff0c;使我们…

置换群,Polya引理和burnside引理(等价类计数问题)

参考文章&#xff1a; 等价类计数问题 Burnside引理&Plya定理 Burnside引理与Polya定理 置换群和Burnside引理&#xff0c;Polya定理 概念引入&#xff1a; 离散数学应该学过置换群的相关概念&#xff0c;置换本质就是映射&#xff0c;可以理解成一个正方形绕其中心逆时针…

时隔两年的重大更新,微软发布.NET Framework 4.8

微软发布了.NET Framework 4.8&#xff0c;这是两年来.NET 框架的第一个重大更新。新版本包含多个产品领域的改进&#xff0c;其中包括 Windows 窗体、WCF、WorkFlow 和 WPF。它还在基类库&#xff08;BCL&#xff09;和通用语言运行时&#xff08;CLR&#xff09;中有一些更新…

EventStore文件存储设计

背景ENode是一个CQRSEvent Sourcing架构的开发框架&#xff0c;Event Sourcing需要持久化事件&#xff0c;事件可以持久化在DB&#xff0c;但是DB由于面向的是CRUD场景&#xff0c;是针对数据会不断修改或删除的场景&#xff0c;所以内部实现会比较复杂&#xff0c;性能也相对比…

.NET Core 如何禁止.resx文件自动生成Designer.cs

点击上方蓝字关注“汪宇杰博客”在 Visual Studio 中&#xff0c;如果我们在一个 .NET Core 工程里加入了一个资源文件&#xff08;.resx&#xff09;&#xff0c;那么你会发现有个对应的 .Designer.cs 文件被自动生成了&#xff0c;每次资源文件的内容有变化&#xff0c;这个设…

P1450 [HAOI2008]硬币购物

P1450 [HAOI2008]硬币购物 题意&#xff1a; 共有 4 种硬币。面值分别为c1,c2,c3,c4c_1,c_2,c_3,c_4c1​,c2​,c3​,c4​。 某人去商店买东西&#xff0c;去了 n 次&#xff0c;对于每次购买&#xff0c;他带了 did_idi​枚 i 种硬币&#xff0c;想购买 s 的价值的东西。请问…

.net core百万设备连接服务和硬件需求测试

随着物联网的普及&#xff0c;服务应用将面对大量物联设备处理&#xff1b;早期.NET在通讯上的处理能力一直给人的印像并不怎样&#xff0c;但net core经历过大量的优化后在各个模块的处理性能都有着比较出色的提升&#xff0c;针对网络方向的处理模块也有着显著的提升。以下主…

字符串匹配(多模式匹配篇)

字符串匹配&#xff08;多模式匹配篇&#xff09;摘要&#xff1a;问题的提出&#xff1a;众所周知&#xff0c;KMP算法在O&#xff08;n&#xff09;的时间中solve单模式串匹配问题。但怎样solve多模式串匹配问题呢&#xff1f;Solve&#xff1a;本文用简要记叙了使用trie树&a…

.net core编写转发服务

我有个小伙伴问我&#xff0c;他需要写一个转发服务的他有很多功能要通过他的服务转发~技术栈又不一定asp.net core&#xff0c;我就想起泥水老前辈的BeetleX.FastHttpApi中午午休&#xff0c;折腾了一会儿前辈&#xff0c;问清楚了FastHttpApi如何配置控制器依赖注入和控制器的…

数据结构(终极线段树篇)

数据结构&#xff08;终极线段树篇&#xff09; 摘要&#xff1a; 问题的提出&#xff1a;如何解决多样化的区间操作问题&#xff1f; solve&#xff1a;线段树&#xff01;&#xff01;&#xff01; 关键字&#xff1a; 线段树&#xff0c;可持久化线段树&#xff0c;权值线段…

.NET Core 3.0之深入源码理解Configuration(一)

微软在.NET Core里设计出了全新的配置体系&#xff0c;并以非常灵活、可扩展的方式实现。从其源码来看&#xff0c;其运行机制大致是&#xff0c;根据其Source&#xff0c;创建一个Builder实例&#xff0c;并会向其添加Provider&#xff0c;在我们使用配置信息的时候&#xff0…

摊还分析

摊还分析 1何为摊还分析&#xff1f; 摊还分析主要求解数据结构维护序列执行的所有操作的平均时间&#xff0c;来评价操作的代价&#xff0c;从而保证最坏情况下每个操作的平均性能。 2聚合分析 2.1何为聚合分析&#xff1f; 若长度为n的操作序列最坏情况下所花费时间为T(…

Bigraph Extension

Bigraph Extension 题意&#xff1a; 有2n个点&#xff0c;n为偶数&#xff0c;n个点属于集合A&#xff0c;n个点属于集合B。起初在途中有m个无向边&#xff0c;边的两侧端点分别在两个集合里&#xff0c;任何两个边都没有公共交点。 现在你可以执行任意次操作&#xff1a; 在…

微服务划分的姿势

我们知道微服务是一种理念&#xff0c;没有确切的定义和边界&#xff0c;好比设计原则&#xff0c;是属于抽象的概念。在定义不明确的情况下谈划分也是一种各说各话&#xff0c;具体问题需要具体分析&#xff0c;所以这篇文章谈到的划分也不是绝对标准&#xff0c;仅供参考。有…

点(树)分治

0.引言 对于树上问题&#xff0c;有许多特殊的求解方法&#xff0c;如&#xff1a;树链剖分。点分治算法也是其中之一&#xff0c;常用于解决树上路径问题。 1.0.问题的引入 给定一棵树&#xff0c;求这棵树的直径&#xff08;树上最长链长度&#xff0c;n<10^5&#xff…

斜率优化Convex Hull Trick

斜率优化 一、简单DP 首先从一道简单题引入。 [IOI2002]任务安排 Description N个任务排成一个序列在一台机器上等待完成&#xff08;顺序不得改变&#xff09;&#xff0c;这N个任务被分成若干批&#xff0c;每批包含相邻的若干任务。从时刻0开始&#xff0c;这些任务被分…

分布式部署携程Apollo构建配置中心

一、开场白在系统设计里我们有很多配置希望独立于系统之外&#xff0c;而又能够被系统实时读取。但是在传统的系统设计里&#xff0c;配置信息通常是耦合在系统内的&#xff0c;比如.net里通常会放在App.config或者web.config里&#xff0c;.net core则是appsettings.json里&am…

[COCI 2017-2018-2]-San

[COCI 2017-2018-2]-San san(1s64M) 游戏世界中有N个楼从左到右排列&#xff0c;从左到右编号为1到N&#xff0c;第i幢楼的高度为Hi,楼上的金币数为Gi,游戏可以从任意一个楼开始且包涵几步。每一步玩家可以从当前位置向右跳&#xff08;可以跳过一些楼&#xff09;但必须跳到…

领域模型架构 eShopOnWeb项目分析 上

一.概述本篇继续探讨web应用架构&#xff0c;讲基于DDD风格下最初的领域模型架构&#xff0c;不同于DDD风格下CQRS架构&#xff0c;二者架构主要区别是领域层的变化。 架构的演变是从领域模型到CQRS, 一开始DDD是用领域模型的分层架构&#xff0c;用单一的领域模型处理业务逻辑…