Semaphore/util/config.go

276 lines
6.5 KiB
Go
Raw Normal View History

2016-01-05 00:32:53 +01:00
package util
import (
"encoding/base64"
2016-01-05 00:32:53 +01:00
"encoding/json"
2020-11-28 22:49:44 +01:00
"errors"
2016-01-05 00:32:53 +01:00
"fmt"
"net/url"
2016-03-16 22:49:43 +01:00
"os"
2016-04-24 20:11:43 +02:00
"path"
2016-03-16 22:49:43 +01:00
"io"
"strings"
"github.com/gorilla/securecookie"
2016-01-05 00:32:53 +01:00
)
// Cookie is a runtime generated secure cookie used for authentication
var Cookie *securecookie.SecureCookie
// WebHostURL is the public route to the semaphore server
2017-05-20 16:14:36 +02:00
var WebHostURL *url.URL
2016-01-05 00:32:53 +01:00
2020-11-28 22:49:44 +01:00
type DbDriver int
const (
DbDriverMySQL DbDriver = iota
DbDriverBolt
2021-08-24 12:51:21 +02:00
DbDriverPostgres
2020-11-28 22:49:44 +01:00
)
type DbConfig struct {
Dialect DbDriver `json:"-"`
Hostname string `json:"host"`
Username string `json:"user"`
Password string `json:"pass"`
DbName string `json:"name"`
2016-01-05 00:32:53 +01:00
}
type ldapMappings struct {
DN string `json:"dn"`
Mail string `json:"mail"`
UID string `json:"uid"`
CN string `json:"cn"`
}
//ConfigType mapping between Config and the json file that sets it
type ConfigType struct {
MySQL DbConfig `json:"mysql"`
BoltDb DbConfig `json:"bolt"`
2021-08-24 17:20:34 +02:00
Postgres DbConfig `json:"pgsql"`
2020-11-28 22:49:44 +01:00
2016-01-05 00:32:53 +01:00
// Format `:port_num` eg, :3000
2018-05-14 21:37:07 +02:00
// if : is missing it will be corrected
2017-05-20 16:14:36 +02:00
Port string `json:"port"`
2016-01-05 00:32:53 +01:00
2018-05-14 21:37:07 +02:00
// Interface ip, put in front of the port.
// defaults to empty
Interface string `json:"interface"`
// semaphore stores ephemeral projects here
2016-01-05 00:32:53 +01:00
TmpPath string `json:"tmp_path"`
// cookie hashing & encryption
CookieHash string `json:"cookie_hash"`
CookieEncryption string `json:"cookie_encryption"`
2017-02-22 09:46:42 +01:00
2017-04-18 16:36:09 +02:00
// email alerting
2017-02-22 09:46:42 +01:00
EmailSender string `json:"email_sender"`
EmailHost string `json:"email_host"`
EmailPort string `json:"email_port"`
2017-04-18 16:36:09 +02:00
// web host
2017-02-22 09:46:42 +01:00
WebHost string `json:"web_host"`
2017-04-18 16:36:09 +02:00
// ldap settings
LdapBindDN string `json:"ldap_binddn"`
LdapBindPassword string `json:"ldap_bindpassword"`
LdapServer string `json:"ldap_server"`
LdapSearchDN string `json:"ldap_searchdn"`
LdapSearchFilter string `json:"ldap_searchfilter"`
LdapMappings ldapMappings `json:"ldap_mappings"`
2017-04-04 13:49:00 +02:00
2017-04-18 16:36:09 +02:00
// telegram alerting
2017-03-22 08:22:09 +01:00
TelegramChat string `json:"telegram_chat"`
TelegramToken string `json:"telegram_token"`
// task concurrency
ConcurrencyMode string `json:"concurrency_mode"`
MaxParallelTasks int `json:"max_parallel_tasks"`
// configType field ordering with bools at end reduces struct size
// (maligned check)
// feature switches
EmailAlert bool `json:"email_alert"`
TelegramAlert bool `json:"telegram_alert"`
LdapEnable bool `json:"ldap_enable"`
LdapNeedTLS bool `json:"ldap_needtls"`
2016-01-05 00:32:53 +01:00
}
//Config exposes the application configuration storage for use in the application
var Config *ConfigType
// ToJSON returns a JSON string of the config
2021-08-25 22:12:19 +02:00
func (conf *ConfigType) ToJSON() ([]byte, error) {
return json.MarshalIndent(&conf, " ", "\t")
}
2016-01-05 00:32:53 +01:00
// ConfigInit reads in cli flags, and switches actions appropriately on them
2021-08-25 22:12:19 +02:00
func ConfigInit(configPath string) {
loadConfig(configPath)
validateConfig()
var encryption []byte
hash, _ := base64.StdEncoding.DecodeString(Config.CookieHash)
if len(Config.CookieEncryption) > 0 {
encryption, _ = base64.StdEncoding.DecodeString(Config.CookieEncryption)
}
Cookie = securecookie.New(hash, encryption)
2017-05-20 16:14:36 +02:00
WebHostURL, _ = url.Parse(Config.WebHost)
2017-05-20 16:25:41 +02:00
if len(WebHostURL.String()) == 0 {
WebHostURL = nil
}
2016-01-05 00:32:53 +01:00
}
2021-08-25 22:12:19 +02:00
func loadConfig(configPath string) {
//If the configPath option has been set try to load and decode it
var usedPath string
2021-08-25 22:12:19 +02:00
if len(configPath) > 0 {
path := configPath
file, err := os.Open(path)
exitOnConfigError(err)
decodeConfig(file)
usedPath = path
} else {
// if no configPath look in the cwd
cwd, err := os.Getwd()
exitOnConfigError(err)
defaultPath := path.Join(cwd, "config.json")
file, err := os.Open(defaultPath)
exitOnConfigError(err)
decodeConfig(file)
usedPath = defaultPath
}
fmt.Println("Using config file: " + usedPath)
}
func validateConfig() {
validatePort()
if len(Config.TmpPath) == 0 {
Config.TmpPath = "/tmp/semaphore"
}
if Config.MaxParallelTasks < 1 {
Config.MaxParallelTasks = 10
}
}
func validatePort() {
//TODO - why do we do this only with this variable?
if len(os.Getenv("PORT")) > 0 {
Config.Port = ":" + os.Getenv("PORT")
}
if len(Config.Port) == 0 {
Config.Port = ":3000"
}
if !strings.HasPrefix(Config.Port, ":") {
Config.Port = ":" + Config.Port
}
}
func exitOnConfigError(err error) {
if err != nil {
fmt.Println("Cannot Find configuration! Use -config parameter to point to a JSON file generated by -setup.\n\n Hint: have you run `-setup` ?")
os.Exit(1)
}
}
func decodeConfig(file io.Reader) {
if err := json.NewDecoder(file).Decode(&Config); err != nil {
fmt.Println("Could not decode configuration!")
panic(err)
}
}
2021-08-24 17:20:34 +02:00
// String returns dialect name for GORP.
// TODO: It should be moved to sql package.
2020-11-28 22:49:44 +01:00
func (d DbDriver) String() string {
2021-08-24 17:20:34 +02:00
return [...]string{
"mysql",
"", // didn't support by BoltDB
"postgres",
}[d]
2020-11-28 22:49:44 +01:00
}
func (d *DbConfig) IsPresent() bool {
2020-11-28 22:49:44 +01:00
return d.Hostname != ""
}
func (d *DbConfig) HasSupportMultipleDatabases() bool {
2020-12-04 09:39:56 +01:00
return true
2020-11-28 22:49:44 +01:00
}
func (d *DbConfig) GetConnectionString(includeDbName bool) (connectionString string, err error) {
switch d.Dialect {
case DbDriverBolt:
connectionString = d.Hostname
2020-11-28 22:49:44 +01:00
case DbDriverMySQL:
if includeDbName {
connectionString = fmt.Sprintf(
"%s:%s@tcp(%s)/%s?parseTime=true&interpolateParams=true",
d.Username,
d.Password,
d.Hostname,
d.DbName)
} else {
connectionString = fmt.Sprintf(
"%s:%s@tcp(%s)/?parseTime=true&interpolateParams=true",
2020-11-28 22:49:44 +01:00
d.Username,
d.Password,
d.Hostname)
}
2021-08-24 17:20:34 +02:00
case DbDriverPostgres:
if includeDbName {
connectionString = fmt.Sprintf(
"postgres://%s:%s@%s/%s",
d.Username,
d.Password,
d.Hostname,
d.DbName)
} else {
connectionString = fmt.Sprintf(
"postgres://%s:%s@%s/",
d.Username,
d.Password,
d.Hostname)
}
2020-11-28 22:49:44 +01:00
default:
err = fmt.Errorf("unsupported database driver: %s", d.Dialect)
}
return
}
func (conf *ConfigType) GetDBConfig() (dbConfig DbConfig, err error) {
switch {
case conf.MySQL.IsPresent():
2020-11-28 22:49:44 +01:00
dbConfig = conf.MySQL
dbConfig.Dialect = DbDriverMySQL
case conf.BoltDb.IsPresent():
dbConfig = conf.BoltDb
dbConfig.Dialect = DbDriverBolt
2021-08-24 17:20:34 +02:00
case conf.Postgres.IsPresent():
dbConfig = conf.Postgres
dbConfig.Dialect = DbDriverPostgres
2020-11-28 22:49:44 +01:00
default:
err = errors.New("database configuration not found")
}
return
}
//GenerateCookieSecrets generates cookie secret during setup
func (conf *ConfigType) GenerateCookieSecrets() {
hash := securecookie.GenerateRandomKey(32)
encryption := securecookie.GenerateRandomKey(32)
2016-01-05 00:32:53 +01:00
conf.CookieHash = base64.StdEncoding.EncodeToString(hash)
conf.CookieEncryption = base64.StdEncoding.EncodeToString(encryption)
2016-01-05 00:32:53 +01:00
}