Change error computation on JSON Unmarshal and create subtests on table test scenarios (#801)
* Change error computation on JSON Unmarshall The [Unmarshall function][1] on the encoding/JSON default library returns different errors for different go versions. On Go 1.12, the version used currently on the CI system it returns `json: cannot unmarshal number into Go struct field PostgresSpec.teamId of type string`. On Go 1.13.5 it returns `json: cannot unmarshal number into Go struct field PostgresSpec.spec.teamId of type string`. The new version includes more details of the whole structure being unmarshelled. This commit introduces the same error but one level deeper on the JSON structure. It creates consistency across different Go versions. [1]: https://godoc.org/encoding/json#Unmarshal * Create subtests on table test scenarios The Run method of T allows defining subtests creating hierarchical tests. It provides better visibility of tests in case of failure. More details on https://golang.org/pkg/testing/. This commit converts each test scenario on pkg/apis/acid.zalan.do/v1/util_test.go to subtests, providing a better visibility and the debugging environment when working with tests. The following code snippet shows an error during test execution with subtests: ``` --- FAIL: TestUnmarshalMaintenanceWindow (0.00s) --- FAIL: TestUnmarshalMaintenanceWindow/expect_error_as_'From'_is_later_than_'To' (0.00s) ``` It included a `about` field on test scenarios describing the test purpose and/or it expected output. When a description was provided with comments it was moved to the about field.
This commit is contained in:
		
							parent
							
								
									7fb163252c
								
							
						
					
					
						commit
						fddaf0fb73
					
				|  | @ -13,127 +13,139 @@ import ( | ||||||
| ) | ) | ||||||
| 
 | 
 | ||||||
| var parseTimeTests = []struct { | var parseTimeTests = []struct { | ||||||
| 	in  string | 	about string | ||||||
| 	out metav1.Time | 	in    string | ||||||
| 	err error | 	out   metav1.Time | ||||||
|  | 	err   error | ||||||
| }{ | }{ | ||||||
| 	{"16:08", mustParseTime("16:08"), nil}, | 	{"parse common time with minutes", "16:08", mustParseTime("16:08"), nil}, | ||||||
| 	{"11:00", mustParseTime("11:00"), nil}, | 	{"parse time with zeroed minutes", "11:00", mustParseTime("11:00"), nil}, | ||||||
| 	{"23:59", mustParseTime("23:59"), nil}, | 	{"parse corner case last minute of the day", "23:59", mustParseTime("23:59"), nil}, | ||||||
| 
 | 
 | ||||||
| 	{"26:09", metav1.Now(), errors.New(`parsing time "26:09": hour out of range`)}, | 	{"expect error as hour is out of range", "26:09", metav1.Now(), errors.New(`parsing time "26:09": hour out of range`)}, | ||||||
| 	{"23:69", metav1.Now(), errors.New(`parsing time "23:69": minute out of range`)}, | 	{"expect error as minute is out of range", "23:69", metav1.Now(), errors.New(`parsing time "23:69": minute out of range`)}, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var parseWeekdayTests = []struct { | var parseWeekdayTests = []struct { | ||||||
| 	in  string | 	about string | ||||||
| 	out time.Weekday | 	in    string | ||||||
| 	err error | 	out   time.Weekday | ||||||
|  | 	err   error | ||||||
| }{ | }{ | ||||||
| 	{"Wed", time.Wednesday, nil}, | 	{"parse common weekday", "Wed", time.Wednesday, nil}, | ||||||
| 	{"Sunday", time.Weekday(0), errors.New("incorrect weekday")}, | 	{"expect error as weekday is invalid", "Sunday", time.Weekday(0), errors.New("incorrect weekday")}, | ||||||
| 	{"", time.Weekday(0), errors.New("incorrect weekday")}, | 	{"expect error as weekday is empty", "", time.Weekday(0), errors.New("incorrect weekday")}, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var clusterNames = []struct { | var clusterNames = []struct { | ||||||
|  | 	about       string | ||||||
| 	in          string | 	in          string | ||||||
| 	inTeam      string | 	inTeam      string | ||||||
| 	clusterName string | 	clusterName string | ||||||
| 	err         error | 	err         error | ||||||
| }{ | }{ | ||||||
| 	{"acid-test", "acid", "test", nil}, | 	{"common team and cluster name", "acid-test", "acid", "test", nil}, | ||||||
| 	{"test-my-name", "test", "my-name", nil}, | 	{"cluster name with hyphen", "test-my-name", "test", "my-name", nil}, | ||||||
| 	{"my-team-another-test", "my-team", "another-test", nil}, | 	{"cluster and team name with hyphen", "my-team-another-test", "my-team", "another-test", nil}, | ||||||
| 	{"------strange-team-cluster", "-----", "strange-team-cluster", | 	{"expect error as cluster name is just hyphens", "------strange-team-cluster", "-----", "strange-team-cluster", | ||||||
| 		errors.New(`name must confirm to DNS-1035, regex used for validation is "^[a-z]([-a-z0-9]*[a-z0-9])?$"`)}, | 		errors.New(`name must confirm to DNS-1035, regex used for validation is "^[a-z]([-a-z0-9]*[a-z0-9])?$"`)}, | ||||||
| 	{"fooobar-fooobarfooobarfooobarfooobarfooobarfooobarfooobarfooobar", "fooobar", "", | 	{"expect error as cluster name is too long", "fooobar-fooobarfooobarfooobarfooobarfooobarfooobarfooobarfooobar", "fooobar", "", | ||||||
| 		errors.New("name cannot be longer than 58 characters")}, | 		errors.New("name cannot be longer than 58 characters")}, | ||||||
| 	{"acid-test", "test", "", errors.New("name must match {TEAM}-{NAME} format")}, | 	{"expect error as cluster name does not match {TEAM}-{NAME} format", "acid-test", "test", "", errors.New("name must match {TEAM}-{NAME} format")}, | ||||||
| 	{"-test", "", "", errors.New("team name is empty")}, | 	{"expect error as team and cluster name are empty", "-test", "", "", errors.New("team name is empty")}, | ||||||
| 	{"-test", "-", "", errors.New("name must match {TEAM}-{NAME} format")}, | 	{"expect error as cluster name is empty and team name is a hyphen", "-test", "-", "", errors.New("name must match {TEAM}-{NAME} format")}, | ||||||
| 	{"", "-", "", errors.New("cluster name must match {TEAM}-{NAME} format. Got cluster name '', team name '-'")}, | 	{"expect error as cluster name is empty, team name is a hyphen and cluster name is empty", "", "-", "", errors.New("cluster name must match {TEAM}-{NAME} format. Got cluster name '', team name '-'")}, | ||||||
| 	{"-", "-", "", errors.New("cluster name must match {TEAM}-{NAME} format. Got cluster name '-', team name '-'")}, | 	{"expect error as cluster and team name are hyphens", "-", "-", "", errors.New("cluster name must match {TEAM}-{NAME} format. Got cluster name '-', team name '-'")}, | ||||||
| 	// user may specify the team part of the full cluster name differently from the team name returned by the Teams API
 | 	// user may specify the team part of the full cluster name differently from the team name returned by the Teams API
 | ||||||
| 	// in the case the actual Teams API name is long enough, this will fail the check
 | 	// in the case the actual Teams API name is long enough, this will fail the check
 | ||||||
| 	{"foo-bar", "qwerty", "", errors.New("cluster name must match {TEAM}-{NAME} format. Got cluster name 'foo-bar', team name 'qwerty'")}, | 	{"expect error as team name does not match", "foo-bar", "qwerty", "", errors.New("cluster name must match {TEAM}-{NAME} format. Got cluster name 'foo-bar', team name 'qwerty'")}, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var cloneClusterDescriptions = []struct { | var cloneClusterDescriptions = []struct { | ||||||
| 	in  *CloneDescription | 	about string | ||||||
| 	err error | 	in    *CloneDescription | ||||||
|  | 	err   error | ||||||
| }{ | }{ | ||||||
| 	{&CloneDescription{"foo+bar", "", "NotEmpty", "", "", "", "", nil}, nil}, | 	{"cluster name invalid but EndTimeSet is not empty", &CloneDescription{"foo+bar", "", "NotEmpty", "", "", "", "", nil}, nil}, | ||||||
| 	{&CloneDescription{"foo+bar", "", "", "", "", "", "", nil}, | 	{"expect error as cluster name does not match DNS-1035", &CloneDescription{"foo+bar", "", "", "", "", "", "", nil}, | ||||||
| 		errors.New(`clone cluster name must confirm to DNS-1035, regex used for validation is "^[a-z]([-a-z0-9]*[a-z0-9])?$"`)}, | 		errors.New(`clone cluster name must confirm to DNS-1035, regex used for validation is "^[a-z]([-a-z0-9]*[a-z0-9])?$"`)}, | ||||||
| 	{&CloneDescription{"foobar123456789012345678901234567890123456789012345678901234567890", "", "", "", "", "", "", nil}, | 	{"expect error as cluster name is too long", &CloneDescription{"foobar123456789012345678901234567890123456789012345678901234567890", "", "", "", "", "", "", nil}, | ||||||
| 		errors.New("clone cluster name must be no longer than 63 characters")}, | 		errors.New("clone cluster name must be no longer than 63 characters")}, | ||||||
| 	{&CloneDescription{"foobar", "", "", "", "", "", "", nil}, nil}, | 	{"common cluster name", &CloneDescription{"foobar", "", "", "", "", "", "", nil}, nil}, | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| var maintenanceWindows = []struct { | var maintenanceWindows = []struct { | ||||||
| 	in  []byte | 	about string | ||||||
| 	out MaintenanceWindow | 	in    []byte | ||||||
| 	err error | 	out   MaintenanceWindow | ||||||
| }{{[]byte(`"Tue:10:00-20:00"`), | 	err   error | ||||||
|  | }{{"regular scenario", | ||||||
|  | 	[]byte(`"Tue:10:00-20:00"`), | ||||||
| 	MaintenanceWindow{ | 	MaintenanceWindow{ | ||||||
| 		Everyday:  false, | 		Everyday:  false, | ||||||
| 		Weekday:   time.Tuesday, | 		Weekday:   time.Tuesday, | ||||||
| 		StartTime: mustParseTime("10:00"), | 		StartTime: mustParseTime("10:00"), | ||||||
| 		EndTime:   mustParseTime("20:00"), | 		EndTime:   mustParseTime("20:00"), | ||||||
| 	}, nil}, | 	}, nil}, | ||||||
| 	{[]byte(`"Mon:10:00-10:00"`), | 	{"starts and ends at the same time", | ||||||
|  | 		[]byte(`"Mon:10:00-10:00"`), | ||||||
| 		MaintenanceWindow{ | 		MaintenanceWindow{ | ||||||
| 			Everyday:  false, | 			Everyday:  false, | ||||||
| 			Weekday:   time.Monday, | 			Weekday:   time.Monday, | ||||||
| 			StartTime: mustParseTime("10:00"), | 			StartTime: mustParseTime("10:00"), | ||||||
| 			EndTime:   mustParseTime("10:00"), | 			EndTime:   mustParseTime("10:00"), | ||||||
| 		}, nil}, | 		}, nil}, | ||||||
| 	{[]byte(`"Sun:00:00-00:00"`), | 	{"starts and ends 00:00 on sunday", | ||||||
|  | 		[]byte(`"Sun:00:00-00:00"`), | ||||||
| 		MaintenanceWindow{ | 		MaintenanceWindow{ | ||||||
| 			Everyday:  false, | 			Everyday:  false, | ||||||
| 			Weekday:   time.Sunday, | 			Weekday:   time.Sunday, | ||||||
| 			StartTime: mustParseTime("00:00"), | 			StartTime: mustParseTime("00:00"), | ||||||
| 			EndTime:   mustParseTime("00:00"), | 			EndTime:   mustParseTime("00:00"), | ||||||
| 		}, nil}, | 		}, nil}, | ||||||
| 	{[]byte(`"01:00-10:00"`), | 	{"without day indication should define to sunday", | ||||||
|  | 		[]byte(`"01:00-10:00"`), | ||||||
| 		MaintenanceWindow{ | 		MaintenanceWindow{ | ||||||
| 			Everyday:  true, | 			Everyday:  true, | ||||||
| 			Weekday:   time.Sunday, | 			Weekday:   time.Sunday, | ||||||
| 			StartTime: mustParseTime("01:00"), | 			StartTime: mustParseTime("01:00"), | ||||||
| 			EndTime:   mustParseTime("10:00"), | 			EndTime:   mustParseTime("10:00"), | ||||||
| 		}, nil}, | 		}, nil}, | ||||||
| 	{[]byte(`"Mon:12:00-11:00"`), MaintenanceWindow{}, errors.New(`'From' time must be prior to the 'To' time`)}, | 	{"expect error as 'From' is later than 'To'", []byte(`"Mon:12:00-11:00"`), MaintenanceWindow{}, errors.New(`'From' time must be prior to the 'To' time`)}, | ||||||
| 	{[]byte(`"Wed:33:00-00:00"`), MaintenanceWindow{}, errors.New(`could not parse start time: parsing time "33:00": hour out of range`)}, | 	{"expect error as 'From' is later than 'To' with 00:00 corner case", []byte(`"Mon:10:00-00:00"`), MaintenanceWindow{}, errors.New(`'From' time must be prior to the 'To' time`)}, | ||||||
| 	{[]byte(`"Wed:00:00-26:00"`), MaintenanceWindow{}, errors.New(`could not parse end time: parsing time "26:00": hour out of range`)}, | 	{"expect error as 'From' time is not valid", []byte(`"Wed:33:00-00:00"`), MaintenanceWindow{}, errors.New(`could not parse start time: parsing time "33:00": hour out of range`)}, | ||||||
| 	{[]byte(`"Sunday:00:00-00:00"`), MaintenanceWindow{}, errors.New(`could not parse weekday: incorrect weekday`)}, | 	{"expect error as 'To' time is not valid", []byte(`"Wed:00:00-26:00"`), MaintenanceWindow{}, errors.New(`could not parse end time: parsing time "26:00": hour out of range`)}, | ||||||
| 	{[]byte(`":00:00-10:00"`), MaintenanceWindow{}, errors.New(`could not parse weekday: incorrect weekday`)}, | 	{"expect error as weekday is not valid", []byte(`"Sunday:00:00-00:00"`), MaintenanceWindow{}, errors.New(`could not parse weekday: incorrect weekday`)}, | ||||||
| 	{[]byte(`"Mon:10:00-00:00"`), MaintenanceWindow{}, errors.New(`'From' time must be prior to the 'To' time`)}, | 	{"expect error as weekday is empty", []byte(`":00:00-10:00"`), MaintenanceWindow{}, errors.New(`could not parse weekday: incorrect weekday`)}, | ||||||
| 	{[]byte(`"Mon:00:00:00-10:00:00"`), MaintenanceWindow{}, errors.New(`incorrect maintenance window format`)}, | 	{"expect error as maintenance window set seconds", []byte(`"Mon:00:00:00-10:00:00"`), MaintenanceWindow{}, errors.New(`incorrect maintenance window format`)}, | ||||||
| 	{[]byte(`"Mon:00:00"`), MaintenanceWindow{}, errors.New("incorrect maintenance window format")}, | 	{"expect error as 'To' time set seconds", []byte(`"Mon:00:00-00:00:00"`), MaintenanceWindow{}, errors.New("could not parse end time: incorrect time format")}, | ||||||
| 	{[]byte(`"Mon:00:00-00:00:00"`), MaintenanceWindow{}, errors.New("could not parse end time: incorrect time format")}} | 	{"expect error as 'To' time is missing", []byte(`"Mon:00:00"`), MaintenanceWindow{}, errors.New("incorrect maintenance window format")}} | ||||||
| 
 | 
 | ||||||
| var postgresStatus = []struct { | var postgresStatus = []struct { | ||||||
| 	in  []byte | 	about string | ||||||
| 	out PostgresStatus | 	in    []byte | ||||||
| 	err error | 	out   PostgresStatus | ||||||
|  | 	err   error | ||||||
| }{ | }{ | ||||||
| 	{[]byte(`{"PostgresClusterStatus":"Running"}`), | 	{"cluster running", []byte(`{"PostgresClusterStatus":"Running"}`), | ||||||
| 		PostgresStatus{PostgresClusterStatus: ClusterStatusRunning}, nil}, | 		PostgresStatus{PostgresClusterStatus: ClusterStatusRunning}, nil}, | ||||||
| 	{[]byte(`{"PostgresClusterStatus":""}`), | 	{"cluster status undefined", []byte(`{"PostgresClusterStatus":""}`), | ||||||
| 		PostgresStatus{PostgresClusterStatus: ClusterStatusUnknown}, nil}, | 		PostgresStatus{PostgresClusterStatus: ClusterStatusUnknown}, nil}, | ||||||
| 	{[]byte(`"Running"`), | 	{"cluster running without full JSON format", []byte(`"Running"`), | ||||||
| 		PostgresStatus{PostgresClusterStatus: ClusterStatusRunning}, nil}, | 		PostgresStatus{PostgresClusterStatus: ClusterStatusRunning}, nil}, | ||||||
| 	{[]byte(`""`), | 	{"cluster status empty", []byte(`""`), | ||||||
| 		PostgresStatus{PostgresClusterStatus: ClusterStatusUnknown}, nil}} | 		PostgresStatus{PostgresClusterStatus: ClusterStatusUnknown}, nil}} | ||||||
| 
 | 
 | ||||||
|  | var tmp postgresqlCopy | ||||||
| var unmarshalCluster = []struct { | var unmarshalCluster = []struct { | ||||||
|  | 	about   string | ||||||
| 	in      []byte | 	in      []byte | ||||||
| 	out     Postgresql | 	out     Postgresql | ||||||
| 	marshal []byte | 	marshal []byte | ||||||
| 	err     error | 	err     error | ||||||
| }{ | }{ | ||||||
| 	// example with simple status field
 |  | ||||||
| 	{ | 	{ | ||||||
|  | 		about: "example with simple status field", | ||||||
| 		in: []byte(`{ | 		in: []byte(`{ | ||||||
| 	  "kind": "Postgresql","apiVersion": "acid.zalan.do/v1", | 	  "kind": "Postgresql","apiVersion": "acid.zalan.do/v1", | ||||||
| 	  "metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": 100}}`), | 	  "metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": 100}}`), | ||||||
|  | @ -147,12 +159,14 @@ var unmarshalCluster = []struct { | ||||||
| 			}, | 			}, | ||||||
| 			Status: PostgresStatus{PostgresClusterStatus: ClusterStatusInvalid}, | 			Status: PostgresStatus{PostgresClusterStatus: ClusterStatusInvalid}, | ||||||
| 			// This error message can vary between Go versions, so compute it for the current version.
 | 			// This error message can vary between Go versions, so compute it for the current version.
 | ||||||
| 			Error: json.Unmarshal([]byte(`{"teamId": 0}`), &PostgresSpec{}).Error(), | 			Error: json.Unmarshal([]byte(`{ | ||||||
|  | 				"kind": "Postgresql","apiVersion": "acid.zalan.do/v1", | ||||||
|  | 				"metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": 100}}`), &tmp).Error(), | ||||||
| 		}, | 		}, | ||||||
| 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":"Invalid"}`), | 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":"Invalid"}`), | ||||||
| 		err:     nil}, | 		err:     nil}, | ||||||
| 	// example with /status subresource
 |  | ||||||
| 	{ | 	{ | ||||||
|  | 		about: "example with /status subresource", | ||||||
| 		in: []byte(`{ | 		in: []byte(`{ | ||||||
| 	  "kind": "Postgresql","apiVersion": "acid.zalan.do/v1", | 	  "kind": "Postgresql","apiVersion": "acid.zalan.do/v1", | ||||||
| 	  "metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": 100}}`), | 	  "metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": 100}}`), | ||||||
|  | @ -166,13 +180,14 @@ var unmarshalCluster = []struct { | ||||||
| 			}, | 			}, | ||||||
| 			Status: PostgresStatus{PostgresClusterStatus: ClusterStatusInvalid}, | 			Status: PostgresStatus{PostgresClusterStatus: ClusterStatusInvalid}, | ||||||
| 			// This error message can vary between Go versions, so compute it for the current version.
 | 			// This error message can vary between Go versions, so compute it for the current version.
 | ||||||
| 			Error: json.Unmarshal([]byte(`{"teamId": 0}`), &PostgresSpec{}).Error(), | 			Error: json.Unmarshal([]byte(`{ | ||||||
|  | 				"kind": "Postgresql","apiVersion": "acid.zalan.do/v1", | ||||||
|  | 				"metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": 100}}`), &tmp).Error(), | ||||||
| 		}, | 		}, | ||||||
| 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":{"PostgresClusterStatus":"Invalid"}}`), | 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":{"PostgresClusterStatus":"Invalid"}}`), | ||||||
| 		err:     nil}, | 		err:     nil}, | ||||||
| 	// example with detailed input manifest
 |  | ||||||
| 	// and deprecated pod_priority_class_name -> podPriorityClassName
 |  | ||||||
| 	{ | 	{ | ||||||
|  | 		about: "example with detailed input manifest and deprecated pod_priority_class_name -> podPriorityClassName", | ||||||
| 		in: []byte(`{ | 		in: []byte(`{ | ||||||
| 	  "kind": "Postgresql", | 	  "kind": "Postgresql", | ||||||
| 	  "apiVersion": "acid.zalan.do/v1", | 	  "apiVersion": "acid.zalan.do/v1", | ||||||
|  | @ -321,9 +336,9 @@ var unmarshalCluster = []struct { | ||||||
| 		}, | 		}, | ||||||
| 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"9.6","parameters":{"log_statement":"all","max_connections":"10","shared_buffers":"32MB"}},"pod_priority_class_name":"spilo-pod-priority","volume":{"size":"5Gi","storageClass":"SSD", "subPath": "subdir"},"enableShmVolume":false,"patroni":{"initdb":{"data-checksums":"true","encoding":"UTF8","locale":"en_US.UTF-8"},"pg_hba":["hostssl all all 0.0.0.0/0 md5","host    all all 0.0.0.0/0 md5"],"ttl":30,"loop_wait":10,"retry_timeout":10,"maximum_lag_on_failover":33554432,"slots":{"permanent_logical_1":{"database":"foo","plugin":"pgoutput","type":"logical"}}},"resources":{"requests":{"cpu":"10m","memory":"50Mi"},"limits":{"cpu":"300m","memory":"3000Mi"}},"teamId":"acid","allowedSourceRanges":["127.0.0.1/32"],"numberOfInstances":2,"users":{"zalando":["superuser","createdb"]},"maintenanceWindows":["Mon:01:00-06:00","Sat:00:00-04:00","05:00-05:15"],"clone":{"cluster":"acid-batman"}},"status":{"PostgresClusterStatus":""}}`), | 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"9.6","parameters":{"log_statement":"all","max_connections":"10","shared_buffers":"32MB"}},"pod_priority_class_name":"spilo-pod-priority","volume":{"size":"5Gi","storageClass":"SSD", "subPath": "subdir"},"enableShmVolume":false,"patroni":{"initdb":{"data-checksums":"true","encoding":"UTF8","locale":"en_US.UTF-8"},"pg_hba":["hostssl all all 0.0.0.0/0 md5","host    all all 0.0.0.0/0 md5"],"ttl":30,"loop_wait":10,"retry_timeout":10,"maximum_lag_on_failover":33554432,"slots":{"permanent_logical_1":{"database":"foo","plugin":"pgoutput","type":"logical"}}},"resources":{"requests":{"cpu":"10m","memory":"50Mi"},"limits":{"cpu":"300m","memory":"3000Mi"}},"teamId":"acid","allowedSourceRanges":["127.0.0.1/32"],"numberOfInstances":2,"users":{"zalando":["superuser","createdb"]},"maintenanceWindows":["Mon:01:00-06:00","Sat:00:00-04:00","05:00-05:15"],"clone":{"cluster":"acid-batman"}},"status":{"PostgresClusterStatus":""}}`), | ||||||
| 		err:     nil}, | 		err:     nil}, | ||||||
| 	// example with teamId set in input
 |  | ||||||
| 	{ | 	{ | ||||||
| 		in: []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "teapot-testcluster1"}, "spec": {"teamId": "acid"}}`), | 		about: "example with teamId set in input", | ||||||
|  | 		in:    []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "teapot-testcluster1"}, "spec": {"teamId": "acid"}}`), | ||||||
| 		out: Postgresql{ | 		out: Postgresql{ | ||||||
| 			TypeMeta: metav1.TypeMeta{ | 			TypeMeta: metav1.TypeMeta{ | ||||||
| 				Kind:       "Postgresql", | 				Kind:       "Postgresql", | ||||||
|  | @ -338,9 +353,9 @@ var unmarshalCluster = []struct { | ||||||
| 		}, | 		}, | ||||||
| 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"teapot-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null} ,"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":{"PostgresClusterStatus":"Invalid"}}`), | 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"teapot-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null} ,"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":{"PostgresClusterStatus":"Invalid"}}`), | ||||||
| 		err:     nil}, | 		err:     nil}, | ||||||
| 	// clone example
 |  | ||||||
| 	{ | 	{ | ||||||
| 		in: []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": "acid", "clone": {"cluster": "team-batman"}}}`), | 		about: "example with clone", | ||||||
|  | 		in:    []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": "acid", "clone": {"cluster": "team-batman"}}}`), | ||||||
| 		out: Postgresql{ | 		out: Postgresql{ | ||||||
| 			TypeMeta: metav1.TypeMeta{ | 			TypeMeta: metav1.TypeMeta{ | ||||||
| 				Kind:       "Postgresql", | 				Kind:       "Postgresql", | ||||||
|  | @ -360,9 +375,9 @@ var unmarshalCluster = []struct { | ||||||
| 		}, | 		}, | ||||||
| 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{"cluster":"team-batman"}},"status":{"PostgresClusterStatus":""}}`), | 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{"cluster":"team-batman"}},"status":{"PostgresClusterStatus":""}}`), | ||||||
| 		err:     nil}, | 		err:     nil}, | ||||||
| 	// standby example
 |  | ||||||
| 	{ | 	{ | ||||||
| 		in: []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": "acid", "standby": {"s3_wal_path": "s3://custom/path/to/bucket/"}}}`), | 		about: "standby example", | ||||||
|  | 		in:    []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "acid-testcluster1"}, "spec": {"teamId": "acid", "standby": {"s3_wal_path": "s3://custom/path/to/bucket/"}}}`), | ||||||
| 		out: Postgresql{ | 		out: Postgresql{ | ||||||
| 			TypeMeta: metav1.TypeMeta{ | 			TypeMeta: metav1.TypeMeta{ | ||||||
| 				Kind:       "Postgresql", | 				Kind:       "Postgresql", | ||||||
|  | @ -382,24 +397,28 @@ var unmarshalCluster = []struct { | ||||||
| 		}, | 		}, | ||||||
| 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"standby":{"s3_wal_path":"s3://custom/path/to/bucket/"}},"status":{"PostgresClusterStatus":""}}`), | 		marshal: []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster1","creationTimestamp":null},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"standby":{"s3_wal_path":"s3://custom/path/to/bucket/"}},"status":{"PostgresClusterStatus":""}}`), | ||||||
| 		err:     nil}, | 		err:     nil}, | ||||||
| 	// erroneous examples
 |  | ||||||
| 	{ | 	{ | ||||||
|  | 		about:   "expect error on malformatted JSON", | ||||||
| 		in:      []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1"`), | 		in:      []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1"`), | ||||||
| 		out:     Postgresql{}, | 		out:     Postgresql{}, | ||||||
| 		marshal: []byte{}, | 		marshal: []byte{}, | ||||||
| 		err:     errors.New("unexpected end of JSON input")}, | 		err:     errors.New("unexpected end of JSON input")}, | ||||||
| 	{ | 	{ | ||||||
|  | 		about:   "expect error on JSON with field's value malformatted", | ||||||
| 		in:      []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster","creationTimestamp":qaz},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":{"PostgresClusterStatus":"Invalid"}}`), | 		in:      []byte(`{"kind":"Postgresql","apiVersion":"acid.zalan.do/v1","metadata":{"name":"acid-testcluster","creationTimestamp":qaz},"spec":{"postgresql":{"version":"","parameters":null},"volume":{"size":"","storageClass":""},"patroni":{"initdb":null,"pg_hba":null,"ttl":0,"loop_wait":0,"retry_timeout":0,"maximum_lag_on_failover":0,"slots":null},"resources":{"requests":{"cpu":"","memory":""},"limits":{"cpu":"","memory":""}},"teamId":"acid","allowedSourceRanges":null,"numberOfInstances":0,"users":null,"clone":{}},"status":{"PostgresClusterStatus":"Invalid"}}`), | ||||||
| 		out:     Postgresql{}, | 		out:     Postgresql{}, | ||||||
| 		marshal: []byte{}, | 		marshal: []byte{}, | ||||||
| 		err:     errors.New("invalid character 'q' looking for beginning of value")}} | 		err:     errors.New("invalid character 'q' looking for beginning of value"), | ||||||
|  | 	}, | ||||||
|  | } | ||||||
| 
 | 
 | ||||||
| var postgresqlList = []struct { | var postgresqlList = []struct { | ||||||
| 	in  []byte | 	about string | ||||||
| 	out PostgresqlList | 	in    []byte | ||||||
| 	err error | 	out   PostgresqlList | ||||||
|  | 	err   error | ||||||
| }{ | }{ | ||||||
| 	{[]byte(`{"apiVersion":"v1","items":[{"apiVersion":"acid.zalan.do/v1","kind":"Postgresql","metadata":{"labels":{"team":"acid"},"name":"acid-testcluster42","namespace":"default","resourceVersion":"30446957","selfLink":"/apis/acid.zalan.do/v1/namespaces/default/postgresqls/acid-testcluster42","uid":"857cd208-33dc-11e7-b20a-0699041e4b03"},"spec":{"allowedSourceRanges":["185.85.220.0/22"],"numberOfInstances":1,"postgresql":{"version":"9.6"},"teamId":"acid","volume":{"size":"10Gi"}},"status":{"PostgresClusterStatus":"Running"}}],"kind":"List","metadata":{},"resourceVersion":"","selfLink":""}`), | 	{"expect success", []byte(`{"apiVersion":"v1","items":[{"apiVersion":"acid.zalan.do/v1","kind":"Postgresql","metadata":{"labels":{"team":"acid"},"name":"acid-testcluster42","namespace":"default","resourceVersion":"30446957","selfLink":"/apis/acid.zalan.do/v1/namespaces/default/postgresqls/acid-testcluster42","uid":"857cd208-33dc-11e7-b20a-0699041e4b03"},"spec":{"allowedSourceRanges":["185.85.220.0/22"],"numberOfInstances":1,"postgresql":{"version":"9.6"},"teamId":"acid","volume":{"size":"10Gi"}},"status":{"PostgresClusterStatus":"Running"}}],"kind":"List","metadata":{},"resourceVersion":"","selfLink":""}`), | ||||||
| 		PostgresqlList{ | 		PostgresqlList{ | ||||||
| 			TypeMeta: metav1.TypeMeta{ | 			TypeMeta: metav1.TypeMeta{ | ||||||
| 				Kind:       "List", | 				Kind:       "List", | ||||||
|  | @ -433,15 +452,17 @@ var postgresqlList = []struct { | ||||||
| 			}}, | 			}}, | ||||||
| 		}, | 		}, | ||||||
| 		nil}, | 		nil}, | ||||||
| 	{[]byte(`{"apiVersion":"v1","items":[{"apiVersion":"acid.zalan.do/v1","kind":"Postgresql","metadata":{"labels":{"team":"acid"},"name":"acid-testcluster42","namespace"`), | 	{"expect error on malformatted JSON", []byte(`{"apiVersion":"v1","items":[{"apiVersion":"acid.zalan.do/v1","kind":"Postgresql","metadata":{"labels":{"team":"acid"},"name":"acid-testcluster42","namespace"`), | ||||||
| 		PostgresqlList{}, | 		PostgresqlList{}, | ||||||
| 		errors.New("unexpected end of JSON input")}} | 		errors.New("unexpected end of JSON input")}} | ||||||
| 
 | 
 | ||||||
| var annotations = []struct { | var annotations = []struct { | ||||||
|  | 	about       string | ||||||
| 	in          []byte | 	in          []byte | ||||||
| 	annotations map[string]string | 	annotations map[string]string | ||||||
| 	err         error | 	err         error | ||||||
| }{{ | }{{ | ||||||
|  | 	about:       "common annotations", | ||||||
| 	in:          []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "acid-testcluster1"}, "spec": {"podAnnotations": {"foo": "bar"},"teamId": "acid", "clone": {"cluster": "team-batman"}}}`), | 	in:          []byte(`{"kind": "Postgresql","apiVersion": "acid.zalan.do/v1","metadata": {"name": "acid-testcluster1"}, "spec": {"podAnnotations": {"foo": "bar"},"teamId": "acid", "clone": {"cluster": "team-batman"}}}`), | ||||||
| 	annotations: map[string]string{"foo": "bar"}, | 	annotations: map[string]string{"foo": "bar"}, | ||||||
| 	err:         nil}, | 	err:         nil}, | ||||||
|  | @ -458,230 +479,256 @@ func mustParseTime(s string) metav1.Time { | ||||||
| 
 | 
 | ||||||
| func TestParseTime(t *testing.T) { | func TestParseTime(t *testing.T) { | ||||||
| 	for _, tt := range parseTimeTests { | 	for _, tt := range parseTimeTests { | ||||||
| 		aTime, err := parseTime(tt.in) | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		if err != nil { | 			aTime, err := parseTime(tt.in) | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err != nil { | ||||||
| 				t.Errorf("ParseTime expected error: %v, got: %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("ParseTime expected error: %v, got: %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
|  | 			} else if tt.err != nil { | ||||||
|  | 				t.Errorf("Expected error: %v", tt.err) | ||||||
| 			} | 			} | ||||||
| 			continue |  | ||||||
| 		} else if tt.err != nil { |  | ||||||
| 			t.Errorf("Expected error: %v", tt.err) |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		if aTime != tt.out { | 			if aTime != tt.out { | ||||||
| 			t.Errorf("Expected time: %v, got: %v", tt.out, aTime) | 				t.Errorf("Expected time: %v, got: %v", tt.out, aTime) | ||||||
| 		} | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestWeekdayTime(t *testing.T) { | func TestWeekdayTime(t *testing.T) { | ||||||
| 	for _, tt := range parseWeekdayTests { | 	for _, tt := range parseWeekdayTests { | ||||||
| 		aTime, err := parseWeekday(tt.in) | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		if err != nil { | 			aTime, err := parseWeekday(tt.in) | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err != nil { | ||||||
| 				t.Errorf("ParseWeekday expected error: %v, got: %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("ParseWeekday expected error: %v, got: %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
|  | 			} else if tt.err != nil { | ||||||
|  | 				t.Errorf("Expected error: %v", tt.err) | ||||||
| 			} | 			} | ||||||
| 			continue |  | ||||||
| 		} else if tt.err != nil { |  | ||||||
| 			t.Errorf("Expected error: %v", tt.err) |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		if aTime != tt.out { | 			if aTime != tt.out { | ||||||
| 			t.Errorf("Expected weekday: %v, got: %v", tt.out, aTime) | 				t.Errorf("Expected weekday: %v, got: %v", tt.out, aTime) | ||||||
| 		} | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestClusterAnnotations(t *testing.T) { | func TestClusterAnnotations(t *testing.T) { | ||||||
| 	for _, tt := range annotations { | 	for _, tt := range annotations { | ||||||
| 		var cluster Postgresql | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		err := cluster.UnmarshalJSON(tt.in) | 			var cluster Postgresql | ||||||
| 		if err != nil { | 			err := cluster.UnmarshalJSON(tt.in) | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err != nil { | ||||||
| 				t.Errorf("Unable to marshal cluster with annotations: expected %v got %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("Unable to marshal cluster with annotations: expected %v got %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
| 			} | 			} | ||||||
| 			continue | 			for k, v := range cluster.Spec.PodAnnotations { | ||||||
| 		} | 				found, expected := v, tt.annotations[k] | ||||||
| 		for k, v := range cluster.Spec.PodAnnotations { | 				if found != expected { | ||||||
| 			found, expected := v, tt.annotations[k] | 					t.Errorf("Didn't find correct value for key %v in for podAnnotations:  Expected %v found %v", k, expected, found) | ||||||
| 			if found != expected { | 				} | ||||||
| 				t.Errorf("Didn't find correct value for key %v in for podAnnotations:  Expected %v found %v", k, expected, found) |  | ||||||
| 			} | 			} | ||||||
| 		} | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestClusterName(t *testing.T) { | func TestClusterName(t *testing.T) { | ||||||
| 	for _, tt := range clusterNames { | 	for _, tt := range clusterNames { | ||||||
| 		name, err := extractClusterName(tt.in, tt.inTeam) | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		if err != nil { | 			name, err := extractClusterName(tt.in, tt.inTeam) | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err != nil { | ||||||
| 				t.Errorf("extractClusterName expected error: %v, got: %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("extractClusterName expected error: %v, got: %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
|  | 			} else if tt.err != nil { | ||||||
|  | 				t.Errorf("Expected error: %v", tt.err) | ||||||
| 			} | 			} | ||||||
| 			continue | 			if name != tt.clusterName { | ||||||
| 		} else if tt.err != nil { | 				t.Errorf("Expected cluserName: %q, got: %q", tt.clusterName, name) | ||||||
| 			t.Errorf("Expected error: %v", tt.err) | 			} | ||||||
| 		} | 		}) | ||||||
| 		if name != tt.clusterName { |  | ||||||
| 			t.Errorf("Expected cluserName: %q, got: %q", tt.clusterName, name) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestCloneClusterDescription(t *testing.T) { | func TestCloneClusterDescription(t *testing.T) { | ||||||
| 	for _, tt := range cloneClusterDescriptions { | 	for _, tt := range cloneClusterDescriptions { | ||||||
| 		if err := validateCloneClusterDescription(tt.in); err != nil { | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err := validateCloneClusterDescription(tt.in); err != nil { | ||||||
| 				t.Errorf("testCloneClusterDescription expected error: %v, got: %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("testCloneClusterDescription expected error: %v, got: %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 			} else if tt.err != nil { | ||||||
|  | 				t.Errorf("Expected error: %v", tt.err) | ||||||
| 			} | 			} | ||||||
| 		} else if tt.err != nil { | 		}) | ||||||
| 			t.Errorf("Expected error: %v", tt.err) |  | ||||||
| 		} |  | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestUnmarshalMaintenanceWindow(t *testing.T) { | func TestUnmarshalMaintenanceWindow(t *testing.T) { | ||||||
| 	for _, tt := range maintenanceWindows { | 	for _, tt := range maintenanceWindows { | ||||||
| 		var m MaintenanceWindow | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		err := m.UnmarshalJSON(tt.in) | 			var m MaintenanceWindow | ||||||
| 		if err != nil { | 			err := m.UnmarshalJSON(tt.in) | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err != nil { | ||||||
| 				t.Errorf("MaintenanceWindow unmarshal expected error: %v, got %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("MaintenanceWindow unmarshal expected error: %v, got %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
|  | 			} else if tt.err != nil { | ||||||
|  | 				t.Errorf("Expected error: %v", tt.err) | ||||||
| 			} | 			} | ||||||
| 			continue |  | ||||||
| 		} else if tt.err != nil { |  | ||||||
| 			t.Errorf("Expected error: %v", tt.err) |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		if !reflect.DeepEqual(m, tt.out) { | 			if !reflect.DeepEqual(m, tt.out) { | ||||||
| 			t.Errorf("Expected maintenance window: %#v, got: %#v", tt.out, m) | 				t.Errorf("Expected maintenance window: %#v, got: %#v", tt.out, m) | ||||||
| 		} | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestMarshalMaintenanceWindow(t *testing.T) { | func TestMarshalMaintenanceWindow(t *testing.T) { | ||||||
| 	for _, tt := range maintenanceWindows { | 	for _, tt := range maintenanceWindows { | ||||||
| 		if tt.err != nil { | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 			continue | 			if tt.err != nil { | ||||||
| 		} | 				return | ||||||
|  | 			} | ||||||
| 
 | 
 | ||||||
| 		s, err := tt.out.MarshalJSON() | 			s, err := tt.out.MarshalJSON() | ||||||
| 		if err != nil { | 			if err != nil { | ||||||
| 			t.Errorf("Marshal Error: %v", err) | 				t.Errorf("Marshal Error: %v", err) | ||||||
| 		} | 			} | ||||||
| 
 | 
 | ||||||
| 		if !bytes.Equal(s, tt.in) { | 			if !bytes.Equal(s, tt.in) { | ||||||
| 			t.Errorf("Expected Marshal: %q, got: %q", string(tt.in), string(s)) | 				t.Errorf("Expected Marshal: %q, got: %q", string(tt.in), string(s)) | ||||||
| 		} | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestUnmarshalPostgresStatus(t *testing.T) { | func TestUnmarshalPostgresStatus(t *testing.T) { | ||||||
| 	for _, tt := range postgresStatus { | 	for _, tt := range postgresStatus { | ||||||
| 		var ps PostgresStatus | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		err := ps.UnmarshalJSON(tt.in) |  | ||||||
| 		if err != nil { |  | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { |  | ||||||
| 				t.Errorf("CR status unmarshal expected error: %v, got %v", tt.err, err) |  | ||||||
| 			} |  | ||||||
| 			continue |  | ||||||
| 			//} else if tt.err != nil {
 |  | ||||||
| 			//t.Errorf("Expected error: %v", tt.err)
 |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		if !reflect.DeepEqual(ps, tt.out) { | 			var ps PostgresStatus | ||||||
| 			t.Errorf("Expected status: %#v, got: %#v", tt.out, ps) | 			err := ps.UnmarshalJSON(tt.in) | ||||||
| 		} | 			if err != nil { | ||||||
|  | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("CR status unmarshal expected error: %v, got %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			if !reflect.DeepEqual(ps, tt.out) { | ||||||
|  | 				t.Errorf("Expected status: %#v, got: %#v", tt.out, ps) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestPostgresUnmarshal(t *testing.T) { | func TestPostgresUnmarshal(t *testing.T) { | ||||||
| 	for _, tt := range unmarshalCluster { | 	for _, tt := range unmarshalCluster { | ||||||
| 		var cluster Postgresql | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		err := cluster.UnmarshalJSON(tt.in) | 			var cluster Postgresql | ||||||
| 		if err != nil { | 			err := cluster.UnmarshalJSON(tt.in) | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { | 			if err != nil { | ||||||
| 				t.Errorf("Unmarshal expected error: %v, got: %v", tt.err, err) | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
|  | 					t.Errorf("Unmarshal expected error: %v, got: %v", tt.err, err) | ||||||
|  | 				} | ||||||
|  | 				return | ||||||
|  | 			} else if tt.err != nil { | ||||||
|  | 				t.Errorf("Expected error: %v", tt.err) | ||||||
| 			} | 			} | ||||||
| 			continue |  | ||||||
| 		} else if tt.err != nil { |  | ||||||
| 			t.Errorf("Expected error: %v", tt.err) |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		if !reflect.DeepEqual(cluster, tt.out) { | 			if !reflect.DeepEqual(cluster, tt.out) { | ||||||
| 			t.Errorf("Expected Postgresql: %#v, got %#v", tt.out, cluster) | 				t.Errorf("Expected Postgresql: %#v, got %#v", tt.out, cluster) | ||||||
| 		} | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestMarshal(t *testing.T) { | func TestMarshal(t *testing.T) { | ||||||
| 	for _, tt := range unmarshalCluster { | 	for _, tt := range unmarshalCluster { | ||||||
| 		if tt.err != nil { | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		// Unmarshal and marshal example to capture api changes
 | 			if tt.err != nil { | ||||||
| 		var cluster Postgresql | 				return | ||||||
| 		err := cluster.UnmarshalJSON(tt.marshal) |  | ||||||
| 		if err != nil { |  | ||||||
| 			if tt.err == nil || err.Error() != tt.err.Error() { |  | ||||||
| 				t.Errorf("Backwards compatibility unmarshal expected error: %v, got: %v", tt.err, err) |  | ||||||
| 			} | 			} | ||||||
| 			continue |  | ||||||
| 		} |  | ||||||
| 		expected, err := json.Marshal(cluster) |  | ||||||
| 		if err != nil { |  | ||||||
| 			t.Errorf("Backwards compatibility marshal error: %v", err) |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		m, err := json.Marshal(tt.out) | 			// Unmarshal and marshal example to capture api changes
 | ||||||
| 		if err != nil { | 			var cluster Postgresql | ||||||
| 			t.Errorf("Marshal error: %v", err) | 			err := cluster.UnmarshalJSON(tt.marshal) | ||||||
| 		} | 			if err != nil { | ||||||
| 		if !bytes.Equal(m, expected) { | 				if tt.err == nil || err.Error() != tt.err.Error() { | ||||||
| 			t.Errorf("Marshal Postgresql \nexpected: %q, \ngot:      %q", string(expected), string(m)) | 					t.Errorf("Backwards compatibility unmarshal expected error: %v, got: %v", tt.err, err) | ||||||
| 		} | 				} | ||||||
|  | 				return | ||||||
|  | 			} | ||||||
|  | 			expected, err := json.Marshal(cluster) | ||||||
|  | 			if err != nil { | ||||||
|  | 				t.Errorf("Backwards compatibility marshal error: %v", err) | ||||||
|  | 			} | ||||||
|  | 
 | ||||||
|  | 			m, err := json.Marshal(tt.out) | ||||||
|  | 			if err != nil { | ||||||
|  | 				t.Errorf("Marshal error: %v", err) | ||||||
|  | 			} | ||||||
|  | 			if !bytes.Equal(m, expected) { | ||||||
|  | 				t.Errorf("Marshal Postgresql \nexpected: %q, \ngot:      %q", string(expected), string(m)) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestPostgresMeta(t *testing.T) { | func TestPostgresMeta(t *testing.T) { | ||||||
| 	for _, tt := range unmarshalCluster { | 	for _, tt := range unmarshalCluster { | ||||||
| 		if a := tt.out.GetObjectKind(); a != &tt.out.TypeMeta { | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 			t.Errorf("GetObjectKindMeta \nexpected: %v, \ngot:       %v", tt.out.TypeMeta, a) |  | ||||||
| 		} |  | ||||||
| 
 | 
 | ||||||
| 		if a := tt.out.GetObjectMeta(); reflect.DeepEqual(a, tt.out.ObjectMeta) { | 			if a := tt.out.GetObjectKind(); a != &tt.out.TypeMeta { | ||||||
| 			t.Errorf("GetObjectMeta \nexpected: %v, \ngot:       %v", tt.out.ObjectMeta, a) | 				t.Errorf("GetObjectKindMeta \nexpected: %v, \ngot:       %v", tt.out.TypeMeta, a) | ||||||
| 		} | 			} | ||||||
|  | 
 | ||||||
|  | 			if a := tt.out.GetObjectMeta(); reflect.DeepEqual(a, tt.out.ObjectMeta) { | ||||||
|  | 				t.Errorf("GetObjectMeta \nexpected: %v, \ngot:       %v", tt.out.ObjectMeta, a) | ||||||
|  | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestPostgresListMeta(t *testing.T) { | func TestPostgresListMeta(t *testing.T) { | ||||||
| 	for _, tt := range postgresqlList { | 	for _, tt := range postgresqlList { | ||||||
| 		if tt.err != nil { | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 			continue | 			if tt.err != nil { | ||||||
| 		} | 				return | ||||||
|  | 			} | ||||||
| 
 | 
 | ||||||
| 		if a := tt.out.GetObjectKind(); a != &tt.out.TypeMeta { | 			if a := tt.out.GetObjectKind(); a != &tt.out.TypeMeta { | ||||||
| 			t.Errorf("GetObjectKindMeta expected: %v, got: %v", tt.out.TypeMeta, a) | 				t.Errorf("GetObjectKindMeta expected: %v, got: %v", tt.out.TypeMeta, a) | ||||||
| 		} | 			} | ||||||
| 
 | 
 | ||||||
| 		if a := tt.out.GetListMeta(); reflect.DeepEqual(a, tt.out.ListMeta) { | 			if a := tt.out.GetListMeta(); reflect.DeepEqual(a, tt.out.ListMeta) { | ||||||
| 			t.Errorf("GetObjectMeta expected: %v, got: %v", tt.out.ListMeta, a) | 				t.Errorf("GetObjectMeta expected: %v, got: %v", tt.out.ListMeta, a) | ||||||
| 		} | 			} | ||||||
| 
 | 
 | ||||||
| 		return | 			return | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
| 
 | 
 | ||||||
| func TestPostgresqlClone(t *testing.T) { | func TestPostgresqlClone(t *testing.T) { | ||||||
| 	for _, tt := range unmarshalCluster { | 	for _, tt := range unmarshalCluster { | ||||||
| 		cp := &tt.out | 		t.Run(tt.about, func(t *testing.T) { | ||||||
| 		cp.Error = "" | 			cp := &tt.out | ||||||
| 		clone := cp.Clone() | 			cp.Error = "" | ||||||
| 		if !reflect.DeepEqual(clone, cp) { | 			clone := cp.Clone() | ||||||
| 			t.Errorf("TestPostgresqlClone expected: \n%#v\n, got \n%#v", cp, clone) | 			if !reflect.DeepEqual(clone, cp) { | ||||||
| 		} | 				t.Errorf("TestPostgresqlClone expected: \n%#v\n, got \n%#v", cp, clone) | ||||||
| 
 | 			} | ||||||
|  | 		}) | ||||||
| 	} | 	} | ||||||
| } | } | ||||||
|  |  | ||||||
		Loading…
	
		Reference in New Issue