2020-09-29 19:55:07 +03:00
|
|
|
|
package config
|
2019-05-05 23:36:49 +03:00
|
|
|
|
|
2020-05-24 18:32:26 +03:00
|
|
|
|
import (
|
|
|
|
|
"encoding/json"
|
2020-10-04 11:37:38 +03:00
|
|
|
|
"flag"
|
2020-11-01 22:14:10 +03:00
|
|
|
|
"fmt"
|
2021-01-30 13:17:37 +03:00
|
|
|
|
"github.com/emvi/logbuch"
|
2020-05-24 18:32:26 +03:00
|
|
|
|
"github.com/gorilla/securecookie"
|
2020-10-04 11:37:38 +03:00
|
|
|
|
"github.com/jinzhu/configor"
|
2021-01-22 15:50:46 +03:00
|
|
|
|
"github.com/markbates/pkger"
|
2020-09-29 19:55:07 +03:00
|
|
|
|
"github.com/muety/wakapi/models"
|
2020-05-30 21:41:27 +03:00
|
|
|
|
migrate "github.com/rubenv/sql-migrate"
|
2020-11-01 22:14:10 +03:00
|
|
|
|
"gorm.io/driver/mysql"
|
|
|
|
|
"gorm.io/driver/postgres"
|
|
|
|
|
"gorm.io/driver/sqlite"
|
|
|
|
|
"gorm.io/gorm"
|
2020-05-24 18:32:26 +03:00
|
|
|
|
"io/ioutil"
|
2020-11-22 00:30:56 +03:00
|
|
|
|
"net/http"
|
2020-05-24 18:32:26 +03:00
|
|
|
|
"os"
|
|
|
|
|
"strings"
|
|
|
|
|
)
|
|
|
|
|
|
2020-10-04 12:14:44 +03:00
|
|
|
|
const (
|
2021-02-03 00:52:13 +03:00
|
|
|
|
defaultConfigPath = "config.yml"
|
2020-11-06 19:20:26 +03:00
|
|
|
|
|
|
|
|
|
SQLDialectMysql = "mysql"
|
|
|
|
|
SQLDialectPostgres = "postgres"
|
|
|
|
|
SQLDialectSqlite = "sqlite3"
|
2021-01-17 11:24:09 +03:00
|
|
|
|
|
|
|
|
|
KeyLatestTotalTime = "latest_total_time"
|
|
|
|
|
KeyLatestTotalUsers = "latest_total_users"
|
2020-10-04 12:14:44 +03:00
|
|
|
|
)
|
|
|
|
|
|
2021-01-30 12:54:54 +03:00
|
|
|
|
const (
|
|
|
|
|
WakatimeApiUrl = "https://wakatime.com/api/v1"
|
|
|
|
|
WakatimeApiHeartbeatsEndpoint = "/users/current/heartbeats.bulk"
|
|
|
|
|
WakatimeApiUserEndpoint = "/users/current"
|
|
|
|
|
)
|
|
|
|
|
|
2020-11-08 12:12:49 +03:00
|
|
|
|
var cfg *Config
|
|
|
|
|
var cFlag = flag.String("config", defaultConfigPath, "config file location")
|
2020-05-24 14:41:19 +03:00
|
|
|
|
|
2020-10-04 12:14:44 +03:00
|
|
|
|
type appConfig struct {
|
2021-01-30 11:51:36 +03:00
|
|
|
|
AggregationTime string `yaml:"aggregation_time" default:"02:15" env:"WAKAPI_AGGREGATION_TIME"`
|
|
|
|
|
CountingTime string `yaml:"counting_time" default:"05:15" env:"WAKAPI_COUNTING_TIME"`
|
|
|
|
|
CustomLanguages map[string]string `yaml:"custom_languages"`
|
|
|
|
|
Colors map[string]map[string]string `yaml:"-"`
|
2020-10-04 12:14:44 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type securityConfig struct {
|
|
|
|
|
// this is actually a pepper (https://en.wikipedia.org/wiki/Pepper_(cryptography))
|
|
|
|
|
PasswordSalt string `yaml:"password_salt" default:"" env:"WAKAPI_PASSWORD_SALT"`
|
|
|
|
|
InsecureCookies bool `yaml:"insecure_cookies" default:"false" env:"WAKAPI_INSECURE_COOKIES"`
|
2020-11-22 00:30:56 +03:00
|
|
|
|
CookieMaxAgeSec int `yaml:"cookie_max_age" default:"172800" env:"WAKAPI_COOKIE_MAX_AGE"`
|
2020-10-04 12:14:44 +03:00
|
|
|
|
SecureCookie *securecookie.SecureCookie `yaml:"-"`
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type dbConfig struct {
|
|
|
|
|
Host string `env:"WAKAPI_DB_HOST"`
|
|
|
|
|
Port uint `env:"WAKAPI_DB_PORT"`
|
|
|
|
|
User string `env:"WAKAPI_DB_USER"`
|
|
|
|
|
Password string `env:"WAKAPI_DB_PASSWORD"`
|
|
|
|
|
Name string `default:"wakapi_db.db" env:"WAKAPI_DB_NAME"`
|
2021-01-18 23:34:08 +03:00
|
|
|
|
Dialect string `yaml:"-"`
|
|
|
|
|
Type string `yaml:"dialect" default:"sqlite3" env:"WAKAPI_DB_TYPE"`
|
2020-10-04 12:14:44 +03:00
|
|
|
|
MaxConn uint `yaml:"max_conn" default:"2" env:"WAKAPI_DB_MAX_CONNECTIONS"`
|
2021-01-18 23:34:08 +03:00
|
|
|
|
Ssl bool `default:"false" env:"WAKAPI_DB_SSL"`
|
2020-10-04 12:14:44 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
type serverConfig struct {
|
2020-12-13 00:07:00 +03:00
|
|
|
|
Port int `default:"3000" env:"WAKAPI_PORT"`
|
|
|
|
|
ListenIpV4 string `yaml:"listen_ipv4" default:"127.0.0.1" env:"WAKAPI_LISTEN_IPV4"`
|
|
|
|
|
ListenIpV6 string `yaml:"listen_ipv6" default:"::1" env:"WAKAPI_LISTEN_IPV6"`
|
|
|
|
|
BasePath string `yaml:"base_path" default:"/" env:"WAKAPI_BASE_PATH"`
|
|
|
|
|
TlsCertPath string `yaml:"tls_cert_path" default:"" env:"WAKAPI_TLS_CERT_PATH"`
|
|
|
|
|
TlsKeyPath string `yaml:"tls_key_path" default:"" env:"WAKAPI_TLS_KEY_PATH"`
|
2020-10-04 12:14:44 +03:00
|
|
|
|
}
|
|
|
|
|
|
2019-05-05 23:36:49 +03:00
|
|
|
|
type Config struct {
|
2020-10-04 12:14:44 +03:00
|
|
|
|
Env string `default:"dev" env:"ENVIRONMENT"`
|
|
|
|
|
Version string `yaml:"-"`
|
|
|
|
|
App appConfig
|
|
|
|
|
Security securityConfig
|
|
|
|
|
Db dbConfig
|
|
|
|
|
Server serverConfig
|
2019-05-05 23:36:49 +03:00
|
|
|
|
}
|
2020-02-20 17:39:56 +03:00
|
|
|
|
|
2020-11-22 00:30:56 +03:00
|
|
|
|
func (c *Config) CreateCookie(name, value, path string) *http.Cookie {
|
|
|
|
|
return c.createCookie(name, value, path, c.Security.CookieMaxAgeSec)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *Config) GetClearCookie(name, path string) *http.Cookie {
|
|
|
|
|
return c.createCookie(name, "", path, -1)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *Config) createCookie(name, value, path string, maxAge int) *http.Cookie {
|
|
|
|
|
return &http.Cookie{
|
|
|
|
|
Name: name,
|
|
|
|
|
Value: value,
|
|
|
|
|
Path: path,
|
|
|
|
|
MaxAge: maxAge,
|
|
|
|
|
Secure: !c.Security.InsecureCookies,
|
|
|
|
|
HttpOnly: true,
|
|
|
|
|
SameSite: http.SameSiteStrictMode,
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-02-20 17:39:56 +03:00
|
|
|
|
func (c *Config) IsDev() bool {
|
2020-05-30 13:11:21 +03:00
|
|
|
|
return IsDev(c.Env)
|
|
|
|
|
}
|
|
|
|
|
|
2020-12-13 00:07:00 +03:00
|
|
|
|
func (c *Config) UseTLS() bool {
|
|
|
|
|
return c.Server.TlsCertPath != "" && c.Server.TlsKeyPath != ""
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-29 19:55:07 +03:00
|
|
|
|
func (c *Config) GetMigrationFunc(dbDialect string) models.MigrationFunc {
|
2020-05-30 21:41:27 +03:00
|
|
|
|
switch dbDialect {
|
|
|
|
|
default:
|
|
|
|
|
return func(db *gorm.DB) error {
|
2020-09-29 19:55:07 +03:00
|
|
|
|
db.AutoMigrate(&models.User{})
|
2020-11-03 12:02:39 +03:00
|
|
|
|
db.AutoMigrate(&models.KeyStringValue{})
|
|
|
|
|
db.AutoMigrate(&models.Alias{})
|
2020-11-01 22:14:10 +03:00
|
|
|
|
db.AutoMigrate(&models.Heartbeat{})
|
2020-11-03 12:02:39 +03:00
|
|
|
|
db.AutoMigrate(&models.Summary{})
|
2020-11-01 22:14:10 +03:00
|
|
|
|
db.AutoMigrate(&models.SummaryItem{})
|
|
|
|
|
db.AutoMigrate(&models.LanguageMapping{})
|
2020-05-30 21:41:27 +03:00
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-09-29 19:55:07 +03:00
|
|
|
|
func (c *Config) GetFixturesFunc(dbDialect string) models.MigrationFunc {
|
2020-05-30 21:41:27 +03:00
|
|
|
|
return func(db *gorm.DB) error {
|
2021-01-30 11:51:36 +03:00
|
|
|
|
migrations := &migrate.HttpFileSystemMigrationSource{
|
2021-01-22 15:50:46 +03:00
|
|
|
|
FileSystem: pkger.Dir("/migrations"),
|
2020-05-30 21:41:27 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
migrate.SetIgnoreUnknown(true)
|
2020-11-01 22:14:10 +03:00
|
|
|
|
sqlDb, _ := db.DB()
|
|
|
|
|
n, err := migrate.Exec(sqlDb, dbDialect, migrations, migrate.Up)
|
2020-05-30 21:41:27 +03:00
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Info("applied %d fixtures", n)
|
2020-05-30 21:41:27 +03:00
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2020-11-01 22:14:10 +03:00
|
|
|
|
func (c *dbConfig) GetDialector() gorm.Dialector {
|
|
|
|
|
switch c.Dialect {
|
2020-11-06 19:20:26 +03:00
|
|
|
|
case SQLDialectMysql:
|
2020-11-01 22:14:10 +03:00
|
|
|
|
return mysql.New(mysql.Config{
|
|
|
|
|
DriverName: c.Dialect,
|
|
|
|
|
DSN: mysqlConnectionString(c),
|
|
|
|
|
})
|
2020-11-06 19:20:26 +03:00
|
|
|
|
case SQLDialectPostgres:
|
2020-11-01 22:14:10 +03:00
|
|
|
|
return postgres.New(postgres.Config{
|
2020-11-03 10:30:11 +03:00
|
|
|
|
DSN: postgresConnectionString(c),
|
2020-11-01 22:14:10 +03:00
|
|
|
|
})
|
2020-11-06 19:20:26 +03:00
|
|
|
|
case SQLDialectSqlite:
|
2020-11-01 22:14:10 +03:00
|
|
|
|
return sqlite.Open(sqliteConnectionString(c))
|
|
|
|
|
}
|
|
|
|
|
return nil
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func mysqlConnectionString(config *dbConfig) string {
|
|
|
|
|
//location, _ := time.LoadLocation("Local")
|
|
|
|
|
return fmt.Sprintf("%s:%s@tcp(%s:%d)/%s?charset=utf8&parseTime=true&loc=%s&sql_mode=ANSI_QUOTES",
|
|
|
|
|
config.User,
|
|
|
|
|
config.Password,
|
|
|
|
|
config.Host,
|
|
|
|
|
config.Port,
|
|
|
|
|
config.Name,
|
|
|
|
|
"Local",
|
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func postgresConnectionString(config *dbConfig) string {
|
2021-01-18 23:34:08 +03:00
|
|
|
|
sslmode := "disable"
|
|
|
|
|
if config.Ssl {
|
|
|
|
|
sslmode = "require"
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
return fmt.Sprintf("host=%s port=%d user=%s dbname=%s password=%s sslmode=%s",
|
2020-11-01 22:14:10 +03:00
|
|
|
|
config.Host,
|
|
|
|
|
config.Port,
|
|
|
|
|
config.User,
|
|
|
|
|
config.Name,
|
|
|
|
|
config.Password,
|
2021-01-18 23:34:08 +03:00
|
|
|
|
sslmode,
|
2020-11-01 22:14:10 +03:00
|
|
|
|
)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func sqliteConnectionString(config *dbConfig) string {
|
|
|
|
|
return config.Name
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-07 12:52:49 +03:00
|
|
|
|
func (c *appConfig) GetCustomLanguages() map[string]string {
|
2021-01-30 11:51:36 +03:00
|
|
|
|
return cloneStringMap(c.CustomLanguages, false)
|
2021-01-07 12:52:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *appConfig) GetLanguageColors() map[string]string {
|
2021-01-30 11:51:36 +03:00
|
|
|
|
return cloneStringMap(c.Colors["languages"], true)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *appConfig) GetEditorColors() map[string]string {
|
|
|
|
|
return cloneStringMap(c.Colors["editors"], true)
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func (c *appConfig) GetOSColors() map[string]string {
|
|
|
|
|
return cloneStringMap(c.Colors["operating_systems"], true)
|
2021-01-07 12:52:49 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-05-30 13:11:21 +03:00
|
|
|
|
func IsDev(env string) bool {
|
|
|
|
|
return env == "dev" || env == "development"
|
2020-02-20 17:39:56 +03:00
|
|
|
|
}
|
2020-05-24 18:32:26 +03:00
|
|
|
|
|
2020-05-24 22:42:15 +03:00
|
|
|
|
func readVersion() string {
|
2021-01-22 15:50:46 +03:00
|
|
|
|
file, err := pkger.Open("/version.txt")
|
2020-05-24 22:42:15 +03:00
|
|
|
|
if err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal(err.Error())
|
2020-05-24 22:42:15 +03:00
|
|
|
|
}
|
|
|
|
|
defer file.Close()
|
|
|
|
|
|
|
|
|
|
bytes, err := ioutil.ReadAll(file)
|
|
|
|
|
if err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal(err.Error())
|
2020-05-24 22:42:15 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-22 00:17:32 +03:00
|
|
|
|
return strings.TrimSpace(string(bytes))
|
2020-05-24 22:42:15 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-30 11:51:36 +03:00
|
|
|
|
func readColors() map[string]map[string]string {
|
2020-05-24 18:32:26 +03:00
|
|
|
|
// Read language colors
|
2021-01-30 11:51:36 +03:00
|
|
|
|
// Source:
|
|
|
|
|
// – https://raw.githubusercontent.com/ozh/github-colors/master/colors.json
|
|
|
|
|
// – https://wakatime.com/colors/operating_systems
|
|
|
|
|
// - https://wakatime.com/colors/editors
|
|
|
|
|
// Extracted from Wakatime website with XPath (see below) and did a bit of regex magic after.
|
|
|
|
|
// – $x('//span[@class="editor-icon tip"]/@data-original-title').map(e => e.nodeValue)
|
|
|
|
|
// – $x('//span[@class="editor-icon tip"]/div[1]/text()').map(e => e.nodeValue)
|
|
|
|
|
var colors = make(map[string]map[string]string)
|
2020-05-24 18:32:26 +03:00
|
|
|
|
|
2021-01-22 15:50:46 +03:00
|
|
|
|
file, err := pkger.Open("/data/colors.json")
|
|
|
|
|
if err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal(err.Error())
|
2021-01-22 15:50:46 +03:00
|
|
|
|
}
|
|
|
|
|
defer file.Close()
|
|
|
|
|
bytes, err := ioutil.ReadAll(file)
|
2020-05-24 18:32:26 +03:00
|
|
|
|
if err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal(err.Error())
|
2020-05-24 18:32:26 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-01-30 11:51:36 +03:00
|
|
|
|
if err := json.Unmarshal(bytes, &colors); err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal(err.Error())
|
2020-05-24 18:32:26 +03:00
|
|
|
|
}
|
|
|
|
|
|
2020-10-04 11:37:38 +03:00
|
|
|
|
return colors
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func mustReadConfigLocation() string {
|
|
|
|
|
if _, err := os.Stat(*cFlag); err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal("failed to find config file at '%s'", *cFlag)
|
2020-10-04 11:37:38 +03:00
|
|
|
|
}
|
|
|
|
|
return *cFlag
|
|
|
|
|
}
|
|
|
|
|
|
2021-01-18 23:34:08 +03:00
|
|
|
|
func resolveDbDialect(dbType string) string {
|
|
|
|
|
if dbType == "cockroach" {
|
|
|
|
|
return "postgres"
|
|
|
|
|
}
|
|
|
|
|
return dbType
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-04 11:37:38 +03:00
|
|
|
|
func Set(config *Config) {
|
|
|
|
|
cfg = config
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Get() *Config {
|
|
|
|
|
return cfg
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func Load() *Config {
|
|
|
|
|
config := &Config{}
|
|
|
|
|
|
2020-11-08 12:12:49 +03:00
|
|
|
|
flag.Parse()
|
|
|
|
|
|
2020-10-04 11:37:38 +03:00
|
|
|
|
if err := configor.New(&configor.Config{}).Load(config, mustReadConfigLocation()); err != nil {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal("failed to read config: %v", err)
|
2020-10-04 11:37:38 +03:00
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
config.Version = readVersion()
|
2021-01-30 11:51:36 +03:00
|
|
|
|
config.App.Colors = readColors()
|
2021-01-18 23:34:08 +03:00
|
|
|
|
config.Db.Dialect = resolveDbDialect(config.Db.Type)
|
2020-10-04 11:37:38 +03:00
|
|
|
|
config.Security.SecureCookie = securecookie.New(
|
2020-05-24 18:32:26 +03:00
|
|
|
|
securecookie.GenerateRandomKey(64),
|
|
|
|
|
securecookie.GenerateRandomKey(32),
|
|
|
|
|
)
|
|
|
|
|
|
2020-10-04 11:37:38 +03:00
|
|
|
|
if strings.HasSuffix(config.Server.BasePath, "/") {
|
|
|
|
|
config.Server.BasePath = config.Server.BasePath[:len(config.Server.BasePath)-1]
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
for k, v := range config.App.CustomLanguages {
|
|
|
|
|
if v == "" {
|
|
|
|
|
config.App.CustomLanguages[k] = "unknown"
|
|
|
|
|
}
|
|
|
|
|
}
|
2020-09-29 19:55:07 +03:00
|
|
|
|
|
2020-12-13 00:07:00 +03:00
|
|
|
|
if config.Server.ListenIpV4 == "" && config.Server.ListenIpV6 == "" {
|
2021-01-30 13:17:37 +03:00
|
|
|
|
logbuch.Fatal("either of listen_ipv4 or listen_ipv6 must be set")
|
2020-12-13 00:07:00 +03:00
|
|
|
|
}
|
|
|
|
|
|
2021-02-03 00:52:13 +03:00
|
|
|
|
if config.Db.MaxConn <= 0 {
|
|
|
|
|
logbuch.Fatal("you must allow at least one database connection")
|
|
|
|
|
}
|
|
|
|
|
|
2020-10-04 11:37:38 +03:00
|
|
|
|
Set(config)
|
2020-09-29 19:55:07 +03:00
|
|
|
|
return Get()
|
2020-05-24 18:32:26 +03:00
|
|
|
|
}
|