host/dynamicconfig.go (166 lines of code) (raw):

// Copyright (c) 2017 Uber Technologies, Inc. // // Permission is hereby granted, free of charge, to any person obtaining a copy // of this software and associated documentation files (the "Software"), to deal // in the Software without restriction, including without limitation the rights // to use, copy, modify, merge, publish, distribute, sublicense, and/or sell // copies of the Software, and to permit persons to whom the Software is // furnished to do so, subject to the following conditions: // // The above copyright notice and this permission notice shall be included in // all copies or substantial portions of the Software. // // THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR // IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, // FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE // AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER // LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, // OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN // THE SOFTWARE. package host import ( "sync" "time" "github.com/uber/cadence/common" "github.com/uber/cadence/common/dynamicconfig" "github.com/uber/cadence/common/types" ) var ( // Override values for dynamic configs staticOverrides = map[dynamicconfig.Key]interface{}{ dynamicconfig.FrontendUserRPS: 3000, dynamicconfig.FrontendVisibilityListMaxQPS: 200, dynamicconfig.FrontendESIndexMaxResultWindow: defaultTestValueOfESIndexMaxResultWindow, dynamicconfig.MatchingNumTasklistWritePartitions: 3, dynamicconfig.MatchingNumTasklistReadPartitions: 3, dynamicconfig.TimerProcessorHistoryArchivalSizeLimit: 5 * 1024, dynamicconfig.ReplicationTaskProcessorErrorRetryMaxAttempts: 1, dynamicconfig.AdvancedVisibilityWritingMode: common.AdvancedVisibilityWritingModeOff, dynamicconfig.DecisionHeartbeatTimeout: 5 * time.Second, dynamicconfig.ReplicationTaskFetcherAggregationInterval: 200 * time.Millisecond, dynamicconfig.ReplicationTaskFetcherErrorRetryWait: 50 * time.Millisecond, dynamicconfig.ReplicationTaskProcessorErrorRetryWait: time.Millisecond, dynamicconfig.EnableConsistentQueryByDomain: true, dynamicconfig.MinRetentionDays: 0, dynamicconfig.WorkflowDeletionJitterRange: 1, } ) type dynamicClient struct { sync.RWMutex overrides map[dynamicconfig.Key]interface{} client dynamicconfig.Client } func (d *dynamicClient) GetValue(name dynamicconfig.Key) (interface{}, error) { d.RLock() if val, ok := d.overrides[name]; ok { d.RUnlock() return val, nil } d.RUnlock() return d.client.GetValue(name) } func (d *dynamicClient) GetValueWithFilters(name dynamicconfig.Key, filters map[dynamicconfig.Filter]interface{}) (interface{}, error) { d.RLock() if val, ok := d.overrides[name]; ok { d.RUnlock() return val, nil } d.RUnlock() return d.client.GetValueWithFilters(name, filters) } func (d *dynamicClient) GetIntValue(name dynamicconfig.IntKey, filters map[dynamicconfig.Filter]interface{}) (int, error) { d.RLock() if val, ok := d.overrides[name]; ok { if intVal, ok := val.(int); ok { d.RUnlock() return intVal, nil } } d.RUnlock() return d.client.GetIntValue(name, filters) } func (d *dynamicClient) GetFloatValue(name dynamicconfig.FloatKey, filters map[dynamicconfig.Filter]interface{}) (float64, error) { d.RLock() if val, ok := d.overrides[name]; ok { if floatVal, ok := val.(float64); ok { d.RUnlock() return floatVal, nil } } d.RUnlock() return d.client.GetFloatValue(name, filters) } func (d *dynamicClient) GetBoolValue(name dynamicconfig.BoolKey, filters map[dynamicconfig.Filter]interface{}) (bool, error) { d.RLock() if val, ok := d.overrides[name]; ok { if boolVal, ok := val.(bool); ok { d.RUnlock() return boolVal, nil } } d.RUnlock() return d.client.GetBoolValue(name, filters) } func (d *dynamicClient) GetStringValue(name dynamicconfig.StringKey, filters map[dynamicconfig.Filter]interface{}) (string, error) { d.RLock() if val, ok := d.overrides[name]; ok { if stringVal, ok := val.(string); ok { d.RUnlock() return stringVal, nil } } d.RUnlock() return d.client.GetStringValue(name, filters) } func (d *dynamicClient) GetMapValue(name dynamicconfig.MapKey, filters map[dynamicconfig.Filter]interface{}) (map[string]interface{}, error) { d.RLock() if val, ok := d.overrides[name]; ok { if mapVal, ok := val.(map[string]interface{}); ok { d.RUnlock() return mapVal, nil } } d.RUnlock() return d.client.GetMapValue(name, filters) } func (d *dynamicClient) GetDurationValue(name dynamicconfig.DurationKey, filters map[dynamicconfig.Filter]interface{}) (time.Duration, error) { d.RLock() if val, ok := d.overrides[name]; ok { if durationVal, ok := val.(time.Duration); ok { d.RUnlock() return durationVal, nil } } d.RUnlock() return d.client.GetDurationValue(name, filters) } func (d *dynamicClient) GetListValue(name dynamicconfig.ListKey, filters map[dynamicconfig.Filter]interface{}) ([]interface{}, error) { d.RLock() if val, ok := d.overrides[name]; ok { if listVal, ok := val.([]interface{}); ok { d.RUnlock() return listVal, nil } } d.RUnlock() return d.client.GetListValue(name, filters) } func (d *dynamicClient) UpdateValue(name dynamicconfig.Key, value interface{}) error { if name == dynamicconfig.AdvancedVisibilityWritingMode { // override for es integration tests d.Lock() defer d.Unlock() d.overrides[dynamicconfig.AdvancedVisibilityWritingMode] = value.(string) return nil } else if name == dynamicconfig.EnableReadVisibilityFromES { // override for pinot integration tests d.Lock() defer d.Unlock() d.overrides[dynamicconfig.EnableReadVisibilityFromES] = value.(bool) return nil } return d.client.UpdateValue(name, value) } func (d *dynamicClient) OverrideValue(name dynamicconfig.Key, value interface{}) { d.Lock() defer d.Unlock() d.overrides[name] = value } func (d *dynamicClient) ListValue(name dynamicconfig.Key) ([]*types.DynamicConfigEntry, error) { return d.client.ListValue(name) } func (d *dynamicClient) RestoreValue(name dynamicconfig.Key, filters map[dynamicconfig.Filter]interface{}) error { return d.client.RestoreValue(name, filters) } var _ dynamicconfig.Client = (*dynamicClient)(nil) // newIntegrationConfigClient - returns a dynamic config client for integration testing func newIntegrationConfigClient(client dynamicconfig.Client, overrides map[dynamicconfig.Key]interface{}) *dynamicClient { integrationClient := &dynamicClient{ overrides: make(map[dynamicconfig.Key]interface{}), client: client, } for key, value := range staticOverrides { integrationClient.OverrideValue(key, value) } for key, value := range overrides { integrationClient.OverrideValue(key, value) } return integrationClient }