GO语言Beego框架之WEB安全小系统(3)SQL注入与加解密

版权声明:本文为博主原创文章,未经博主允许不得转载。 https://blog.csdn.net/A657997301/article/details/82594616

SQL注入

  • 攻击原理
    SQL注入是指在后台的数据库操作中,用户的输入本应作为查询或其他sql语句中的某个参数,而攻击者通过精心构造带有sql语句的输入,能够绕过系统防护,达到将输入转换为可执行代码的目的,最终系统执行攻击者的sql代码,造成破坏。

  • 攻击影响
    SQL注入通常被认为有着极大的破坏性,主要原因在于这种攻击方法能够直接进入数据库,执行任意命令。主要的攻击影响包括以下几种:
    1) 对查询约束进行短路 (注入“or 1=1”);
    2) 获取机密信息(注入“UNION SELECT x, y FROM table”);
    3) 数据恶意删除(注入 “DELETE FROM table”);
    4) 获取更大的权限,通向系统控制(利用木马拿到webshell);
    5) 混合攻击方法(与XSS等方法组合,实现更多种的攻击方法)。

  • 防御优先级
    1) 针对SQL注入,首选方案是参数化预编译,但不正确的使用预编译依然可能导致SQL注入问题,例如开发人员使用使用拼接SQL语句的预编译;
    2) 白名单方法也是一种有效防御方法,可以对表名、字段名进行限制,白名单为第二选择;
    3) 在预编码不能使用的场景下,使用转义方法,若有预编码方法,不需要转义;
    4) 黑名单方法,最不建议使用,因为涉及到数据库不同,版本之间也有差异,很难穷举;
    5) 安全配置及纵深防御也是必要的防御手段,能够增加抗攻击性,例如关闭返回错误信息,虽然不能防止攻击,但是能极大增加攻击者的攻击时间。

添加代码

views部分

views 文件夹里新建一个File,命名为SQLController.tpl ,添加如下代码(即在body标签里改成两个表单作SQL查询对比,其他内容与index.tpl无异):

<html>
    <head>
      <title>Beego</title>
      <meta http-equiv="Content-Type" content="text/html; charset=utf-8">
      <link rel="shortcut icon" href="" type="image/x-icon">

      <link href="../static/css/index.css?ver=2" rel="stylesheet" type="text/css">
    </head>
    <body>
        <div class="postform">
            <p> SQL查询注入 </p>
            <form id="user" action="http://127.0.0.1:8080/problems/SQLInjection" method="post">
                名字:<input name="name" type="text"> <br>
                密码:<input name="password" type="text"> <br>
                <input type="submit" value="提交">
            </form>
            <br><br><br><br>
            <p> SQL查询注入防范 </p>
            <form id="user" action="http://127.0.0.1:8080/problems/SafeSQLInjection" method="post">
                名字:<input name="name" type="text"> <br>
                密码:<input name="password" type="text"> <br>
                <input type="submit" value="提交">
            </form>
        </div>
    <script src="/static/js/reload.min.js"></script>
    </body>
</html>

对了,我在index.css作了一点小修改(只是为了出来的表格布局好看点):

.description,
.postform {
    text-align: center;
    font-size: 16px;
    margin-top: 100px; // 加了这一条
}

controllers部分

controllers 文件夹里新建一个go文件,命名为SQLController.go ,添加如下代码(把main.go里的数据库注册部分挪到这了,然后分别定义了两个控制器并重写了其GetPost函数):

package controllers

import (
    _ "github.com/go-sql-driver/mysql"
    "fmt"
    "github.com/astaxie/beego"
    "github.com/astaxie/beego/orm"
        "log"
    "io/ioutil"
    "errors"
    "encoding/pem"
    "crypto/x509"
        "crypto/rsa"
    "crypto/rand"
    "encoding/base64"
)

// 对应数据库的表
type Students struct {
    Id          int     `form:"id"`
    Name        string  `form:"name"`
    Password    string  `form:"password"`
    Sex         string  `form:"sex"`
    Age         int     `form:"age"`
    Tel         string  `form:"tel"`
}

// 注册数据库
func RegisterDB() {
    // 读取数据库配置
    dbhost := beego.AppConfig.String("dbhost")
    dbport := beego.AppConfig.String("dbport")
    dbuser := beego.AppConfig.String("dbuser")
    dbpassword := beego.AppConfig.String("dbpassword")
    db := beego.AppConfig.String("db")

    // 注册mysql Driver
    orm.RegisterDriver("mysql", orm.DRMySQL)
    // 如果使用orm.QuerySeter 进行高级查询的话,必须注册模型
    orm.RegisterModel(new(Students))
    // 构造conn连接
    conn := dbuser + ":" + dbpassword + "@tcp(" + dbhost + ":" + dbport + ")/" + db + "?charset=utf8"
    // 注册数据库连接
    err := orm.RegisterDataBase("default", "mysql", conn)
    if err != nil {
        fmt.Println(err.Error())
    } else {
        fmt.Printf("数据库连接成功!%s\n", conn)
    }
}

// 引入数据模型
func init() {
    // 注册数据库
    RegisterDB()
}

// Cool Jason
// 87654321
// 注入:
// aaa
// xxx' or '1'='1
// SQL注入问题
type SQLController struct {
    beego.Controller
}

func (c *SQLController) Get() {
    c.TplName = "SQLController.tpl"
}

// 获取传过来的SQL参数
func (c *SQLController) Post() {
    orm.Debug = true;
    o := orm.NewOrm()
    o.Using("default") // 默认使用 default,你可以指定为其他数据库

    // 新建一个user
    u := &Students{}
    // 将c,即SQLController,里存储的数据转化到user格式的u变量,注意必须传入地址
    if err := c.ParseForm(u); err != nil {
        log.Fatal("ParseForm err ", err)
    }

    // SliceArg := []string{u.Name, u.Password}
    /** 【错误】直接拼接SQL语句 **/
    sqlStr := "SELECT id, name, password, sex, age, tel FROM students WHERE name = '" + u.Name + "' AND password = '" + u.Password + "'"

    var students [] Students
    num, err := o.Raw(sqlStr).QueryRows(&students)
    if err == nil {
        fmt.Println("user nums: ", num)
        fmt.Println(students)
        /*var j int64
        for j = 0; j < num; j++ {
            fmt.Printf("第%d个:\n", j+1)
            fmt.Printf("ID:%d\n", students[j].Id)
            fmt.Printf("名字:%s\n", students[j].Name)
            fmt.Printf("密码:%s\n", students[j].Password)
            fmt.Printf("性别:%s\n", students[j].Sex)
            fmt.Printf("年龄:%d\n", students[j].Age)
            fmt.Printf("电话:%s\n\n", students[j].Tel)
        }*/
    }
    c.TplName = "SQLController.tpl"
}

// SQL注入防范
type SafeSQLController struct {
    beego.Controller
}

func (c *SafeSQLController) Get() {
    c.TplName = "SQLController.tpl"
}

func (c *SafeSQLController) Post() {
    orm.Debug = true;
    o := orm.NewOrm()
    o.Using("default") // 默认使用 default,你可以指定为其他数据库

    // 新建一个user
    u := &Students{}
    // 将c,即SQLController,里存储的数据转化到user格式的u变量,注意必须传入地址
    if err := c.ParseForm(u); err != nil {
        log.Fatal("ParseForm err ", err)
    }

    // SliceArg := []string{u.Name, u.Password}
    /** 使用预编译参数化查询 **/
    sqlStr := "SELECT id, name, password, sex, age, tel FROM students WHERE name = ? AND password = ?"

    var students [] Students
    // 对SQL语句传入参数
    num, err := o.Raw(sqlStr, u.Name, u.Password).QueryRows(&students)
    if err == nil {
        fmt.Println("user nums: ", num)
        fmt.Println(students)
        /*var j int64
        for j = 0; j < num; j++ {
            fmt.Printf("第%d个:\n", j+1)
            fmt.Printf("ID:%d\n", students[j].Id)
            fmt.Printf("名字:%s\n", students[j].Name)
            fmt.Printf("密码:%s\n", students[j].Password)
            fmt.Printf("性别:%s\n", students[j].Sex)
            fmt.Printf("年龄:%d\n", students[j].Age)
            fmt.Printf("电话:%s\n\n", students[j].Tel)
        }*/
    }
    c.TplName = "SQLController.tpl"
}

MYSQL部分

由于我在Students结构体中新增了Password的字段,因此数据库部分也应该相应地添加上:
这里写图片描述

routers部分

routers/router.go 文件添加如下代码(即为上述两个控制器注册路由):

// SQL注入问题
    beego.Router("/problems/SQLInjection", &controllers.SQLController{})
    beego.Router("/problems/SafeSQLInjection", &controllers.SafeSQLController{})

这样,无论url是访问/problems/SQLInjection 还是/problems/SafeSQLInjection,两种Get请求都能正确渲染SQLController.tpl这个页面,然后当从表单发送Post请求时,一个表单会发送至SQLControllerPost函数响应并处理,而另一个表单会发送至SafeSQLControllerPost函数响应并处理。

上下两个表单的安排实现了SQL注入与SQL注入防范的对照实验作用。

进行实验

现在main.go长这样,运行:
这里写图片描述

在浏览器输入http://127.0.0.1:8080/problems/SQLInjection
这里写图片描述

对了现在的数据库存储的数据如下:
这里写图片描述

正常情况

在“SQL查询注入”的表单里填如下信息并提交:
这里写图片描述

后台显示如下:
这里写图片描述

SQL注入

在“SQL查询注入”的表单里填如下信息并提交:
这里写图片描述

后台显示如下:
这里写图片描述

可以看到,这样输入名字和密码能把数据库里全部用户的全部数据查出

SQL注入防范

在“SQL查询注入防范”的表单里填如下信息并提交:
这里写图片描述

后台显示如下:
这里写图片描述

在这个表单中,上述的注入不成功,查出的数据为空

原因分析

表单的本意设计是输入用户和密码,能在对应数据库中查找到对应的数据信息。然而在SQLControllerPost函数中,错误地直接拿用户的输入来拼接SQL语句,给攻击者提供SQL注入途径。

/** 【错误】直接拼接SQL语句 **/
    sqlStr := "SELECT id, name, password, sex, age, tel FROM students WHERE name = '" + u.Name + "' AND password = '" + u.Password + "'"

在这里,用户的输入对开发者来说就是危险的,不可信的。当攻击者输入”aaa“, “xxx' or '1'='1“,SQL注入便发生了。注入后的SQL语句为:

SELECT id, name, password, sex, age, tel FROM students WHERE name = 'aaa' AND password = 'xxx' or '1'='1'

由于 '1'='1'TRUE,因此name = 'aaa' AND password = 'xxx' or '1'='1'恒为TRUE,在SQL语法中,where后面的条件语句若为TRUE则表示该数据符合该查询,因此恒为TRUE则代表所有行记录都符合该查询,该注入也就把数据库的数据全泄露了。

而这里防范的做法,预编译参数化查询

    /** 使用预编译参数化查询 **/
    sqlStr := "SELECT id, name, password, sex, age, tel FROM students WHERE name = ? AND password = ?"
    var students [] Students
    // 对SQL语句传入参数
    num, err := o.Raw(sqlStr, u.Name, u.Password).QueryRows(&students)

使用参数化查询,通过占位符(上面代码的问号)表示需在运行时确定的参数值,这使得SQL查询的语义逻辑被预先定义,而实际的查询参数值则等到程序运行时再确定(在o.Raw函数中再传入参数)。参数化查询使得数据库能够区分SQL语句中语义逻辑和数据参数,以确保用户输入无法改变预期的SQL查询语义逻辑。

敏感信息加解密

  • 介绍
    敏感数据传输过程中要防止窃取和恶意篡改。使用安全的加密算法加密传输对象可以保护数据,防止对象被非法篡改,保持其完整性。
  • 场景
    在以下场景中,需要对对象密封和数字签名来保证数据安全:
    1) 序列化或者传输敏感数据
    2) 没有使用类似SSL传输通道
    3) 敏感数据需要长久保存,比如在硬盘驱动器上

下载OpenSSL并生成秘钥公钥

  • 下载地址:传送门
    下载后直接安装即可。
    这里写图片描述

  • static目录下新建一个key文件夹:
    这里写图片描述

  • 生成秘钥和公钥文件:
    这里写图片描述

修改代码

views部分

views/SQLController.tpl 的标签内添加如下代码(即在第二个表单后再添加一个表单,负责向数据库添加数据):

            <br><br><br><br>
            <p> SQL添加加密信息 </p>
            <form id="user" action="http://127.0.0.1:8080/problems/EncryptionDecrypt" method="post">
                  ID: <input name="id" type="text"> <br>
                名字:<input name="name" type="text"> <br>
                密码:<input name="password" type="text"> <br>
                性别:<input name="sex" type="text"> <br>
                年龄:<input name="age" type="text"> <br>
                电话:<input name="tel" type="text"> <br>
                <input type="submit" value="提交">
            </form>

controllers部分

controllers/SQLController.go 中添加如下代码(声明一个控制器并重写GetPost函数):

// SQL添加加密信息
type EncryptionDecryptController struct {
    beego.Controller
}

func (c *EncryptionDecryptController) Get() {
    c.TplName = "SQLController.tpl"
}

// 加密
func RsaEncrypt(origData []byte) ([]byte, error) {
    // 读取公钥文件
    publicKey, err := ioutil.ReadFile("static\\key\\public.pem")
    if err != nil {
        return nil, errors.New("Read public key content error.")
    }
    // 解码
    block, _ := pem.Decode(publicKey)
    if block == nil {
        return nil, errors.New("public key error")
    }
    pubInterface, err := x509.ParsePKIXPublicKey(block.Bytes)
    if err != nil {
        return nil, err
    }
    pub := pubInterface.(*rsa.PublicKey)
    res, err := rsa.EncryptPKCS1v15(rand.Reader, pub, origData)
    if err != nil {
        return nil, err
    }
    return res, err
}

// 解密
func RsaDecrypt(ciphertext []byte) ([]byte, error) {
    // 读取私钥文件
    privateKey, err := ioutil.ReadFile("static\\key\\private.pem")
    if err != nil {
        return nil, errors.New("Read private key content error.")
    }
    // 解码
    block, _ := pem.Decode(privateKey)
    if block == nil {
        return nil, errors.New("private key error!")
    }
    priv, err := x509.ParsePKCS1PrivateKey(block.Bytes)
    if err != nil {
        return nil, err
    }
    return rsa.DecryptPKCS1v15(rand.Reader, priv, ciphertext)
}

func (c *EncryptionDecryptController) Post() {
    orm.Debug = true;
    o := orm.NewOrm()
    o.Using("default") // 默认使用 default,你可以指定为其他数据库

    // 新建一个user
    u := Students{}
    // 将c,即MainController,里存储的数据转化到user格式的u变量,注意必须传入地址
    if err := c.ParseForm(&u); err != nil {
        log.Fatal("ParseForm err ", err)
    }

    // 一、MD5加密
    w := md5.New()
    // 将str写入到w中
    io.WriteString(w, u.Password)
    // w.Sum(nil)将w的hash转成[]byte格式
    md5res := fmt.Sprintf("%x", w.Sum(nil))
    fmt.Println(md5res)

    // 二、RSA加密
    signature, err := RsaEncrypt([]byte(u.Password))
    if err != nil {
        fmt.Println("sign sensitive data error.", err)
        return
    }

    // 进行base64编码
    encodeString := base64.StdEncoding.EncodeToString(signature)
    fmt.Println(encodeString)

    // 保存到u中,再把u保存到数据库中
    u.Password = encodeString
    _, error := o.Insert(&u)
    if error == nil {
        fmt.Println("Successful")
    } else {
        fmt.Println("Fail")
    }

    /**
    以下代码为查看数据库密码的操作,先进行base64解码,再进行RSA解密。
    */
    // 进行base64解码
    decodeBytes, err := base64.StdEncoding.DecodeString(encodeString)
    if err != nil {
        log.Fatalln(err)
    }
    // RSA解密看看密码是否相等
    origData, err := RsaDecrypt(decodeBytes)
    if err != nil {
        panic(err)
    }
    // 输出原密码
    fmt.Println(string(origData))

    c.TplName = "SQLController.tpl"
}

routers部分

routers/router.go 中添加如下代码(为加解密的路由器添加路由):

// SQL添加加密信息
    beego.Router("/problems/EncryptionDecrypt", &controllers.EncryptionDecryptController{})

运行

在“SQL添加加密信息”的表单里填如下信息并提交
这里写图片描述

后台显示如下:
这里写图片描述

可以看到:
1) 第一个红框为密码进行MD5加密后的密文
2) 第二个红框为用之前生成的公钥对密码进行RSA加密后生成的密文
3) 第三个红框为用之前生成的私钥对RSA密文进行解密后生成的原密码

数据库显示如下:
这里写图片描述

提交用户数据并保存数据库成功,比如后续若有验证用户身份的系统只需要将用户提交的密码用公钥进行RSA加密然后再进行base64编码数据库里保存的密码密文进行比对即可。

猜你喜欢

转载自blog.csdn.net/A657997301/article/details/82594616