利用Go语言实现轻量级OpenLdap弱密码检测工具
来源:脚本之家
时间:2022-12-29 20:11:14 308浏览 收藏
本篇文章给大家分享《利用Go语言实现轻量级OpenLdap弱密码检测工具》,覆盖了Golang的常见基础知识,其实一个语言的全部知识点一篇文章是不可能说完的,但希望通过这些问题,让读者对自己的掌握程度有一定的认识(B 数),从而弥补自己的不足,更好的掌握它。
1.Go连接LDAP服务
通过go操作的ldap,这里使用到的是go-ldap包,该包基本上实现了ldap v3的基本功能. 比如连接ldap服务、新增、删除、修改用户信息等,支持条件检索的ldap库中存储的数据信息。
2.下载
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查看说明文档
3.准备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并且检查一下服务的是否正常:
4.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
检索BaseDB
为dc=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) } }
但是这样实际就是在攻击自己的服务,这里就会产生两个问题:
- 用户越多,弱密码字典里面的密码越多,检查的次数也就越多,耗时也就越长
- 每次模拟登录,实际上就会创建一个连接,虽然连接认证失败,但是也无疑加重服务器连接创建和销毁的资源损耗,你有调优思路没?
文中关于golang的知识介绍,希望对你的学习有所帮助!若是受益匪浅,那就动动鼠标收藏这篇《利用Go语言实现轻量级OpenLdap弱密码检测工具》文章吧,也可关注golang学习网公众号了解相关技术文章。
-
425 收藏
-
151 收藏
-
438 收藏
-
280 收藏
-
181 收藏
-
371 收藏
-
236 收藏
-
416 收藏
-
- 前端进阶之JavaScript设计模式
- 设计模式是开发人员在软件开发过程中面临一般问题时的解决方案,代表了最佳的实践。本课程的主打内容包括JS常见设计模式以及具体应用场景,打造一站式知识长龙服务,适合有JS基础的同学学习。
- 立即学习 542次学习
-
- GO语言核心编程课程
- 本课程采用真实案例,全面具体可落地,从理论到实践,一步一步将GO核心编程技术、编程思想、底层实现融会贯通,使学习者贴近时代脉搏,做IT互联网时代的弄潮儿。
- 立即学习 507次学习
-
- 简单聊聊mysql8与网络通信
- 如有问题加微信:Le-studyg;在课程中,我们将首先介绍MySQL8的新特性,包括性能优化、安全增强、新数据类型等,帮助学生快速熟悉MySQL8的最新功能。接着,我们将深入解析MySQL的网络通信机制,包括协议、连接管理、数据传输等,让
- 立即学习 497次学习
-
- JavaScript正则表达式基础与实战
- 在任何一门编程语言中,正则表达式,都是一项重要的知识,它提供了高效的字符串匹配与捕获机制,可以极大的简化程序设计。
- 立即学习 487次学习
-
- 从零制作响应式网站—Grid布局
- 本系列教程将展示从零制作一个假想的网络科技公司官网,分为导航,轮播,关于我们,成功案例,服务流程,团队介绍,数据部分,公司动态,底部信息等内容区块。网站整体采用CSSGrid布局,支持响应式,有流畅过渡和展现动画。
- 立即学习 484次学习