Add CRD configuration
With convertion for config, and start tests.
This commit is contained in:
		
							parent
							
								
									c028be493f
								
							
						
					
					
						commit
						7254039f2e
					
				| 
						 | 
					@ -152,6 +152,20 @@ type ScalyrConfiguration struct {
 | 
				
			||||||
	ScalyrMemoryLimit   string `json:"scalyr_memory_limit,omitempty"`
 | 
						ScalyrMemoryLimit   string `json:"scalyr_memory_limit,omitempty"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Defines default configuration for connection pool
 | 
				
			||||||
 | 
					type ConnectionPoolConfiguration struct {
 | 
				
			||||||
 | 
						NumberOfInstances    *int32 `json:"connection_pool_instances_number,omitempty"`
 | 
				
			||||||
 | 
						Schema               string `json:"connection_pool_schema,omitempty"`
 | 
				
			||||||
 | 
						User                 string `json:"connection_pool_user,omitempty"`
 | 
				
			||||||
 | 
						Type                 string `json:"connection_pool_type,omitempty"`
 | 
				
			||||||
 | 
						Image                string `json:"connection_pool_image,omitempty"`
 | 
				
			||||||
 | 
						Mode                 string `json:"connection_pool_mode,omitempty"`
 | 
				
			||||||
 | 
						DefaultCPURequest    string `name:"connection_pool_default_cpu_request,omitempty"`
 | 
				
			||||||
 | 
						DefaultMemoryRequest string `name:"connection_pool_default_memory_request,omitempty"`
 | 
				
			||||||
 | 
						DefaultCPULimit      string `name:"connection_pool_default_cpu_limit,omitempty"`
 | 
				
			||||||
 | 
						DefaultMemoryLimit   string `name:"connection_pool_default_memory_limit,omitempty"`
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// OperatorLogicalBackupConfiguration defines configuration for logical backup
 | 
					// OperatorLogicalBackupConfiguration defines configuration for logical backup
 | 
				
			||||||
type OperatorLogicalBackupConfiguration struct {
 | 
					type OperatorLogicalBackupConfiguration struct {
 | 
				
			||||||
	Schedule          string `json:"logical_backup_schedule,omitempty"`
 | 
						Schedule          string `json:"logical_backup_schedule,omitempty"`
 | 
				
			||||||
| 
						 | 
					@ -188,6 +202,7 @@ type OperatorConfigurationData struct {
 | 
				
			||||||
	LoggingRESTAPI             LoggingRESTAPIConfiguration        `json:"logging_rest_api"`
 | 
						LoggingRESTAPI             LoggingRESTAPIConfiguration        `json:"logging_rest_api"`
 | 
				
			||||||
	Scalyr                     ScalyrConfiguration                `json:"scalyr"`
 | 
						Scalyr                     ScalyrConfiguration                `json:"scalyr"`
 | 
				
			||||||
	LogicalBackup              OperatorLogicalBackupConfiguration `json:"logical_backup"`
 | 
						LogicalBackup              OperatorLogicalBackupConfiguration `json:"logical_backup"`
 | 
				
			||||||
 | 
						ConnectionPool             ConnectionPoolConfiguration        `json:"connection_pool"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
//Duration shortens this frequently used name
 | 
					//Duration shortens this frequently used name
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -165,4 +165,6 @@ type ConnectionPool struct {
 | 
				
			||||||
	Type              *string             `json:"type,omitempty"`
 | 
						Type              *string             `json:"type,omitempty"`
 | 
				
			||||||
	Mode              *string             `json:"mode,omitempty"`
 | 
						Mode              *string             `json:"mode,omitempty"`
 | 
				
			||||||
	PodTemplate       *v1.PodTemplateSpec `json:"podTemplate,omitempty"`
 | 
						PodTemplate       *v1.PodTemplateSpec `json:"podTemplate,omitempty"`
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						Resources `json:"resources,omitempty"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -120,10 +120,39 @@ func (c *Cluster) makeDefaultResources() acidv1.Resources {
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	config := c.OpConfig
 | 
						config := c.OpConfig
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	defaultRequests := acidv1.ResourceDescription{CPU: config.DefaultCPURequest, Memory: config.DefaultMemoryRequest}
 | 
						defaultRequests := acidv1.ResourceDescription{
 | 
				
			||||||
	defaultLimits := acidv1.ResourceDescription{CPU: config.DefaultCPULimit, Memory: config.DefaultMemoryLimit}
 | 
							CPU:    config.Resources.DefaultCPURequest,
 | 
				
			||||||
 | 
							Memory: config.Resources.DefaultMemoryRequest,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						defaultLimits := acidv1.ResourceDescription{
 | 
				
			||||||
 | 
							CPU:    config.Resources.DefaultCPULimit,
 | 
				
			||||||
 | 
							Memory: config.Resources.DefaultMemoryLimit,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return acidv1.Resources{ResourceRequests: defaultRequests, ResourceLimits: defaultLimits}
 | 
						return acidv1.Resources{
 | 
				
			||||||
 | 
							ResourceRequests: defaultRequests,
 | 
				
			||||||
 | 
							ResourceLimits:   defaultLimits,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					// Generate default resource section for connection pool deployment, to be used
 | 
				
			||||||
 | 
					// if nothing custom is specified in the manifest
 | 
				
			||||||
 | 
					func (c *Cluster) makeDefaultConnPoolResources() acidv1.Resources {
 | 
				
			||||||
 | 
						config := c.OpConfig
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						defaultRequests := acidv1.ResourceDescription{
 | 
				
			||||||
 | 
							CPU:    config.ConnectionPool.ConnPoolDefaultCPURequest,
 | 
				
			||||||
 | 
							Memory: config.ConnectionPool.ConnPoolDefaultMemoryRequest,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						defaultLimits := acidv1.ResourceDescription{
 | 
				
			||||||
 | 
							CPU:    config.ConnectionPool.ConnPoolDefaultCPULimit,
 | 
				
			||||||
 | 
							Memory: config.ConnectionPool.ConnPoolDefaultMemoryLimit,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						return acidv1.Resources{
 | 
				
			||||||
 | 
							ResourceRequests: defaultRequests,
 | 
				
			||||||
 | 
							ResourceLimits:   defaultLimits,
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
func generateResourceRequirements(resources acidv1.Resources, defaultResources acidv1.Resources) (*v1.ResourceRequirements, error) {
 | 
					func generateResourceRequirements(resources acidv1.Resources, defaultResources acidv1.Resources) (*v1.ResourceRequirements, error) {
 | 
				
			||||||
| 
						 | 
					@ -765,12 +794,12 @@ func (c *Cluster) generateStatefulSet(spec *acidv1.PostgresSpec) (*appsv1.Statef
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		request := spec.Resources.ResourceRequests.Memory
 | 
							request := spec.Resources.ResourceRequests.Memory
 | 
				
			||||||
		if request == "" {
 | 
							if request == "" {
 | 
				
			||||||
			request = c.OpConfig.DefaultMemoryRequest
 | 
								request = c.OpConfig.Resources.DefaultMemoryRequest
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		limit := spec.Resources.ResourceLimits.Memory
 | 
							limit := spec.Resources.ResourceLimits.Memory
 | 
				
			||||||
		if limit == "" {
 | 
							if limit == "" {
 | 
				
			||||||
			limit = c.OpConfig.DefaultMemoryLimit
 | 
								limit = c.OpConfig.Resources.DefaultMemoryLimit
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		isSmaller, err := util.IsSmallerQuantity(request, limit)
 | 
							isSmaller, err := util.IsSmallerQuantity(request, limit)
 | 
				
			||||||
| 
						 | 
					@ -792,12 +821,12 @@ func (c *Cluster) generateStatefulSet(spec *acidv1.PostgresSpec) (*appsv1.Statef
 | 
				
			||||||
			// TODO #413
 | 
								// TODO #413
 | 
				
			||||||
			sidecarRequest := sidecar.Resources.ResourceRequests.Memory
 | 
								sidecarRequest := sidecar.Resources.ResourceRequests.Memory
 | 
				
			||||||
			if request == "" {
 | 
								if request == "" {
 | 
				
			||||||
				request = c.OpConfig.DefaultMemoryRequest
 | 
									request = c.OpConfig.Resources.DefaultMemoryRequest
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			sidecarLimit := sidecar.Resources.ResourceLimits.Memory
 | 
								sidecarLimit := sidecar.Resources.ResourceLimits.Memory
 | 
				
			||||||
			if limit == "" {
 | 
								if limit == "" {
 | 
				
			||||||
				limit = c.OpConfig.DefaultMemoryLimit
 | 
									limit = c.OpConfig.Resources.DefaultMemoryLimit
 | 
				
			||||||
			}
 | 
								}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
			isSmaller, err := util.IsSmallerQuantity(sidecarRequest, sidecarLimit)
 | 
								isSmaller, err := util.IsSmallerQuantity(sidecarRequest, sidecarLimit)
 | 
				
			||||||
| 
						 | 
					@ -1710,8 +1739,8 @@ func (c *Cluster) generateConnPoolPodTemplate(spec *acidv1.PostgresSpec) (
 | 
				
			||||||
	if podTemplate == nil {
 | 
						if podTemplate == nil {
 | 
				
			||||||
		gracePeriod := int64(c.OpConfig.PodTerminateGracePeriod.Seconds())
 | 
							gracePeriod := int64(c.OpConfig.PodTerminateGracePeriod.Seconds())
 | 
				
			||||||
		resources, err := generateResourceRequirements(
 | 
							resources, err := generateResourceRequirements(
 | 
				
			||||||
			c.Spec.Resources,
 | 
								spec.ConnectionPool.Resources,
 | 
				
			||||||
			c.makeDefaultResources())
 | 
								c.makeDefaultConnPoolResources())
 | 
				
			||||||
 | 
					
 | 
				
			||||||
		effectiveMode := spec.ConnectionPool.Mode
 | 
							effectiveMode := spec.ConnectionPool.Mode
 | 
				
			||||||
		if effectiveMode == nil {
 | 
							if effectiveMode == nil {
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -1,6 +1,7 @@
 | 
				
			||||||
package cluster
 | 
					package cluster
 | 
				
			||||||
 | 
					
 | 
				
			||||||
import (
 | 
					import (
 | 
				
			||||||
 | 
						"errors"
 | 
				
			||||||
	"reflect"
 | 
						"reflect"
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	v1 "k8s.io/api/core/v1"
 | 
						v1 "k8s.io/api/core/v1"
 | 
				
			||||||
| 
						 | 
					@ -451,3 +452,67 @@ func TestSecretVolume(t *testing.T) {
 | 
				
			||||||
		}
 | 
							}
 | 
				
			||||||
	}
 | 
						}
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
					func TestConnPoolPodTemplate(t *testing.T) {
 | 
				
			||||||
 | 
						testName := "Test connection pool pod template generation"
 | 
				
			||||||
 | 
						var cluster = New(
 | 
				
			||||||
 | 
							Config{
 | 
				
			||||||
 | 
								OpConfig: config.Config{
 | 
				
			||||||
 | 
									ProtectedRoles: []string{"admin"},
 | 
				
			||||||
 | 
									Auth: config.Auth{
 | 
				
			||||||
 | 
										SuperUsername:       superUserName,
 | 
				
			||||||
 | 
										ReplicationUsername: replicationUserName,
 | 
				
			||||||
 | 
									},
 | 
				
			||||||
 | 
									ConnectionPool: config.ConnectionPool{
 | 
				
			||||||
 | 
										ConnPoolDefaultCPURequest:    "100m",
 | 
				
			||||||
 | 
										ConnPoolDefaultCPULimit:      "100m",
 | 
				
			||||||
 | 
										ConnPoolDefaultMemoryRequest: "100M",
 | 
				
			||||||
 | 
										ConnPoolDefaultMemoryLimit:   "100M",
 | 
				
			||||||
 | 
									},
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
							}, k8sutil.KubernetesClient{}, acidv1.Postgresql{}, logger)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						var clusterNoDefaultRes = New(
 | 
				
			||||||
 | 
							Config{
 | 
				
			||||||
 | 
								OpConfig: config.Config{
 | 
				
			||||||
 | 
									ProtectedRoles: []string{"admin"},
 | 
				
			||||||
 | 
									Auth: config.Auth{
 | 
				
			||||||
 | 
										SuperUsername:       superUserName,
 | 
				
			||||||
 | 
										ReplicationUsername: replicationUserName,
 | 
				
			||||||
 | 
									},
 | 
				
			||||||
 | 
									ConnectionPool: config.ConnectionPool{},
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
							}, k8sutil.KubernetesClient{}, acidv1.Postgresql{}, logger)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						tests := []struct {
 | 
				
			||||||
 | 
							subTest  string
 | 
				
			||||||
 | 
							spec     *acidv1.PostgresSpec
 | 
				
			||||||
 | 
							expected error
 | 
				
			||||||
 | 
							cluster  *Cluster
 | 
				
			||||||
 | 
						}{
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								subTest: "empty pod template",
 | 
				
			||||||
 | 
								spec: &acidv1.PostgresSpec{
 | 
				
			||||||
 | 
									ConnectionPool: &acidv1.ConnectionPool{},
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
								expected: nil,
 | 
				
			||||||
 | 
								cluster:  cluster,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
							{
 | 
				
			||||||
 | 
								subTest: "no default resources",
 | 
				
			||||||
 | 
								spec: &acidv1.PostgresSpec{
 | 
				
			||||||
 | 
									ConnectionPool: &acidv1.ConnectionPool{},
 | 
				
			||||||
 | 
								},
 | 
				
			||||||
 | 
								expected: errors.New(`could not generate resource requirements: could not fill resource requests: could not parse default CPU quantity: quantities must match the regular expression '^([+-]?[0-9.]+)([eEinumkKMGTP]*[-+]?[0-9]*)$'`),
 | 
				
			||||||
 | 
								cluster:  clusterNoDefaultRes,
 | 
				
			||||||
 | 
							},
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
						for _, tt := range tests {
 | 
				
			||||||
 | 
							_, err := tt.cluster.generateConnPoolPodTemplate(tt.spec)
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
							if err != tt.expected && err.Error() != tt.expected.Error() {
 | 
				
			||||||
 | 
								t.Errorf("%s [%s]: Could not generate pod template,\n %+v, expected\n %+v",
 | 
				
			||||||
 | 
									testName, tt.subTest, err, tt.expected)
 | 
				
			||||||
 | 
							}
 | 
				
			||||||
 | 
						}
 | 
				
			||||||
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -142,5 +142,17 @@ func (c *Controller) importConfigurationFromCRD(fromCRD *acidv1.OperatorConfigur
 | 
				
			||||||
	result.ScalyrCPULimit = fromCRD.Scalyr.ScalyrCPULimit
 | 
						result.ScalyrCPULimit = fromCRD.Scalyr.ScalyrCPULimit
 | 
				
			||||||
	result.ScalyrMemoryLimit = fromCRD.Scalyr.ScalyrMemoryLimit
 | 
						result.ScalyrMemoryLimit = fromCRD.Scalyr.ScalyrMemoryLimit
 | 
				
			||||||
 | 
					
 | 
				
			||||||
 | 
						// connection pool
 | 
				
			||||||
 | 
						result.ConnectionPool.NumberOfInstances = fromCRD.ConnectionPool.NumberOfInstances
 | 
				
			||||||
 | 
						result.ConnectionPool.Schema = fromCRD.ConnectionPool.Schema
 | 
				
			||||||
 | 
						result.ConnectionPool.User = fromCRD.ConnectionPool.User
 | 
				
			||||||
 | 
						result.ConnectionPool.Type = fromCRD.ConnectionPool.Type
 | 
				
			||||||
 | 
						result.ConnectionPool.Image = fromCRD.ConnectionPool.Image
 | 
				
			||||||
 | 
						result.ConnectionPool.Mode = fromCRD.ConnectionPool.Mode
 | 
				
			||||||
 | 
						result.ConnectionPool.ConnPoolDefaultCPURequest = fromCRD.ConnectionPool.DefaultCPURequest
 | 
				
			||||||
 | 
						result.ConnectionPool.ConnPoolDefaultMemoryRequest = fromCRD.ConnectionPool.DefaultMemoryRequest
 | 
				
			||||||
 | 
						result.ConnectionPool.ConnPoolDefaultCPULimit = fromCRD.ConnectionPool.DefaultCPULimit
 | 
				
			||||||
 | 
						result.ConnectionPool.ConnPoolDefaultMemoryLimit = fromCRD.ConnectionPool.DefaultMemoryLimit
 | 
				
			||||||
 | 
					
 | 
				
			||||||
	return result
 | 
						return result
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
| 
						 | 
					@ -91,6 +91,10 @@ type ConnectionPool struct {
 | 
				
			||||||
	Type                         string `name:"connection_pool_type" default:"pgbouncer"`
 | 
						Type                         string `name:"connection_pool_type" default:"pgbouncer"`
 | 
				
			||||||
	Image                        string `name:"connection_pool_image"  default:"pgbouncer:1.0"`
 | 
						Image                        string `name:"connection_pool_image"  default:"pgbouncer:1.0"`
 | 
				
			||||||
	Mode                         string `name:"connection_pool_mode"  default:"session"`
 | 
						Mode                         string `name:"connection_pool_mode"  default:"session"`
 | 
				
			||||||
 | 
						ConnPoolDefaultCPURequest    string `name:"connection_pool_default_cpu_request" default:"100m"`
 | 
				
			||||||
 | 
						ConnPoolDefaultMemoryRequest string `name:"connection_pool_default_memory_request" default:"100Mi"`
 | 
				
			||||||
 | 
						ConnPoolDefaultCPULimit      string `name:"connection_pool_default_cpu_limit" default:"3"`
 | 
				
			||||||
 | 
						ConnPoolDefaultMemoryLimit   string `name:"connection_pool_default_memory_limit" default:"1Gi"`
 | 
				
			||||||
}
 | 
					}
 | 
				
			||||||
 | 
					
 | 
				
			||||||
// Config describes operator config
 | 
					// Config describes operator config
 | 
				
			||||||
| 
						 | 
					
 | 
				
			||||||
		Loading…
	
		Reference in New Issue