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
123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501
  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. has, err = sess.Where("id!=?", user.ID).
  254. And("type=?", user.Type).
  255. And("email=?", strings.ToLower(user.Email)).
  256. Get(new(User))
  257. if err != nil {
  258. return err
  259. } else if has {
  260. return ErrEmailAlreadyUsed{user.Email}
  261. }
  262. if _, err = sess.ID(user.ID).Cols("email").Update(user); err != nil {
  263. return err
  264. }
  265. return sess.Commit()
  266. }
  267. // SearchEmailOrderBy is used to sort the results from SearchEmails()
  268. type SearchEmailOrderBy string
  269. func (s SearchEmailOrderBy) String() string {
  270. return string(s)
  271. }
  272. // Strings for sorting result
  273. const (
  274. SearchEmailOrderByEmail SearchEmailOrderBy = "emails.email ASC, is_primary DESC, sortid ASC"
  275. SearchEmailOrderByEmailReverse SearchEmailOrderBy = "emails.email DESC, is_primary ASC, sortid DESC"
  276. SearchEmailOrderByName SearchEmailOrderBy = "`user`.lower_name ASC, is_primary DESC, sortid ASC"
  277. SearchEmailOrderByNameReverse SearchEmailOrderBy = "`user`.lower_name DESC, is_primary ASC, sortid DESC"
  278. )
  279. // SearchEmailOptions are options to search e-mail addresses for the admin panel
  280. type SearchEmailOptions struct {
  281. ListOptions
  282. Keyword string
  283. SortType SearchEmailOrderBy
  284. IsPrimary util.OptionalBool
  285. IsActivated util.OptionalBool
  286. }
  287. // SearchEmailResult is an e-mail address found in the user or email_address table
  288. type SearchEmailResult struct {
  289. UID int64
  290. Email string
  291. IsActivated bool
  292. IsPrimary bool
  293. // From User
  294. Name string
  295. FullName string
  296. }
  297. // SearchEmails takes options i.e. keyword and part of email name to search,
  298. // it returns results in given range and number of total results.
  299. func SearchEmails(opts *SearchEmailOptions) ([]*SearchEmailResult, int64, error) {
  300. // Unfortunately, UNION support for SQLite in xorm is currently broken, so we must
  301. // build the SQL ourselves.
  302. where := make([]string, 0, 5)
  303. args := make([]interface{}, 0, 5)
  304. emailsSQL := "(SELECT id as sortid, uid, email, is_activated, 0 as is_primary " +
  305. "FROM email_address " +
  306. "UNION ALL " +
  307. "SELECT id as sortid, id AS uid, email, is_active AS is_activated, 1 as is_primary " +
  308. "FROM `user` " +
  309. "WHERE type = ?) AS emails"
  310. args = append(args, UserTypeIndividual)
  311. if len(opts.Keyword) > 0 {
  312. // Note: % can be injected in the Keyword parameter, but it won't do any harm.
  313. where = append(where, "(lower(`user`.full_name) LIKE ? OR `user`.lower_name LIKE ? OR emails.email LIKE ?)")
  314. likeStr := "%" + strings.ToLower(opts.Keyword) + "%"
  315. args = append(args, likeStr)
  316. args = append(args, likeStr)
  317. args = append(args, likeStr)
  318. }
  319. switch {
  320. case opts.IsPrimary.IsTrue():
  321. where = append(where, "emails.is_primary = ?")
  322. args = append(args, true)
  323. case opts.IsPrimary.IsFalse():
  324. where = append(where, "emails.is_primary = ?")
  325. args = append(args, false)
  326. }
  327. switch {
  328. case opts.IsActivated.IsTrue():
  329. where = append(where, "emails.is_activated = ?")
  330. args = append(args, true)
  331. case opts.IsActivated.IsFalse():
  332. where = append(where, "emails.is_activated = ?")
  333. args = append(args, false)
  334. }
  335. var whereStr string
  336. if len(where) > 0 {
  337. whereStr = "WHERE " + strings.Join(where, " AND ")
  338. }
  339. joinSQL := "FROM " + emailsSQL + " INNER JOIN `user` ON `user`.id = emails.uid " + whereStr
  340. count, err := x.SQL("SELECT count(*) "+joinSQL, args...).Count()
  341. if err != nil {
  342. return nil, 0, fmt.Errorf("Count: %v", err)
  343. }
  344. orderby := opts.SortType.String()
  345. if orderby == "" {
  346. orderby = SearchEmailOrderByEmail.String()
  347. }
  348. querySQL := "SELECT emails.uid, emails.email, emails.is_activated, emails.is_primary, " +
  349. "`user`.name, `user`.full_name " + joinSQL + " ORDER BY " + orderby
  350. opts.setDefaultValues()
  351. rows, err := x.SQL(querySQL, args...).Rows(new(SearchEmailResult))
  352. if err != nil {
  353. return nil, 0, fmt.Errorf("Emails: %v", err)
  354. }
  355. // Page manually because xorm can't handle Limit() with raw SQL
  356. defer rows.Close()
  357. emails := make([]*SearchEmailResult, 0, opts.PageSize)
  358. skip := (opts.Page - 1) * opts.PageSize
  359. for rows.Next() {
  360. var email SearchEmailResult
  361. if err := rows.Scan(&email); err != nil {
  362. return nil, 0, err
  363. }
  364. if skip > 0 {
  365. skip--
  366. continue
  367. }
  368. emails = append(emails, &email)
  369. if len(emails) == opts.PageSize {
  370. break
  371. }
  372. }
  373. return emails, count, err
  374. }
  375. // ActivateUserEmail will change the activated state of an email address,
  376. // either primary (in the user table) or secondary (in the email_address table)
  377. func ActivateUserEmail(userID int64, email string, primary, activate bool) (err error) {
  378. sess := x.NewSession()
  379. defer sess.Close()
  380. if err = sess.Begin(); err != nil {
  381. return err
  382. }
  383. if primary {
  384. // Activate/deactivate a user's primary email address
  385. user := User{ID: userID, Email: email}
  386. if has, err := sess.Get(&user); err != nil {
  387. return err
  388. } else if !has {
  389. return fmt.Errorf("no such user: %d (%s)", userID, email)
  390. }
  391. if user.IsActive == activate {
  392. // Already in the desired state; no action
  393. return nil
  394. }
  395. if activate {
  396. if used, err := isEmailActive(sess, email, userID, 0); err != nil {
  397. return fmt.Errorf("isEmailActive(): %v", err)
  398. } else if used {
  399. return ErrEmailAlreadyUsed{Email: email}
  400. }
  401. }
  402. user.IsActive = activate
  403. if user.Rands, err = GetUserSalt(); err != nil {
  404. return fmt.Errorf("generate salt: %v", err)
  405. }
  406. if err = updateUserCols(sess, &user, "is_active", "rands"); err != nil {
  407. return fmt.Errorf("updateUserCols(): %v", err)
  408. }
  409. } else {
  410. // Activate/deactivate a user's secondary email address
  411. // First check if there's another user active with the same address
  412. addr := EmailAddress{UID: userID, Email: email}
  413. if has, err := sess.Get(&addr); err != nil {
  414. return err
  415. } else if !has {
  416. return fmt.Errorf("no such email: %d (%s)", userID, email)
  417. }
  418. if addr.IsActivated == activate {
  419. // Already in the desired state; no action
  420. return nil
  421. }
  422. if activate {
  423. if used, err := isEmailActive(sess, email, 0, addr.ID); err != nil {
  424. return fmt.Errorf("isEmailActive(): %v", err)
  425. } else if used {
  426. return ErrEmailAlreadyUsed{Email: email}
  427. }
  428. }
  429. if err = addr.updateActivation(sess, activate); err != nil {
  430. return fmt.Errorf("updateActivation(): %v", err)
  431. }
  432. }
  433. return sess.Commit()
  434. }