utils.go 2.9 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162
  1. package model
  2. import (
  3. "database/sql/driver"
  4. "errors"
  5. "fmt"
  6. "strconv"
  7. "strings"
  8. "gorm.io/gorm"
  9. "gorm.io/gorm/clause"
  10. )
  11. func NotFoundError(errMsg ...string) error {
  12. return fmt.Errorf("%s %w", strings.Join(errMsg, " "), gorm.ErrRecordNotFound)
  13. }
  14. func HandleNotFound(err error, errMsg ...string) error {
  15. if err != nil && errors.Is(err, gorm.ErrRecordNotFound) {
  16. return NotFoundError(strings.Join(errMsg, " "))
  17. }
  18. return err
  19. }
  20. // Helper function to handle update results
  21. func HandleUpdateResult(result *gorm.DB, entityName string) error {
  22. if result.Error != nil {
  23. return HandleNotFound(result.Error, entityName)
  24. }
  25. if result.RowsAffected == 0 {
  26. return NotFoundError(entityName)
  27. }
  28. return nil
  29. }
  30. func OnConflictDoNothing() *gorm.DB {
  31. return DB.Clauses(clause.OnConflict{
  32. DoNothing: true,
  33. })
  34. }
  35. func IgnoreNotFound(err error) error {
  36. if err != nil && errors.Is(err, gorm.ErrRecordNotFound) {
  37. return nil
  38. }
  39. return err
  40. }
  41. type ZeroNullFloat64 float64
  42. func (znf ZeroNullFloat64) Value() (driver.Value, error) {
  43. if znf == 0 {
  44. return nil, nil
  45. }
  46. return float64(znf), nil
  47. }
  48. func (znf *ZeroNullFloat64) Scan(value any) error {
  49. if value == nil {
  50. *znf = 0
  51. return nil
  52. }
  53. switch v := value.(type) {
  54. case string:
  55. vf, err := strconv.ParseFloat(v, 64)
  56. if err != nil {
  57. return err
  58. }
  59. *znf = ZeroNullFloat64(vf)
  60. case int:
  61. *znf = ZeroNullFloat64(v)
  62. case int64:
  63. *znf = ZeroNullFloat64(v)
  64. case float32:
  65. *znf = ZeroNullFloat64(v)
  66. case float64:
  67. *znf = ZeroNullFloat64(v)
  68. default:
  69. return fmt.Errorf("unsupported type: %T", v)
  70. }
  71. return nil
  72. }
  73. type ZeroNullInt64 int64
  74. func (zni ZeroNullInt64) Value() (driver.Value, error) {
  75. if zni == 0 {
  76. return nil, nil
  77. }
  78. return int64(zni), nil
  79. }
  80. func (zni *ZeroNullInt64) Scan(value any) error {
  81. if value == nil {
  82. *zni = 0
  83. return nil
  84. }
  85. switch v := value.(type) {
  86. case int:
  87. *zni = ZeroNullInt64(v)
  88. case int64:
  89. *zni = ZeroNullInt64(v)
  90. default:
  91. return fmt.Errorf("unsupported type: %T", v)
  92. }
  93. return nil
  94. }
  95. type EmptyNullString string
  96. func (ns EmptyNullString) String() string {
  97. return string(ns)
  98. }
  99. // Scan implements the [Scanner] interface.
  100. func (ns *EmptyNullString) Scan(value any) error {
  101. if value == nil {
  102. *ns = ""
  103. return nil
  104. }
  105. switch v := value.(type) {
  106. case []byte:
  107. *ns = EmptyNullString(v)
  108. case string:
  109. *ns = EmptyNullString(v)
  110. default:
  111. return fmt.Errorf("unsupported type: %T", v)
  112. }
  113. return nil
  114. }
  115. // Value implements the [driver.Valuer] interface.
  116. func (ns EmptyNullString) Value() (driver.Value, error) {
  117. if ns == "" {
  118. return nil, nil
  119. }
  120. return string(ns), nil
  121. }
  122. func String2Int(keyword string) int {
  123. if keyword == "" {
  124. return 0
  125. }
  126. i, err := strconv.Atoi(keyword)
  127. if err != nil {
  128. return 0
  129. }
  130. return i
  131. }
  132. func toLimitOffset(page int, perPage int) (limit int, offset int) {
  133. page--
  134. if page < 0 {
  135. page = 0
  136. }
  137. if perPage <= 0 {
  138. perPage = 10
  139. } else if perPage > 100 {
  140. perPage = 100
  141. }
  142. return perPage, page * perPage
  143. }