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.

session_delete.go 7.1 kB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255
  1. // Copyright 2016 The Xorm Authors. All rights reserved.
  2. // Use of this source code is governed by a BSD-style
  3. // license that can be found in the LICENSE file.
  4. package xorm
  5. import (
  6. "errors"
  7. "fmt"
  8. "strconv"
  9. "xorm.io/xorm/caches"
  10. "xorm.io/xorm/schemas"
  11. )
  12. var (
  13. // ErrNeedDeletedCond delete needs less one condition error
  14. ErrNeedDeletedCond = errors.New("Delete action needs at least one condition")
  15. // ErrNotImplemented not implemented
  16. ErrNotImplemented = errors.New("Not implemented")
  17. )
  18. func (session *Session) cacheDelete(table *schemas.Table, tableName, sqlStr string, args ...interface{}) error {
  19. if table == nil ||
  20. session.tx != nil {
  21. return ErrCacheFailed
  22. }
  23. for _, filter := range session.engine.dialect.Filters() {
  24. sqlStr = filter.Do(sqlStr)
  25. }
  26. newsql := session.statement.ConvertIDSQL(sqlStr)
  27. if newsql == "" {
  28. return ErrCacheFailed
  29. }
  30. cacher := session.engine.cacherMgr.GetCacher(tableName)
  31. pkColumns := table.PKColumns()
  32. ids, err := caches.GetCacheSql(cacher, tableName, newsql, args)
  33. if err != nil {
  34. resultsSlice, err := session.queryBytes(newsql, args...)
  35. if err != nil {
  36. return err
  37. }
  38. ids = make([]schemas.PK, 0)
  39. if len(resultsSlice) > 0 {
  40. for _, data := range resultsSlice {
  41. var id int64
  42. var pk schemas.PK = make([]interface{}, 0)
  43. for _, col := range pkColumns {
  44. if v, ok := data[col.Name]; !ok {
  45. return errors.New("no id")
  46. } else if col.SQLType.IsText() {
  47. pk = append(pk, string(v))
  48. } else if col.SQLType.IsNumeric() {
  49. id, err = strconv.ParseInt(string(v), 10, 64)
  50. if err != nil {
  51. return err
  52. }
  53. pk = append(pk, id)
  54. } else {
  55. return errors.New("not supported primary key type")
  56. }
  57. }
  58. ids = append(ids, pk)
  59. }
  60. }
  61. }
  62. for _, id := range ids {
  63. session.engine.logger.Debugf("[cache] delete cache obj: %v, %v", tableName, id)
  64. sid, err := id.ToString()
  65. if err != nil {
  66. return err
  67. }
  68. cacher.DelBean(tableName, sid)
  69. }
  70. session.engine.logger.Debugf("[cache] clear cache table: %v", tableName)
  71. cacher.ClearIds(tableName)
  72. return nil
  73. }
  74. // Delete records, bean's non-empty fields are conditions
  75. func (session *Session) Delete(bean interface{}) (int64, error) {
  76. if session.isAutoClose {
  77. defer session.Close()
  78. }
  79. if session.statement.LastError != nil {
  80. return 0, session.statement.LastError
  81. }
  82. if err := session.statement.SetRefBean(bean); err != nil {
  83. return 0, err
  84. }
  85. // handle before delete processors
  86. for _, closure := range session.beforeClosures {
  87. closure(bean)
  88. }
  89. cleanupProcessorsClosures(&session.beforeClosures)
  90. if processor, ok := interface{}(bean).(BeforeDeleteProcessor); ok {
  91. processor.BeforeDelete()
  92. }
  93. condSQL, condArgs, err := session.statement.GenConds(bean)
  94. if err != nil {
  95. return 0, err
  96. }
  97. pLimitN := session.statement.LimitN
  98. if len(condSQL) == 0 && (pLimitN == nil || *pLimitN == 0) {
  99. return 0, ErrNeedDeletedCond
  100. }
  101. var tableNameNoQuote = session.statement.TableName()
  102. var tableName = session.engine.Quote(tableNameNoQuote)
  103. var table = session.statement.RefTable
  104. var deleteSQL string
  105. if len(condSQL) > 0 {
  106. deleteSQL = fmt.Sprintf("DELETE FROM %v WHERE %v", tableName, condSQL)
  107. } else {
  108. deleteSQL = fmt.Sprintf("DELETE FROM %v", tableName)
  109. }
  110. var orderSQL string
  111. if len(session.statement.OrderStr) > 0 {
  112. orderSQL += fmt.Sprintf(" ORDER BY %s", session.statement.OrderStr)
  113. }
  114. if pLimitN != nil && *pLimitN > 0 {
  115. limitNValue := *pLimitN
  116. orderSQL += fmt.Sprintf(" LIMIT %d", limitNValue)
  117. }
  118. if len(orderSQL) > 0 {
  119. switch session.engine.dialect.URI().DBType {
  120. case schemas.POSTGRES:
  121. inSQL := fmt.Sprintf("ctid IN (SELECT ctid FROM %s%s)", tableName, orderSQL)
  122. if len(condSQL) > 0 {
  123. deleteSQL += " AND " + inSQL
  124. } else {
  125. deleteSQL += " WHERE " + inSQL
  126. }
  127. case schemas.SQLITE:
  128. inSQL := fmt.Sprintf("rowid IN (SELECT rowid FROM %s%s)", tableName, orderSQL)
  129. if len(condSQL) > 0 {
  130. deleteSQL += " AND " + inSQL
  131. } else {
  132. deleteSQL += " WHERE " + inSQL
  133. }
  134. // TODO: how to handle delete limit on mssql?
  135. case schemas.MSSQL:
  136. return 0, ErrNotImplemented
  137. default:
  138. deleteSQL += orderSQL
  139. }
  140. }
  141. var realSQL string
  142. argsForCache := make([]interface{}, 0, len(condArgs)*2)
  143. if session.statement.GetUnscoped() || table.DeletedColumn() == nil { // tag "deleted" is disabled
  144. realSQL = deleteSQL
  145. copy(argsForCache, condArgs)
  146. argsForCache = append(condArgs, argsForCache...)
  147. } else {
  148. // !oinume! sqlStrForCache and argsForCache is needed to behave as executing "DELETE FROM ..." for caches.
  149. copy(argsForCache, condArgs)
  150. argsForCache = append(condArgs, argsForCache...)
  151. deletedColumn := table.DeletedColumn()
  152. realSQL = fmt.Sprintf("UPDATE %v SET %v = ? WHERE %v",
  153. session.engine.Quote(session.statement.TableName()),
  154. session.engine.Quote(deletedColumn.Name),
  155. condSQL)
  156. if len(orderSQL) > 0 {
  157. switch session.engine.dialect.URI().DBType {
  158. case schemas.POSTGRES:
  159. inSQL := fmt.Sprintf("ctid IN (SELECT ctid FROM %s%s)", tableName, orderSQL)
  160. if len(condSQL) > 0 {
  161. realSQL += " AND " + inSQL
  162. } else {
  163. realSQL += " WHERE " + inSQL
  164. }
  165. case schemas.SQLITE:
  166. inSQL := fmt.Sprintf("rowid IN (SELECT rowid FROM %s%s)", tableName, orderSQL)
  167. if len(condSQL) > 0 {
  168. realSQL += " AND " + inSQL
  169. } else {
  170. realSQL += " WHERE " + inSQL
  171. }
  172. // TODO: how to handle delete limit on mssql?
  173. case schemas.MSSQL:
  174. return 0, ErrNotImplemented
  175. default:
  176. realSQL += orderSQL
  177. }
  178. }
  179. // !oinume! Insert nowTime to the head of session.statement.Params
  180. condArgs = append(condArgs, "")
  181. paramsLen := len(condArgs)
  182. copy(condArgs[1:paramsLen], condArgs[0:paramsLen-1])
  183. val, t := session.engine.nowTime(deletedColumn)
  184. condArgs[0] = val
  185. var colName = deletedColumn.Name
  186. session.afterClosures = append(session.afterClosures, func(bean interface{}) {
  187. col := table.GetColumn(colName)
  188. setColumnTime(bean, col, t)
  189. })
  190. }
  191. if cacher := session.engine.GetCacher(tableNameNoQuote); cacher != nil && session.statement.UseCache {
  192. session.cacheDelete(table, tableNameNoQuote, deleteSQL, argsForCache...)
  193. }
  194. session.statement.RefTable = table
  195. res, err := session.exec(realSQL, condArgs...)
  196. if err != nil {
  197. return 0, err
  198. }
  199. // handle after delete processors
  200. if session.isAutoCommit {
  201. for _, closure := range session.afterClosures {
  202. closure(bean)
  203. }
  204. if processor, ok := interface{}(bean).(AfterDeleteProcessor); ok {
  205. processor.AfterDelete()
  206. }
  207. } else {
  208. lenAfterClosures := len(session.afterClosures)
  209. if lenAfterClosures > 0 {
  210. if value, has := session.afterDeleteBeans[bean]; has && value != nil {
  211. *value = append(*value, session.afterClosures...)
  212. } else {
  213. afterClosures := make([]func(interface{}), lenAfterClosures)
  214. copy(afterClosures, session.afterClosures)
  215. session.afterDeleteBeans[bean] = &afterClosures
  216. }
  217. } else {
  218. if _, ok := interface{}(bean).(AfterDeleteProcessor); ok {
  219. session.afterDeleteBeans[bean] = nil
  220. }
  221. }
  222. }
  223. cleanupProcessorsClosures(&session.afterClosures)
  224. // --
  225. return res.RowsAffected()
  226. }