mirror of
https://github.com/minio/minio.git
synced 2025-01-11 23:13:23 -05:00
Better validation of all config file fields (#6090)
Add Validate() to serverConfig to call it at server startup and in Admin SetConfig handler to minimize errors scenario after server restart.
This commit is contained in:
parent
758a80e39b
commit
e8a008f5b5
@ -712,7 +712,7 @@ func (a adminAPIHandlers) SetConfigHandler(w http.ResponseWriter, r *http.Reques
|
|||||||
err = json.Unmarshal(configBytes, &config)
|
err = json.Unmarshal(configBytes, &config)
|
||||||
if err != nil {
|
if err != nil {
|
||||||
logger.LogIf(ctx, err)
|
logger.LogIf(ctx, err)
|
||||||
writeErrorResponseJSON(w, toAPIErrorCode(err), r.URL)
|
writeCustomErrorResponseJSON(w, ErrAdminConfigBadJSON, err.Error(), r.URL)
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -727,6 +727,11 @@ func (a adminAPIHandlers) SetConfigHandler(w http.ResponseWriter, r *http.Reques
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if err := config.Validate(); err != nil {
|
||||||
|
writeCustomErrorResponseJSON(w, ErrAdminConfigBadJSON, err.Error(), r.URL)
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
// Write config received from request onto a temporary file on
|
// Write config received from request onto a temporary file on
|
||||||
// all nodes.
|
// all nodes.
|
||||||
tmpFileName := fmt.Sprintf(minioConfigTmpFormat, mustGetUUID())
|
tmpFileName := fmt.Sprintf(minioConfigTmpFormat, mustGetUUID())
|
||||||
|
@ -38,22 +38,24 @@ import (
|
|||||||
|
|
||||||
var (
|
var (
|
||||||
configJSON = []byte(`{
|
configJSON = []byte(`{
|
||||||
"version": "13",
|
"version": "26",
|
||||||
"credential": {
|
"credential": {
|
||||||
"accessKey": "minio",
|
"accessKey": "minio",
|
||||||
"secretKey": "minio123"
|
"secretKey": "minio123"
|
||||||
},
|
},
|
||||||
"region": "us-west-1",
|
"region": "",
|
||||||
"logger": {
|
"browser": "on",
|
||||||
"console": {
|
"worm": "off",
|
||||||
"enable": true,
|
"domain": "",
|
||||||
"level": "fatal"
|
"storageclass": {
|
||||||
|
"standard": "",
|
||||||
|
"rrs": ""
|
||||||
},
|
},
|
||||||
"file": {
|
"cache": {
|
||||||
"enable": false,
|
"drives": [],
|
||||||
"fileName": "",
|
"expiry": 90,
|
||||||
"level": ""
|
"maxuse": 80,
|
||||||
}
|
"exclude": []
|
||||||
},
|
},
|
||||||
"notify": {
|
"notify": {
|
||||||
"amqp": {
|
"amqp": {
|
||||||
@ -63,6 +65,7 @@ var (
|
|||||||
"exchange": "",
|
"exchange": "",
|
||||||
"routingKey": "",
|
"routingKey": "",
|
||||||
"exchangeType": "",
|
"exchangeType": "",
|
||||||
|
"deliveryMode": 0,
|
||||||
"mandatory": false,
|
"mandatory": false,
|
||||||
"immediate": false,
|
"immediate": false,
|
||||||
"durable": false,
|
"durable": false,
|
||||||
@ -71,6 +74,47 @@ var (
|
|||||||
"autoDeleted": false
|
"autoDeleted": false
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
|
"elasticsearch": {
|
||||||
|
"1": {
|
||||||
|
"enable": false,
|
||||||
|
"format": "",
|
||||||
|
"url": "",
|
||||||
|
"index": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"kafka": {
|
||||||
|
"1": {
|
||||||
|
"enable": false,
|
||||||
|
"brokers": null,
|
||||||
|
"topic": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"mqtt": {
|
||||||
|
"1": {
|
||||||
|
"enable": false,
|
||||||
|
"broker": "",
|
||||||
|
"topic": "",
|
||||||
|
"qos": 0,
|
||||||
|
"clientId": "",
|
||||||
|
"username": "",
|
||||||
|
"password": "",
|
||||||
|
"reconnectInterval": 0,
|
||||||
|
"keepAliveInterval": 0
|
||||||
|
}
|
||||||
|
},
|
||||||
|
"mysql": {
|
||||||
|
"1": {
|
||||||
|
"enable": false,
|
||||||
|
"format": "",
|
||||||
|
"dsnString": "",
|
||||||
|
"table": "",
|
||||||
|
"host": "",
|
||||||
|
"port": "",
|
||||||
|
"user": "",
|
||||||
|
"password": "",
|
||||||
|
"database": ""
|
||||||
|
}
|
||||||
|
},
|
||||||
"nats": {
|
"nats": {
|
||||||
"1": {
|
"1": {
|
||||||
"enable": false,
|
"enable": false,
|
||||||
@ -90,24 +134,10 @@ var (
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"elasticsearch": {
|
|
||||||
"1": {
|
|
||||||
"enable": false,
|
|
||||||
"url": "",
|
|
||||||
"index": ""
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"redis": {
|
|
||||||
"1": {
|
|
||||||
"enable": false,
|
|
||||||
"address": "",
|
|
||||||
"password": "",
|
|
||||||
"key": ""
|
|
||||||
}
|
|
||||||
},
|
|
||||||
"postgresql": {
|
"postgresql": {
|
||||||
"1": {
|
"1": {
|
||||||
"enable": false,
|
"enable": false,
|
||||||
|
"format": "",
|
||||||
"connectionString": "",
|
"connectionString": "",
|
||||||
"table": "",
|
"table": "",
|
||||||
"host": "",
|
"host": "",
|
||||||
@ -117,11 +147,13 @@ var (
|
|||||||
"database": ""
|
"database": ""
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"kafka": {
|
"redis": {
|
||||||
"1": {
|
"1": {
|
||||||
"enable": false,
|
"enable": false,
|
||||||
"brokers": null,
|
"format": "",
|
||||||
"topic": ""
|
"address": "",
|
||||||
|
"password": "",
|
||||||
|
"key": ""
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
"webhook": {
|
"webhook": {
|
||||||
|
@ -23,6 +23,7 @@ import (
|
|||||||
"reflect"
|
"reflect"
|
||||||
"sync"
|
"sync"
|
||||||
|
|
||||||
|
"github.com/miekg/dns"
|
||||||
"github.com/minio/minio/cmd/logger"
|
"github.com/minio/minio/cmd/logger"
|
||||||
|
|
||||||
"github.com/minio/minio/pkg/auth"
|
"github.com/minio/minio/pkg/auth"
|
||||||
@ -129,6 +130,84 @@ func (s *serverConfig) GetCacheConfig() CacheConfig {
|
|||||||
return s.Cache
|
return s.Cache
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func (s *serverConfig) Validate() error {
|
||||||
|
if s.Version != serverConfigVersion {
|
||||||
|
return fmt.Errorf("configuration version mismatch. Expected: ‘%s’, Got: ‘%s’", serverConfigVersion, s.Version)
|
||||||
|
}
|
||||||
|
|
||||||
|
// Validate credential fields only when
|
||||||
|
// they are not set via the environment
|
||||||
|
// Error out if global is env credential is not set and config has invalid credential
|
||||||
|
if !globalIsEnvCreds && !s.Credential.IsValid() {
|
||||||
|
return errors.New("invalid credential in config file")
|
||||||
|
}
|
||||||
|
|
||||||
|
// Region: nothing to validate
|
||||||
|
// Browser, Worm, Cache and StorageClass values are already validated during json unmarshal
|
||||||
|
|
||||||
|
if s.Domain != "" {
|
||||||
|
if _, ok := dns.IsDomainName(s.Domain); !ok {
|
||||||
|
return errors.New("invalid domain name")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.AMQP {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("amqp: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.Elasticsearch {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("elasticsearch: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.Kafka {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("kafka: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.MQTT {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("mqtt: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.MySQL {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("mysql: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.NATS {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("nats: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.PostgreSQL {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("postgreSQL: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.Redis {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("redis: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
for _, v := range s.Notify.Webhook {
|
||||||
|
if err := v.Validate(); err != nil {
|
||||||
|
return fmt.Errorf("webhook: %s", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// Save config file to corresponding backend
|
// Save config file to corresponding backend
|
||||||
func Save(configFile string, data interface{}) error {
|
func Save(configFile string, data interface{}) error {
|
||||||
return quick.SaveConfig(data, configFile, globalEtcdClient)
|
return quick.SaveConfig(data, configFile, globalEtcdClient)
|
||||||
@ -318,15 +397,8 @@ func getValidConfig() (*serverConfig, error) {
|
|||||||
return nil, err
|
return nil, err
|
||||||
}
|
}
|
||||||
|
|
||||||
if srvCfg.Version != serverConfigVersion {
|
if err = srvCfg.Validate(); err != nil {
|
||||||
return nil, fmt.Errorf("configuration version mismatch. Expected: ‘%s’, Got: ‘%s’", serverConfigVersion, srvCfg.Version)
|
return nil, err
|
||||||
}
|
|
||||||
|
|
||||||
// Validate credential fields only when
|
|
||||||
// they are not set via the environment
|
|
||||||
// Error out if global is env credential is not set and config has invalid credential
|
|
||||||
if !globalIsEnvCreds && !srvCfg.Credential.IsValid() {
|
|
||||||
return nil, errors.New("invalid credential in config file " + getConfigFile())
|
|
||||||
}
|
}
|
||||||
|
|
||||||
return srvCfg, nil
|
return srvCfg, nil
|
||||||
|
@ -174,55 +174,55 @@ func TestValidateConfig(t *testing.T) {
|
|||||||
{`{"version": "` + v + `", "browser": "on", "browser": "on", "region":"us-east-1", "credential" : {"accessKey":"minio", "secretKey":"minio123"}}`, false},
|
{`{"version": "` + v + `", "browser": "on", "browser": "on", "region":"us-east-1", "credential" : {"accessKey":"minio", "secretKey":"minio123"}}`, false},
|
||||||
|
|
||||||
// Test 11 - Test AMQP
|
// Test 11 - Test AMQP
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "amqp": { "1": { "enable": true, "url": "", "exchange": "", "routingKey": "", "exchangeType": "", "mandatory": false, "immediate": false, "durable": false, "internal": false, "noWait": false, "autoDeleted": false }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "amqp": { "1": { "enable": true, "url": "", "exchange": "", "routingKey": "", "exchangeType": "", "mandatory": false, "immediate": false, "durable": false, "internal": false, "noWait": false, "autoDeleted": false }}}}`, false},
|
||||||
|
|
||||||
// Test 12 - Test NATS
|
// Test 12 - Test NATS
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "nats": { "1": { "enable": true, "address": "", "subject": "", "username": "", "password": "", "token": "", "secure": false, "pingInterval": 0, "streaming": { "enable": false, "clusterID": "", "clientID": "", "async": false, "maxPubAcksInflight": 0 } } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "nats": { "1": { "enable": true, "address": "", "subject": "", "username": "", "password": "", "token": "", "secure": false, "pingInterval": 0, "streaming": { "enable": false, "clusterID": "", "clientID": "", "async": false, "maxPubAcksInflight": 0 } } }}}`, false},
|
||||||
|
|
||||||
// Test 13 - Test ElasticSearch
|
// Test 13 - Test ElasticSearch
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "elasticsearch": { "1": { "enable": true, "url": "", "index": "" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "elasticsearch": { "1": { "enable": true, "url": "", "index": "" } }}}`, false},
|
||||||
|
|
||||||
// Test 14 - Test Redis
|
// Test 14 - Test Redis
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "redis": { "1": { "enable": true, "address": "", "password": "", "key": "" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "redis": { "1": { "enable": true, "address": "", "password": "", "key": "" } }}}`, false},
|
||||||
|
|
||||||
// Test 15 - Test PostgreSQL
|
// Test 15 - Test PostgreSQL
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "postgresql": { "1": { "enable": true, "connectionString": "", "table": "", "host": "", "port": "", "user": "", "password": "", "database": "" }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "postgresql": { "1": { "enable": true, "connectionString": "", "table": "", "host": "", "port": "", "user": "", "password": "", "database": "" }}}}`, false},
|
||||||
|
|
||||||
// Test 16 - Test Kafka
|
// Test 16 - Test Kafka
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "kafka": { "1": { "enable": true, "brokers": null, "topic": "" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "kafka": { "1": { "enable": true, "brokers": null, "topic": "" } }}}`, false},
|
||||||
|
|
||||||
// Test 17 - Test Webhook
|
// Test 17 - Test Webhook
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "webhook": { "1": { "enable": true, "endpoint": "" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "webhook": { "1": { "enable": true, "endpoint": "" } }}}`, false},
|
||||||
|
|
||||||
// Test 18 - Test MySQL
|
// Test 18 - Test MySQL
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mysql": { "1": { "enable": true, "dsnString": "", "table": "", "host": "", "port": "", "user": "", "password": "", "database": "" }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mysql": { "1": { "enable": true, "dsnString": "", "table": "", "host": "", "port": "", "user": "", "password": "", "database": "" }}}}`, false},
|
||||||
|
|
||||||
// Test 19 - Test Format for MySQL
|
// Test 19 - Test Format for MySQL
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mysql": { "1": { "enable": true, "dsnString": "", "format": "invalid", "table": "xxx", "host": "10.0.0.1", "port": "3306", "user": "abc", "password": "pqr", "database": "test1" }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mysql": { "1": { "enable": true, "dsnString": "", "format": "invalid", "table": "xxx", "host": "10.0.0.1", "port": "3306", "user": "abc", "password": "pqr", "database": "test1" }}}}`, false},
|
||||||
|
|
||||||
// Test 20 - Test valid Format for MySQL
|
// Test 20 - Test valid Format for MySQL
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mysql": { "1": { "enable": true, "dsnString": "", "format": "namespace", "table": "xxx", "host": "10.0.0.1", "port": "3306", "user": "abc", "password": "pqr", "database": "test1" }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mysql": { "1": { "enable": true, "dsnString": "", "format": "namespace", "table": "xxx", "host": "10.0.0.1", "port": "3306", "user": "abc", "password": "pqr", "database": "test1" }}}}`, true},
|
||||||
|
|
||||||
// Test 21 - Test Format for PostgreSQL
|
// Test 21 - Test Format for PostgreSQL
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "postgresql": { "1": { "enable": true, "connectionString": "", "format": "invalid", "table": "xxx", "host": "myhost", "port": "5432", "user": "abc", "password": "pqr", "database": "test1" }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "postgresql": { "1": { "enable": true, "connectionString": "", "format": "invalid", "table": "xxx", "host": "myhost", "port": "5432", "user": "abc", "password": "pqr", "database": "test1" }}}}`, false},
|
||||||
|
|
||||||
// Test 22 - Test valid Format for PostgreSQL
|
// Test 22 - Test valid Format for PostgreSQL
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "postgresql": { "1": { "enable": true, "connectionString": "", "format": "namespace", "table": "xxx", "host": "myhost", "port": "5432", "user": "abc", "password": "pqr", "database": "test1" }}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "postgresql": { "1": { "enable": true, "connectionString": "", "format": "namespace", "table": "xxx", "host": "myhost", "port": "5432", "user": "abc", "password": "pqr", "database": "test1" }}}}`, true},
|
||||||
|
|
||||||
// Test 23 - Test Format for ElasticSearch
|
// Test 23 - Test Format for ElasticSearch
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "elasticsearch": { "1": { "enable": true, "format": "invalid", "url": "example.com", "index": "myindex" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "elasticsearch": { "1": { "enable": true, "format": "invalid", "url": "example.com", "index": "myindex" } }}}`, false},
|
||||||
|
|
||||||
// Test 24 - Test valid Format for ElasticSearch
|
// Test 24 - Test valid Format for ElasticSearch
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "elasticsearch": { "1": { "enable": true, "format": "namespace", "url": "example.com", "index": "myindex" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "elasticsearch": { "1": { "enable": true, "format": "namespace", "url": "example.com", "index": "myindex" } }}}`, true},
|
||||||
|
|
||||||
// Test 25 - Test Format for Redis
|
// Test 25 - Test Format for Redis
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "redis": { "1": { "enable": true, "format": "invalid", "address": "example.com:80", "password": "xxx", "key": "key1" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "redis": { "1": { "enable": true, "format": "invalid", "address": "example.com:80", "password": "xxx", "key": "key1" } }}}`, false},
|
||||||
|
|
||||||
// Test 26 - Test valid Format for Redis
|
// Test 26 - Test valid Format for Redis
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "redis": { "1": { "enable": true, "format": "namespace", "address": "example.com:80", "password": "xxx", "key": "key1" } }}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "redis": { "1": { "enable": true, "format": "namespace", "address": "example.com:80", "password": "xxx", "key": "key1" } }}}`, true},
|
||||||
|
|
||||||
// Test 27 - Test MQTT
|
// Test 27 - Test MQTT
|
||||||
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mqtt": { "1": { "enable": true, "broker": "", "topic": "", "qos": 0, "clientId": "", "username": "", "password": ""}}}}`, true},
|
{`{"version": "` + v + `", "credential": { "accessKey": "minio", "secretKey": "minio123" }, "region": "us-east-1", "browser": "on", "notify": { "mqtt": { "1": { "enable": true, "broker": "", "topic": "", "qos": 0, "clientId": "", "username": "", "password": ""}}}}`, false},
|
||||||
}
|
}
|
||||||
|
|
||||||
for i, testCase := range testCases {
|
for i, testCase := range testCases {
|
||||||
|
@ -18,6 +18,7 @@ package cmd
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"path/filepath"
|
"path/filepath"
|
||||||
"strings"
|
"strings"
|
||||||
|
|
||||||
@ -44,6 +45,15 @@ func (cfg *CacheConfig) UnmarshalJSON(data []byte) (err error) {
|
|||||||
if err = json.Unmarshal(data, _cfg); err != nil {
|
if err = json.Unmarshal(data, _cfg); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
|
||||||
|
if _cfg.Expiry < 0 {
|
||||||
|
return errors.New("config expiry value should not be negative")
|
||||||
|
}
|
||||||
|
|
||||||
|
if _cfg.MaxUse < 0 {
|
||||||
|
return errors.New("config max use value should not be null or negative")
|
||||||
|
}
|
||||||
|
|
||||||
if _, err = parseCacheDrives(_cfg.Drives); err != nil {
|
if _, err = parseCacheDrives(_cfg.Drives); err != nil {
|
||||||
return err
|
return err
|
||||||
}
|
}
|
||||||
|
@ -43,6 +43,17 @@ type AMQPArgs struct {
|
|||||||
AutoDeleted bool `json:"autoDeleted"`
|
AutoDeleted bool `json:"autoDeleted"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate AMQP arguments
|
||||||
|
func (a *AMQPArgs) Validate() error {
|
||||||
|
if !a.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if _, err := amqp.ParseURI(a.URL.String()); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// AMQPTarget - AMQP target
|
// AMQPTarget - AMQP target
|
||||||
type AMQPTarget struct {
|
type AMQPTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -18,8 +18,10 @@ package target
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"context"
|
"context"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
@ -36,6 +38,26 @@ type ElasticsearchArgs struct {
|
|||||||
Index string `json:"index"`
|
Index string `json:"index"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate ElasticsearchArgs fields
|
||||||
|
func (a ElasticsearchArgs) Validate() error {
|
||||||
|
if !a.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if a.URL.IsEmpty() {
|
||||||
|
return errors.New("empty URL")
|
||||||
|
}
|
||||||
|
if a.Format != "" {
|
||||||
|
f := strings.ToLower(a.Format)
|
||||||
|
if f != event.NamespaceFormat && f != event.AccessFormat {
|
||||||
|
return errors.New("format value unrecognized")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if a.Index == "" {
|
||||||
|
return errors.New("empty index value")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// ElasticsearchTarget - Elasticsearch target.
|
// ElasticsearchTarget - Elasticsearch target.
|
||||||
type ElasticsearchTarget struct {
|
type ElasticsearchTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -18,6 +18,7 @@ package target
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
@ -33,6 +34,22 @@ type KafkaArgs struct {
|
|||||||
Topic string `json:"topic"`
|
Topic string `json:"topic"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate KafkaArgs fields
|
||||||
|
func (k KafkaArgs) Validate() error {
|
||||||
|
if !k.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if len(k.Brokers) == 0 {
|
||||||
|
return errors.New("no broker address found")
|
||||||
|
}
|
||||||
|
for _, b := range k.Brokers {
|
||||||
|
if _, err := xnet.ParseHost(b.String()); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// KafkaTarget - Kafka target.
|
// KafkaTarget - Kafka target.
|
||||||
type KafkaTarget struct {
|
type KafkaTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -20,6 +20,7 @@ import (
|
|||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"crypto/x509"
|
"crypto/x509"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"net/url"
|
"net/url"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
@ -42,6 +43,23 @@ type MQTTArgs struct {
|
|||||||
RootCAs *x509.CertPool `json:"-"`
|
RootCAs *x509.CertPool `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate MQTTArgs fields
|
||||||
|
func (m MQTTArgs) Validate() error {
|
||||||
|
if !m.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
u, err := xnet.ParseURL(m.Broker.String())
|
||||||
|
if err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
switch u.Scheme {
|
||||||
|
case "ws", "wss", "tcp", "ssl", "tls", "tcps":
|
||||||
|
default:
|
||||||
|
return errors.New("unknown protocol in broker address")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// MQTTTarget - MQTT target.
|
// MQTTTarget - MQTT target.
|
||||||
type MQTTTarget struct {
|
type MQTTTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -58,6 +58,8 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strconv"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/go-sql-driver/mysql"
|
"github.com/go-sql-driver/mysql"
|
||||||
@ -88,6 +90,42 @@ type MySQLArgs struct {
|
|||||||
Database string `json:"database"`
|
Database string `json:"database"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate MySQLArgs fields
|
||||||
|
func (m MySQLArgs) Validate() error {
|
||||||
|
if !m.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.Format != "" {
|
||||||
|
f := strings.ToLower(m.Format)
|
||||||
|
if f != event.NamespaceFormat && f != event.AccessFormat {
|
||||||
|
return fmt.Errorf("unrecognized format")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.Table == "" {
|
||||||
|
return fmt.Errorf("table unspecified")
|
||||||
|
}
|
||||||
|
|
||||||
|
if m.DSN != "" {
|
||||||
|
if _, err := mysql.ParseDSN(m.DSN); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Some fields need to be specified when DSN is unspecified
|
||||||
|
if m.Port == "" {
|
||||||
|
return fmt.Errorf("unspecified port")
|
||||||
|
}
|
||||||
|
if _, err := strconv.Atoi(m.Port); err != nil {
|
||||||
|
return fmt.Errorf("invalid port")
|
||||||
|
}
|
||||||
|
if m.Database == "" {
|
||||||
|
return fmt.Errorf("database unspecified")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// MySQLTarget - MySQL target.
|
// MySQLTarget - MySQL target.
|
||||||
type MySQLTarget struct {
|
type MySQLTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -18,6 +18,7 @@ package target
|
|||||||
|
|
||||||
import (
|
import (
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
@ -45,6 +46,32 @@ type NATSArgs struct {
|
|||||||
} `json:"streaming"`
|
} `json:"streaming"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate NATSArgs fields
|
||||||
|
func (n NATSArgs) Validate() error {
|
||||||
|
if !n.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if n.Address.IsEmpty() {
|
||||||
|
return errors.New("empty address")
|
||||||
|
}
|
||||||
|
|
||||||
|
if n.Subject == "" {
|
||||||
|
return errors.New("empty subject")
|
||||||
|
}
|
||||||
|
|
||||||
|
if n.Streaming.Enable {
|
||||||
|
if n.Streaming.ClusterID == "" {
|
||||||
|
return errors.New("empty cluster id")
|
||||||
|
}
|
||||||
|
if n.Streaming.ClientID == "" {
|
||||||
|
return errors.New("empty client id")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// NATSTarget - NATS target.
|
// NATSTarget - NATS target.
|
||||||
type NATSTarget struct {
|
type NATSTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -58,10 +58,11 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strconv"
|
||||||
"strings"
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
_ "github.com/lib/pq" // Register postgres driver
|
"github.com/lib/pq" // Register postgres driver
|
||||||
"github.com/minio/minio/pkg/event"
|
"github.com/minio/minio/pkg/event"
|
||||||
xnet "github.com/minio/minio/pkg/net"
|
xnet "github.com/minio/minio/pkg/net"
|
||||||
)
|
)
|
||||||
@ -89,6 +90,41 @@ type PostgreSQLArgs struct {
|
|||||||
Database string `json:"database"` // default: same as user
|
Database string `json:"database"` // default: same as user
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate PostgreSQLArgs fields
|
||||||
|
func (p PostgreSQLArgs) Validate() error {
|
||||||
|
if !p.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if p.Table == "" {
|
||||||
|
return fmt.Errorf("empty table name")
|
||||||
|
}
|
||||||
|
if p.Format != "" {
|
||||||
|
f := strings.ToLower(p.Format)
|
||||||
|
if f != event.NamespaceFormat && f != event.AccessFormat {
|
||||||
|
return fmt.Errorf("unrecognized format value")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if p.ConnectionString != "" {
|
||||||
|
if _, err := pq.ParseURL(p.ConnectionString); err != nil {
|
||||||
|
return err
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
// Some fields need to be specified when ConnectionString is unspecified
|
||||||
|
if p.Port == "" {
|
||||||
|
return fmt.Errorf("unspecified port")
|
||||||
|
}
|
||||||
|
if _, err := strconv.Atoi(p.Port); err != nil {
|
||||||
|
return fmt.Errorf("invalid port")
|
||||||
|
}
|
||||||
|
if p.Database == "" {
|
||||||
|
return fmt.Errorf("database unspecified")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// PostgreSQLTarget - PostgreSQL target.
|
// PostgreSQLTarget - PostgreSQL target.
|
||||||
type PostgreSQLTarget struct {
|
type PostgreSQLTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -20,6 +20,7 @@ import (
|
|||||||
"encoding/json"
|
"encoding/json"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
|
"strings"
|
||||||
"time"
|
"time"
|
||||||
|
|
||||||
"github.com/garyburd/redigo/redis"
|
"github.com/garyburd/redigo/redis"
|
||||||
@ -36,6 +37,26 @@ type RedisArgs struct {
|
|||||||
Key string `json:"key"`
|
Key string `json:"key"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate RedisArgs fields
|
||||||
|
func (r RedisArgs) Validate() error {
|
||||||
|
if !r.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.Format != "" {
|
||||||
|
f := strings.ToLower(r.Format)
|
||||||
|
if f != event.NamespaceFormat && f != event.AccessFormat {
|
||||||
|
return fmt.Errorf("unrecognized format")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if r.Key == "" {
|
||||||
|
return fmt.Errorf("empty key")
|
||||||
|
}
|
||||||
|
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// RedisTarget - Redis target.
|
// RedisTarget - Redis target.
|
||||||
type RedisTarget struct {
|
type RedisTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
@ -21,6 +21,7 @@ import (
|
|||||||
"crypto/tls"
|
"crypto/tls"
|
||||||
"crypto/x509"
|
"crypto/x509"
|
||||||
"encoding/json"
|
"encoding/json"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net"
|
"net"
|
||||||
"net/http"
|
"net/http"
|
||||||
@ -38,6 +39,17 @@ type WebhookArgs struct {
|
|||||||
RootCAs *x509.CertPool `json:"-"`
|
RootCAs *x509.CertPool `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// Validate WebhookArgs fields
|
||||||
|
func (w WebhookArgs) Validate() error {
|
||||||
|
if !w.Enable {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if w.Endpoint.IsEmpty() {
|
||||||
|
return errors.New("endpoint empty")
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
// WebhookTarget - Webhook target.
|
// WebhookTarget - Webhook target.
|
||||||
type WebhookTarget struct {
|
type WebhookTarget struct {
|
||||||
id event.TargetID
|
id event.TargetID
|
||||||
|
Loading…
Reference in New Issue
Block a user