sqlite.go 3.5 KB

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