给你的web应用上MFA

什么是MFA

虚拟MFAMulti-FactorAuthentication多因素认证,是需要一部智能手机并安装虚拟MFA应用程序即可在账户上加上一层安全保险

常见的MFA应用程序,

手机类型 MFA应用程序
iPhone Google Authenticator
Android 谷歌动态口令 ,身份宝,洋葱
Windows Phone 身份验证器
Blackberry Google Authenticator

MFA应用程序的算法TOTP

TOTP(Time-Based One-Time Password基于时间的一次性密码),其核心内容包括以下三点:

  1. 共同密钥
  2. 共同时间
  3. 共同签署方法

什么系统需要用到

想要安全,什么系统都可以用

以上基本上从网络搜集的资料,接下来看看MFA加到web应用的效果

同样支持自动扫描,手动添加,二选一添加就可以了,添加之后,输入连续2组不同的动态密码,确认后,系统就会开启MFA认证,这样该用户登录系统就会多一次认证

需要保证:安装MFA应用程序手机的时间需要与服务器一致,不然动态密码无法一致

image.png
image.png
image.png
image.png

代码片段

//二维码图片生成
package controllers

import (
    "github.com/boombuler/barcode"
    "github.com/boombuler/barcode/qr"
    "image"
    "image/png"
    "log"
    "os"
)

func WritePng(filename string, img image.Image) string {
    file, err := os.Create("./static/images/" + filename)
    if err != nil {
        log.Fatal(err)
    }
    err = png.Encode(file, img)
    if err != nil {
        log.Fatal(err)
    }
    file.Close()
    log.Println(file.Name())
    return "/static/images/" + filename
}

func GetQrCode(str string, filename string) string {
    code, err := qr.Encode(str, qr.L, qr.Unicode)
    if err != nil {
        log.Fatal(err)
    }
    log.Println("Encoded data:", code.Content())
    if str != code.Content() {
        log.Fatal("data differs")
    }
    code, err = barcode.Scale(code, 200, 200)
    if err != nil {
        log.Fatal(err)
    }
    return WritePng(filename+".png", code)
}
//TOTP相关
package controllers

import (
    "crypto/hmac"
    "crypto/sha1"
    "encoding/base32"
    "fmt"
    "math/rand"
    "strings"
    "time"
)

const (
    length = 16
)

func toBytes(value int64) []byte {
    var result []byte
    mask := int64(0xFF)
    shifts := [8]uint16{56, 48, 40, 32, 24, 16, 8, 0}
    for _, shift := range shifts {
        result = append(result, byte((value>>shift)&mask))
    }
    return result
}

func toUint32(bytes []byte) uint32 {
    return (uint32(bytes[0]) << 24) + (uint32(bytes[1]) << 16) +
        (uint32(bytes[2]) << 8) + uint32(bytes[3])
}

//密钥生成
func GetSecret() string {
    str := "234567abcdefghijklmnopqrstuvwxyz"
    bytes := []byte(str)
    result := []byte{}
    r := rand.New(rand.NewSource(time.Now().UnixNano()))
    for i := 0; i < length; i++ {
        result = append(result, bytes[r.Intn(len(bytes))])
    }
    return string(result)
}

func oneTimePassword(key []byte, value []byte) uint32 {
    // sign the value using HMAC-SHA1
    hmacSha1 := hmac.New(sha1.New, key)
    hmacSha1.Write(value)
    hash := hmacSha1.Sum(nil)

    // We're going to use a subset of the generated hash.
    // Using the last nibble (half-byte) to choose the index to start from.
    // This number is always appropriate as it's maximum decimal 15, the hash will
    // have the maximum index 19 (20 bytes of SHA1) and we need 4 bytes.
    offset := hash[len(hash)-1] & 0x0F

    // get a 32-bit (4-byte) chunk from the hash starting at offset
    hashParts := hash[offset : offset+4]

    // ignore the most significant bit as per RFC 4226
    hashParts[0] = hashParts[0] & 0x7F

    number := toUint32(hashParts)

    // size to 6 digits
    // one million is the first number with 7 digits so the remainder
    // of the division will always return < 7 digits
    pwd := number % 1000000

    return pwd
}

//动态6位密码
func Totp(secret string, ago int64) string {
    keynospaces := strings.Replace(secret, " ", "", -1)
    keynospacesupper := strings.ToUpper(keynospaces)
    key, err := base32.StdEncoding.DecodeString(keynospacesupper)
    if err != nil {
        fmt.Println(err)
    }
    epochsecond := time.Now().Unix()
    epochsecond -= ago //ago可以为0,也可以为30,这样可以应付2组密码的情况
    pwd := oneTimePassword(key, toBytes(epochsecond/30))

    secondsRemaining := 30 - (epochsecond % 30)
    //fmt.Sprintf("%06d (%d second(s) remaining)\n", pwd, secondsRemaining)
    fmt.Println(secondsRemaining) //这个secondsRemaining没有用到,只是打印下
    return fmt.Sprintf("%06d", pwd)

}

//二维码包含内容
func Getotpauth(name, secret, issuer string) string {
    otpauth := "otpauth://totp/" + "testwd" + ":" + name + "?secret=" + secret + "&issuer=" + issuer
    return otpauth
}

//MFA开启后,登录后验证MFA页面
func (self *UserController) MfaVerifyPage() {
    self.TplName = "user/mfapwd.html"
}
//验证MFA动态密码是否正确
func (self *UserController) MfaVerify() {
    email := self.GetSession("email")
    code := self.Input().Get("code")
    if email == nil {
        self.Data["islogin"] = false
        self.Ctx.Redirect(302, "/")
    } else {
        _, user := models.FindUserByEmail(email.(string))
        if code == Totp(user.Secret, 0) {
            self.SetSession("uid", user.Id)
            msg := map[string]interface{}{"code": 0, "msg": "success"}
            self.Data["json"] = &msg
            self.ServeJSON()
        } else {
            msg := map[string]interface{}{"code": 1, "msg": "invalid code"}
            self.Data["json"] = &msg
            self.ServeJSON()
        }
    }

}

func (self *UserController) MFAPage() {
    uid := self.GetSession("uid")
    if uid == nil {
        self.Data["islogin"] = false
        self.Ctx.Redirect(302, "/")
    } else {
        user := models.FindUserDetialById(uid.(int))
                //随机密钥
        secret := GetSecret()
        qrdata := Getotpauth(user.Nickname, secret, "测试村")
        //把邮箱md5加密成字符串,当作二维码文件名,这样文件名应该是每个用户只有一个
        //不会因为用户多次刷新而生成不必要的文件,以防造成空间浪费
        md5ctx := md5.New()
        md5ctx.Write([]byte(user.Email))
        filename := fmt.Sprintf("%x", md5ctx.Sum(nil))
        fmt.Println(filename, "pppp")
        self.Data["qrimg"] = GetQrCode(qrdata, filename)
        if user.Mfa != true {
            user.Secret = secret
            models.UpdateUser(&user)
            self.Data["secret"] = secret
        } else {
                      //开启MFA,密钥对用户不可见
            self.Data["secret"] = "****************"
        }

        self.Data["islogin"] = true
        self.Data["userinfo"] = user
        self.Data["IsMFA"] = true
        self.TplName = "user/mfa.html"
    }
}

//开启MFA
func (self *UserController) SetMfa() {
    code1, code2 := self.Input().Get("code1"), self.Input().Get("code2")
    uid := self.GetSession("uid")
    if uid == nil {
        self.Data["islogin"] = false
        self.Ctx.Redirect(302, "/")
    } else {
        user := models.FindUserDetialById(uid.(int))
        fmt.Println(code1, code2)
        fmt.Println(Totp(user.Secret, 30), Totp(user.Secret, 0))
        if code1 == Totp(user.Secret, 30) && code2 == Totp(user.Secret, 0) {
            user.Mfa = true
            models.UpdateUser(&user)
            msg := map[string]interface{}{"code": 0, "msg": "success"}
            self.Data["json"] = &msg
            self.ServeJSON()

        } else {
            msg := map[string]interface{}{"code": 1, "msg": "无效密码"}
            self.Data["json"] = &msg
            self.ServeJSON()
        }
    }
}

//关闭MFA,关闭也需要验证动态密码
func (self *UserController) CloseMfa() {
    code1, code2 := self.Input().Get("code1"), self.Input().Get("code2")
    uid := self.GetSession("uid")
    if uid == nil {
        self.Data["islogin"] = false
        self.Ctx.Redirect(302, "/")
    } else {
        user := models.FindUserDetialById(uid.(int))
        fmt.Println(code1, code2)
        fmt.Println(Totp(user.Secret, 30), Totp(user.Secret, 0))
        if code1 == Totp(user.Secret, 30) && code2 == Totp(user.Secret, 0) {
            user.Mfa = false
            models.UpdateUser(&user)
            msg := map[string]interface{}{"code": 0, "msg": "success"}
            self.Data["json"] = &msg
            self.ServeJSON()

        } else {
            msg := map[string]interface{}{"code": 1, "msg": "无效密码"}
            self.Data["json"] = &msg
            self.ServeJSON()
        }
    }
}
最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 203,937评论 6 478
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 85,503评论 2 381
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 150,712评论 0 337
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 54,668评论 1 276
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 63,677评论 5 366
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 48,601评论 1 281
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 37,975评论 3 396
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 36,637评论 0 258
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 40,881评论 1 298
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 35,621评论 2 321
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 37,710评论 1 329
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 33,387评论 4 319
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 38,971评论 3 307
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 29,947评论 0 19
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 31,189评论 1 260
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 44,805评论 2 349
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 42,449评论 2 342

推荐阅读更多精彩内容