IT数码 购物 网址 头条 软件 日历 阅读 图书馆
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
图片批量下载器
↓批量下载图片,美女图库↓
图片自动播放器
↓图片自动播放器↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁
 
   -> 系统运维 -> 用 Go 写的轻量级 OpenLdap 弱密码检测工具 -> 正文阅读

[系统运维]用 Go 写的轻量级 OpenLdap 弱密码检测工具

Go连接LDAP服务

通过go操作的ldap,这里使用到的是go-ldap包,该包基本上实现了ldap v3的基本功能. 比如连接ldap服务、新增、删除、修改用户信息等,支持条件检索的ldap库中存储的数据信息。

下载

go get github.com/go-ldap/ldap/v3
go get github.com/wxnacy/wgo/arrays

使用go-ldap包,可以在gopkg.in/ldap.v3@v3.1.0#section-readme查看说明文档

准备LDAP环境

这里通过docker-compose运行一个临时的ldap实验环境,

version: "3"
services:
  ldap:
    image: osixia/openldap:latest
    container_name: openldap
    hostname: openldap
    restart: always
    environment:
      - "LDAP_ORGANISATION=devopsman"
      - "LDAP_DOMAIN=devopsman.cn"
      - "LDAP_BASE_DN=dc=devopsman,dc=cn"
      - "LDAP_ADMIN_PASSWORD=admin123"
    ports:
      - 389:389
      - 636:636

可以按需修改对应的环境变量信息.可以在hub.docker.com找到指定版本的镜像信息. 现在创建一下openldap并且检查一下服务的是否正常:

GO-LDAP案例实践

创建用户

在pkg.go.dev文档中查看,有一个Add方法可以完成创建用户的操作,但是需要一个AddRequest参数,而NewAddRequest方法可以返回AddRequest,于是按照此思路梳理一下。

  • 首先要建立与openldap之间的连接,验证账号是否正常,同时此账号要有创建的权限。
// LoginBind  connection ldap server and binding ldap server
func LoginBind(ldapUser, ldapPassword string) (*ldap.Conn, error) {
	l, err := ldap.DialURL(ldapURL)
	if err != nil {
		return nil, err
	}
	_, err = l.SimpleBind(&ldap.SimpleBindRequest{
		Username: fmt.Sprintf("cn=%s,dc=devopsman,dc=cn", ldapUser),
		Password: ldapPassword,
	})

	if err != nil {
		fmt.Println("ldap password is error: ", ldap.LDAPResultInvalidCredentials)
		return nil, err
	}
	fmt.Println(ldapUser,"登录成功")
	return l, nil
}
  • 其次,创建用户,需要准备用户的姓名、密码、sn、uid、gid等信息,可以创建一个struct结构
type User struct {
	username    string
	password    string
	telephone   string
	emailSuffix string
	snUsername  string
	uid         string
	gid         string
}

通过go-ldap包提供的NewAddRequest方法,可以返回新增请求

func (user *User) addUser(conn *ldap.Conn) error {
	ldaprow := ldap.NewAddRequest(fmt.Sprintf("cn=%s,dc=devopsman,dc=cn", user.username), nil)
	ldaprow.Attribute("userPassword", []string{user.password})
	ldaprow.Attribute("homeDirectory", []string{fmt.Sprintf("/home/%s", user.username)})
	ldaprow.Attribute("cn", []string{user.username})
	ldaprow.Attribute("uid", []string{user.username})
	ldaprow.Attribute("objectClass", []string{"shadowAccount", "posixAccount", "account"})
	ldaprow.Attribute("uidNumber", []string{"2201"})
	ldaprow.Attribute("gidNumber", []string{"2201"})
	ldaprow.Attribute("loginShell", []string{"/bin/bash"})

	if err := conn.Add(ldaprow); err != nil {
		return err
	}
	return nil
}

最后,我们就可以通过实例化User这个对象,完成用户的创建了:

func main() {
	con, err := LoginBind("admin", "admin123")
	fmt.Println(con.IsClosing())
	if err != nil {
		fmt.Println("V")
		fmt.Println(err)
	}
	var user User
	user.username="marionxue"
	user.password="admin123"
	user.snUsername="Marionxue"
	user.uid="1000"
	user.gid="1000"
	user.emailSuffix="@qq.com"

	if err=user.addUser(con);err!=nil{
		fmt.Println(err)
	}
	fmt.Println(user.username,"创建完成!")
}

最后运行就可以创建用户

...
/private/var/folders/jl/9zk5nj316rlg_0svp07w6btc0000gn/T/GoLand/___go_build_github_com_marionxue_go30_tools_go_openldap
admin登录成功
marionxue 创建完成!

遍历用户

遍历用户依旧需要与openLDAP建立连接,因此我们复用LoginBind函数,创建一个获取账号的函数GetEmployees

func GetEmployees(con *ldap.Conn) ([]string, error) {
	var employees []string
	sql := ldap.NewSearchRequest("dc=devopsman,dc=cn",
		ldap.ScopeWholeSubtree,
		ldap.NeverDerefAliases,
		0,
		0,
		false,
		"(objectClass=*)",
		[]string{"dn", "cn", "objectClass"},
		nil)

	cur, err := con.Search(sql)
	if err != nil {
		return nil, err
	}

	if len(cur.Entries) > 0 {
		for _, item := range cur.Entries {
			cn := item.GetAttributeValues("cn")
			for _, iCn := range cn {
				employees = append(employees, strings.Split(iCn, "[")[0])
			}
		}
		return employees, nil
	}
	return nil, nil
}

我们通过NewSearchRequest检索BaseDBdc=devopsman,dc=cn下的账号信息,最后将用户名cn打印出来

func main() {
	con, err := LoginBind("admin", "admin123")
	if err != nil {
		fmt.Println("V")
		fmt.Println(err)
	}
	employees, err := GetEmployees(con)
	if err != nil {
		fmt.Println(err)
	}
	for _, employe := range employees {
		fmt.Println(employe)

	}
}

结果就是我们前面创建的一个用户

marionxue

删除账号

同样的思路,然后创建一个删除方法delUser

// delUser 删除用户
func (user *User) delUser(conn *ldap.Conn) error{
	ldaprow := ldap.NewDelRequest(fmt.Sprintf("cn=%s,dc=devopsman,dc=cn",user.username),nil)

	if err:= conn.Del(ldaprow);err!=nil{
		return err
	}
	return nil
}

然后在main函数中调用

func main() {
	con, err := LoginBind("admin", "admin123")
	if err != nil {
		fmt.Println("V")
		fmt.Println(err)
	}
	employees, err := GetEmployees(con)
	if err != nil {
		fmt.Println(err)
	}
	var user User
	user.username="marionxue"

	if err:=user.delUser(con);err!=nil{
		fmt.Println("用户删除失败")
	}
	fmt.Println(user.username,"用户删除成功!")
}

运行结果:

admin登录成功
marionxue 用户删除成功!

弱密码检查

默认情况下,在ldap中创建用户,并没有密码复杂度的约束,因此对已存在ldap服务中使用弱密码的账号有什么好办法能获取出来吗?ldap的账号一旦创建,就看不到密码了,如果用弱密码字典模拟登录的话,是否可行呢?

创建一个检查密码的函数CheckPassword,通过逐行读取弱密码词典的数据进行的模拟登录,从而找到ldap中使用弱密码的账号:

func CheckPassword(employe string) {
	// 遍历的弱密码字典
	f, err := os.Open("~/dict.txt")
	if err != nil {
		fmt.Println("reading dict.txt error: ", err)
	}
	defer f.Close()
	scanner := bufio.NewScanner(f)
	for scanner.Scan() {
		weakpassword := scanner.Text()
		_, err := LoginBind(employe, weakpassword)
		if err == nil {
			fmt.Println(employe + " 使用的密码为: " + weakpassword)
		}
	}
	if err := scanner.Err(); err != nil {
		fmt.Println(err)
	}
	fmt.Println(employe + " check have aleardy finished. and the password is stronger well.")

}

结合前面说的遍历账号,拿到所有的账号的信息,然后模拟登录,如果命中了弱密码字典中的密码,就打印出来

func main() {
	con, err := LoginBind("admin", "admin123")
	if err != nil {
		fmt.Println("V")
		fmt.Println(err)
	}
	employees, err := GetEmployees(con)
	if err != nil {
		fmt.Println(err)
	}
	Whitelist := []string{"zhangsan","lisi"}
	for _, employe := range employees {
		fmt.Println("Starting check: ", employe)
		index := arrays.ContainsString(Whitelist, employe)
		if index == -1 {
			CheckPassword(employe)
		} else {
			fmt.Println(employe + " in whitelist. skiping...")
		}
		fmt.Println(employe)
	}
}

但是这样实际就是在攻击自己的服务,这里就会产生两个问题:

  1. 用户越多,弱密码字典里面的密码越多,检查的次数也就越多,耗时也就越长
  2. 每次模拟登录,实际上就会创建一个连接,虽然连接认证失败,但是也无疑加重服务器连接创建和销毁的资源损耗,你有调优思路没?

📌📌📌 欢迎关注【云原生生态圈】微信订阅号评论区留言来讨论。😄

  系统运维 最新文章
配置小型公司网络WLAN基本业务(AC通过三层
如何在交付运维过程中建立风险底线意识,提
快速传输大文件,怎么通过网络传大文件给对
从游戏服务端角度分析移动同步(状态同步)
MySQL使用MyCat实现分库分表
如何用DWDM射频光纤技术实现200公里外的站点
国内顺畅下载k8s.gcr.io的镜像
自动化测试appium
ctfshow ssrf
Linux操作系统学习之实用指令(Centos7/8均
上一篇文章      下一篇文章      查看所有文章
加:2021-09-30 12:19:58  更:2021-09-30 12:21:30 
 
开发: C++知识库 Java知识库 JavaScript Python PHP知识库 人工智能 区块链 大数据 移动开发 嵌入式 开发工具 数据结构与算法 开发测试 游戏开发 网络协议 系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程
数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁

360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 -2025/1/4 17:02:20-

图片自动播放器
↓图片自动播放器↓
TxT小说阅读器
↓语音阅读,小说下载,古典文学↓
一键清除垃圾
↓轻轻一点,清除系统垃圾↓
图片批量下载器
↓批量下载图片,美女图库↓
  网站联系: qq:121756557 email:121756557@qq.com  IT数码