store_sql.go 6.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227
  1. package chunk
  2. import (
  3. "database/sql"
  4. "encoding/binary"
  5. "encoding/json"
  6. "github.com/flashmob/go-guerrilla/backends"
  7. "github.com/flashmob/go-guerrilla/mail/smtp"
  8. "net"
  9. )
  10. type sqlConfig struct {
  11. EmailTable string `json:"chunksaver_email_table,omitempty"`
  12. ChunkTable string `json:"chunksaver_chunk_table,omitempty"`
  13. Driver string `json:"chunksaver_sql_driver,omitempty"`
  14. DSN string `json:"chunksaver_sql_dsn,omitempty"`
  15. PrimaryHost string `json:"chunksaver_primary_mail_host,omitempty"`
  16. }
  17. // StoreSQL implements the Storage interface
  18. type StoreSQL struct {
  19. config *sqlConfig
  20. statements map[string]*sql.Stmt
  21. db *sql.DB
  22. }
  23. func (s *StoreSQL) connect() (*sql.DB, error) {
  24. var err error
  25. if s.db, err = sql.Open(s.config.Driver, s.config.DSN); err != nil {
  26. backends.Log().Error("cannot open database: ", err)
  27. return nil, err
  28. }
  29. // do we have permission to access the table?
  30. _, err = s.db.Query("SELECT mail_id FROM " + s.config.EmailTable + " LIMIT 1")
  31. if err != nil {
  32. return nil, err
  33. }
  34. return s.db, err
  35. }
  36. func (s *StoreSQL) prepareSql() error {
  37. if s.statements == nil {
  38. s.statements = make(map[string]*sql.Stmt)
  39. }
  40. // begin inserting an email (before saving chunks)
  41. if stmt, err := s.db.Prepare(`INSERT INTO ` +
  42. s.config.EmailTable +
  43. ` (from, helo, recipient, ipv4_addr, ipv6_addr, return_path, is_tls, is_8bit)
  44. VALUES(?, ?, ?, ?, ?, ?, ?, ?)`); err != nil {
  45. return err
  46. } else {
  47. s.statements["insertEmail"] = stmt
  48. }
  49. // insert a chunk of email's data
  50. if stmt, err := s.db.Prepare(`INSERT INTO ` +
  51. s.config.ChunkTable +
  52. ` (data, hash)
  53. VALUES(?, ?)`); err != nil {
  54. return err
  55. } else {
  56. s.statements["insertChunk"] = stmt
  57. }
  58. // finalize the email (the connection closed)
  59. if stmt, err := s.db.Prepare(`
  60. UPDATE ` + s.config.EmailTable + `
  61. SET size=?, parts_info = ?, subject, delivery_id = ?, to = ?
  62. WHERE mail_id = ? `); err != nil {
  63. return err
  64. } else {
  65. s.statements["finalizeEmail"] = stmt
  66. }
  67. // Check the existence of a chunk (the reference_count col is incremented if it exists)
  68. // This means we can avoid re-inserting an existing chunk, only update its reference_count
  69. // check the "affected rows" count after executing query
  70. if stmt, err := s.db.Prepare(`
  71. UPDATE ` + s.config.ChunkTable + `
  72. SET reference_count=reference_count+1
  73. WHERE hash = ? `); err != nil {
  74. return err
  75. } else {
  76. s.statements["chunkReferenceIncr"] = stmt
  77. }
  78. // If the reference_count is 0 then it means the chunk has been deleted
  79. // Chunks are soft-deleted for now, hard-deleted by another sweeper query as they become stale.
  80. if stmt, err := s.db.Prepare(`
  81. UPDATE ` + s.config.ChunkTable + `
  82. SET reference_count=reference_count-1
  83. WHERE hash = ? AND reference_count > 0`); err != nil {
  84. return err
  85. } else {
  86. s.statements["chunkReferenceDecr"] = stmt
  87. }
  88. // fetch an email
  89. if stmt, err := s.db.Prepare(`
  90. SELECT *
  91. from ` + s.config.EmailTable + `
  92. where mail_id=?`); err != nil {
  93. return err
  94. } else {
  95. s.statements["selectMail"] = stmt
  96. }
  97. // fetch a chunk
  98. if stmt, err := s.db.Prepare(`
  99. SELECT *
  100. from ` + s.config.ChunkTable + `
  101. where hash=?`); err != nil {
  102. return err
  103. } else {
  104. s.statements["selectChunk"] = stmt
  105. }
  106. // TODO sweep old chunks
  107. // TODO sweep incomplete emails
  108. return nil
  109. }
  110. // OpenMessage implements the Storage interface
  111. func (s *StoreSQL) OpenMessage(from string, helo string, recipient string, ipAddress net.IPAddr, returnPath string, isTLS bool, transport smtp.TransportType) (mailID uint64, err error) {
  112. // if it's ipv4 then we want ipv6 to be 0, and vice-versa
  113. var ip4 uint32
  114. ip6 := make([]byte, 16)
  115. if ip := ipAddress.IP.To4(); ip != nil {
  116. ip4 = binary.BigEndian.Uint32(ip)
  117. } else {
  118. _ = copy(ip6, ipAddress.IP)
  119. }
  120. r, err := s.statements["insertEmail"].Exec(from, helo, recipient, ip4, ip6, returnPath, isTLS, transport)
  121. if err != nil {
  122. return 0, err
  123. }
  124. id, err := r.LastInsertId()
  125. if err != nil {
  126. return 0, err
  127. }
  128. return uint64(id), err
  129. }
  130. // AddChunk implements the Storage interface
  131. func (s *StoreSQL) AddChunk(data []byte, hash []byte) error {
  132. // attempt to increment the reference_count (it means the chunk is already in there)
  133. r, err := s.statements["chunkReferenceIncr"].Exec(hash)
  134. if err != nil {
  135. return err
  136. }
  137. affected, err := r.RowsAffected()
  138. if err != nil {
  139. return err
  140. }
  141. if affected == 0 {
  142. // chunk isn't in there, let's insert it
  143. _, err := s.statements["insertChunk"].Exec(data, hash)
  144. if err != nil {
  145. return err
  146. }
  147. }
  148. return nil
  149. }
  150. // CloseMessage implements the Storage interface
  151. func (s *StoreSQL) CloseMessage(mailID uint64, size int64, partsInfo *PartsInfo, subject string, deliveryID string, to string, from string) error {
  152. partsInfoJson, err := json.Marshal(partsInfo)
  153. if err != nil {
  154. return err
  155. }
  156. _, err = s.statements["finalizeEmail"].Exec(size, partsInfoJson, subject, deliveryID, to, mailID)
  157. if err != nil {
  158. return err
  159. }
  160. return nil
  161. }
  162. // Initialize loads the specific database config, connects to the db, prepares statements
  163. func (s *StoreSQL) Initialize(cfg backends.BackendConfig) error {
  164. configType := backends.BaseConfig(&sqlConfig{})
  165. bcfg, err := backends.Svc.ExtractConfig(backends.ConfigStreamProcessors, "chunksaver", cfg, configType)
  166. if err != nil {
  167. return err
  168. }
  169. s.config = bcfg.(*sqlConfig)
  170. s.db, err = s.connect()
  171. if err != nil {
  172. return err
  173. }
  174. err = s.prepareSql()
  175. if err != nil {
  176. return err
  177. }
  178. return nil
  179. }
  180. // Shutdown implements the Storage interface
  181. func (s *StoreSQL) Shutdown() (err error) {
  182. defer func() {
  183. closeErr := s.db.Close()
  184. if closeErr != err {
  185. backends.Log().WithError(err).Error("failed to close sql database")
  186. err = closeErr
  187. }
  188. }()
  189. for i := range s.statements {
  190. if err = s.statements[i].Close(); err != nil {
  191. backends.Log().WithError(err).Error("failed to close sql statement")
  192. }
  193. }
  194. return err
  195. }
  196. // GetEmail implements the Storage interface
  197. func (s *StoreSQL) GetEmail(mailID uint64) (*Email, error) {
  198. return &Email{}, nil
  199. }
  200. // GetChunk implements the Storage interface
  201. func (s *StoreSQL) GetChunks(hash ...HashKey) ([]*Chunk, error) {
  202. result := make([]*Chunk, 0, len(hash))
  203. return result, nil
  204. }