gitea源码

v151.go 5.9KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198
  1. // Copyright 2020 The Gitea Authors. All rights reserved.
  2. // SPDX-License-Identifier: MIT
  3. package v1_13
  4. import (
  5. "context"
  6. "errors"
  7. "fmt"
  8. "strings"
  9. "code.gitea.io/gitea/modules/log"
  10. "code.gitea.io/gitea/modules/setting"
  11. "xorm.io/xorm"
  12. "xorm.io/xorm/schemas"
  13. )
  14. func SetDefaultPasswordToArgon2(x *xorm.Engine) error {
  15. switch {
  16. case setting.Database.Type.IsMySQL():
  17. _, err := x.Exec("ALTER TABLE `user` ALTER passwd_hash_algo SET DEFAULT 'argon2';")
  18. return err
  19. case setting.Database.Type.IsPostgreSQL():
  20. _, err := x.Exec("ALTER TABLE `user` ALTER COLUMN passwd_hash_algo SET DEFAULT 'argon2';")
  21. return err
  22. case setting.Database.Type.IsMSSQL():
  23. // need to find the constraint and drop it, then recreate it.
  24. sess := x.NewSession()
  25. defer sess.Close()
  26. if err := sess.Begin(); err != nil {
  27. return err
  28. }
  29. res, err := sess.QueryString("SELECT [name] FROM sys.default_constraints WHERE parent_object_id=OBJECT_ID(?) AND COL_NAME(parent_object_id, parent_column_id)=?;", "user", "passwd_hash_algo")
  30. if err != nil {
  31. return err
  32. }
  33. if len(res) > 0 {
  34. constraintName := res[0]["name"]
  35. log.Error("Results of select constraint: %s", constraintName)
  36. _, err := sess.Exec("ALTER TABLE [user] DROP CONSTRAINT " + constraintName)
  37. if err != nil {
  38. return err
  39. }
  40. _, err = sess.Exec("ALTER TABLE [user] ADD CONSTRAINT " + constraintName + " DEFAULT 'argon2' FOR passwd_hash_algo")
  41. if err != nil {
  42. return err
  43. }
  44. } else {
  45. _, err := sess.Exec("ALTER TABLE [user] ADD DEFAULT('argon2') FOR passwd_hash_algo")
  46. if err != nil {
  47. return err
  48. }
  49. }
  50. return sess.Commit()
  51. case setting.Database.Type.IsSQLite3():
  52. // drop through
  53. default:
  54. log.Fatal("Unrecognized DB")
  55. }
  56. tables, err := x.DBMetas()
  57. if err != nil {
  58. return err
  59. }
  60. // Now for SQLite we have to recreate the table
  61. var table *schemas.Table
  62. tableName := "user"
  63. for _, table = range tables {
  64. if table.Name == tableName {
  65. break
  66. }
  67. }
  68. if table == nil || table.Name != tableName {
  69. type User struct {
  70. PasswdHashAlgo string `xorm:"NOT NULL DEFAULT 'argon2'"`
  71. }
  72. return x.Sync(new(User))
  73. }
  74. column := table.GetColumn("passwd_hash_algo")
  75. if column == nil {
  76. type User struct {
  77. PasswdHashAlgo string `xorm:"NOT NULL DEFAULT 'argon2'"`
  78. }
  79. return x.Sync(new(User))
  80. }
  81. tempTableName := "tmp_recreate__user"
  82. column.Default = "'argon2'"
  83. createTableSQL, _, err := x.Dialect().CreateTableSQL(context.Background(), x.DB(), table, tempTableName)
  84. if err != nil {
  85. return err
  86. }
  87. sess := x.NewSession()
  88. defer sess.Close()
  89. if err := sess.Begin(); err != nil {
  90. return err
  91. }
  92. if _, err := sess.Exec(createTableSQL); err != nil {
  93. log.Error("Unable to create table %s. Error: %v\n", tempTableName, err, createTableSQL)
  94. return err
  95. }
  96. for _, index := range table.Indexes {
  97. if _, err := sess.Exec(x.Dialect().CreateIndexSQL(tempTableName, index)); err != nil {
  98. log.Error("Unable to create indexes on temporary table %s. Error: %v", tempTableName, err)
  99. return err
  100. }
  101. }
  102. newTableColumns := table.Columns()
  103. if len(newTableColumns) == 0 {
  104. return errors.New("no columns in new table")
  105. }
  106. hasID := false
  107. for _, column := range newTableColumns {
  108. hasID = hasID || (column.IsPrimaryKey && column.IsAutoIncrement)
  109. }
  110. sqlStringBuilder := &strings.Builder{}
  111. _, _ = sqlStringBuilder.WriteString("INSERT INTO `")
  112. _, _ = sqlStringBuilder.WriteString(tempTableName)
  113. _, _ = sqlStringBuilder.WriteString("` (`")
  114. _, _ = sqlStringBuilder.WriteString(newTableColumns[0].Name)
  115. _, _ = sqlStringBuilder.WriteString("`")
  116. for _, column := range newTableColumns[1:] {
  117. _, _ = sqlStringBuilder.WriteString(", `")
  118. _, _ = sqlStringBuilder.WriteString(column.Name)
  119. _, _ = sqlStringBuilder.WriteString("`")
  120. }
  121. _, _ = sqlStringBuilder.WriteString(")")
  122. _, _ = sqlStringBuilder.WriteString(" SELECT ")
  123. if newTableColumns[0].Default != "" {
  124. _, _ = sqlStringBuilder.WriteString("COALESCE(`")
  125. _, _ = sqlStringBuilder.WriteString(newTableColumns[0].Name)
  126. _, _ = sqlStringBuilder.WriteString("`, ")
  127. _, _ = sqlStringBuilder.WriteString(newTableColumns[0].Default)
  128. _, _ = sqlStringBuilder.WriteString(")")
  129. } else {
  130. _, _ = sqlStringBuilder.WriteString("`")
  131. _, _ = sqlStringBuilder.WriteString(newTableColumns[0].Name)
  132. _, _ = sqlStringBuilder.WriteString("`")
  133. }
  134. for _, column := range newTableColumns[1:] {
  135. if column.Default != "" {
  136. _, _ = sqlStringBuilder.WriteString(", COALESCE(`")
  137. _, _ = sqlStringBuilder.WriteString(column.Name)
  138. _, _ = sqlStringBuilder.WriteString("`, ")
  139. _, _ = sqlStringBuilder.WriteString(column.Default)
  140. _, _ = sqlStringBuilder.WriteString(")")
  141. } else {
  142. _, _ = sqlStringBuilder.WriteString(", `")
  143. _, _ = sqlStringBuilder.WriteString(column.Name)
  144. _, _ = sqlStringBuilder.WriteString("`")
  145. }
  146. }
  147. _, _ = sqlStringBuilder.WriteString(" FROM `")
  148. _, _ = sqlStringBuilder.WriteString(tableName)
  149. _, _ = sqlStringBuilder.WriteString("`")
  150. if _, err := sess.Exec(sqlStringBuilder.String()); err != nil {
  151. log.Error("Unable to set copy data in to temp table %s. Error: %v", tempTableName, err)
  152. return err
  153. }
  154. // SQLite will drop all the constraints on the old table
  155. if _, err := sess.Exec(fmt.Sprintf("DROP TABLE `%s`", tableName)); err != nil {
  156. log.Error("Unable to drop old table %s. Error: %v", tableName, err)
  157. return err
  158. }
  159. for _, index := range table.Indexes {
  160. if _, err := sess.Exec(x.Dialect().DropIndexSQL(tempTableName, index)); err != nil {
  161. log.Error("Unable to drop indexes on temporary table %s. Error: %v", tempTableName, err)
  162. return err
  163. }
  164. }
  165. if _, err := sess.Exec(fmt.Sprintf("ALTER TABLE `%s` RENAME TO `%s`", tempTableName, tableName)); err != nil {
  166. log.Error("Unable to rename %s to %s. Error: %v", tempTableName, tableName, err)
  167. return err
  168. }
  169. for _, index := range table.Indexes {
  170. if _, err := sess.Exec(x.Dialect().CreateIndexSQL(tableName, index)); err != nil {
  171. log.Error("Unable to recreate indexes on table %s. Error: %v", tableName, err)
  172. return err
  173. }
  174. }
  175. return sess.Commit()
  176. }