login.go 4.4 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157
  1. package controllers
  2. import (
  3. "math/rand"
  4. "net"
  5. "sync"
  6. "time"
  7. "ehang.io/nps/lib/common"
  8. "ehang.io/nps/lib/file"
  9. "ehang.io/nps/server"
  10. "github.com/astaxie/beego"
  11. )
  12. type LoginController struct {
  13. beego.Controller
  14. }
  15. var ipRecord sync.Map
  16. type record struct {
  17. hasLoginFailTimes int
  18. lastLoginTime time.Time
  19. }
  20. func (self *LoginController) Index() {
  21. // Try login implicitly, will succeed if it's configured as no-auth(empty username&password).
  22. webBaseUrl := beego.AppConfig.String("web_base_url")
  23. if self.doLogin("", "", false) {
  24. self.Redirect(webBaseUrl+"/index/index", 302)
  25. }
  26. self.Data["web_base_url"] = webBaseUrl
  27. self.Data["register_allow"], _ = beego.AppConfig.Bool("allow_user_register")
  28. self.TplName = "login/index.html"
  29. }
  30. func (self *LoginController) Verify() {
  31. username := self.GetString("username")
  32. password := self.GetString("password")
  33. if self.doLogin(username, password, true) {
  34. self.Data["json"] = map[string]interface{}{"status": 1, "msg": "login success"}
  35. } else {
  36. self.Data["json"] = map[string]interface{}{"status": 0, "msg": "username or password incorrect"}
  37. }
  38. self.ServeJSON()
  39. }
  40. func (self *LoginController) doLogin(username, password string, explicit bool) bool {
  41. clearIprecord()
  42. ip, _, _ := net.SplitHostPort(self.Ctx.Request.RemoteAddr)
  43. if v, ok := ipRecord.Load(ip); ok {
  44. vv := v.(*record)
  45. if (time.Now().Unix() - vv.lastLoginTime.Unix()) >= 60 {
  46. vv.hasLoginFailTimes = 0
  47. }
  48. if vv.hasLoginFailTimes >= 10 {
  49. return false
  50. }
  51. }
  52. var auth bool
  53. if password == beego.AppConfig.String("web_password") && username == beego.AppConfig.String("web_username") {
  54. self.SetSession("isAdmin", true)
  55. self.DelSession("clientId")
  56. self.DelSession("username")
  57. auth = true
  58. server.Bridge.Register.Store(common.GetIpByAddr(self.Ctx.Input.IP()), time.Now().Add(time.Hour*time.Duration(2)))
  59. }
  60. b, err := beego.AppConfig.Bool("allow_user_login")
  61. if err == nil && b && !auth {
  62. file.GetDb().JsonDb.Clients.Range(func(key, value interface{}) bool {
  63. v := value.(*file.Client)
  64. if !v.Status || v.NoDisplay {
  65. return true
  66. }
  67. if v.WebUserName == "" && v.WebPassword == "" {
  68. if username != "user" || v.VerifyKey != password {
  69. return true
  70. } else {
  71. auth = true
  72. }
  73. }
  74. if !auth && v.WebPassword == password && v.WebUserName == username {
  75. auth = true
  76. }
  77. if auth {
  78. self.SetSession("isAdmin", false)
  79. self.SetSession("clientId", v.Id)
  80. self.SetSession("username", v.WebUserName)
  81. return false
  82. }
  83. return true
  84. })
  85. }
  86. if auth {
  87. self.SetSession("auth", true)
  88. ipRecord.Delete(ip)
  89. return true
  90. }
  91. if v, load := ipRecord.LoadOrStore(ip, &record{hasLoginFailTimes: 1, lastLoginTime: time.Now()}); load && explicit {
  92. vv := v.(*record)
  93. vv.lastLoginTime = time.Now()
  94. vv.hasLoginFailTimes += 1
  95. ipRecord.Store(ip, vv)
  96. }
  97. return false
  98. }
  99. func (self *LoginController) Register() {
  100. if self.Ctx.Request.Method == "GET" {
  101. self.Data["web_base_url"] = beego.AppConfig.String("web_base_url")
  102. self.TplName = "login/register.html"
  103. } else {
  104. if b, err := beego.AppConfig.Bool("allow_user_register"); err != nil || !b {
  105. self.Data["json"] = map[string]interface{}{"status": 0, "msg": "register is not allow"}
  106. self.ServeJSON()
  107. return
  108. }
  109. if self.GetString("username") == "" || self.GetString("password") == "" || self.GetString("username") == beego.AppConfig.String("web_username") {
  110. self.Data["json"] = map[string]interface{}{"status": 0, "msg": "please check your input"}
  111. self.ServeJSON()
  112. return
  113. }
  114. t := &file.Client{
  115. Id: int(file.GetDb().JsonDb.GetClientId()),
  116. Status: true,
  117. Cnf: &file.Config{},
  118. WebUserName: self.GetString("username"),
  119. WebPassword: self.GetString("password"),
  120. Flow: &file.Flow{},
  121. }
  122. if err := file.GetDb().NewClient(t); err != nil {
  123. self.Data["json"] = map[string]interface{}{"status": 0, "msg": err.Error()}
  124. } else {
  125. self.Data["json"] = map[string]interface{}{"status": 1, "msg": "register success"}
  126. }
  127. self.ServeJSON()
  128. }
  129. }
  130. func (self *LoginController) Out() {
  131. self.SetSession("auth", false)
  132. self.Redirect(beego.AppConfig.String("web_base_url")+"/login/index", 302)
  133. }
  134. func clearIprecord() {
  135. rand.Seed(time.Now().UnixNano())
  136. x := rand.Intn(100)
  137. if x == 1 {
  138. ipRecord.Range(func(key, value interface{}) bool {
  139. v := value.(*record)
  140. if time.Now().Unix()-v.lastLoginTime.Unix() >= 60 {
  141. ipRecord.Delete(key)
  142. }
  143. return true
  144. })
  145. }
  146. }