自拍偷在线精品自拍偷,亚洲欧美中文日韩v在线观看不卡

用 Go 寫(xiě)的輕量級(jí) OpenLdap 弱密碼檢測(cè)工具

開(kāi)發(fā) 后端
通過(guò)go操作的ldap,這里使用到的是go-ldap[1]包,該包基本上實(shí)現(xiàn)了ldap v3的基本功能. 比如連接ldap服務(wù)、新增、刪除、修改用戶信息等,支持條件檢索的ldap庫(kù)中存儲(chǔ)的數(shù)據(jù)信息。

[[426727]]

1 Go連接LDAP服務(wù)

通過(guò)go操作的ldap,這里使用到的是go-ldap[1]包,該包基本上實(shí)現(xiàn)了ldap v3的基本功能. 比如連接ldap服務(wù)、新增、刪除、修改用戶信息等,支持條件檢索的ldap庫(kù)中存儲(chǔ)的數(shù)據(jù)信息。

2 下載

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

使用go-ldap包,可以在gopkg.in/ldap.v3@v3.1.0#section-readme[2]查看說(shuō)明文檔

3 準(zhǔn)備LDAP環(huán)境

這里通過(guò)docker-compose運(yùn)行一個(gè)臨時(shí)的ldap實(shí)驗(yàn)環(huán)境,

  1. version: "3" 
  2. services: 
  3.   ldap: 
  4.     image: osixia/openldap:latest 
  5.     container_name: openldap 
  6.     hostname: openldap 
  7.     restart: always 
  8.     environment: 
  9.       - "LDAP_ORGANISATION=devopsman" 
  10.       - "LDAP_DOMAIN=devopsman.cn" 
  11.       - "LDAP_BASE_DN=dc=devopsman,dc=cn" 
  12.       - "LDAP_ADMIN_PASSWORD=admin123" 
  13.     ports: 
  14.       - 389:389 
  15.       - 636:636 

可以按需修改對(duì)應(yīng)的環(huán)境變量信息.可以在hub.docker.com[3]找到指定版本的鏡像信息. 現(xiàn)在創(chuàng)建一下openldap并且檢查一下服務(wù)的是否正常:

4 GO-LDAP案例實(shí)踐

創(chuàng)建用戶

在pkg.go.dev文檔中查看,有一個(gè)Add方法可以完成創(chuàng)建用戶的操作,但是需要一個(gè)AddRequest參數(shù),而NewAddRequest方法可以返回AddRequest,于是按照此思路梳理一下。

首先要建立與openldap之間的連接,驗(yàn)證賬號(hào)是否正常,同時(shí)此賬號(hào)要有創(chuàng)建的權(quán)限。

  1. // LoginBind  connection ldap server and binding ldap server 
  2. func LoginBind(ldapUser, ldapPassword string) (*ldap.Conn, error) { 
  3.  l, err := ldap.DialURL(ldapURL) 
  4.  if err != nil { 
  5.   return nil, err 
  6.  } 
  7.  _, err = l.SimpleBind(&ldap.SimpleBindRequest{ 
  8.   Username: fmt.Sprintf("cn=%s,dc=devopsman,dc=cn", ldapUser), 
  9.   Password: ldapPassword, 
  10.  }) 
  11.  
  12.  if err != nil { 
  13.   fmt.Println("ldap password is error: ", ldap.LDAPResultInvalidCredentials) 
  14.   return nil, err 
  15.  } 
  16.  fmt.Println(ldapUser,"登錄成功"
  17.  return l, nil 

其次,創(chuàng)建用戶,需要準(zhǔn)備用戶的姓名、密碼、sn、uid、gid等信息,可以創(chuàng)建一個(gè)struct結(jié)構(gòu)

  1. type User struct { 
  2.  username    string 
  3.  password    string 
  4.  telephone   string 
  5.  emailSuffix string 
  6.  snUsername  string 
  7.  uid         string 
  8.  gid         string 

通過(guò)go-ldap包提供的NewAddRequest方法,可以返回新增請(qǐng)求

  1. func (user *User) addUser(conn *ldap.Conn) error { 
  2.  ldaprow := ldap.NewAddRequest(fmt.Sprintf("cn=%s,dc=devopsman,dc=cn"user.username), nil) 
  3.  ldaprow.Attribute("userPassword", []string{user.password}) 
  4.  ldaprow.Attribute("homeDirectory", []string{fmt.Sprintf("/home/%s"user.username)}) 
  5.  ldaprow.Attribute("cn", []string{user.username}) 
  6.  ldaprow.Attribute("uid", []string{user.username}) 
  7.  ldaprow.Attribute("objectClass", []string{"shadowAccount""posixAccount""account"}) 
  8.  ldaprow.Attribute("uidNumber", []string{"2201"}) 
  9.  ldaprow.Attribute("gidNumber", []string{"2201"}) 
  10.  ldaprow.Attribute("loginShell", []string{"/bin/bash"}) 
  11.  
  12.  if err := conn.Add(ldaprow); err != nil { 
  13.   return err 
  14.  } 
  15.  return nil 

最后,我們就可以通過(guò)實(shí)例化User這個(gè)對(duì)象,完成用戶的創(chuàng)建了:

  1. func main() { 
  2.  con, err := LoginBind("admin""admin123"
  3.  fmt.Println(con.IsClosing()) 
  4.  if err != nil { 
  5.   fmt.Println("V"
  6.   fmt.Println(err) 
  7.  } 
  8.  var user User 
  9.  user.username="marionxue" 
  10.  user.password="admin123" 
  11.  user.snUsername="Marionxue" 
  12.  user.uid="1000" 
  13.  user.gid="1000" 
  14.  user.emailSuffix="@qq.com" 
  15.  
  16.  if err=user.addUser(con);err!=nil{ 
  17.   fmt.Println(err) 
  18.  } 
  19.  fmt.Println(user.username,"創(chuàng)建完成!"

最后運(yùn)行就可以創(chuàng)建用戶

  1. ... 
  2. /private/var/folders/jl/9zk5nj316rlg_0svp07w6btc0000gn/T/GoLand/___go_build_github_com_marionxue_go30_tools_go_openldap 
  3. admin登錄成功 
  4. marionxue 創(chuàng)建完成! 

遍歷用戶

遍歷用戶依舊需要與openLDAP建立連接,因此我們復(fù)用LoginBind函數(shù),創(chuàng)建一個(gè)獲取賬號(hào)的函數(shù)GetEmployees

  1. func GetEmployees(con *ldap.Conn) ([]string, error) { 
  2.  var employees []string 
  3.  sql := ldap.NewSearchRequest("dc=devopsman,dc=cn"
  4.   ldap.ScopeWholeSubtree, 
  5.   ldap.NeverDerefAliases, 
  6.   0, 
  7.   0, 
  8.   false
  9.   "(objectClass=*)"
  10.   []string{"dn""cn""objectClass"}, 
  11.   nil) 
  12.  
  13.  cur, err := con.Search(sql) 
  14.  if err != nil { 
  15.   return nil, err 
  16.  } 
  17.  
  18.  if len(cur.Entries) > 0 { 
  19.   for _, item := range cur.Entries { 
  20.    cn := item.GetAttributeValues("cn"
  21.    for _, iCn := range cn { 
  22.     employees = append(employees, strings.Split(iCn, "[")[0]) 
  23.    } 
  24.   } 
  25.   return employees, nil 
  26.  } 
  27.  return nil, nil 

我們通過(guò)NewSearchRequest檢索BaseDB為dc=devopsman,dc=cn下的賬號(hào)信息,最后將用戶名cn打印出來(lái)

  1. func main() { 
  2.  con, err := LoginBind("admin""admin123"
  3.  if err != nil { 
  4.   fmt.Println("V"
  5.   fmt.Println(err) 
  6.  } 
  7.  employees, err := GetEmployees(con) 
  8.  if err != nil { 
  9.   fmt.Println(err) 
  10.  } 
  11.  for _, employe := range employees { 
  12.   fmt.Println(employe) 
  13.  
  14.  } 

結(jié)果就是我們前面創(chuàng)建的一個(gè)用戶

  1. marionxue 

刪除賬號(hào)

同樣的思路,然后創(chuàng)建一個(gè)刪除方法delUser

  1. // delUser 刪除用戶 
  2. func (user *User) delUser(conn *ldap.Conn) error{ 
  3.  ldaprow := ldap.NewDelRequest(fmt.Sprintf("cn=%s,dc=devopsman,dc=cn",user.username),nil) 
  4.  
  5.  if err:= conn.Del(ldaprow);err!=nil{ 
  6.   return err 
  7.  } 
  8.  return nil 

然后在main函數(shù)中調(diào)用

  1. func main() { 
  2.  con, err := LoginBind("admin""admin123"
  3.  if err != nil { 
  4.   fmt.Println("V"
  5.   fmt.Println(err) 
  6.  } 
  7.  employees, err := GetEmployees(con) 
  8.  if err != nil { 
  9.   fmt.Println(err) 
  10.  } 
  11.  var user User 
  12.  user.username="marionxue" 
  13.  
  14.  if err:=user.delUser(con);err!=nil{ 
  15.   fmt.Println("用戶刪除失敗"
  16.  } 
  17.  fmt.Println(user.username,"用戶刪除成功!"

運(yùn)行結(jié)果:

  1. admin登錄成功 
  2. marionxue 用戶刪除成功! 

弱密碼檢查

默認(rèn)情況下,在ldap中創(chuàng)建用戶,并沒(méi)有密碼復(fù)雜度的約束,因此對(duì)已存在ldap服務(wù)中使用弱密碼的賬號(hào)有什么好辦法能獲取出來(lái)嗎?ldap的賬號(hào)一旦創(chuàng)建,就看不到密碼了,如果用弱密碼字典模擬登錄的話,是否可行呢?

創(chuàng)建一個(gè)檢查密碼的函數(shù)CheckPassword,通過(guò)逐行讀取弱密碼詞典的數(shù)據(jù)進(jìn)行的模擬登錄,從而找到ldap中使用弱密碼的賬號(hào):

  1. func CheckPassword(employe string) { 
  2.  // 遍歷的弱密碼字典 
  3.  f, err := os.Open("~/dict.txt"
  4.  if err != nil { 
  5.   fmt.Println("reading dict.txt error: ", err) 
  6.  } 
  7.  defer f.Close() 
  8.  scanner := bufio.NewScanner(f) 
  9.  for scanner.Scan() { 
  10.   weakpassword := scanner.Text() 
  11.   _, err := LoginBind(employe, weakpassword) 
  12.   if err == nil { 
  13.    fmt.Println(employe + " 使用的密碼為: " + weakpassword) 
  14.   } 
  15.  } 
  16.  if err := scanner.Err(); err != nil { 
  17.   fmt.Println(err) 
  18.  } 
  19.  fmt.Println(employe + " check have aleardy finished. and the password is stronger well."
  20.  

結(jié)合前面說(shuō)的遍歷賬號(hào),拿到所有的賬號(hào)的信息,然后模擬登錄,如果命中了弱密碼字典中的密碼,就打印出來(lái)

  1. func main() { 
  2.  con, err := LoginBind("admin""admin123"
  3.  if err != nil { 
  4.   fmt.Println("V"
  5.   fmt.Println(err) 
  6.  } 
  7.  employees, err := GetEmployees(con) 
  8.  if err != nil { 
  9.   fmt.Println(err) 
  10.  } 
  11.  Whitelist := []string{"zhangsan","lisi"
  12.  for _, employe := range employees { 
  13.   fmt.Println("Starting check: ", employe) 
  14.   index := arrays.ContainsString(Whitelist, employe) 
  15.   if index == -1 { 
  16.    CheckPassword(employe) 
  17.   } else { 
  18.    fmt.Println(employe + " in whitelist. skiping..."
  19.   } 
  20.   fmt.Println(employe) 
  21.  } 

但是這樣實(shí)際就是在攻擊自己的服務(wù),這里就會(huì)產(chǎn)生兩個(gè)問(wèn)題:

用戶越多,弱密碼字典里面的密碼越多,檢查的次數(shù)也就越多,耗時(shí)也就越長(zhǎng)

每次模擬登錄,實(shí)際上就會(huì)創(chuàng)建一個(gè)連接,雖然連接認(rèn)證失敗,但是也無(wú)疑加重服務(wù)器連接創(chuàng)建和銷(xiāo)毀的資源損耗,你有調(diào)優(yōu)思路沒(méi)?

參考資料

[1]go-ldap: https://github.com/go-ldap/ldap

[2]go-ldap v3@3.1.0: https://pkg.go.dev/gopkg.in/ldap.v3@v3.1.0#section-readme

[3]osixia/openldap鏡像倉(cāng)庫(kù): https://hub.docker.com/r/osixia/openldap/tags

 

責(zé)任編輯:姜華 來(lái)源: 云原生生態(tài)圈
相關(guān)推薦

2014-06-06 10:01:31

2022-07-15 16:39:19

PythonWhoosh工具

2023-12-22 14:07:00

Go輕量級(jí)Goroutines

2016-03-31 15:25:09

2022-06-06 15:18:41

開(kāi)源GiteaDrone

2020-06-19 15:38:08

分析工具GoatCounter開(kāi)發(fā)

2014-04-15 17:03:00

2025-01-17 13:53:11

AI大模型檢測(cè)工具

2009-06-03 14:15:34

2023-11-06 13:08:45

2015-03-13 09:10:29

2021-04-14 08:20:46

Lighthouse工具性能檢測(cè)

2023-03-03 10:21:17

2024-08-26 14:32:43

2010-06-04 10:09:29

Linux 性能檢測(cè)

2010-06-04 10:30:15

Linux 性能檢測(cè)

2011-01-11 13:58:32

WebLog ExpeWEB服務(wù)器流量記錄

2009-12-17 17:31:10

2021-12-13 16:16:42

Java開(kāi)發(fā)工具

2021-07-01 05:17:52

Windows 11操作系統(tǒng)微軟
點(diǎn)贊
收藏

51CTO技術(shù)棧公眾號(hào)