You can not select more than 25 topics Topics must start with a chinese character,a letter or number, can include dashes ('-') and can be up to 35 characters long.

user_mail.go 13 kB

8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
8 years ago
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491
  1. // Copyright 2016 The Gogs Authors. All rights reserved.
  2. // Copyright 2020 The Gitea Authors. All rights reserved.
  3. // Use of this source code is governed by a MIT-style
  4. // license that can be found in the LICENSE file.
  5. package models
  6. import (
  7. "errors"
  8. "fmt"
  9. "strings"
  10. "code.gitea.io/gitea/modules/log"
  11. "code.gitea.io/gitea/modules/setting"
  12. "code.gitea.io/gitea/modules/util"
  13. "xorm.io/builder"
  14. )
  15. var (
  16. // ErrEmailAddressNotExist email address not exist
  17. ErrEmailAddressNotExist = errors.New("Email address does not exist")
  18. )
  19. // EmailAddress is the list of all email addresses of a user. Can contain the
  20. // primary email address, but is not obligatory.
  21. type EmailAddress struct {
  22. ID int64 `xorm:"pk autoincr"`
  23. UID int64 `xorm:"INDEX NOT NULL"`
  24. Email string `xorm:"UNIQUE NOT NULL"`
  25. IsActivated bool
  26. IsPrimary bool `xorm:"-"`
  27. }
  28. // GetEmailAddresses returns all email addresses belongs to given user.
  29. func GetEmailAddresses(uid int64) ([]*EmailAddress, error) {
  30. emails := make([]*EmailAddress, 0, 5)
  31. if err := x.
  32. Where("uid=?", uid).
  33. Find(&emails); err != nil {
  34. return nil, err
  35. }
  36. u, err := GetUserByID(uid)
  37. if err != nil {
  38. return nil, err
  39. }
  40. isPrimaryFound := false
  41. for _, email := range emails {
  42. if email.Email == u.Email {
  43. isPrimaryFound = true
  44. email.IsPrimary = true
  45. } else {
  46. email.IsPrimary = false
  47. }
  48. }
  49. // We always want the primary email address displayed, even if it's not in
  50. // the email address table (yet).
  51. if !isPrimaryFound {
  52. emails = append(emails, &EmailAddress{
  53. Email: u.Email,
  54. IsActivated: u.IsActive,
  55. IsPrimary: true,
  56. })
  57. }
  58. return emails, nil
  59. }
  60. // GetEmailAddressByID gets a user's email address by ID
  61. func GetEmailAddressByID(uid, id int64) (*EmailAddress, error) {
  62. // User ID is required for security reasons
  63. email := &EmailAddress{ID: id, UID: uid}
  64. if has, err := x.Get(email); err != nil {
  65. return nil, err
  66. } else if !has {
  67. return nil, nil
  68. }
  69. return email, nil
  70. }
  71. // GetEmailAddressByIDAndEmail gets a user's email address by ID and email
  72. func GetEmailAddressByIDAndEmail(uid int64, emailAddr string) (*EmailAddress, error) {
  73. // User ID is required for security reasons
  74. email := &EmailAddress{UID: uid, Email: emailAddr}
  75. if has, err := x.Get(email); err != nil {
  76. return nil, err
  77. } else if !has {
  78. return nil, nil
  79. }
  80. return email, nil
  81. }
  82. func isEmailActive(e Engine, email string, userID, emailID int64) (bool, error) {
  83. if len(email) == 0 {
  84. return true, nil
  85. }
  86. // Can't filter by boolean field unless it's explicit
  87. cond := builder.NewCond()
  88. cond = cond.And(builder.Eq{"email": email}, builder.Neq{"id": emailID})
  89. if setting.Service.RegisterEmailConfirm {
  90. // Inactive (unvalidated) addresses don't count as active if email validation is required
  91. cond = cond.And(builder.Eq{"is_activated": true})
  92. }
  93. em := EmailAddress{}
  94. if has, err := e.Where(cond).Get(&em); has || err != nil {
  95. if has {
  96. log.Info("isEmailActive('%s',%d,%d) found duplicate in email ID %d", email, userID, emailID, em.ID)
  97. }
  98. return has, err
  99. }
  100. // Can't filter by boolean field unless it's explicit
  101. cond = builder.NewCond()
  102. cond = cond.And(builder.Eq{"email": email}, builder.Neq{"id": userID})
  103. if setting.Service.RegisterEmailConfirm {
  104. cond = cond.And(builder.Eq{"is_active": true})
  105. }
  106. us := User{}
  107. if has, err := e.Where(cond).Get(&us); has || err != nil {
  108. if has {
  109. log.Info("isEmailActive('%s',%d,%d) found duplicate in user ID %d", email, userID, emailID, us.ID)
  110. }
  111. return has, err
  112. }
  113. return false, nil
  114. }
  115. func isEmailUsed(e Engine, email string) (bool, error) {
  116. if len(email) == 0 {
  117. return true, nil
  118. }
  119. return e.Get(&EmailAddress{Email: email})
  120. }
  121. // IsEmailUsed returns true if the email has been used.
  122. func IsEmailUsed(email string) (bool, error) {
  123. return isEmailUsed(x, email)
  124. }
  125. func addEmailAddress(e Engine, email *EmailAddress) error {
  126. email.Email = strings.ToLower(strings.TrimSpace(email.Email))
  127. used, err := isEmailUsed(e, email.Email)
  128. if err != nil {
  129. return err
  130. } else if used {
  131. return ErrEmailAlreadyUsed{email.Email}
  132. }
  133. _, err = e.Insert(email)
  134. return err
  135. }
  136. // AddEmailAddress adds an email address to given user.
  137. func AddEmailAddress(email *EmailAddress) error {
  138. return addEmailAddress(x, email)
  139. }
  140. // AddEmailAddresses adds an email address to given user.
  141. func AddEmailAddresses(emails []*EmailAddress) error {
  142. if len(emails) == 0 {
  143. return nil
  144. }
  145. // Check if any of them has been used
  146. for i := range emails {
  147. emails[i].Email = strings.ToLower(strings.TrimSpace(emails[i].Email))
  148. used, err := IsEmailUsed(emails[i].Email)
  149. if err != nil {
  150. return err
  151. } else if used {
  152. return ErrEmailAlreadyUsed{emails[i].Email}
  153. }
  154. }
  155. if _, err := x.Insert(emails); err != nil {
  156. return fmt.Errorf("Insert: %v", err)
  157. }
  158. return nil
  159. }
  160. // Activate activates the email address to given user.
  161. func (email *EmailAddress) Activate() error {
  162. sess := x.NewSession()
  163. defer sess.Close()
  164. if err := sess.Begin(); err != nil {
  165. return err
  166. }
  167. if err := email.updateActivation(sess, true); err != nil {
  168. return err
  169. }
  170. return sess.Commit()
  171. }
  172. func (email *EmailAddress) updateActivation(e Engine, activate bool) error {
  173. user, err := getUserByID(e, email.UID)
  174. if err != nil {
  175. return err
  176. }
  177. if user.Rands, err = GetUserSalt(); err != nil {
  178. return err
  179. }
  180. email.IsActivated = activate
  181. if _, err := e.ID(email.ID).Cols("is_activated").Update(email); err != nil {
  182. return err
  183. }
  184. return updateUserCols(e, user, "rands")
  185. }
  186. // DeleteEmailAddress deletes an email address of given user.
  187. func DeleteEmailAddress(email *EmailAddress) (err error) {
  188. var deleted int64
  189. // ask to check UID
  190. var address = EmailAddress{
  191. UID: email.UID,
  192. }
  193. if email.ID > 0 {
  194. deleted, err = x.ID(email.ID).Delete(&address)
  195. } else {
  196. deleted, err = x.
  197. Where("email=?", email.Email).
  198. Delete(&address)
  199. }
  200. if err != nil {
  201. return err
  202. } else if deleted != 1 {
  203. return ErrEmailAddressNotExist
  204. }
  205. return nil
  206. }
  207. // DeleteEmailAddresses deletes multiple email addresses
  208. func DeleteEmailAddresses(emails []*EmailAddress) (err error) {
  209. for i := range emails {
  210. if err = DeleteEmailAddress(emails[i]); err != nil {
  211. return err
  212. }
  213. }
  214. return nil
  215. }
  216. // MakeEmailPrimary sets primary email address of given user.
  217. func MakeEmailPrimary(email *EmailAddress) error {
  218. has, err := x.Get(email)
  219. if err != nil {
  220. return err
  221. } else if !has {
  222. return ErrEmailNotExist
  223. }
  224. if !email.IsActivated {
  225. return ErrEmailNotActivated
  226. }
  227. user := &User{ID: email.UID}
  228. has, err = x.Get(user)
  229. if err != nil {
  230. return err
  231. } else if !has {
  232. return ErrUserNotExist{email.UID, "", 0}
  233. }
  234. // Make sure the former primary email doesn't disappear.
  235. formerPrimaryEmail := &EmailAddress{UID: user.ID, Email: user.Email}
  236. has, err = x.Get(formerPrimaryEmail)
  237. if err != nil {
  238. return err
  239. }
  240. sess := x.NewSession()
  241. defer sess.Close()
  242. if err = sess.Begin(); err != nil {
  243. return err
  244. }
  245. if !has {
  246. formerPrimaryEmail.UID = user.ID
  247. formerPrimaryEmail.IsActivated = user.IsActive
  248. if _, err = sess.Insert(formerPrimaryEmail); err != nil {
  249. return err
  250. }
  251. }
  252. user.Email = email.Email
  253. if _, err = sess.ID(user.ID).Cols("email").Update(user); err != nil {
  254. return err
  255. }
  256. return sess.Commit()
  257. }
  258. // SearchEmailOrderBy is used to sort the results from SearchEmails()
  259. type SearchEmailOrderBy string
  260. func (s SearchEmailOrderBy) String() string {
  261. return string(s)
  262. }
  263. // Strings for sorting result
  264. const (
  265. SearchEmailOrderByEmail SearchEmailOrderBy = "emails.email ASC, is_primary DESC, sortid ASC"
  266. SearchEmailOrderByEmailReverse SearchEmailOrderBy = "emails.email DESC, is_primary ASC, sortid DESC"
  267. SearchEmailOrderByName SearchEmailOrderBy = "`user`.lower_name ASC, is_primary DESC, sortid ASC"
  268. SearchEmailOrderByNameReverse SearchEmailOrderBy = "`user`.lower_name DESC, is_primary ASC, sortid DESC"
  269. )
  270. // SearchEmailOptions are options to search e-mail addresses for the admin panel
  271. type SearchEmailOptions struct {
  272. ListOptions
  273. Keyword string
  274. SortType SearchEmailOrderBy
  275. IsPrimary util.OptionalBool
  276. IsActivated util.OptionalBool
  277. }
  278. // SearchEmailResult is an e-mail address found in the user or email_address table
  279. type SearchEmailResult struct {
  280. UID int64
  281. Email string
  282. IsActivated bool
  283. IsPrimary bool
  284. // From User
  285. Name string
  286. FullName string
  287. }
  288. // SearchEmails takes options i.e. keyword and part of email name to search,
  289. // it returns results in given range and number of total results.
  290. func SearchEmails(opts *SearchEmailOptions) ([]*SearchEmailResult, int64, error) {
  291. // Unfortunately, UNION support for SQLite in xorm is currently broken, so we must
  292. // build the SQL ourselves.
  293. where := make([]string, 0, 5)
  294. args := make([]interface{}, 0, 5)
  295. emailsSQL := "(SELECT id as sortid, uid, email, is_activated, 0 as is_primary " +
  296. "FROM email_address " +
  297. "UNION ALL " +
  298. "SELECT id as sortid, id AS uid, email, is_active AS is_activated, 1 as is_primary " +
  299. "FROM `user` " +
  300. "WHERE type = ?) AS emails"
  301. args = append(args, UserTypeIndividual)
  302. if len(opts.Keyword) > 0 {
  303. // Note: % can be injected in the Keyword parameter, but it won't do any harm.
  304. where = append(where, "(lower(`user`.full_name) LIKE ? OR `user`.lower_name LIKE ? OR emails.email LIKE ?)")
  305. likeStr := "%" + strings.ToLower(opts.Keyword) + "%"
  306. args = append(args, likeStr)
  307. args = append(args, likeStr)
  308. args = append(args, likeStr)
  309. }
  310. switch {
  311. case opts.IsPrimary.IsTrue():
  312. where = append(where, "emails.is_primary = ?")
  313. args = append(args, true)
  314. case opts.IsPrimary.IsFalse():
  315. where = append(where, "emails.is_primary = ?")
  316. args = append(args, false)
  317. }
  318. switch {
  319. case opts.IsActivated.IsTrue():
  320. where = append(where, "emails.is_activated = ?")
  321. args = append(args, true)
  322. case opts.IsActivated.IsFalse():
  323. where = append(where, "emails.is_activated = ?")
  324. args = append(args, false)
  325. }
  326. var whereStr string
  327. if len(where) > 0 {
  328. whereStr = "WHERE " + strings.Join(where, " AND ")
  329. }
  330. joinSQL := "FROM " + emailsSQL + " INNER JOIN `user` ON `user`.id = emails.uid " + whereStr
  331. count, err := x.SQL("SELECT count(*) "+joinSQL, args...).Count()
  332. if err != nil {
  333. return nil, 0, fmt.Errorf("Count: %v", err)
  334. }
  335. orderby := opts.SortType.String()
  336. if orderby == "" {
  337. orderby = SearchEmailOrderByEmail.String()
  338. }
  339. querySQL := "SELECT emails.uid, emails.email, emails.is_activated, emails.is_primary, " +
  340. "`user`.name, `user`.full_name " + joinSQL + " ORDER BY " + orderby
  341. opts.setDefaultValues()
  342. rows, err := x.SQL(querySQL, args...).Rows(new(SearchEmailResult))
  343. if err != nil {
  344. return nil, 0, fmt.Errorf("Emails: %v", err)
  345. }
  346. // Page manually because xorm can't handle Limit() with raw SQL
  347. defer rows.Close()
  348. emails := make([]*SearchEmailResult, 0, opts.PageSize)
  349. skip := (opts.Page - 1) * opts.PageSize
  350. for rows.Next() {
  351. var email SearchEmailResult
  352. if err := rows.Scan(&email); err != nil {
  353. return nil, 0, err
  354. }
  355. if skip > 0 {
  356. skip--
  357. continue
  358. }
  359. emails = append(emails, &email)
  360. if len(emails) == opts.PageSize {
  361. break
  362. }
  363. }
  364. return emails, count, err
  365. }
  366. // ActivateUserEmail will change the activated state of an email address,
  367. // either primary (in the user table) or secondary (in the email_address table)
  368. func ActivateUserEmail(userID int64, email string, primary, activate bool) (err error) {
  369. sess := x.NewSession()
  370. defer sess.Close()
  371. if err = sess.Begin(); err != nil {
  372. return err
  373. }
  374. if primary {
  375. // Activate/deactivate a user's primary email address
  376. user := User{ID: userID, Email: email}
  377. if has, err := sess.Get(&user); err != nil {
  378. return err
  379. } else if !has {
  380. return fmt.Errorf("no such user: %d (%s)", userID, email)
  381. }
  382. if user.IsActive == activate {
  383. // Already in the desired state; no action
  384. return nil
  385. }
  386. if activate {
  387. if used, err := isEmailActive(sess, email, userID, 0); err != nil {
  388. return fmt.Errorf("isEmailActive(): %v", err)
  389. } else if used {
  390. return ErrEmailAlreadyUsed{Email: email}
  391. }
  392. }
  393. user.IsActive = activate
  394. if user.Rands, err = GetUserSalt(); err != nil {
  395. return fmt.Errorf("generate salt: %v", err)
  396. }
  397. if err = updateUserCols(sess, &user, "is_active", "rands"); err != nil {
  398. return fmt.Errorf("updateUserCols(): %v", err)
  399. }
  400. } else {
  401. // Activate/deactivate a user's secondary email address
  402. // First check if there's another user active with the same address
  403. addr := EmailAddress{UID: userID, Email: email}
  404. if has, err := sess.Get(&addr); err != nil {
  405. return err
  406. } else if !has {
  407. return fmt.Errorf("no such email: %d (%s)", userID, email)
  408. }
  409. if addr.IsActivated == activate {
  410. // Already in the desired state; no action
  411. return nil
  412. }
  413. if activate {
  414. if used, err := isEmailActive(sess, email, 0, addr.ID); err != nil {
  415. return fmt.Errorf("isEmailActive(): %v", err)
  416. } else if used {
  417. return ErrEmailAlreadyUsed{Email: email}
  418. }
  419. }
  420. if err = addr.updateActivation(sess, activate); err != nil {
  421. return fmt.Errorf("updateActivation(): %v", err)
  422. }
  423. }
  424. return sess.Commit()
  425. }