2021-02-15 20:37:16 +01:00
package utils
import (
2021-06-13 12:03:36 +02:00
"bufio"
2021-02-15 20:37:16 +01:00
"bytes"
2022-02-06 15:37:50 +01:00
"database/sql"
2022-02-07 21:20:09 +01:00
"encoding/json"
2021-02-20 19:09:41 +01:00
"fmt"
2021-02-20 22:43:27 +01:00
"io"
2021-02-15 20:37:16 +01:00
"log"
2021-02-20 22:43:27 +01:00
"os"
2021-02-15 20:37:16 +01:00
"os/exec"
2021-06-13 12:03:36 +02:00
"strings"
2021-05-30 13:28:42 +02:00
"time"
2021-02-20 19:09:41 +01:00
2022-02-07 21:20:09 +01:00
"github.com/TylerBrock/colorjson"
2021-02-17 12:10:15 +01:00
"github.com/joho/godotenv"
2021-02-15 20:37:16 +01:00
)
2021-02-20 22:43:27 +01:00
// ExecAndStream : Runs a terminal command, but streams progress instead of outputting. Ideal for long lived process that need to be logged.
2021-06-13 12:03:36 +02:00
func ExecAndStream ( path string , command string , args [ ] string ) {
2021-02-20 22:43:27 +01:00
cmd := exec . Command ( command , args ... )
var stdoutBuf , stderrBuf bytes . Buffer
cmd . Stdout = io . MultiWriter ( os . Stdout , & stdoutBuf )
cmd . Stderr = io . MultiWriter ( os . Stderr , & stderrBuf )
2021-06-13 12:03:36 +02:00
cmd . Dir = path
2021-02-20 22:43:27 +01:00
err := cmd . Run ( )
if err != nil {
2021-06-13 12:03:36 +02:00
fmt . Printf ( "cmd.Run() failed with %s\n" , err )
2021-02-20 22:43:27 +01:00
}
outStr , errStr := string ( stdoutBuf . Bytes ( ) ) , string ( stderrBuf . Bytes ( ) )
fmt . Printf ( "\nout:\n%s\nerr:\n%s\n" , outStr , errStr )
}
2021-02-15 20:37:16 +01:00
// Exec : Runs a terminal Command, catches and logs errors, returns the result.
2021-06-13 12:03:36 +02:00
func Exec ( path string , command string , args [ ] string ) string {
2021-02-15 20:37:16 +01:00
cmd := exec . Command ( command , args ... )
var out bytes . Buffer
var stderr bytes . Buffer
cmd . Stdout = & out
cmd . Stderr = & stderr
2021-06-13 12:03:36 +02:00
cmd . Dir = path
2021-02-15 20:37:16 +01:00
err := cmd . Run ( )
if err != nil {
2021-02-17 13:34:53 +01:00
// TODO: Deal with possibility of error in command, allow explicit error handling and return proper formatted stderr
2021-02-21 00:55:59 +01:00
// log.Println(fmt.Sprint(err) + ": " + stderr.String()) // ... Silence...
2021-02-15 20:37:16 +01:00
}
2021-02-17 01:08:10 +01:00
2021-02-21 00:55:59 +01:00
// log.Println("Result: " + out.String()) // ... Silence ...
2021-02-15 20:37:16 +01:00
2021-06-13 12:03:36 +02:00
return strings . Trim ( out . String ( ) , " \n" )
2021-02-17 01:08:10 +01:00
}
2021-06-13 12:03:36 +02:00
// Exec : Runs a terminal Command, returns the result as a *bufio.Scanner type, split in lines and ready to parse.
func ExecAndGetLines ( path string , command string , args [ ] string ) * bufio . Scanner {
cmdOutput := Exec ( path , command , args )
cmdOutputReader := strings . NewReader ( cmdOutput )
scanner := bufio . NewScanner ( cmdOutputReader )
scanner . Split ( bufio . ScanLines )
return scanner
}
2021-02-17 01:08:10 +01:00
// DeleteEmptySlices : Given a string array, delete empty entries.
func DeleteEmptySlices ( s [ ] string ) [ ] string {
var r [ ] string
for _ , str := range s {
if str != "" {
r = append ( r , str )
}
}
return r
2021-02-15 20:37:16 +01:00
}
2021-02-17 12:10:15 +01:00
2021-05-02 10:52:32 +02:00
// GetSQLiteDbConnectionDetails : Returns the necessary string as connection info for SQL.db()
func GetSQLiteDbConnectionDetails ( ) string {
var apiEnv map [ string ] string
apiEnv , err := godotenv . Read ( GetPath ( "apiEnvFileLocation" ) )
if err != nil {
log . Fatal ( "Error loading .env file" )
}
return apiEnv [ "SQLITE_DATABASE" ] // Will read from api project edgebox.env file
}
2022-02-14 18:23:08 +01:00
// GetFormattedDateTime: Given a Time, Returns a string that is formatted to be passed around the application, including correctly inserting SQLite Datetime field using sql.Prepare.
2022-02-06 18:10:02 +01:00
func GetFormattedDateTime ( t time . Time ) string {
2021-05-30 13:28:42 +02:00
// This date is used to indicate the layout.
const datetimeLayout = "2006-01-02 15:04:05"
formatedDatetime := t . Format ( datetimeLayout )
return formatedDatetime
}
2021-02-17 12:35:22 +01:00
// GetPath : Returns either the hardcoded path, or a overwritten value via .env file at project root. Register paths here for seamless working code between dev and prod environments ;)
func GetPath ( pathKey string ) string {
// Read whole of .env file to map.
var env map [ string ] string
env , err := godotenv . Read ( )
2021-06-13 12:03:36 +02:00
var targetPath string
2021-02-17 12:35:22 +01:00
if err != nil {
2021-06-13 12:03:36 +02:00
targetPath = ""
2021-02-17 12:35:22 +01:00
}
switch pathKey {
2022-02-06 15:37:50 +01:00
case "cloudEnvFileLocation" :
if env [ "CLOUD_ENV_FILE_LOCATION" ] != "" {
targetPath = env [ "CLOUD_ENV_FILE_LOCATION" ]
} else {
targetPath = "/home/system/components/edgeboxctl/cloud.env"
}
2021-02-17 12:35:22 +01:00
case "apiEnvFileLocation" :
if env [ "API_ENV_FILE_LOCATION" ] != "" {
targetPath = env [ "API_ENV_FILE_LOCATION" ]
} else {
targetPath = "/home/system/components/api/edgebox.env"
}
2022-02-06 15:37:50 +01:00
case "apiPath" :
if env [ "API_PATH" ] != "" {
targetPath = env [ "APT_PATH" ]
} else {
targetPath = "/home/system/components/api/"
}
2021-02-17 12:35:22 +01:00
case "edgeAppsPath" :
if env [ "EDGEAPPS_PATH" ] != "" {
targetPath = env [ "EDGEAPPS_PATH" ]
} else {
targetPath = "/home/system/components/apps/"
}
case "wsPath" :
if env [ "WS_PATH" ] != "" {
targetPath = env [ "WS_PATH" ]
} else {
targetPath = "/home/system/components/ws/"
}
default :
log . Printf ( "path_key %s nonexistant in GetPath().\n" , pathKey )
}
return targetPath
}
2022-02-06 15:37:50 +01:00
// WriteOption : Writes a key value pair option into the api shared database
func WriteOption ( optionKey string , optionValue string ) {
db , err := sql . Open ( "sqlite3" , GetSQLiteDbConnectionDetails ( ) )
if err != nil {
log . Fatal ( err . Error ( ) )
}
statement , err := db . Prepare ( "REPLACE into option (name, value, created, updated) VALUES (?, ?, ?, ?);" ) // Prepare SQL Statement
if err != nil {
log . Fatal ( err . Error ( ) )
}
2022-02-06 18:10:02 +01:00
formatedDatetime := GetFormattedDateTime ( time . Now ( ) )
2022-02-06 15:37:50 +01:00
_ , err = statement . Exec ( optionKey , optionValue , formatedDatetime , formatedDatetime ) // Execute SQL Statement
if err != nil {
log . Fatal ( err . Error ( ) )
}
db . Close ( )
}
2022-02-07 21:20:09 +01:00
2022-02-14 18:23:08 +01:00
// IsSystemReady : Checks hability of the service to execute commands (Only after "edgebox --build" is ran at least once via SSH, or if built for distribution)
func IsSystemReady ( ) bool {
2022-02-14 21:33:10 +01:00
_ , err := os . Stat ( GetPath ( "wsPath" ) + ".ready" )
2022-02-14 18:23:08 +01:00
return ! os . IsNotExist ( err )
}
// IsDatabaseReady : Checks is it can successfully connect to the task queue db
func IsDatabaseReady ( ) bool {
return false
}
2022-02-07 21:20:09 +01:00
func ColorJsonString ( jsonString string ) string {
var obj map [ string ] interface { }
json . Unmarshal ( [ ] byte ( jsonString ) , & obj )
resultJSON , _ := colorjson . Marshal ( obj )
return string ( resultJSON )
}