sqlite.go 4.0 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165
  1. package database
  2. import (
  3. "database/sql"
  4. "errors"
  5. "os"
  6. "path/filepath"
  7. _ "github.com/mattn/go-sqlite3" // need to blank import this package
  8. )
  9. // == sqlite ==
  10. const dbFilename = "netmaker.db"
  11. // SqliteDB is the db object for sqlite database connections
  12. var SqliteDB *sql.DB
  13. // SQLITE_FUNCTIONS - contains a map of the functions for sqlite
  14. var SQLITE_FUNCTIONS = map[string]interface{}{
  15. INIT_DB: initSqliteDB,
  16. CREATE_TABLE: sqliteCreateTable,
  17. INSERT: sqliteInsert,
  18. INSERT_PEER: sqliteInsertPeer,
  19. DELETE: sqliteDeleteRecord,
  20. DELETE_ALL: sqliteDeleteAllRecords,
  21. FETCH_ALL: sqliteFetchRecords,
  22. CLOSE_DB: sqliteCloseDB,
  23. isConnected: sqliteConnected,
  24. }
  25. func initSqliteDB() error {
  26. // == create db file if not present ==
  27. if _, err := os.Stat("data"); os.IsNotExist(err) {
  28. os.Mkdir("data", 0700)
  29. }
  30. dbFilePath := filepath.Join("data", dbFilename)
  31. if _, err := os.Stat(dbFilePath); os.IsNotExist(err) {
  32. os.Create(dbFilePath)
  33. }
  34. // == "connect" the database ==
  35. var dbOpenErr error
  36. SqliteDB, dbOpenErr = sql.Open("sqlite3", dbFilePath)
  37. if dbOpenErr != nil {
  38. return dbOpenErr
  39. }
  40. const q = `
  41. PRAGMA foreign_keys = OFF;
  42. PRAGMA synchronous = NORMAL;
  43. PRAGMA journal_mode = 'WAL';
  44. PRAGMA busy_timeout = 10000;
  45. `
  46. _, dbOpenErr = SqliteDB.Exec(q)
  47. if dbOpenErr != nil {
  48. return dbOpenErr
  49. }
  50. // if _, dbOpenErr = SqliteDB.Exec("PRAGMA synchronous = NORMAL"); dbOpenErr != nil {
  51. // return dbOpenErr
  52. // }
  53. // if _, dbOpenErr = SqliteDB.Exec("PRAGMA busy_timeout = 10000"); dbOpenErr != nil {
  54. // return dbOpenErr
  55. // }
  56. // if _, dbOpenErr = SqliteDB.Exec("PRAGMA foreign_keys = false"); dbOpenErr != nil {
  57. // return dbOpenErr
  58. // }
  59. // if _, dbOpenErr = SqliteDB.Exec("PRAGMA journal_mode = WAL"); dbOpenErr != nil {
  60. // return dbOpenErr
  61. // }
  62. SqliteDB.SetMaxOpenConns(1)
  63. return nil
  64. }
  65. func sqliteCreateTable(tableName string) error {
  66. statement, err := SqliteDB.Prepare("CREATE TABLE IF NOT EXISTS " + tableName + " (key TEXT NOT NULL UNIQUE PRIMARY KEY, value TEXT)")
  67. if err != nil {
  68. return err
  69. }
  70. defer statement.Close()
  71. _, err = statement.Exec()
  72. if err != nil {
  73. return err
  74. }
  75. return nil
  76. }
  77. func sqliteInsert(key string, value string, tableName string) error {
  78. if key != "" && value != "" && IsJSONString(value) {
  79. insertSQL := "INSERT OR REPLACE INTO " + tableName + " (key, value) VALUES (?, ?)"
  80. statement, err := SqliteDB.Prepare(insertSQL)
  81. if err != nil {
  82. return err
  83. }
  84. defer statement.Close()
  85. _, err = statement.Exec(key, value)
  86. if err != nil {
  87. return err
  88. }
  89. return nil
  90. }
  91. return errors.New("invalid insert " + key + " : " + value)
  92. }
  93. func sqliteInsertPeer(key string, value string) error {
  94. if key != "" && value != "" && IsJSONString(value) {
  95. err := sqliteInsert(key, value, PEERS_TABLE_NAME)
  96. if err != nil {
  97. return err
  98. }
  99. return nil
  100. }
  101. return errors.New("invalid peer insert " + key + " : " + value)
  102. }
  103. func sqliteDeleteRecord(tableName string, key string) error {
  104. deleteSQL := "DELETE FROM " + tableName + " WHERE key = \"" + key + "\""
  105. statement, err := SqliteDB.Prepare(deleteSQL)
  106. if err != nil {
  107. return err
  108. }
  109. defer statement.Close()
  110. if _, err = statement.Exec(); err != nil {
  111. return err
  112. }
  113. return nil
  114. }
  115. func sqliteDeleteAllRecords(tableName string) error {
  116. deleteSQL := "DELETE FROM " + tableName
  117. statement, err := SqliteDB.Prepare(deleteSQL)
  118. if err != nil {
  119. return err
  120. }
  121. defer statement.Close()
  122. if _, err = statement.Exec(); err != nil {
  123. return err
  124. }
  125. return nil
  126. }
  127. func sqliteFetchRecords(tableName string) (map[string]string, error) {
  128. row, err := SqliteDB.Query("SELECT * FROM " + tableName + " ORDER BY key")
  129. if err != nil {
  130. return nil, err
  131. }
  132. records := make(map[string]string)
  133. defer row.Close()
  134. for row.Next() { // Iterate and fetch the records from result cursor
  135. var key string
  136. var value string
  137. row.Scan(&key, &value)
  138. records[key] = value
  139. }
  140. if len(records) == 0 {
  141. return nil, errors.New(NO_RECORDS)
  142. }
  143. return records, nil
  144. }
  145. func sqliteCloseDB() {
  146. SqliteDB.Close()
  147. }
  148. func sqliteConnected() bool {
  149. stats := SqliteDB.Stats()
  150. return stats.OpenConnections > 0
  151. }