config.go 8.5 KB
Newer Older
1 2 3 4 5 6 7
package config

import (
	"fmt"
	"github.com/emicklei/go-restful"
	"github.com/spf13/viper"
	"k8s.io/apimachinery/pkg/runtime/schema"
8
	authoptions "kubesphere.io/kubesphere/pkg/apiserver/authentication/options"
9
	"kubesphere.io/kubesphere/pkg/apiserver/runtime"
J
Jeff 已提交
10
	"kubesphere.io/kubesphere/pkg/simple/client/alerting"
Z
zryfish 已提交
11
	"kubesphere.io/kubesphere/pkg/simple/client/cache"
R
runzexia 已提交
12
	"kubesphere.io/kubesphere/pkg/simple/client/devops/jenkins"
13 14
	"kubesphere.io/kubesphere/pkg/simple/client/k8s"
	"kubesphere.io/kubesphere/pkg/simple/client/ldap"
G
Guangzhe Huang 已提交
15
	"kubesphere.io/kubesphere/pkg/simple/client/logging/elasticsearch"
G
Guangzhe Huang 已提交
16
	"kubesphere.io/kubesphere/pkg/simple/client/monitoring/prometheus"
17
	"kubesphere.io/kubesphere/pkg/simple/client/mysql"
J
Jeff 已提交
18
	"kubesphere.io/kubesphere/pkg/simple/client/notification"
19
	"kubesphere.io/kubesphere/pkg/simple/client/openpitrix"
Z
zryfish 已提交
20
	"kubesphere.io/kubesphere/pkg/simple/client/s3"
21 22 23
	"kubesphere.io/kubesphere/pkg/simple/client/servicemesh"
	"kubesphere.io/kubesphere/pkg/simple/client/sonarqube"
	"net/http"
J
Jeff 已提交
24 25
	"reflect"
	"strings"
26 27
)

J
Jeff 已提交
28 29 30 31 32 33 34 35 36 37 38 39 40 41 42 43 44 45 46 47 48 49 50
// Package config saves configuration for running KubeSphere components
//
// Config can be configured from command line flags and configuration file.
// Command line flags hold higher priority than configuration file. But if
// component Endpoint/Host/APIServer was left empty, all of that component
// command line flags will be ignored, use configuration file instead.
// For example, we have configuration file
//
// mysql:
//   host: mysql.kubesphere-system.svc
//   username: root
//   password: password
//
// At the same time, have command line flags like following:
//
// --mysql-host mysql.openpitrix-system.svc --mysql-username king --mysql-password 1234
//
// We will use `king:1234@mysql.openpitrix-system.svc` from command line flags rather
// than `root:password@mysql.kubesphere-system.svc` from configuration file,
// cause command line has higher priority. But if command line flags like following:
//
// --mysql-username root --mysql-password password
//
51
// we will `root:password@mysql.kubesphere-system.svc` as input, cause
J
Jeff 已提交
52 53 54
// mysql-host is missing in command line flags, all other mysql command line flags
// will be ignored.

55
const (
J
Jeff 已提交
56
	// DefaultConfigurationName is the default name of configuration
57
	defaultConfigurationName = "kubesphere"
J
Jeff 已提交
58 59

	// DefaultConfigurationPath the default location of the configuration file
60
	defaultConfigurationPath = "/etc/kubesphere"
61 62
)

63
// Config defines everything needed for apiserver to deal with external services
64
type Config struct {
65 66 67 68 69 70 71 72 73 74 75 76
	MySQLOptions          *mysql.Options                     `json:"mysql,omitempty" yaml:"mysql,omitempty" mapstructure:"mysql"`
	DevopsOptions         *jenkins.Options                   `json:"devops,omitempty" yaml:"devops,omitempty" mapstructure:"devops"`
	SonarQubeOptions      *sonarqube.Options                 `json:"sonarqube,omitempty" yaml:"sonarQube,omitempty" mapstructure:"sonarqube"`
	KubernetesOptions     *k8s.KubernetesOptions             `json:"kubernetes,omitempty" yaml:"kubernetes,omitempty" mapstructure:"kubernetes"`
	ServiceMeshOptions    *servicemesh.Options               `json:"servicemesh,omitempty" yaml:"servicemesh,omitempty" mapstructure:"servicemesh"`
	LdapOptions           *ldap.Options                      `json:"ldap,omitempty" yaml:"ldap,omitempty" mapstructure:"ldap"`
	RedisOptions          *cache.Options                     `json:"redis,omitempty" yaml:"redis,omitempty" mapstructure:"redis"`
	S3Options             *s3.Options                        `json:"s3,omitempty" yaml:"s3,omitempty" mapstructure:"s3"`
	OpenPitrixOptions     *openpitrix.Options                `json:"openpitrix,omitempty" yaml:"openpitrix,omitempty" mapstructure:"openpitrix"`
	MonitoringOptions     *prometheus.Options                `json:"monitoring,omitempty" yaml:"monitoring,omitempty" mapstructure:"monitoring"`
	LoggingOptions        *elasticsearch.Options             `json:"logging,omitempty" yaml:"logging,omitempty" mapstructure:"logging"`
	AuthenticationOptions *authoptions.AuthenticationOptions `json:"authentication,omitempty" yaml:"authentication,omitempty" mapstructure:"authentication"`
J
Jeff 已提交
77 78
	// Options used for enabling components, not actually used now. Once we switch Alerting/Notification API to kubesphere,
	// we can add these options to kubesphere command lines
Z
zryfish 已提交
79 80
	AlertingOptions     *alerting.Options     `json:"alerting,omitempty" yaml:"alerting,omitempty" mapstructure:"alerting"`
	NotificationOptions *notification.Options `json:"notification,omitempty" yaml:"notification,omitempty" mapstructure:"notification"`
81 82
}

83 84
// newConfig creates a default non-empty Config
func New() *Config {
85
	return &Config{
86 87 88 89 90 91 92 93 94 95 96 97 98 99
		MySQLOptions:          mysql.NewMySQLOptions(),
		DevopsOptions:         jenkins.NewDevopsOptions(),
		SonarQubeOptions:      sonarqube.NewSonarQubeOptions(),
		KubernetesOptions:     k8s.NewKubernetesOptions(),
		ServiceMeshOptions:    servicemesh.NewServiceMeshOptions(),
		LdapOptions:           ldap.NewOptions(),
		RedisOptions:          cache.NewRedisOptions(),
		S3Options:             s3.NewS3Options(),
		OpenPitrixOptions:     openpitrix.NewOptions(),
		MonitoringOptions:     prometheus.NewPrometheusOptions(),
		AlertingOptions:       alerting.NewAlertingOptions(),
		NotificationOptions:   notification.NewNotificationOptions(),
		LoggingOptions:        elasticsearch.NewElasticSearchOptions(),
		AuthenticationOptions: authoptions.NewAuthenticateOptions(),
100 101 102
	}
}

103 104 105 106 107
// TryLoadFromDisk loads configuration from default location after server startup
// return nil error if configuration file not exists
func TryLoadFromDisk() (*Config, error) {
	viper.SetConfigName(defaultConfigurationName)
	viper.AddConfigPath(defaultConfigurationPath)
108

109 110
	// Load from current working directory, only used for debugging
	viper.AddConfigPath(".")
111

112 113 114 115 116 117
	if err := viper.ReadInConfig(); err != nil {
		if _, ok := err.(viper.ConfigFileNotFoundError); ok {
			return nil, err
		} else {
			return nil, fmt.Errorf("error parsing configuration file %s", err)
		}
J
Jeff 已提交
118 119
	}

120
	conf := New()
121

122 123
	if err := viper.Unmarshal(conf); err != nil {
		return nil, err
J
Jeff 已提交
124 125
	}

126 127
	return conf, nil
}
128

129 130 131 132 133 134
// InstallAPI installs api for config
func (conf *Config) InstallAPI(c *restful.Container) {
	ws := runtime.NewWebService(schema.GroupVersion{
		Group:   "",
		Version: "v1alpha1",
	})
135

136 137 138
	ws.Route(ws.GET("/configz").
		To(func(request *restful.Request, response *restful.Response) {
			conf.stripEmptyOptions()
139
			response.WriteAsJson(conf.ToMap())
140 141 142 143 144
		}).
		Doc("Get system components configuration").
		Produces(restful.MIME_JSON).
		Writes(Config{}).
		Returns(http.StatusOK, "ok", Config{}))
145

146 147
	c.Add(ws)
}
148

149 150
// convertToMap simply converts config to map[string]bool
// to hide sensitive information
151 152
func (conf *Config) ToMap() map[string]bool {
	conf.stripEmptyOptions()
153
	result := make(map[string]bool, 0)
154

155 156
	if conf == nil {
		return result
157 158
	}

159
	c := reflect.Indirect(reflect.ValueOf(conf))
160

161 162 163 164 165
	for i := 0; i < c.NumField(); i++ {
		name := strings.Split(c.Type().Field(i).Tag.Get("json"), ",")[0]
		if strings.HasPrefix(name, "-") {
			continue
		}
166

167 168 169 170 171
		if c.Field(i).IsNil() {
			result[name] = false
		} else {
			result[name] = true
		}
172
	}
173 174

	return result
175
}
J
Jeff 已提交
176

Z
zryfish 已提交
177
// Remove invalid options before serializing to json or yaml
178 179 180
func (conf *Config) stripEmptyOptions() {
	if conf.MySQLOptions != nil && conf.MySQLOptions.Host == "" {
		conf.MySQLOptions = nil
J
Jeff 已提交
181 182
	}

Z
zryfish 已提交
183
	if conf.RedisOptions != nil && conf.RedisOptions.Host == "" {
184
		conf.RedisOptions = nil
J
Jeff 已提交
185 186
	}

187 188
	if conf.DevopsOptions != nil && conf.DevopsOptions.Host == "" {
		conf.DevopsOptions = nil
J
Jeff 已提交
189 190
	}

191 192 193
	if conf.MonitoringOptions != nil && conf.MonitoringOptions.Endpoint == "" &&
		conf.MonitoringOptions.SecondaryEndpoint == "" {
		conf.MonitoringOptions = nil
J
Jeff 已提交
194 195
	}

196 197
	if conf.SonarQubeOptions != nil && conf.SonarQubeOptions.Host == "" {
		conf.SonarQubeOptions = nil
J
Jeff 已提交
198 199
	}

200 201
	if conf.LdapOptions != nil && conf.LdapOptions.Host == "" {
		conf.LdapOptions = nil
J
Jeff 已提交
202 203
	}

204 205
	if conf.OpenPitrixOptions != nil && conf.OpenPitrixOptions.IsEmpty() {
		conf.OpenPitrixOptions = nil
J
Jeff 已提交
206 207
	}

208 209 210 211
	if conf.ServiceMeshOptions != nil && conf.ServiceMeshOptions.IstioPilotHost == "" &&
		conf.ServiceMeshOptions.ServicemeshPrometheusHost == "" &&
		conf.ServiceMeshOptions.JaegerQueryHost == "" {
		conf.ServiceMeshOptions = nil
J
Jeff 已提交
212 213
	}

214 215
	if conf.S3Options != nil && conf.S3Options.Endpoint == "" {
		conf.S3Options = nil
J
Jeff 已提交
216 217
	}

218 219
	if conf.AlertingOptions != nil && conf.AlertingOptions.Endpoint == "" {
		conf.AlertingOptions = nil
J
Jeff 已提交
220 221
	}

222 223
	if conf.LoggingOptions != nil && conf.LoggingOptions.Host == "" {
		conf.LoggingOptions = nil
J
Jeff 已提交
224 225
	}

226 227
	if conf.NotificationOptions != nil && conf.NotificationOptions.Endpoint == "" {
		conf.NotificationOptions = nil
J
Jeff 已提交
228 229
	}

J
Jeff 已提交
230
}