From 6bddde196435e19b1143063ae4fce14011e7a33e Mon Sep 17 00:00:00 2001 From: Mike Baynton Date: Thu, 26 Sep 2019 15:13:14 -0500 Subject: [PATCH] [CORE-1576] Commands to set and unset primary domain. (#2011) * Commands to set and unset primary domain. WIP: needs tests. * Tests for setting/resetting primary domain and its commands. * Rename command to remove primary domain to `domain:primary:reset` for consistency. * Refactor set_primary_domain workflow maker out of a Command to Models\Environment class. * CS fix * Filter pantheonsite.io domains from interactive choice list. While the platform supports setting these domains as primary, it seems highly unlikely that a user would do this intentionally, and more likely that if it were chosen from a multiple choice list, that it was just fat-fingered. For better UX, they are therefore excluded. Users that really mean to do this can do so by typing/passing the full pantheonsite.io domain non-interactively to the domain:primary:set command. * Add primary domain to Domain model and domain:list command. * UX review: Change command to domain:primary:unset, log completion. * Changes that are clear from @TeslaDethray's requests. * Rename domain:primary:set to domain:primary:add per consensus of product and technical leadership. * Best guess at new model refactor request * Fix failing tests in older minor versions of php 7.0 - 7.3. It seems substr_compare issues a warning iff the first parameter is the empty string in these earlier minor releases; manual states it was fixed in 7.2.18. Since a) it's been fixed and b) empty strings are not expected real-world input, I've simply removed that input from the test coverage. * Behat tests for add and remove primary domain * Make behat.yml a bit more yaml-compliant (% not allowed as beginning of string) * Restore Consolidation\\Filter\\Hooks\\FilterHooks to command list * Changelog for 2.2.0 * Fix stated version in changelog --- CHANGELOG.md | 6 + src/Commands/Domain/ListCommand.php | 1 + src/Commands/Domain/Primary/AddCommand.php | 98 ++++++++ src/Commands/Domain/Primary/RemoveCommand.php | 43 ++++ src/Models/Domain.php | 1 + src/Models/Environment.php | 8 + src/Models/PrimaryDomain.php | 54 +++++ src/Terminus.php | 3 + tests/config/behat.yml | 4 +- tests/features/domain-primary.feature | 28 +++ tests/fixtures/domain-primary-add.yml | 223 ++++++++++++++++++ .../Domain/Primary/AddCommandTest.php | 54 +++++ .../Primary/PrimaryDomainCommandsTestBase.php | 81 +++++++ .../Domain/Primary/RemoveCommandTest.php | 28 +++ tests/unit_tests/Models/DomainTest.php | 27 ++- 15 files changed, 656 insertions(+), 3 deletions(-) create mode 100644 src/Commands/Domain/Primary/AddCommand.php create mode 100644 src/Commands/Domain/Primary/RemoveCommand.php create mode 100644 src/Models/PrimaryDomain.php create mode 100644 tests/features/domain-primary.feature create mode 100644 tests/fixtures/domain-primary-add.yml create mode 100644 tests/unit_tests/Commands/Domain/Primary/AddCommandTest.php create mode 100644 tests/unit_tests/Commands/Domain/Primary/PrimaryDomainCommandsTestBase.php create mode 100644 tests/unit_tests/Commands/Domain/Primary/RemoveCommandTest.php diff --git a/CHANGELOG.md b/CHANGELOG.md index 88eff76be..b845b9a83 100644 --- a/CHANGELOG.md +++ b/CHANGELOG.md @@ -1,6 +1,12 @@ # Change Log All notable changes to this project will be documented in this file. This project adheres to [Semantic Versioning](http://semver.org) +## MASTER +### Added +- Added `primary` field to the output of `domain:list`. (#2011) +- Added `domain:primary:add` command to set a domain as primary, causing traffic to redirect to it. (#2011) +- Added `domain:primary:remove` command to remove a domain's primary designation. (#2011) + ## 2.1.0 - 2019-09-03 ### Added - Added `--filter` option to `backup:list` command (#1992) diff --git a/src/Commands/Domain/ListCommand.php b/src/Commands/Domain/ListCommand.php index 250f2c616..cdd7a8ebf 100644 --- a/src/Commands/Domain/ListCommand.php +++ b/src/Commands/Domain/ListCommand.php @@ -29,6 +29,7 @@ class ListCommand extends TerminusCommand implements SiteAwareInterface * @field-labels * id: Domain/ID * type: Type + * primary: Is Primary * deletable: Is Deletable * status: status * @return RowsOfFields diff --git a/src/Commands/Domain/Primary/AddCommand.php b/src/Commands/Domain/Primary/AddCommand.php new file mode 100644 index 000000000..e39da83b9 --- /dev/null +++ b/src/Commands/Domain/Primary/AddCommand.php @@ -0,0 +1,98 @@ + + */ + public function add($site_env, $domain) + { + /** + * @var $site Site + * @var $env Environment + */ + list($site, $env) = $this->getSiteEnv($site_env); + + // The primary domain is set via a workflow so as to use workflow logging to track changes & update policy docs. + $workflow = $env->getPrimaryDomainModel()->setPrimaryDomain($domain); + $this->processWorkflow($workflow); + $this->log()->notice( + 'Set {domain} as primary for {site}.{env}', + ['domain' => $domain, 'site' => $site->get('name'), 'env' => $env->id,] + ); + } + + /** + * Prompt the user for the domain, if it was not specified. + * + * n.b. This hook is not called in --no-interaction mode. + * + * @param InputInterface $input + * @param OutputInterface $output + * @param AnnotationData $annotationData + * + * @hook interact domain:primary:add + */ + public function interact(InputInterface $input, OutputInterface $output, AnnotationData $annotationData) + { + $domain = $input->getArgument('domain'); + if (empty($domain)) { + /** + * @var $site Site + * @var $env Environment + */ + list($site, $env) = $this->getSiteEnv($input->getArgument('site_env')); + $domains = $this->filterPlatformDomains($env->getDomains()->ids()); + sort($domains); + + if (!empty($domains)) { + $domain = $this->io()->choice('Select the primary domain for this site', $domains); + $input->setArgument('domain', $domain); + } + } + } + + /** + * Filters strings ending in the platform domain from an array. + * + * @param $domains + * @return array + */ + public function filterPlatformDomains($domains) + { + return array_filter($domains, function ($domain) { + return substr_compare($domain, self::PLATFORM_DOMAIN, -strlen(self::PLATFORM_DOMAIN)) !== 0; + }); + } +} diff --git a/src/Commands/Domain/Primary/RemoveCommand.php b/src/Commands/Domain/Primary/RemoveCommand.php new file mode 100644 index 000000000..9e93e8597 --- /dev/null +++ b/src/Commands/Domain/Primary/RemoveCommand.php @@ -0,0 +1,43 @@ +getSiteEnv($site_env); + + $workflow = $env->getPrimaryDomainModel()->removePrimaryDomain(); + $this->processWorkflow($workflow); + $this->log()->notice( + 'Primary domain has been removed from {site}.{env}', + ['site' => $site->get('name'), 'env' => $env->id,] + ); + } +} diff --git a/src/Models/Domain.php b/src/Models/Domain.php index c63f04729..a280a6137 100755 --- a/src/Models/Domain.php +++ b/src/Models/Domain.php @@ -65,6 +65,7 @@ public function serialize() 'status' => in_array($this->get('status'), ['ok', 'okay',]) ? 'OK' : $this->get('status'), 'status_message' => $this->get('status_message'), 'deletable' => (boolean)$this->get('deletable'), + 'primary' => (boolean)$this->get('primary'), ]; } } diff --git a/src/Models/Environment.php b/src/Models/Environment.php index ebbebd6af..321ac2928 100755 --- a/src/Models/Environment.php +++ b/src/Models/Environment.php @@ -469,6 +469,14 @@ public function getDomains() return $this->domains; } + /** + * @return PrimaryDomain + */ + public function getPrimaryDomainModel() + { + return $this->getContainer()->get(PrimaryDomain::class, [$this]); + } + /** * Gets the Drush version of this environment * diff --git a/src/Models/PrimaryDomain.php b/src/Models/PrimaryDomain.php new file mode 100644 index 000000000..a398c2e74 --- /dev/null +++ b/src/Models/PrimaryDomain.php @@ -0,0 +1,54 @@ +setEnvironment($environment); + } + + /** + * Builds a Workflow to set the primary domain for this environment. + * + * @param string $domain A domain name attached to this environment. + * + * @return Workflow + */ + public function setPrimaryDomain($domain) + { + return $this->workflowFactory($domain); + } + + /** + * Builds a workflow to remove the primary domain from this environment. + * + * @return Workflow + */ + public function removePrimaryDomain() + { + return $this->workflowFactory(null); + } + + /** + * @param string $domain + * @return Workflow + */ + protected function workflowFactory($domain) + { + return $this->getEnvironment()->getWorkflows()->create( + 'set_primary_domain', + [ + 'environment' => $this->getEnvironment()->id, + 'params' => ['primary_domain' => $domain] + ] + ); + } +} diff --git a/src/Terminus.php b/src/Terminus.php index c5952b1b6..555af9134 100644 --- a/src/Terminus.php +++ b/src/Terminus.php @@ -147,6 +147,8 @@ private function addBuiltInCommandsAndHooks() 'Pantheon\\Terminus\\Commands\\Domain\\DNSCommand', 'Pantheon\\Terminus\\Commands\\Domain\\ListCommand', 'Pantheon\\Terminus\\Commands\\Domain\\LookupCommand', + 'Pantheon\\Terminus\\Commands\\Domain\\Primary\\AddCommand', + 'Pantheon\\Terminus\\Commands\\Domain\\Primary\\RemoveCommand', 'Pantheon\\Terminus\\Commands\\Domain\\RemoveCommand', 'Pantheon\\Terminus\\Commands\\Env\\ClearCacheCommand', 'Pantheon\\Terminus\\Commands\\Env\\CloneContentCommand', @@ -367,6 +369,7 @@ private function configureModulesAndCollections($container) $container->add(\Pantheon\Terminus\Models\OrganizationUserMembership::class); $container->add(\Pantheon\Terminus\Models\PaymentMethod::class); $container->add(\Pantheon\Terminus\Models\Plan::class); + $container->add(\Pantheon\Terminus\Models\PrimaryDomain::class); $container->add(\Pantheon\Terminus\Models\Profile::class); $container->add(\Pantheon\Terminus\Models\Redis::class); $container->add(\Pantheon\Terminus\Models\SSHKey::class); diff --git a/tests/config/behat.yml b/tests/config/behat.yml index f4905889d..e35873359 100644 --- a/tests/config/behat.yml +++ b/tests/config/behat.yml @@ -1,8 +1,8 @@ default: - autoload: [ %paths.base%/../features/bootstrap ] + autoload: [ '%paths.base%/../features/bootstrap' ] suites: default: - paths: [ %paths.base%/../features ] + paths: [ '%paths.base%/../features' ] contexts: - Pantheon\Terminus\FeatureTests\FeatureContext: parameters: diff --git a/tests/features/domain-primary.feature b/tests/features/domain-primary.feature new file mode 100644 index 000000000..a0bf3b925 --- /dev/null +++ b/tests/features/domain-primary.feature @@ -0,0 +1,28 @@ +Feature: Adding a primary domain to an environment + In order to redirect all visitors to my preferred domain + As a user + I need to be able to manage primary domains attached to my site's environments + + Background: I am authenticated and have a site named [[test_site_name]] + Given I am authenticated + And a site named "[[test_site_name]]" + + @vcr domain-primary-add.yml + Scenario: Adding a primary domain to an environment + When I run "terminus domain:primary:add [[test_site_name]].live testdomain.com" + Then I should get "." + And I should get "." + Then I should get: + """ + Set testdomain.com as primary for [[test_site_name]].live + """ + + @vcr domain-primary-add.yml + Scenario: Removing a primary domain from an environment + When I run "terminus domain:primary:remove [[test_site_name]].live" + Then I should get "." + And I should get "." + Then I should get: + """ + Primary domain has been removed from [[test_site_name]].live + """ diff --git a/tests/fixtures/domain-primary-add.yml b/tests/fixtures/domain-primary-add.yml new file mode 100644 index 000000000..9dd1f0701 --- /dev/null +++ b/tests/fixtures/domain-primary-add.yml @@ -0,0 +1,223 @@ + +- + request: + method: POST + url: 'https://onebox/api/authorize/machine-token' + headers: + Host: onebox + Expect: null + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00001-0123-4567-89ab-cdef01234567:Jdg4ilJ9O3O0QIR1eBKUb' + Accept: null + body: '{"machine_token":"111111111111111111111111111111111111111111111","client":"terminus"}' + response: + status: + http_version: '1.1' + code: '200' + message: OK + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:01 GMT' + Content-Type: 'application/json; charset=utf-8' + Content-Length: '182' + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00002-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + Strict-Transport-Security: max-age=31536000 + body: '{"session":"11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt","expires_at":1571784241,"user_id":"11111111-1111-1111-1111-111111111111"}' +- + request: + method: GET + url: 'https://onebox/api/site-names/behat-tests' + headers: + Host: onebox + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt' + Accept: null + response: + status: + http_version: '1.1' + code: '200' + message: OK + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:02 GMT' + Content-Type: 'application/json; charset=utf-8' + Transfer-Encoding: chunked + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00004-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + Strict-Transport-Security: max-age=31536000 + body: '{"id": "aaa00005-0123-4567-89ab-cdef01234567", "name": "behat-tests"}' +- + request: + method: GET + url: 'https://onebox/api/sites/aaa00005-0123-4567-89ab-cdef01234567?site_state=true' + headers: + Host: onebox + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt' + Accept: null + response: + status: + http_version: '1.1' + code: '200' + message: OK + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:02 GMT' + Content-Type: application/json + Content-Length: '81208' + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00006-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + Strict-Transport-Security: max-age=31536000 + body: '{"created": 1565815807, "created_by_user_id": "11111111-1111-1111-1111-111111111111", "current_num_domains": 4, "domain_lookup_rax": 1568401902, "framework": "wordpress", "holder_id": "aaa00007-0123-4567-89ab-cdef01234567", "holder_type": "organization", "instrument": "aaa00008-0123-4567-89ab-cdef01234567", "last_code_push": {"timestamp": "2019-09-17T15:14:30", "user_uuid": null}, "name": "behat-tests", "organization": "aaa00007-0123-4567-89ab-cdef01234567", "owner": "11111111-1111-1111-1111-111111111111", "php_version": "55", "plan_name": "Basic", "preferred_availability_zone": "us-central1-a", "preferred_zone": "us-central1", "purchased_at": 1565816602, "service_level": "basic_small", "service_level_updated_at": 1565816602, "upstream": {"repository_branch": "master", "machine_name": "wordpress", "product_id": "aaa00009-0123-4567-89ab-cdef01234567", "url": "https://github.com/pantheon-systems/WordPress", "label": "WordPress", "organization_id": "", "framework": "wordpress", "branch": "master", "repository_url": "https://github.com/pantheon-systems/WordPress", "type": "core", "id": "aaa00009-0123-4567-89ab-cdef01234567"}, "label": "Mikes onboarding site", "id": "aaa00005-0123-4567-89ab-cdef01234567", "preferred_zone_label": "United States", "holder": {"base_domain": null, "billing_url": "", "change_management": true, "change_service_url": "https://www.pantheon.io", "custom_upstreams": true, "email_domain": null, "experimental_agency_organization": "no", "instrument": "aaa00008-0123-4567-89ab-cdef01234567", "is_discoverable": false, "maxdevsites": "20", "multidev": true, "name": "Pantheon Employees", "org_logo": "", "org_logo_height": 85, "org_logo_width": 128, "requires_onboarding": false, "secure_runtime_access_enableable": true, "service_level": "flagship_silver", "show_org_name_header": "yes", "terms_of_service": "", "use_org_instrument": true, "id": "aaa00007-0123-4567-89ab-cdef01234567", "key": "aaa00007-0123-4567-89ab-cdef01234567", "machine_name": "pantheon-employees", "has_multidev": false, "has_change_management": false, "profile": {"machine_name": "pantheon-employees", "change_service_url": "https://www.pantheon.io", "name": "Pantheon Employees", "email_domain": null, "org_logo_width": 128, "org_logo_height": 85, "base_domain": null, "billing_url": "", "terms_of_service": "", "org_logo": ""}, "settings": {"is_discoverable": false, "show_org_name_header": "yes", "service_level": "flagship_silver", "base_domain": null, "email_domain": null}}, "settings": {"allow_domains": true, "preferred_availability_zone": "us-central1-a", "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "stunnel": false, "min_backups": 0, "owner": "11111111-1111-1111-1111-111111111111", "secure_runtime_access": false, "pingdom": 0, "allow_indexserver": false, "created_by_user_id": "11111111-1111-1111-1111-111111111111", "failover_appserver": 0, "migration_started_at": null, "cacheserver": 0, "instrument": "aaa00008-0123-4567-89ab-cdef01234567", "on_server_development": false, "drush_version": 5, "migration_method": null, "current_num_domains": 4, "appserver": 1, "allow_read_slaves": false, "preferred_zone": "us-central1", "php_version": "55", "php_channel": "stable", "allow_cacheserver": false, "upstream": {"url": "https://github.com/pantheon-systems/WordPress", "product_id": "aaa00009-0123-4567-89ab-cdef01234567", "branch": "master"}, "ssl_enabled": null, "plan_name": "Basic", "fileserver": 1, "service_level": "basic_small", "dedicated_ip": null, "dbserver": 1, "migration_origin_url": null, "domain_lookup_rax": 1568401902, "purchased_at": 1565816602, "framework": "wordpress", "max_total_domains": 5, "key": "aaa00005-0123-4567-89ab-cdef01234567", "max_num_cdes": 10, "migration_completed_at": null, "guilty_of_abuse": null, "indexserver": 0, "pingdom_chance": 0, "holder_id": "aaa00007-0123-4567-89ab-cdef01234567", "name": "behat-tests", "created": 1565815807, "max_backups": 7, "last_code_push": {"timestamp": "2019-09-17T15:14:30", "user_uuid": null}, "holder_type": "organization", "replica_verification_strategy": "pt-heartbeat", "id": "aaa00005-0123-4567-89ab-cdef01234567", "organization": "aaa00007-0123-4567-89ab-cdef01234567", "pingdom_manually_enabled": false, "service_level_updated_at": 1565816602}, "base_domain": null, "attributes": {"label": "Mikes onboarding site", "m3_ui": true}, "add_ons": []}' +- + request: + method: GET + url: 'https://onebox/api/sites/aaa00005-0123-4567-89ab-cdef01234567/environments' + headers: + Host: onebox + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt' + Accept: null + response: + status: + http_version: '1.1' + code: '200' + message: OK + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:06 GMT' + Content-Type: 'application/json; charset=utf-8' + Transfer-Encoding: chunked + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00010-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + Strict-Transport-Security: max-age=31536000 + body: '{"test": {"dns_zone": "pantheonsite.io", "environment_created": 1565815814, "environment_variables": {"php_version": 7.3, "live_primary_domain": "testdomain.com"}, "php_version": "73", "randseed": "DRUMT5CLDWCROHKHA7SEI74GI3PBHT97", "styx_cluster": "styx-fe2.pantheon.io", "target_commit": "ce711a18d37c8d8430468b8b32df2a86e86d1d23", "target_ref": "refs/tags/pantheon_test_2", "key": "aaa00005-0123-4567-89ab-cdef01234567!test", "is_initialized": true, "styx_clusters_for_cache_clear": ["styx-fe3-europe-west4.pantheon.io", "styx-fe4-europe-west4.pantheon.io", "edge.live.getpantheon.com", "styx-fe2-australia-southeast1.pantheon.io", "styx-fe4.pantheon.io", "styx-fe3-australia-southeast1.pantheon.io", "styx-fe1.pantheon.io", "styx-fe2.pantheon.io", "styx-fe3-northamerica-northeast1.pantheon.io", "styx-fe4-australia-southeast1.pantheon.io", "styx-fe1-australia-southeast1.pantheon.io", "styx-fe2-europe-west4.pantheon.io", "styx-fe3.pantheon.io", "styx-01.pantheon.io", "styx-fe2-northamerica-northeast1.pantheon.io", "styx-fe1-europe-west4.pantheon.io", "styx-fe1-northamerica-northeast1.pantheon.io", "styx-fe4-northamerica-northeast1.pantheon.io"], "lock": {"username": null, "password": null, "locked": false}}, "live": {"dns_zone": "pantheonsite.io", "environment_created": 1565815816, "environment_variables": {"php_version": 7.3, "live_primary_domain": "testdomain.com"}, "maintenance": {"enabled": false}, "php_version": "73", "randseed": "3RCQ34T46NI7UZ7CADC5ZK8W5QH8UL0N", "styx_cluster": "styx-fe2.pantheon.io", "target_commit": "ce711a18d37c8d8430468b8b32df2a86e86d1d23", "target_ref": "refs/tags/pantheon_live_2", "key": "aaa00005-0123-4567-89ab-cdef01234567!live", "is_initialized": true, "styx_clusters_for_cache_clear": ["styx-fe3-europe-west4.pantheon.io", "styx-fe4-europe-west4.pantheon.io", "edge.live.getpantheon.com", "styx-fe2-australia-southeast1.pantheon.io", "styx-fe4.pantheon.io", "styx-fe3-australia-southeast1.pantheon.io", "styx-fe1.pantheon.io", "styx-fe2.pantheon.io", "styx-fe3-northamerica-northeast1.pantheon.io", "styx-fe4-australia-southeast1.pantheon.io", "styx-fe1-australia-southeast1.pantheon.io", "styx-fe2-europe-west4.pantheon.io", "styx-fe3.pantheon.io", "styx-01.pantheon.io", "styx-fe2-northamerica-northeast1.pantheon.io", "styx-fe1-europe-west4.pantheon.io", "styx-fe1-northamerica-northeast1.pantheon.io", "styx-fe4-northamerica-northeast1.pantheon.io"], "lock": {"username": null, "password": null, "locked": false}}, "dev": {"diffstat": {}, "dns_zone": "pantheonsite.io", "environment_created": 1565815809, "environment_variables": {"php_version": 7.3, "live_primary_domain": "testdomain.com"}, "on_server_development": false, "php_version": "73", "quicksilver_configuration": {}, "randseed": "XSRNW5DUNQAYXI1HC6MHEKPE04RTINXM", "styx_cluster": "styx-fe1.pantheon.io", "target_commit": "ab644624256c60e9fd5c7ad35b758461deb8dec3", "target_ref": "refs/heads/master", "key": "aaa00005-0123-4567-89ab-cdef01234567!dev", "is_initialized": true, "styx_clusters_for_cache_clear": ["styx-fe3-europe-west4.pantheon.io", "styx-fe4-europe-west4.pantheon.io", "edge.live.getpantheon.com", "styx-fe2-australia-southeast1.pantheon.io", "styx-fe4.pantheon.io", "styx-fe3-australia-southeast1.pantheon.io", "styx-fe1.pantheon.io", "styx-fe2.pantheon.io", "styx-fe3-northamerica-northeast1.pantheon.io", "styx-fe4-australia-southeast1.pantheon.io", "styx-fe1-australia-southeast1.pantheon.io", "styx-fe2-europe-west4.pantheon.io", "styx-fe3.pantheon.io", "styx-01.pantheon.io", "styx-fe2-northamerica-northeast1.pantheon.io", "styx-fe1-europe-west4.pantheon.io", "styx-fe1-northamerica-northeast1.pantheon.io", "styx-fe4-northamerica-northeast1.pantheon.io"], "lock": {"username": null, "password": null, "locked": false}}}' +- + request: + method: POST + url: 'https://onebox/api/sites/aaa00005-0123-4567-89ab-cdef01234567/environments/live/workflows' + headers: + Host: onebox + Expect: null + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt' + Accept: null + body: '{"type":"set_primary_domain","params":{"primary_domain":"testdomain.com"}}' + response: + status: + http_version: '1.1' + code: '202' + message: Accepted + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:09 GMT' + Content-Type: 'application/json; charset=utf-8' + Transfer-Encoding: chunked + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00011-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + body: '{"environment_id": "live", "params": {"primary_domain": "testdomain.com"}, "role": "owner", "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "started_at": 1569365048.462987, "task_ids": ["aaa00012-0123-4567-89ab-cdef01234567", "aaa00013-0123-4567-89ab-cdef01234567"], "trace_id": "aaa00011-0123-4567-89ab-cdef01234567", "type": "set_primary_domain", "user_id": "11111111-1111-1111-1111-111111111111", "waiting_for_task_id": "aaa00013-0123-4567-89ab-cdef01234567", "id": "aaa00014-0123-4567-89ab-cdef01234567", "key": "1569362400", "keep_forever": false, "phase": "started", "queued_time": null, "run_time": null, "created_at": 1569365047.698356, "reason": "", "environment": "live", "final_task_id": null, "result": null, "total_time": null, "active_description": "Setting the primary domain for environment \"live\"", "description": "Set the primary domain for environment \"live\"", "step": 2, "has_operation_log_output": false, "number_of_tasks": 2, "trace_log_url": "https://app.logz.io/#/dashboard/kibana?kibanaRoute=discover%3F_a%3D(query:(query_string:(analyze_wildcard:!t,query:%27trace_id:aaa00011-0123-4567-89ab-cdef01234567%27)))%26_g%3D(refreshInterval:(display:Off,pause:!f,value:0),time:(from:%272019-09-24T22:39:07.698356Z%27,mode:quick,to:%27now%27))", "user": {"user_id": "11111111-1111-1111-1111-111111111111", "auth0_user_id": "samlp|pantheonokta|devuser@pantheon.io", "created_at": 1565656598, "destination_organization_id": null, "is_registered": true, "created_organization_id": null, "password": "SCRUBBED", "email": "devuser@pantheon.io"}, "user_email": "devuser@pantheon.io", "waiting_for_task": {"environment": "live", "fn_name": "trigger_task", "initialized_at": 1569365048.84577, "params": {"environment": "live", "domains": ["live-behat-tests.pantheonsite.io", "testdomain.com"], "task_type": "save_policy_docs", "site_id": "aaa00005-0123-4567-89ab-cdef01234567"}, "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "trace_id": "aaa00011-0123-4567-89ab-cdef01234567", "user_id": "11111111-1111-1111-1111-111111111111", "workflow_id": "aaa00014-0123-4567-89ab-cdef01234567", "id": "aaa00013-0123-4567-89ab-cdef01234567", "key": "1569362400", "responses": [], "queued_time": null, "host": null, "result": "succeeded", "phase": "initialized", "created_at": 1569365048.311134, "allow_concurrent": false, "run_time": null, "total_time": null, "reason": "", "error_details": "", "internal_reason": "", "trace_log_url": "https://app.logz.io/#/dashboard/kibana?kibanaRoute=discover%3F_a%3D(query:(query_string:(analyze_wildcard:!t,query:%27trace_id:aaa00011-0123-4567-89ab-cdef01234567%27)))%26_g%3D(refreshInterval:(display:Off,pause:!f,value:0),time:(from:%272019-09-24T22:39:08.311134Z%27,mode:quick,to:%27now%27))", "type": "save_policy_docs", "build_url": null, "messages": {}}}' +# The "remove" request is the same as an add request, but with the requested primary domain null. +- + request: + method: POST + url: 'https://onebox/api/sites/aaa00005-0123-4567-89ab-cdef01234567/environments/live/workflows' + headers: + Host: onebox + Expect: null + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt' + Accept: null + body: '{"type":"set_primary_domain","params":{"primary_domain":null}}' + response: + status: + http_version: '1.1' + code: '202' + message: Accepted + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:09 GMT' + Content-Type: 'application/json; charset=utf-8' + Transfer-Encoding: chunked + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00011-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + body: '{"environment_id": "live", "params": {"primary_domain": null}, "role": "owner", "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "started_at": 1569365048.462987, "task_ids": ["aaa00012-0123-4567-89ab-cdef01234567", "aaa00013-0123-4567-89ab-cdef01234567"], "trace_id": "aaa00011-0123-4567-89ab-cdef01234567", "type": "set_primary_domain", "user_id": "11111111-1111-1111-1111-111111111111", "waiting_for_task_id": "aaa00013-0123-4567-89ab-cdef01234567", "id": "aaa00014-0123-4567-89ab-cdef01234567", "key": "1569362400", "keep_forever": false, "phase": "started", "queued_time": null, "run_time": null, "created_at": 1569365047.698356, "reason": "", "environment": "live", "final_task_id": null, "result": null, "total_time": null, "active_description": "Setting the primary domain for environment \"live\"", "description": "Set the primary domain for environment \"live\"", "step": 2, "has_operation_log_output": false, "number_of_tasks": 2, "trace_log_url": "https://app.logz.io/#/dashboard/kibana?kibanaRoute=discover%3F_a%3D(query:(query_string:(analyze_wildcard:!t,query:%27trace_id:aaa00011-0123-4567-89ab-cdef01234567%27)))%26_g%3D(refreshInterval:(display:Off,pause:!f,value:0),time:(from:%272019-09-24T22:39:07.698356Z%27,mode:quick,to:%27now%27))", "user": {"user_id": "11111111-1111-1111-1111-111111111111", "auth0_user_id": "samlp|pantheonokta|devuser@pantheon.io", "created_at": 1565656598, "destination_organization_id": null, "is_registered": true, "created_organization_id": null, "password": "SCRUBBED", "email": "devuser@pantheon.io"}, "user_email": "devuser@pantheon.io", "waiting_for_task": {"environment": "live", "fn_name": "trigger_task", "initialized_at": 1569365048.84577, "params": {"environment": "live", "domains": ["live-behat-tests.pantheonsite.io", "testdomain.com"], "task_type": "save_policy_docs", "site_id": "aaa00005-0123-4567-89ab-cdef01234567"}, "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "trace_id": "aaa00011-0123-4567-89ab-cdef01234567", "user_id": "11111111-1111-1111-1111-111111111111", "workflow_id": "aaa00014-0123-4567-89ab-cdef01234567", "id": "aaa00013-0123-4567-89ab-cdef01234567", "key": "1569362400", "responses": [], "queued_time": null, "host": null, "result": "succeeded", "phase": "initialized", "created_at": 1569365048.311134, "allow_concurrent": false, "run_time": null, "total_time": null, "reason": "", "error_details": "", "internal_reason": "", "trace_log_url": "https://app.logz.io/#/dashboard/kibana?kibanaRoute=discover%3F_a%3D(query:(query_string:(analyze_wildcard:!t,query:%27trace_id:aaa00011-0123-4567-89ab-cdef01234567%27)))%26_g%3D(refreshInterval:(display:Off,pause:!f,value:0),time:(from:%272019-09-24T22:39:08.311134Z%27,mode:quick,to:%27now%27))", "type": "save_policy_docs", "build_url": null, "messages": {}}}' +- + request: + method: GET + url: 'https://onebox/api/sites/aaa00005-0123-4567-89ab-cdef01234567/workflows/aaa00014-0123-4567-89ab-cdef01234567' + headers: + Host: onebox + Accept-Encoding: null + Content-type: application/json + User-Agent: 'Terminus/2.1.1-dev (php_version=7.2.19-0ubuntu0.18.04.2&script=bin/terminus)' + Authorization: 'Bearer 11111111-1111-1111-1111-111111111111:aaa00003-0123-4567-89ab-cdef01234567:vqfLeZlGZnTQmbEwNZ4dt' + Accept: null + response: + status: + http_version: '1.1' + code: '200' + message: OK + headers: + Server: nginx + Date: 'Tue, 24 Sep 2019 22:44:09 GMT' + Content-Type: 'application/json; charset=utf-8' + Transfer-Encoding: chunked + Connection: keep-alive + X-Pantheon-Trace-Id: aaa00015-0123-4567-89ab-cdef01234567 + X-Frame-Options: deny + Access-Control-Allow-Methods: GET + Access-Control-Allow-Headers: 'Origin, Content-Type, Accept' + Cache-Control: 'private, max-age=0, no-cache, no-store' + Pragma: no-cache + Vary: Accept-Encoding + Strict-Transport-Security: max-age=31536000 + body: '{"environment_id": "live", "final_task_id": "aaa00013-0123-4567-89ab-cdef01234567", "finished_at": 1569365049.595287, "params": {"primary_domain": "testdomain.com"}, "reason": "", "result": "succeeded", "role": "owner", "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "started_at": 1569365048.462987, "task_ids": ["aaa00012-0123-4567-89ab-cdef01234567", "aaa00013-0123-4567-89ab-cdef01234567"], "trace_id": "aaa00011-0123-4567-89ab-cdef01234567", "type": "set_primary_domain", "user_id": "11111111-1111-1111-1111-111111111111", "waiting_for_task_id": null, "id": "aaa00014-0123-4567-89ab-cdef01234567", "key": "aaa00005-0123-4567-89ab-cdef01234567", "keep_forever": false, "phase": "finished", "queued_time": null, "run_time": 1.1323001384735107, "created_at": 1569365047.698356, "environment": "live", "total_time": 1.8969311714172363, "active_description": "Set the primary domain for environment \"live\"", "description": "Set the primary domain for environment \"live\"", "step": 2, "has_operation_log_output": false, "number_of_tasks": 2, "trace_log_url": "https://app.logz.io/#/dashboard/kibana?kibanaRoute=discover%3F_a%3D(query:(query_string:(analyze_wildcard:!t,query:%27trace_id:aaa00011-0123-4567-89ab-cdef01234567%27)))%26_g%3D(refreshInterval:(display:Off,pause:!f,value:0),time:(from:%272019-09-24T22:39:07.698356Z%27,mode:quick,to:%272019-09-24T22:49:09.595287Z%27))", "user": {"user_id": "11111111-1111-1111-1111-111111111111", "auth0_user_id": "samlp|pantheonokta|devuser@pantheon.io", "created_at": 1565656598, "destination_organization_id": null, "is_registered": true, "created_organization_id": null, "password": "SCRUBBED", "email": "devuser@pantheon.io"}, "user_email": "devuser@pantheon.io", "final_task": {"environment": "live", "finished_at": 1569365049.521994, "fn_name": "trigger_task", "initialized_at": 1569365048.84577, "params": {"environment": "live", "domains": ["live-behat-tests.pantheonsite.io", "testdomain.com"], "task_type": "save_policy_docs", "site_id": "aaa00005-0123-4567-89ab-cdef01234567"}, "queued_at": 1569365048.920202, "responses": [{"code": 200, "body": "Saved policy doc associated with domains: [u''live-behat-tests.pantheonsite.io'', u''testdomain.com'']", "error_details": "", "internal_reason": ""}], "result": "succeeded", "site_id": "aaa00005-0123-4567-89ab-cdef01234567", "started_at": 1569365048.92021, "trace_id": "aaa00011-0123-4567-89ab-cdef01234567", "user_id": "11111111-1111-1111-1111-111111111111", "workflow_id": "aaa00014-0123-4567-89ab-cdef01234567", "id": "aaa00013-0123-4567-89ab-cdef01234567", "key": "1569362400", "queued_time": 7.867813110351562e-06, "host": null, "phase": "finished", "created_at": 1569365048.311134, "allow_concurrent": false, "run_time": 0.6017842292785645, "total_time": 1.210860013961792, "reason": "", "error_details": "", "internal_reason": "", "trace_log_url": "https://app.logz.io/#/dashboard/kibana?kibanaRoute=discover%3F_a%3D(query:(query_string:(analyze_wildcard:!t,query:%27trace_id:aaa00011-0123-4567-89ab-cdef01234567%27)))%26_g%3D(refreshInterval:(display:Off,pause:!f,value:0),time:(from:%272019-09-24T22:39:08.311134Z%27,mode:quick,to:%272019-09-24T22:49:09.521994Z%27))", "type": "save_policy_docs", "build_url": null, "messages": {"2019-09-24T22:44:09.816436": {"message": "Saved policy doc associated with domains: [u''live-behat-tests.pantheonsite.io'', u''testdomain.com'']", "level": "INFO"}}}}' diff --git a/tests/unit_tests/Commands/Domain/Primary/AddCommandTest.php b/tests/unit_tests/Commands/Domain/Primary/AddCommandTest.php new file mode 100644 index 000000000..47c0a2bb1 --- /dev/null +++ b/tests/unit_tests/Commands/Domain/Primary/AddCommandTest.php @@ -0,0 +1,54 @@ +prepareTestSetReset( + $domain, + 'Set {domain} as primary for {site}.{env}', + ['domain' => $domain, 'site' => $this->site->get('name'), 'env' => $this->environment->id] + ); + + $out = $this->command->add("$site_name.{$this->environment->id}", $domain); + $this->assertNull($out); + } + + public function testFilterPlatformDomains() + { + $sys_under_test = new AddCommand(); + $this->assertEquals( + array_values([ + 'x', + 'something.com', + 'averylong.domain.ofsomeosrt.tld', + ]), + array_values($sys_under_test->filterPlatformDomains( + [ + 'x', + 'something.com', + 'dev-mikes-testsite.pantheonsite.io', + 'averylong.domain.ofsomeosrt.tld', + ] + )) + ); + } +} diff --git a/tests/unit_tests/Commands/Domain/Primary/PrimaryDomainCommandsTestBase.php b/tests/unit_tests/Commands/Domain/Primary/PrimaryDomainCommandsTestBase.php new file mode 100644 index 000000000..8151ec9cc --- /dev/null +++ b/tests/unit_tests/Commands/Domain/Primary/PrimaryDomainCommandsTestBase.php @@ -0,0 +1,81 @@ +session = $this->getMockBuilder(Session::class) + ->disableOriginalConstructor() + ->getMock(); + + $this->command = $this->getSystemUnderTest(); + $this->command->setLogger($this->logger); + $this->command->setSites($this->sites); + $this->command->setSession($this->session); + $this->command->setConfig($this->getConfig()); + $this->command->setContainer($this->getContainer()); + + $this->environment->id = 'env_id'; + } + + protected function prepareTestSetReset($domain, $logTemplate, $logParams) + { + $workflow = $this->getMockBuilder(Workflow::class) + ->disableOriginalConstructor() + ->getMock(); + + $primaryDomainModel = $this->getMockBuilder(PrimaryDomain::class) + ->disableOriginalConstructor() + ->setMethods(['setPrimaryDomain', 'removePrimaryDomain']) + ->getMock(); + + if ($domain != null) { + $primaryDomainModel + ->expects($this->once()) + ->method('setPrimaryDomain') + ->with($this->equalTo($domain)) + ->willReturn($workflow); + } else { + $primaryDomainModel + ->expects($this->once()) + ->method('removePrimaryDomain') + ->willReturn($workflow); + } + + $this->environment->expects($this->once()) + ->method('getPrimaryDomainModel') + ->willReturn($primaryDomainModel); + + $this->expectWorkflowProcessing(); + + $this->logger->expects($this->once()) + ->method('log') + ->with( + $this->equalTo('notice'), + $this->equalTo($logTemplate), + $this->equalTo($logParams) + ); + } +} diff --git a/tests/unit_tests/Commands/Domain/Primary/RemoveCommandTest.php b/tests/unit_tests/Commands/Domain/Primary/RemoveCommandTest.php new file mode 100644 index 000000000..2b36ea3ad --- /dev/null +++ b/tests/unit_tests/Commands/Domain/Primary/RemoveCommandTest.php @@ -0,0 +1,28 @@ +prepareTestSetReset( + $domain, + 'Primary domain has been removed from {site}.{env}', + ['site' => $this->site->get('name'), 'env' => $this->environment->id] + ); + + $out = $this->command->remove("$site_name.{$this->environment->id}", $domain); + $this->assertNull($out); + } +} diff --git a/tests/unit_tests/Models/DomainTest.php b/tests/unit_tests/Models/DomainTest.php index 484505e7e..e940f67ac 100644 --- a/tests/unit_tests/Models/DomainTest.php +++ b/tests/unit_tests/Models/DomainTest.php @@ -122,7 +122,7 @@ public function testGetDNSRecords() $this->assertEquals($records, $domain->getDNSRecords()); } - + public function testSerialize() { $data = [ @@ -131,6 +131,30 @@ public function testSerialize() 'status' => 'status', 'status_message' => 'status message', 'deletable' => false, + 'primary' => true, + 'extraneous' => 'info', + ]; + $expected = [ + 'type' => 'platform', + 'id' => 'live-mysite.pantheonsite.io', + 'status' => 'status', + 'status_message' => 'status message', + 'deletable' => false, + 'primary' => true, + ]; + + $domain = $this->_createDomain($data); + $actual = $domain->serialize(); + $this->assertEquals($expected, $actual); + } + + public function testSerializeSparse() + { + $data = [ + 'type' => 'platform', + 'id' => 'live-mysite.pantheonsite.io', + 'status' => 'status', + 'status_message' => 'status message', 'extraneous' => 'info', ]; $expected = [ @@ -139,6 +163,7 @@ public function testSerialize() 'status' => 'status', 'status_message' => 'status message', 'deletable' => false, + 'primary' => false, ]; $domain = $this->_createDomain($data);