package datamaps import ( "database/sql" "errors" "fmt" "log" "os" "time" // Needed for the sqlite3 driver _ "github.com/mattn/go-sqlite3" ) // SetupDB creates the intitial database func SetupDB(path string) (*sql.DB, error) { stmtBase := `DROP TABLE IF EXISTS datamap; CREATE TABLE datamap(id INTEGER PRIMARY KEY, name TEXT, date_created TEXT); DROP TABLE IF EXISTS datamap_line; CREATE TABLE datamap_line( id INTEGER PRIMARY KEY, dm_id INTEGER, key TEXT NOT NULL, sheet TEXT NOT NULL, cellref TEXT, FOREIGN KEY (dm_id) REFERENCES datamap(id) ON DELETE CASCADE ); ` os.Create(path) db, err := sql.Open("sqlite3", path) if err != nil { return db, errors.New("Cannot open that damn database file") } // We probably don't need pragma here but we have it for later. pragma := "PRAGMA foreign_keys = ON;" _, err = db.Exec(pragma) if err != nil { // log.Printf("%q: %s\n", err, pragma) return nil, err } _, err = db.Exec(stmtBase) if err != nil { // log.Printf("%q: %s\n", err, stmt_base) return nil, err } return db, nil } // DatamapToDB takes a slice of DatamapLine and writes it to a sqlite3 db file. // func DatafmapToDB(d_path string, data []DatamapLine, dm_name string, dm_path string) error { func DatamapToDB(opts *Options) error { fmt.Printf("Importing datamap file %s and naming it %s.\n", opts.DMPath, opts.DMName) data, err := ReadDML(opts.DMPath) if err != nil { log.Fatal(err) } d, err := sql.Open("sqlite3", opts.DBPath) if err != nil { return errors.New("Cannot open that damn database file") } tx, err := d.Begin() if err != nil { return err } pragma := "PRAGMA foreign_keys = ON;" _, err = d.Exec(pragma) if err != nil { log.Printf("%q: %s\n", err, pragma) return err } stmtDm, err := tx.Prepare("INSERT INTO datamap (name, date_created) VALUES(?,?)") if err != nil { return err } res, err := stmtDm.Exec(opts.DMName, time.Now()) if err != nil { return err } lastId, err := res.LastInsertId() if err != nil { return err } stmtDml, err := tx.Prepare("INSERT INTO datamap_line (dm_id, key, sheet, cellref) VALUES(?,?,?,?);") if err != nil { return err } defer stmtDm.Close() defer stmtDml.Close() for _, dml := range data { _, err = stmtDml.Exec(lastId, dml.Key, dml.Sheet, dml.Cellref) if err != nil { return err } } err = tx.Commit() if err != nil { return err } return nil }