From 095d8ec2562fc876c4ffffb968aeb57a1bd6efdd Mon Sep 17 00:00:00 2001 From: Panos Xynos Date: Mon, 2 Oct 2023 17:02:35 +0100 Subject: [PATCH] [185765111] Broker can accept TLS connections * Includes integration test covering selfsigned TLS --- CONFIGURATION.md | 174 +++--- Makefile | 6 +- ci/blackbox/rdsbroker_test.go | 2 +- ci/helpers/brokerapi_helper.go | 31 +- ci/integration.yml | 1 + ci/tls/config.json | 566 ++++++++++++++++++ ci/tls/tls_suite_test.go | 17 + ci/tls/tls_test.go | 139 +++++ config-sample.json | 105 +++- config/config.go | 23 +- config/tls_config.go | 45 ++ main.go | 17 +- release/jobs/rds-broker/spec | 14 + .../templates/config/rds-config.json.erb | 9 + scripts/run-broker-tls.sh | 40 ++ .../ginkgo/v2/extensions/table/deprecated.go | 3 - vendor/modules.txt | 1 - 17 files changed, 1063 insertions(+), 130 deletions(-) create mode 100644 ci/tls/config.json create mode 100644 ci/tls/tls_suite_test.go create mode 100644 ci/tls/tls_test.go create mode 100644 config/tls_config.go create mode 100755 scripts/run-broker-tls.sh delete mode 100644 vendor/github.com/onsi/ginkgo/v2/extensions/table/deprecated.go diff --git a/CONFIGURATION.md b/CONFIGURATION.md index 19eb0994..08870c97 100644 --- a/CONFIGURATION.md +++ b/CONFIGURATION.md @@ -4,30 +4,42 @@ A sample configuration can be found at [config-sample.json](https://github.com/a ## General Configuration -| Option | Required | Type | Description -|:-----------------------|:--------:|:--------|:----------- -| port | N | Integer | The TCP port to listen on. Defaults to 3000 if unspecified. -| log_level | Y | String | Broker Log Level (DEBUG, INFO, ERROR, FATAL) -| username | Y | String | Broker Auth Username -| password | Y | String | Broker Auth Password -| run_housekeeping | N | Boolean | Whether to run housekeeping tasks (including master password rotation, and snapshot cleanups). This should be set to true on exactly one instance in your deployment. -| cron_schedule | Y | String | Schedule for cron jobs. A crontab-like expression with seconds precision (e.g. '0 0 * * * *' or '@hourly'), with fields: 'second minute hour dom month dow' -| keep_snapshots_for_days| Y | Integer | Number of days to keep old RDS snapshots for -| rds_config | Y | Hash | [RDS Broker configuration](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-broker-configuration) +| Option | Required | Type | Description | +| :---------------------- | :------: | :------ | :-------------------------------------------------------------------------------------------------------------------------------------------------------------------- | +| port | N | Integer | The TCP port to listen on. Defaults to 3000 if unspecified. | +| host | N | String | The hostname or ip address that the broker accepts connections. Defaults to '0.0.0.0' if unspecified. | +| log_level | Y | String | Broker Log Level (DEBUG, INFO, ERROR, FATAL) | +| username | Y | String | Broker Auth Username | +| password | Y | String | Broker Auth Password | +| run_housekeeping | N | Boolean | Whether to run housekeeping tasks (including master password rotation, and snapshot cleanups). This should be set to true on exactly one instance in your deployment. | +| cron_schedule | Y | String | Schedule for cron jobs. A crontab-like expression with seconds precision (e.g. '0 0 * * * *' or '@hourly'), with fields: 'second minute hour dom month dow' | +| keep_snapshots_for_days | Y | Integer | Number of days to keep old RDS snapshots for | +| rds_config | Y | Hash | [RDS Broker configuration](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-broker-configuration) | +| tls | N | Hash | [RDS Broker configuration](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-broker-tls-configuration) | ## RDS Broker Configuration -| Option | Required | Type | Description -|:-------------------------------|:--------:|:------- |:----------- -| region | Y | String | RDS Region -| db_prefix | Y | String | Prefix to add to RDS DB Identifiers -| allow_user_provision_parameters| N | Boolean | Allow users to send arbitrary parameters on provision calls (defaults to `false`) -| allow_user_update_parameters | N | Boolean | Allow users to send arbitrary parameters on update calls (defaults to `false`) -| allow_user_bind_parameters | N | Boolean | Allow users to send arbitrary parameters on bind calls (defaults to `false`) -| catalog | Y | Hash | [RDS Broker catalog](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-broker-catalog) -| master_password_seed | Y | String | Seed to generate DB instances master passwords -| aws_tag_cache_seconds | N | Integer | Cache expiry time of AWS Tags cache (in seconds) -| broker_name | Y | String | RDS broker name used to tag instances for identification +| Option | Required | Type | Description | +| :------------------------------ | :------: | :------ | :---------------------------------------------------------------------------------------------------------------- | +| region | Y | String | RDS Region | +| db_prefix | Y | String | Prefix to add to RDS DB Identifiers | +| allow_user_provision_parameters | N | Boolean | Allow users to send arbitrary parameters on provision calls (defaults to `false`) | +| allow_user_update_parameters | N | Boolean | Allow users to send arbitrary parameters on update calls (defaults to `false`) | +| allow_user_bind_parameters | N | Boolean | Allow users to send arbitrary parameters on bind calls (defaults to `false`) | +| catalog | Y | Hash | [RDS Broker catalog](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-broker-catalog) | +| master_password_seed | Y | String | Seed to generate DB instances master passwords | +| aws_tag_cache_seconds | N | Integer | Cache expiry time of AWS Tags cache (in seconds) | +| broker_name | Y | String | RDS broker name used to tag instances for identification | + +## RDS Broker TLS Configuration + +> If the configuration is provided all fields are required. +> To try how this works `scripts/run-broker-tls.sh` will run RDS broker locally with self signed certificate. + +| Option | Required | Type | Description | +| :---------- | :------: | :----- | :----------------------------------------- | +| certificate | Y | String | The certificate to use for TLS connection. | +| private_key | Y | String | The private key to use for TLS connection. | ### Note When the seed is changed and the broker restarted, the instances master passwords will be updated. @@ -38,74 +50,74 @@ Please refer to the [Catalog Documentation](https://docs.cloudfoundry.org/servic ### Catalog -| Option | Required | Type | Description -|:---------|:--------:|:--------- |:----------- -| services | N | []Service | A list of [Services](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#service) +| Option | Required | Type | Description | +| :------- | :------: | :-------- | :----------------------------------------------------------------------------------------------------- | +| services | N | []Service | A list of [Services](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#service) | ### Service -| Option | Required | Type | Description -|:------------------------------|:--------:|:------------- |:----------- -| id | Y | String | An identifier used to correlate this service in future requests to the catalog -| name | Y | String | The CLI-friendly name of the service that will appear in the catalog. All lowercase, no spaces -| description | Y | String | A short description of the service that will appear in the catalog -| tags | N | []String | A list of service tags -| metadata.displayName | N | String | The name of the service to be displayed in graphical clients -| metadata.imageUrl | N | String | The URL to an image -| metadata.longDescription | N | String | Long description -| metadata.providerDisplayName | N | String | The name of the upstream entity providing the actual service -| metadata.documentationUrl | N | String | Link to documentation page for service -| metadata.supportUrl | N | String | Link to support for the service -| requires | N | []String | A list of permissions that the user would have to give the service, if they provision it (only `syslog_drain` is supported) -| plan_updateable | N | Boolean | Whether the service supports upgrade/downgrade for some plans -| plans | N | []ServicePlan | A list of [Plans](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#service-plan) for this service -| dashboard_client.id | N | String | The id of the Oauth2 client that the service intends to use -| dashboard_client.secret | N | String | A secret for the dashboard client -| dashboard_client.redirect_uri | N | String | A domain for the service dashboard that will be whitelisted by the UAA to enable SSO +| Option | Required | Type | Description | +| :---------------------------- | :------: | :------------ | :-------------------------------------------------------------------------------------------------------------------------- | +| id | Y | String | An identifier used to correlate this service in future requests to the catalog | +| name | Y | String | The CLI-friendly name of the service that will appear in the catalog. All lowercase, no spaces | +| description | Y | String | A short description of the service that will appear in the catalog | +| tags | N | []String | A list of service tags | +| metadata.displayName | N | String | The name of the service to be displayed in graphical clients | +| metadata.imageUrl | N | String | The URL to an image | +| metadata.longDescription | N | String | Long description | +| metadata.providerDisplayName | N | String | The name of the upstream entity providing the actual service | +| metadata.documentationUrl | N | String | Link to documentation page for service | +| metadata.supportUrl | N | String | Link to support for the service | +| requires | N | []String | A list of permissions that the user would have to give the service, if they provision it (only `syslog_drain` is supported) | +| plan_updateable | N | Boolean | Whether the service supports upgrade/downgrade for some plans | +| plans | N | []ServicePlan | A list of [Plans](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#service-plan) for this service | +| dashboard_client.id | N | String | The id of the Oauth2 client that the service intends to use | +| dashboard_client.secret | N | String | A secret for the dashboard client | +| dashboard_client.redirect_uri | N | String | A domain for the service dashboard that will be whitelisted by the UAA to enable SSO | ### Service Plan -| Option | Required | Type | Description -|:---------------------|:--------:|:------------- |:----------- -| id | Y | String | An identifier used to correlate this plan in future requests to the catalog -| name | Y | String | The CLI-friendly name of the plan that will appear in the catalog. All lowercase, no spaces -| description | Y | String | A short description of the plan that will appear in the catalog -| metadata.bullets | N | []String | Features of this plan, to be displayed in a bulleted-list -| metadata.costs | N | Cost Object | An array-of-objects that describes the costs of a service, in what currency, and the unit of measure -| metadata.displayName | N | String | Name of the plan to be display in graphical clients -| free | N | Boolean | This field allows the plan to be limited by the non_basic_services_allowed field in a Cloud Foundry Quota -| rds_properties | Y | RDSProperties | [RDS Properties](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-properties) +| Option | Required | Type | Description | +| :------------------- | :------: | :------------ | :-------------------------------------------------------------------------------------------------------- | +| id | Y | String | An identifier used to correlate this plan in future requests to the catalog | +| name | Y | String | The CLI-friendly name of the plan that will appear in the catalog. All lowercase, no spaces | +| description | Y | String | A short description of the plan that will appear in the catalog | +| metadata.bullets | N | []String | Features of this plan, to be displayed in a bulleted-list | +| metadata.costs | N | Cost Object | An array-of-objects that describes the costs of a service, in what currency, and the unit of measure | +| metadata.displayName | N | String | Name of the plan to be display in graphical clients | +| free | N | Boolean | This field allows the plan to be limited by the non_basic_services_allowed field in a Cloud Foundry Quota | +| rds_properties | Y | RDSProperties | [RDS Properties](https://github.com/alphagov/paas-rds-broker/blob/master/CONFIGURATION.md#rds-properties) | ## RDS Properties Please refer to the [Amazon Relational Database Service Documentation](https://aws.amazon.com/documentation/rds/) for more details about these properties. -| Option | Required | Type | Description -|:--------------------------------|:--------:|:--------- |:----------- -| allocated_storage | Y | Integer | The amount of storage (in gigabytes) to be initially allocated for the database instances (between `5` and `6144`) -| auto_minor_version_upgrade | N | Boolean | Enable or disable automatic upgrades to new minor versions as they are released (defaults to `false`) -| availability_zone | N | String | The Availability Zone that database instances will be created in -| backup_retention_period | N | Integer | The number of days that Amazon RDS should retain automatic backups of DB instances (between `0` and `35`) -| character_set_name | N | String | For supported engines, indicates that DB instances should be associated with the specified CharacterSet -| copy_tags_to_snapshot | N | Boolean | Enable or disable copying all tags from DB instances to snapshots -| db_instance_class | Y | String | The name of the DB Instance Class -| db_security_groups | N | []String | The security group(s) names that have rules authorizing connections from applications that need to access the data stored in the DB instance -| db_subnet_group_name | N | String | The DB subnet group name that defines which subnets and IP ranges the DB instance can use in the VPC -| engine | Y | String | The name of the Database Engine (only `mariadb`, `mysql` and `postgres` are supported) -| engine_version | Y | String | The version number of the Database Engine -| engine_family | Y | String | The family name for the engine and version, as reported by RDS -| iops | N | Integer | The amount of Provisioned IOPS to be initially allocated for DB instances when using `io1` storage type -| kms_key_id | N | String | The KMS key identifier for encrypted DB instances -| license_model | N | String | License model information for DB instances (`license-included`, `bring-your-own-license`, `general-public-license`) -| multi_az | N | Boolean | Enable or disable Multi-AZ deployment for high availability DB Instances -| option_group_name | N | String | The DB option group name that enables any optional functionality you want the DB instances to support -| port | N | Integer | The TCP/IP port DB instances will use for application connections -| preferred_backup_window | N | String | The daily time range during which automated backups are created if automated backups are enabled -| preferred_maintenance_window | N | String | The weekly time range during which system maintenance can occur -| publicly_accessible | N | Boolean | Specify if DB instances will be publicly accessible -| skip_final_snapshot | N | Boolean | Determines whether a final DB snapshot is created before the DB instances are deleted -| storage_encrypted | N | Boolean | Specifies whether DB instances are encrypted -| storage_type | N | String | The storage type to be associated with DB instances (`standard`, `gp2`, `io1`) -| vpc_security_group_ids | N | []String | VPC security group(s) IDs that have rules authorizing connections from applications that need to access the data stored in DB instances -| allowed_extensions | Y | []String | The set of Postgres extensions which can be enabled -| default_extensions | Y | []String | The set of Postgres extensions which are enabled by default. Each of these must also be in the `allowed_extensions` list. +| Option | Required | Type | Description | +| :--------------------------- | :------: | :------- | :------------------------------------------------------------------------------------------------------------------------------------------- | +| allocated_storage | Y | Integer | The amount of storage (in gigabytes) to be initially allocated for the database instances (between `5` and `6144`) | +| auto_minor_version_upgrade | N | Boolean | Enable or disable automatic upgrades to new minor versions as they are released (defaults to `false`) | +| availability_zone | N | String | The Availability Zone that database instances will be created in | +| backup_retention_period | N | Integer | The number of days that Amazon RDS should retain automatic backups of DB instances (between `0` and `35`) | +| character_set_name | N | String | For supported engines, indicates that DB instances should be associated with the specified CharacterSet | +| copy_tags_to_snapshot | N | Boolean | Enable or disable copying all tags from DB instances to snapshots | +| db_instance_class | Y | String | The name of the DB Instance Class | +| db_security_groups | N | []String | The security group(s) names that have rules authorizing connections from applications that need to access the data stored in the DB instance | +| db_subnet_group_name | N | String | The DB subnet group name that defines which subnets and IP ranges the DB instance can use in the VPC | +| engine | Y | String | The name of the Database Engine (only `mariadb`, `mysql` and `postgres` are supported) | +| engine_version | Y | String | The version number of the Database Engine | +| engine_family | Y | String | The family name for the engine and version, as reported by RDS | +| iops | N | Integer | The amount of Provisioned IOPS to be initially allocated for DB instances when using `io1` storage type | +| kms_key_id | N | String | The KMS key identifier for encrypted DB instances | +| license_model | N | String | License model information for DB instances (`license-included`, `bring-your-own-license`, `general-public-license`) | +| multi_az | N | Boolean | Enable or disable Multi-AZ deployment for high availability DB Instances | +| option_group_name | N | String | The DB option group name that enables any optional functionality you want the DB instances to support | +| port | N | Integer | The TCP/IP port DB instances will use for application connections | +| preferred_backup_window | N | String | The daily time range during which automated backups are created if automated backups are enabled | +| preferred_maintenance_window | N | String | The weekly time range during which system maintenance can occur | +| publicly_accessible | N | Boolean | Specify if DB instances will be publicly accessible | +| skip_final_snapshot | N | Boolean | Determines whether a final DB snapshot is created before the DB instances are deleted | +| storage_encrypted | N | Boolean | Specifies whether DB instances are encrypted | +| storage_type | N | String | The storage type to be associated with DB instances (`standard`, `gp2`, `io1`) | +| vpc_security_group_ids | N | []String | VPC security group(s) IDs that have rules authorizing connections from applications that need to access the data stored in DB instances | +| allowed_extensions | Y | []String | The set of Postgres extensions which can be enabled | +| default_extensions | Y | []String | The set of Postgres extensions which are enabled by default. Each of these must also be in the `allowed_extensions` list. | diff --git a/Makefile b/Makefile index aa2cbf7d..ba0d11fd 100644 --- a/Makefile +++ b/Makefile @@ -5,6 +5,10 @@ MYSQL_PASSWORD=toor integration: go run github.com/onsi/ginkgo/v2/ginkgo --timeout=5h --nodes=4 -r ci/blackbox --slowSpecThreshold=1800 -stream -failFast +.PHONY: tls_integration +tls_integration: + go run github.com/onsi/ginkgo/v2/ginkgo --timeout=5h -r ci/tls --slowSpecThreshold=1800 -stream -failFast + .PHONY: unit unit: test_unit test_all_sql @@ -14,7 +18,7 @@ test_unit: .PHONY: test_all_sql test_all_sql: test_postgres test_mysql .PHONY: test_postgres -test_postgres: run_postgres_sql_tests start_postgres_10 run_postgres_sql_tests stop_postgres_10 start_postgres_11 run_postgres_sql_tests stop_postgres_11 start_postgres_12 run_postgres_sql_tests stop_postgres_12 start_postgres_13 run_postgres_sql_tests stop_postgres_13 +test_postgres: start_postgres_10 run_postgres_sql_tests stop_postgres_10 start_postgres_11 run_postgres_sql_tests stop_postgres_11 start_postgres_12 run_postgres_sql_tests stop_postgres_12 start_postgres_13 run_postgres_sql_tests stop_postgres_13 .PHONY: test_mysql test_mysql: start_mysql_80 run_mysql_sql_tests stop_mysql_80 start_mysql_57 run_mysql_sql_tests stop_mysql_57 diff --git a/ci/blackbox/rdsbroker_test.go b/ci/blackbox/rdsbroker_test.go index db2898b6..b05dd03a 100644 --- a/ci/blackbox/rdsbroker_test.go +++ b/ci/blackbox/rdsbroker_test.go @@ -1163,7 +1163,7 @@ func startNewBroker(rdsBrokerConfig *config.Config, brokerName string) (*gexec.S // Wait for it to be listening Eventually(rdsBrokerSession, 10*time.Second).Should(And( gbytes.Say("rds-broker.start"), - gbytes.Say(fmt.Sprintf(`{"port":%d}`, rdsBrokerPort)), + gbytes.Say(fmt.Sprintf(`{"address":"0.0.0.0:%d","host":"0.0.0.0","port":%d,"tls":false}`, rdsBrokerPort, rdsBrokerPort)), )) rdsBrokerUrl := fmt.Sprintf("http://localhost:%d", rdsBrokerPort) diff --git a/ci/helpers/brokerapi_helper.go b/ci/helpers/brokerapi_helper.go index 4b8c3fda..383ce2c9 100644 --- a/ci/helpers/brokerapi_helper.go +++ b/ci/helpers/brokerapi_helper.go @@ -2,6 +2,7 @@ package helpers import ( "bytes" + "crypto/tls" "encoding/json" "fmt" "io" @@ -47,23 +48,37 @@ func BodyBytes(resp *http.Response) ([]byte, error) { } type BrokerAPIClient struct { - Url string - Username string - Password string - AcceptsIncomplete bool + Url string + Username string + Password string + AcceptsIncomplete bool + AcceptsSelfSignedCerts bool } func NewBrokerAPIClient(Url string, Username string, Password string) *BrokerAPIClient { return &BrokerAPIClient{ - Url: Url, - Username: Username, - Password: Password, + Url: Url, + Username: Username, + Password: Password, + AcceptsSelfSignedCerts: true, } } func (b *BrokerAPIClient) doRequest(action string, path string, body io.Reader, params ...uriParam) (*http.Response, error) { - client := &http.Client{} + var client *http.Client + if b.AcceptsSelfSignedCerts { + client = &http.Client{ + Transport: &http.Transport{ + TLSClientConfig: &tls.Config{ + InsecureSkipVerify: true, + }, + }, + } + } else { + client = &http.Client{} + + } req, err := http.NewRequest(action, b.Url+path, body) if err != nil { diff --git a/ci/integration.yml b/ci/integration.yml index 43f11b87..82df6d04 100644 --- a/ci/integration.yml +++ b/ci/integration.yml @@ -15,4 +15,5 @@ run: - -c - | cd src/github.com/alphagov/paas-rds-broker + make tls_integration make integration diff --git a/ci/tls/config.json b/ci/tls/config.json new file mode 100644 index 00000000..3537a7c2 --- /dev/null +++ b/ci/tls/config.json @@ -0,0 +1,566 @@ +{ + "log_level": "DEBUG", + "username": "username", + "password": "password", + "cron_schedule": "*/5 * * * *", + "keep_snapshots_for_days": 7, + "tls": { + "certificate": "__from_maketarget__", + "private_key": "__from_maketarget__" + }, + "rds_config": { + "region": "us-east-1", + "db_prefix": "cf", + "allow_user_provision_parameters": true, + "allow_user_update_parameters": true, + "allow_user_bind_parameters": true, + "master_password_seed": "jK5Q45Gx2q4I", + "broker_name": "GDS-prod-1", + "catalog": { + "services": [ + { + "id": "ce71b484-d542-40f7-9dd4-5526e38c81ba", + "name": "rdsmysql", + "description": "RDS MySQL service", + "bindable": true, + "tags": [ + "mysql", + "relational" + ], + "metadata": { + "displayName": "RDS MySQL", + "imageUrl": "", + "longDescription": "A RDS MySQL service", + "providerDisplayName": "Amazon Web Services", + "documentationUrl": "https://aws.amazon.com/documentation/rds/", + "supportUrl": "https://forums.aws.amazon.com/forum.jspa?forumID=60" + }, + "plan_updateable": true, + "plans": [ + { + "id": "326b78b0-a8ab-4cc0-8657-79c9c0ac8126", + "name": "5.5-medium", + "description": "RDS MySQL 5.5 (db.m3.medium, 10Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.180 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated MySQL 5.5 server", + "MySQL 5.5", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.medium", + "engine": "MySQL", + "engine_version": "5.5.42", + "engine_family": "mysql5.5", + "allocated_storage": 10, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [], + "default_extensions": [] + } + }, + { + "id": "729d81e7-29a0-4709-bdf2-3317a1468291", + "name": "5.5-large", + "description": "RDS MySQL 5.5 (db.m3.large, 50Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.370 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated MySQL 5.5 server", + "MySQL 5.5", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.large", + "engine": "MySQL", + "engine_version": "5.5.42", + "engine_family": "mysql5.5", + "allocated_storage": 50, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [], + "default_extensions": [] + } + }, + { + "id": "499e9ff5-28a7-43eb-bf93-7bc69b61712b", + "name": "5.5-xlarge", + "description": "RDS MySQL 5.5 (db.m3.xlarge, 100Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.740 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated MySQL 5.5 server", + "MySQL 5.5", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.xlarge", + "engine": "MySQL", + "engine_version": "5.5.42", + "engine_family": "mysql5.5", + "allocated_storage": 100, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [], + "default_extensions": [] + } + }, + { + "id": "5b8282cf-a669-4ffc-b426-c169a7bbfc71", + "name": "5.6-medium", + "description": "RDS MySQL 5.6 (db.m3.medium, 10Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.180 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated MySQL 5.6 server", + "MySQL 5.6", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.medium", + "engine": "MySQL", + "engine_version": "5.6.23", + "engine_family": "mysql5.6", + "allocated_storage": 10, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [], + "default_extensions": [] + } + }, + { + "id": "b4943c34-fd33-47a6-8f0b-eb4f462fd746", + "name": "5.6-large", + "description": "RDS MySQL 5.6 (db.m3.large, 50Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.370 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated MySQL 5.6 server", + "MySQL 5.6", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.large", + "engine": "MySQL", + "engine_version": "5.6.23", + "engine_family": "mysql5.6", + "allocated_storage": 50, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [], + "default_extensions": [] + } + }, + { + "id": "7e47cd05-625e-415d-bafd-09fbb0eb9ed8", + "name": "5.6-xlarge", + "description": "RDS MySQL 5.6 (db.m3.xlarge, 100Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.740 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated MySQL 5.6 server", + "MySQL 5.6", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.xlarge", + "engine": "MySQL", + "engine_version": "5.6.23", + "engine_family": "mysql5.6", + "allocated_storage": 100, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [], + "default_extensions": [] + } + } + ] + }, + { + "id": "a2c9adda-6511-462c-9934-b3fd8236e9f0", + "name": "rdspostgres", + "description": "RDS PostgreSQL service", + "bindable": true, + "tags": [ + "postgres", + "relational" + ], + "metadata": { + "displayName": "RDS PostgreSQL", + "imageUrl": "", + "longDescription": "RDS PostgreSQL service", + "providerDisplayName": "Amazon Web Services", + "documentationUrl": "https://aws.amazon.com/documentation/rds/", + "supportUrl": "https://forums.aws.amazon.com/forum.jspa?forumID=60" + }, + "plan_updateable": true, + "plans": [ + { + "id": "d42fc3cc-1341-4aa3-866e-01bc5243dc3e", + "name": "9.6-medium", + "description": "RDS PostgreSQL 9.6 (db.m3.medium, 10Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.180 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated PostgreSQL 9.6 server", + "PostgreSQL 9.6", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.medium", + "engine": "postgres", + "engine_version": "9.6", + "engine_family": "postgres9.6", + "allocated_storage": 10, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] + } + }, + { + "id": "80768f31-5c2c-40e8-8135-59fe3d710dc3", + "name": "9.6-large", + "description": "RDS PostgreSQL 9.6 (db.m3.large, 50Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.370 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated PostgreSQL 9.6 server", + "PostgreSQL 9.6", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.large", + "engine": "postgres", + "engine_version": "9.6", + "engine_family": "postgres9.6", + "allocated_storage": 50, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] + } + }, + { + "id": "347dfd90-4f0c-42ed-a92c-1d8096d373d1", + "name": "9.6-xlarge", + "description": "RDS PostgreSQL 9.6 (db.m3.xlarge, 100Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.740 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated PostgreSQL 9.6 server", + "PostgreSQL 9.6", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.xlarge", + "engine": "postgres", + "engine_version": "9.6", + "engine_family": "postgres9.6", + "allocated_storage": 100, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] + } + }, + { + "id": "d03b544e-3be5-4aca-bb3b-11544247f313", + "name": "9.5-medium", + "description": "RDS PostgreSQL 9.5 (db.m3.medium, 10Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.180 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated PostgreSQL 9.5 server", + "PostgreSQL 9.5", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.medium", + "engine": "postgres", + "engine_version": "9.5", + "engine_family": "postgres9.5", + "allocated_storage": 10, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] + } + }, + { + "id": "65cfcd68-625d-4635-a828-4ccef50af986", + "name": "9.5-large", + "description": "RDS PostgreSQL 9.5 (db.m3.large, 50Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.370 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated PostgreSQL 9.5 server", + "PostgreSQL 9.5", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.large", + "engine": "postgres", + "engine_version": "9.5", + "engine_family": "postgres9.5", + "allocated_storage": 50, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] + } + }, + { + "id": "8f8d8ed7-ddd3-4b1f-befa-976cf7fc4116", + "name": "9.5-xlarge", + "description": "RDS PostgreSQL 9.5 (db.m3.xlarge, 100Gb)", + "metadata": { + "costs": [ + { + "amount": { + "usd": 0.740 + }, + "unit": "HOUR" + } + ], + "bullets": [ + "Dedicated PostgreSQL 9.5 server", + "PostgreSQL 9.5", + "AWS RDS" + ] + }, + "free": false, + "rds_properties": { + "db_instance_class": "db.m3.xlarge", + "engine": "postgres", + "engine_version": "9.5", + "engine_family": "postgres9.5", + "allocated_storage": 100, + "auto_minor_version_upgrade": true, + "multi_az": true, + "publicly_accessible": true, + "copy_tags_to_snapshot": true, + "skip_final_snapshot": false, + "db_security_groups": [ + "default" + ], + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] + } + } + ] + } + ], + "exclude_engines": [ + { + "engine": "postgres", + "engine_version": "^9\\.3\\." + }, + { + "engine": "postgres", + "engine_version": "^9\\.4\\.[0-8]$" + }, + { + "engine": "postgres", + "engine_version": "^9\\.5\\.[0-4]$" + } + ] + } + } +} diff --git a/ci/tls/tls_suite_test.go b/ci/tls/tls_suite_test.go new file mode 100644 index 00000000..64a0545c --- /dev/null +++ b/ci/tls/tls_suite_test.go @@ -0,0 +1,17 @@ +package tls_test + +import ( + "testing" + + . "github.com/onsi/ginkgo/v2" + . "github.com/onsi/gomega" +) + +type SuiteData struct { + RdsBrokerPath string +} + +func TestTls(t *testing.T) { + RegisterFailHandler(Fail) + RunSpecs(t, "Tls Suite") +} diff --git a/ci/tls/tls_test.go b/ci/tls/tls_test.go new file mode 100644 index 00000000..684133b7 --- /dev/null +++ b/ci/tls/tls_test.go @@ -0,0 +1,139 @@ +package tls_test + +import ( + "crypto/rand" + "crypto/rsa" + "crypto/x509" + "crypto/x509/pkix" + "encoding/json" + "encoding/pem" + "fmt" + "math/big" + "os" + "os/exec" + "path/filepath" + "time" + + . "github.com/onsi/ginkgo/v2" + . "github.com/onsi/gomega" + uuid "github.com/satori/go.uuid" + + . "github.com/alphagov/paas-rds-broker/ci/helpers" + "github.com/alphagov/paas-rds-broker/config" + "github.com/onsi/gomega/gbytes" + "github.com/onsi/gomega/gexec" + "github.com/phayes/freeport" +) + +var ( + rdsBrokerSession *gexec.Session + brokerAPIClient *BrokerAPIClient + brokerName string + rdsBrokerPath string +) + +var _ = Describe("TLS Support", func() { + BeforeEach(func() { + var err error + + // Give a different Broker Name in each execution, to avoid conflicts + brokerName = fmt.Sprintf( + "%s-%s", + "rdsbroker-integration-test", + uuid.NewV4().String(), + ) + rdsBrokerPath, err = gexec.Build("github.com/alphagov/paas-rds-broker") + Expect(err).ShouldNot(HaveOccurred()) + + jsonConfigFilename := "config.json" + jsonFilePath, err := filepath.Abs(jsonConfigFilename) + Expect(err).ShouldNot(HaveOccurred()) + + jsonData, err := os.ReadFile(jsonFilePath) + Expect(err).ShouldNot(HaveOccurred()) + + var testConfig config.Config + err = json.Unmarshal(jsonData, &testConfig) + Expect(err).ShouldNot(HaveOccurred()) + + rdsBrokerSession, brokerAPIClient, _ = startNewBroker(&testConfig, brokerName) + }) + + AfterEach(func() { + rdsBrokerSession.Kill() + }) + + Describe("Services", func() { + It("returns the proper CatalogResponse", func() { + _, err := brokerAPIClient.GetCatalog() + Expect(err).ToNot(HaveOccurred()) + }) + }) +}) + +func startNewBroker(rdsBrokerConfig *config.Config, brokerName string) (*gexec.Session, *BrokerAPIClient, *RDSClient) { + configFile, err := os.CreateTemp("", "rds-broker") + Expect(err).ToNot(HaveOccurred()) + defer os.Remove(configFile.Name()) + + newRDSBrokerConfig := *rdsBrokerConfig + // start the broker in a random port + rdsBrokerPort := freeport.GetPort() + newRDSBrokerConfig.Port = rdsBrokerPort + + newRDSBrokerConfig.RDSConfig.BrokerName = "tls_broker" + + priv, err := rsa.GenerateKey(rand.Reader, 2048) + Expect(err).NotTo(HaveOccurred()) + + template := x509.Certificate{ + SerialNumber: big.NewInt(1), + Subject: pkix.Name{ + Organization: []string{"Acme Co"}, + }, + NotBefore: time.Now(), + NotAfter: time.Now().Add(time.Hour), + IsCA: true, + KeyUsage: x509.KeyUsageCertSign | x509.KeyUsageDigitalSignature, + ExtKeyUsage: []x509.ExtKeyUsage{x509.ExtKeyUsageServerAuth}, + } + + derBytes, err := x509.CreateCertificate(rand.Reader, &template, &template, &priv.PublicKey, priv) + Expect(err).NotTo(HaveOccurred()) + + certPEM := pem.EncodeToMemory(&pem.Block{Type: "CERTIFICATE", Bytes: derBytes}) + privBytes := x509.MarshalPKCS1PrivateKey(priv) + keyPEM := pem.EncodeToMemory(&pem.Block{Type: "RSA PRIVATE KEY", Bytes: privBytes}) + + newRDSBrokerConfig.TLS = &config.TLSConfig{ + Certificate: string(certPEM), + PrivateKey: string(keyPEM), + } + + configJSON, err := json.Marshal(&newRDSBrokerConfig) + Expect(err).ToNot(HaveOccurred()) + Expect(os.WriteFile(configFile.Name(), configJSON, 0644)).To(Succeed()) + Expect(configFile.Close()).To(Succeed()) + + command := exec.Command(rdsBrokerPath, + fmt.Sprintf("-config=%s", configFile.Name()), + ) + fmt.Printf("\n\n\n----------%s\n%s-----------\n\n", rdsBrokerPath, configFile.Name()) + rdsBrokerSession, err := gexec.Start(command, GinkgoWriter, GinkgoWriter) + Expect(err).ShouldNot(HaveOccurred()) + + // Wait for it to be listening + Eventually(rdsBrokerSession, 10*time.Second).Should(And( + gbytes.Say("rds-broker.start"), + gbytes.Say(fmt.Sprintf(`{"address":"0.0.0.0:%d","host":"0.0.0.0","port":%d,"tls":true}`, rdsBrokerPort, rdsBrokerPort)), + )) + + rdsBrokerUrl := fmt.Sprintf("https://localhost:%d", rdsBrokerPort) + + brokerAPIClient := NewBrokerAPIClient(rdsBrokerUrl, rdsBrokerConfig.Username, rdsBrokerConfig.Password) + rdsClient, err := NewRDSClient(rdsBrokerConfig.RDSConfig.Region, rdsBrokerConfig.RDSConfig.DBPrefix) + + Expect(err).ToNot(HaveOccurred()) + + return rdsBrokerSession, brokerAPIClient, rdsClient +} diff --git a/config-sample.json b/config-sample.json index fd22d513..3537a7c2 100644 --- a/config-sample.json +++ b/config-sample.json @@ -4,6 +4,10 @@ "password": "password", "cron_schedule": "*/5 * * * *", "keep_snapshots_for_days": 7, + "tls": { + "certificate": "__from_maketarget__", + "private_key": "__from_maketarget__" + }, "rds_config": { "region": "us-east-1", "db_prefix": "cf", @@ -62,7 +66,7 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" @@ -100,7 +104,7 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" @@ -138,7 +142,7 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" @@ -176,7 +180,7 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" @@ -214,7 +218,7 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" @@ -252,7 +256,7 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" @@ -311,16 +315,21 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" ], - "allowed_extensions": ["postgis", "pg_stat_statements"], - "default_extensions": ["postgis"] + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] } }, - { + { "id": "80768f31-5c2c-40e8-8135-59fe3d710dc3", "name": "9.6-large", "description": "RDS PostgreSQL 9.6 (db.m3.large, 50Gb)", @@ -349,16 +358,21 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" ], - "allowed_extensions": ["postgis", "pg_stat_statements"], - "default_extensions": ["postgis"] + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] } }, - { + { "id": "347dfd90-4f0c-42ed-a92c-1d8096d373d1", "name": "9.6-xlarge", "description": "RDS PostgreSQL 9.6 (db.m3.xlarge, 100Gb)", @@ -387,13 +401,18 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" ], - "allowed_extensions": ["postgis", "pg_stat_statements"], - "default_extensions": ["postgis"] + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] } }, { @@ -425,16 +444,21 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" ], - "allowed_extensions": ["postgis", "pg_stat_statements"], - "default_extensions": ["postgis"] + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] } }, - { + { "id": "65cfcd68-625d-4635-a828-4ccef50af986", "name": "9.5-large", "description": "RDS PostgreSQL 9.5 (db.m3.large, 50Gb)", @@ -463,15 +487,20 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" ], - "allowed_extensions": ["postgis", "pg_stat_statements"], - "default_extensions": ["postgis"] + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] } - }, + }, { "id": "8f8d8ed7-ddd3-4b1f-befa-976cf7fc4116", "name": "9.5-xlarge", @@ -501,22 +530,36 @@ "auto_minor_version_upgrade": true, "multi_az": true, "publicly_accessible": true, - "copy_tags_to_snapshot":true, + "copy_tags_to_snapshot": true, "skip_final_snapshot": false, "db_security_groups": [ "default" ], - "allowed_extensions": ["postgis", "pg_stat_statements"], - "default_extensions": ["postgis"] + "allowed_extensions": [ + "postgis", + "pg_stat_statements" + ], + "default_extensions": [ + "postgis" + ] } } ] } ], "exclude_engines": [ - {"engine": "postgres", "engine_version": "^9\\.3\\."}, - {"engine": "postgres", "engine_version": "^9\\.4\\.[0-8]$"}, - {"engine": "postgres", "engine_version": "^9\\.5\\.[0-4]$"} + { + "engine": "postgres", + "engine_version": "^9\\.3\\." + }, + { + "engine": "postgres", + "engine_version": "^9\\.4\\.[0-8]$" + }, + { + "engine": "postgres", + "engine_version": "^9\\.5\\.[0-4]$" + } ] } } diff --git a/config/config.go b/config/config.go index 1ac2eb68..c8b30668 100644 --- a/config/config.go +++ b/config/config.go @@ -9,15 +9,22 @@ import ( "github.com/alphagov/paas-rds-broker/rdsbroker" ) +const ( + DefaultPort = 3000 + DefaultHost = "0.0.0.0" +) + type Config struct { Port int `json:"port"` LogLevel string `json:"log_level"` Username string `json:"username"` Password string `json:"password"` + Host string `json:"host"` RunHousekeeping bool `json:"run_housekeeping"` KeepSnapshotsForDays int `json:"keep_snapshots_for_days"` CronSchedule string `json:"cron_schedule"` RDSConfig *rdsbroker.Config `json:"rds_config"` + TLS *TLSConfig `json:"tls"` } func LoadConfig(configFile string) (config *Config, err error) { @@ -46,11 +53,18 @@ func LoadConfig(configFile string) (config *Config, err error) { func (c *Config) FillDefaults() { if c.Port == 0 { - c.Port = 3000 + c.Port = DefaultPort + } + if c.Host == "" { + c.Host = DefaultHost } c.RDSConfig.FillDefaults() } +func (c Config) TLSEnabled() bool { + return c.TLS != nil +} + func (c Config) Validate() error { if c.LogLevel == "" { return errors.New("Must provide a non-empty LogLevel") @@ -76,5 +90,12 @@ func (c Config) Validate() error { return fmt.Errorf("Validating RDS configuration: %s", err) } + if c.TLS != nil { + tlsValidation := c.TLS.validate() + if tlsValidation != nil { + return tlsValidation + } + } + return nil } diff --git a/config/tls_config.go b/config/tls_config.go new file mode 100644 index 00000000..b3380535 --- /dev/null +++ b/config/tls_config.go @@ -0,0 +1,45 @@ +package config + +import ( + "crypto/tls" + "fmt" +) + +type TLSConfig struct { + Certificate string `json:"certificate"` + PrivateKey string `json:"private_key"` +} + +// GenerateTLSConfig produces a tls.Config structure out of TLSConfig. +// Aiming to be used while configuring a TLS client or server. +func (t *TLSConfig) GenerateTLSConfig() (*tls.Config, error) { + certificate, err := tls.X509KeyPair([]byte(t.Certificate), []byte(t.PrivateKey)) + if err != nil { + return nil, err + } + + return &tls.Config{ + Certificates: []tls.Certificate{certificate}, + + MinVersion: tls.VersionTLS12, + CurvePreferences: []tls.CurveID{tls.CurveP521, tls.CurveP384, tls.CurveP256}, + + PreferServerCipherSuites: true, + CipherSuites: []uint16{ + tls.TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384, + tls.TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA, + tls.TLS_RSA_WITH_AES_256_GCM_SHA384, + tls.TLS_RSA_WITH_AES_256_CBC_SHA, + }, + }, nil +} + +func (t *TLSConfig) validate() error { + if t.Certificate == "" { + return fmt.Errorf("Config error: TLS certificate required") + } + if t.PrivateKey == "" { + return fmt.Errorf("Config error: TLS private key required") + } + return nil +} diff --git a/main.go b/main.go index 94a3c8a9..02b5c445 100644 --- a/main.go +++ b/main.go @@ -1,6 +1,7 @@ package main import ( + "crypto/tls" "flag" "fmt" "log" @@ -97,15 +98,25 @@ func startHTTPServer( ) error { server := buildHTTPHandler(serviceBroker, logger, cfg) + listenAddress := fmt.Sprintf("%s:%d", cfg.Host, cfg.Port) // We don't use http.ListenAndServe here so that the "start" log message is // logged after the socket is listening. This log message is used by the // tests to wait until the broker is ready. - listener, err := net.Listen("tcp", fmt.Sprintf(":%d", cfg.Port)) + listener, err := net.Listen("tcp", listenAddress) if err != nil { - return fmt.Errorf("failed to listen on port %d: %s", cfg.Port, err) + return fmt.Errorf("failed to listen on address %s: %s", listenAddress, err) + } + if cfg.TLSEnabled() { + tlsConfig, err := cfg.TLS.GenerateTLSConfig() + if err != nil { + log.Fatalf("Error configuring TLS: %s", err) + } + listener = tls.NewListener(listener, tlsConfig) + logger.Info("start", lager.Data{"port": cfg.Port, "tls": true, "host": cfg.Host, "address": listenAddress}) + } else { + logger.Info("start", lager.Data{"port": cfg.Port, "tls": false, "host": cfg.Host, "address": listenAddress}) } - logger.Info("start", lager.Data{"port": cfg.Port}) return http.Serve(listener, server) } diff --git a/release/jobs/rds-broker/spec b/release/jobs/rds-broker/spec index b7337e24..56ba8dae 100644 --- a/release/jobs/rds-broker/spec +++ b/release/jobs/rds-broker/spec @@ -17,6 +17,9 @@ properties: rds-broker.port: description: "Broker Listen Port" default: 80 + rds-broker.host: + description: "Broker Listen hostname or IP" + example: "0.0.0.0" rds-broker.username: description: "Broker Auth Username" default: "rds-broker" @@ -60,3 +63,14 @@ properties: rds-broker.keep_snapshots_for_days: description: "Number of days to keep old RDS snapshots for" default: 35 + rds-broker.tls: + description: "Certificate and private key for TLS listener" + example: | + certificate: | + -----BEGIN CERTIFICATE----- + ... + -----END CERTIFICATE----- + private_key: | + -----BEGIN EXAMPLE RSA PRIVATE KEY----- + ... + -----END EXAMPLE RSA PRIVATE KEY----- diff --git a/release/jobs/rds-broker/templates/config/rds-config.json.erb b/release/jobs/rds-broker/templates/config/rds-config.json.erb index 019ffda4..2632640c 100644 --- a/release/jobs/rds-broker/templates/config/rds-config.json.erb +++ b/release/jobs/rds-broker/templates/config/rds-config.json.erb @@ -1,5 +1,8 @@ { "port": <%= p('rds-broker.port') %>, +<% if_p('rds-broker.host') do |host| -%> + "host": "<%= host %>", +<% end -%> "log_level": "<%= p('rds-broker.log_level') %>", "username": "<%= p('rds-broker.username') %>", "password": "<%= p('rds-broker.password') %>", @@ -7,6 +10,12 @@ "cron_schedule": "<%= p('rds-broker.cron_schedule') %>", "keep_snapshots_for_days": <%= p('rds-broker.keep_snapshots_for_days') %>, "state_encryption_key": "<%= p('rds-broker.state_encryption_key') %>", +<% if_p('rds-broker.tls') do |tls| -%> + "tls": { + "certificate": "<%= tls.fetch("certificate").gsub(/\n/, '\\n') %>", + "private_key": "<%= tls.fetch("private_key").gsub(/\n/, '\\n') %>" + }, +<% end -%> "rds_config": { "region": "<%= p('rds-broker.aws_region') %>", "db_prefix": "<%= p('rds-broker.db_prefix') %>", diff --git a/scripts/run-broker-tls.sh b/scripts/run-broker-tls.sh new file mode 100755 index 00000000..190d63c9 --- /dev/null +++ b/scripts/run-broker-tls.sh @@ -0,0 +1,40 @@ +#!/usr/bin/env bash + +set -euo pipefail + +SCRIPT_DIR=$(cd "$(dirname "$0")" && pwd) + +_tmp_dir="$(mktemp -d)" +trap 'rm -rf "${_tmp_dir}"' EXIT + +pushd "${_tmp_dir}" > /dev/null + echo "Generating TLS Certificates" + # Generate the CA Key and Certificate for signing Client Certs + openssl req -x509 -newkey rsa:1024 -nodes \ + -keyout CAkey.pem -out CAcert.pem \ + -sha256 -days 3650 -nodes \ + -subj "/C=UK/ST=London/L=London/O=PAASDEV/CN=broker-ca.local" &> /dev/null + + # Generate the Server Key, and CSR and sign with the CA Certificate + openssl req -new -newkey rsa:1024 -nodes \ + -keyout key.pem -out req.pem \ + -subj "/C=UK/ST=London/L=London/O=PAASDEV/CN=broker.local" &> /dev/null + openssl x509 -req \ + -in req.pem -CA CAcert.pem -CAkey CAkey.pem \ + -extfile <(printf "subjectAltName=IP:127.0.0.1") \ + -CAcreateserial -out cert.pem -days 30 -sha256 #&> /dev/null + + # rm CAkey.pem req.pem + + cert_value="$(sed -e ':a' -e 'N' -e '$!ba' -e 's/\n/\\n/g' "cert.pem")" + key_value="$(sed -e ':a' -e 'N' -e '$!ba' -e 's/\n/\\n/g' "key.pem")" + ca_cert_value="$(sed -e ':a' -e 'N' -e '$!ba' -e 's/\n/\\n/g' "CAcert.pem")" +popd > /dev/null + +echo "TLS Certificates Generated in ${_tmp_dir}" +echo "Example curl command: curl --cacert ${_tmp_dir}/CAcert.pem https://127.0.0.1:3000" + +jq \ + '.tls.certificate = "'"${cert_value}"'" | .tls.private_key = "'"${key_value}"'" | .tls.ca = "'"${ca_cert_value}"'"' \ + "${SCRIPT_DIR}/../config-sample.json" > "${_tmp_dir}/config.json" +go run main.go --config "${_tmp_dir}/config.json" diff --git a/vendor/github.com/onsi/ginkgo/v2/extensions/table/deprecated.go b/vendor/github.com/onsi/ginkgo/v2/extensions/table/deprecated.go deleted file mode 100644 index 6f4be6ab..00000000 --- a/vendor/github.com/onsi/ginkgo/v2/extensions/table/deprecated.go +++ /dev/null @@ -1,3 +0,0 @@ -package table - -type TableSupportHasBeenMovedToTheCoreGinkgoDSL struct{} diff --git a/vendor/modules.txt b/vendor/modules.txt index 81e2c4e2..ec6d0ea6 100644 --- a/vendor/modules.txt +++ b/vendor/modules.txt @@ -92,7 +92,6 @@ github.com/lib/pq/scram ## explicit; go 1.18 github.com/onsi/ginkgo/v2 github.com/onsi/ginkgo/v2/config -github.com/onsi/ginkgo/v2/extensions/table github.com/onsi/ginkgo/v2/formatter github.com/onsi/ginkgo/v2/ginkgo github.com/onsi/ginkgo/v2/ginkgo/build