2017-09-03 04:50:31 +00:00
/ *
*
* Gosora MySQL Interface
2019-02-23 06:55:34 +00:00
* Copyright Azareal 2017 - 2020
2017-09-03 04:50:31 +00:00
*
* /
2017-10-14 07:39:22 +00:00
package install
2017-07-12 11:05:18 +00:00
import (
2022-02-21 03:53:13 +00:00
"bytes"
"database/sql"
"fmt"
"io/ioutil"
"os"
"path/filepath"
"strconv"
"strings"
qgen "git.tuxpa.in/a/gosora/query_gen"
_ "github.com/go-sql-driver/mysql"
2017-07-12 11:05:18 +00:00
)
2017-09-03 04:50:31 +00:00
//var dbCollation string = "utf8mb4_general_ci"
2017-07-12 11:05:18 +00:00
2017-10-14 07:39:22 +00:00
func init ( ) {
2022-02-21 03:53:13 +00:00
adapters [ "mysql" ] = & MysqlInstaller { dbHost : "" }
2017-07-12 11:05:18 +00:00
}
2017-10-14 07:39:22 +00:00
type MysqlInstaller struct {
2022-02-21 03:53:13 +00:00
db * sql . DB
dbHost string
dbUsername string
dbPassword string
dbName string
dbPort string
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) SetConfig ( dbHost string , dbUsername string , dbPassword string , dbName string , dbPort string ) {
2022-02-21 03:53:13 +00:00
ins . dbHost = dbHost
ins . dbUsername = dbUsername
ins . dbPassword = dbPassword
ins . dbName = dbName
ins . dbPort = dbPort
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) Name ( ) string {
2022-02-21 03:53:13 +00:00
return "mysql"
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) DefaultPort ( ) string {
2022-02-21 03:53:13 +00:00
return "3306"
2017-10-14 07:39:22 +00:00
}
2017-11-13 05:22:37 +00:00
func ( ins * MysqlInstaller ) dbExists ( dbName string ) ( bool , error ) {
2022-02-21 03:53:13 +00:00
var waste string
err := ins . db . QueryRow ( "SHOW DATABASES LIKE '" + dbName + "'" ) . Scan ( & waste )
if err != nil && err != sql . ErrNoRows {
return false , err
} else if err == sql . ErrNoRows {
return false , nil
}
return true , nil
2017-11-13 05:22:37 +00:00
}
2017-10-14 07:39:22 +00:00
func ( ins * MysqlInstaller ) InitDatabase ( ) ( err error ) {
2022-02-21 03:53:13 +00:00
_dbPassword := ins . dbPassword
if _dbPassword != "" {
_dbPassword = ":" + _dbPassword
}
db , err := sql . Open ( "mysql" , ins . dbUsername + _dbPassword + "@tcp(" + ins . dbHost + ":" + ins . dbPort + ")/" )
if err != nil {
return err
}
// Make sure that the connection is alive..
err = db . Ping ( )
if err != nil {
return err
}
fmt . Println ( "Successfully connected to the database" )
ins . db = db
ok , err := ins . dbExists ( ins . dbName )
if err != nil {
return err
}
if ! ok {
fmt . Println ( "Unable to find the database. Attempting to create it" )
_ , err = db . Exec ( "CREATE DATABASE IF NOT EXISTS " + ins . dbName )
if err != nil {
return err
}
fmt . Println ( "The database was successfully created" )
}
/ * fmt . Println ( "Switching to database " , ins . dbName )
_ , err = db . Exec ( "USE " + ins . dbName )
if err != nil {
return err
} * /
db . Close ( )
db , err = sql . Open ( "mysql" , ins . dbUsername + _dbPassword + "@tcp(" + ins . dbHost + ":" + ins . dbPort + ")/" + ins . dbName )
if err != nil {
return err
}
// Make sure that the connection is alive..
err = db . Ping ( )
if err != nil {
return err
}
fmt . Println ( "Successfully connected to the database" )
// Ready the query builder
ins . db = db
qgen . Builder . SetConn ( db )
return qgen . Builder . SetAdapter ( "mysql" )
2017-07-12 11:05:18 +00:00
}
2019-05-19 06:28:33 +00:00
func ( ins * MysqlInstaller ) createTable ( f os . FileInfo ) error {
2022-02-21 03:53:13 +00:00
table := strings . TrimPrefix ( f . Name ( ) , "query_" )
ext := filepath . Ext ( table )
if ext != ".sql" {
return nil
}
table = strings . TrimSuffix ( table , ext )
// ? - This is mainly here for tests, although it might allow the installer to overwrite a production database, so we might want to proceed with caution
q := "DROP TABLE IF EXISTS `" + table + "`;"
_ , err := ins . db . Exec ( q )
if err != nil {
fmt . Println ( "Failed query:" , q )
fmt . Println ( "e:" , err )
return err
}
data , err := ioutil . ReadFile ( "./schema/mysql/" + f . Name ( ) )
if err != nil {
return err
}
data = bytes . TrimSpace ( data )
q = string ( data )
_ , err = ins . db . Exec ( q )
if err != nil {
fmt . Println ( "Failed query:" , q )
fmt . Println ( "e:" , err )
return err
}
fmt . Printf ( "Created table '%s'\n" , table )
return nil
2019-05-06 04:04:00 +00:00
}
2017-10-21 00:27:47 +00:00
func ( ins * MysqlInstaller ) TableDefs ( ) ( err error ) {
2022-02-21 03:53:13 +00:00
fmt . Println ( "Creating the tables" )
files , err := ioutil . ReadDir ( "./schema/mysql/" )
if err != nil {
return err
}
_ , err = ins . db . Exec ( "SET FOREIGN_KEY_CHECKS = 0;" )
if err != nil {
return err
}
for _ , f := range files {
if ! strings . HasPrefix ( f . Name ( ) , "query_" ) {
continue
}
err := ins . createTable ( f )
if err != nil {
return err
}
}
_ , err = ins . db . Exec ( "SET FOREIGN_KEY_CHECKS = 1;" )
return err
2017-07-12 11:05:18 +00:00
}
2017-09-22 05:36:15 +00:00
// ? - Moved this here since it was breaking the installer, we need to add this at some point
/* TODO: Implement the html-attribute setting type before deploying this */
/*INSERT INTO settings(`name`,`content`,`type`) VALUES ('meta_desc','','html-attribute');*/
2017-10-14 07:39:22 +00:00
func ( ins * MysqlInstaller ) InitialData ( ) error {
2022-02-21 03:53:13 +00:00
fmt . Println ( "Seeding the tables" )
data , err := ioutil . ReadFile ( "./schema/mysql/inserts.sql" )
if err != nil {
return err
}
data = bytes . TrimSpace ( data )
statements := bytes . Split ( data , [ ] byte ( ";" ) )
for key , sBytes := range statements {
statement := string ( sBytes )
if statement == "" {
continue
}
statement += ";"
fmt . Println ( "Executing query #" + strconv . Itoa ( key ) + " " + statement )
_ , err = ins . db . Exec ( statement )
if err != nil {
return err
}
}
return nil
2017-07-12 11:05:18 +00:00
}
2017-10-14 07:39:22 +00:00
func ( ins * MysqlInstaller ) CreateAdmin ( ) error {
2022-02-21 03:53:13 +00:00
return createAdmin ( )
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) DBHost ( ) string {
2022-02-21 03:53:13 +00:00
return ins . dbHost
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) DBUsername ( ) string {
2022-02-21 03:53:13 +00:00
return ins . dbUsername
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) DBPassword ( ) string {
2022-02-21 03:53:13 +00:00
return ins . dbPassword
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) DBName ( ) string {
2022-02-21 03:53:13 +00:00
return ins . dbName
2017-10-14 07:39:22 +00:00
}
func ( ins * MysqlInstaller ) DBPort ( ) string {
2022-02-21 03:53:13 +00:00
return ins . dbPort
2017-10-14 07:39:22 +00:00
}