From 53d0e3922161a83b6a09cc4bc7b30bcd1dea9cbe Mon Sep 17 00:00:00 2001 From: Andy Baugh Date: Thu, 25 Jul 2024 18:36:45 +0000 Subject: [PATCH] RE-34: Move up the recording of the active remote MySQL profile Fixes an issue where it was not recorded when the `try` block actually succeeded. I also golfed things down a bit where I could. Notably, the unit test I had to modify died along the critical path until I added a Test::MockFile invocation RE /var/cpanel/elevate-mysql-profile. I am pretty sure that this along with my manual testing should ensure this functions properly going forward. Changelog: Fix minor bug in original RE-34 fix where it would not always record the active profile. --- elevate-cpanel | 55 ++++++++++----------- lib/Elevate/Components/DatabaseUpgrade.pm | 58 +++++++++++------------ t/components-DatabaseUpgrade.t | 38 +++++++++++---- 3 files changed, 83 insertions(+), 68 deletions(-) diff --git a/elevate-cpanel b/elevate-cpanel index ff9bd6e6a..094ab3cec 100755 --- a/elevate-cpanel +++ b/elevate-cpanel @@ -5470,55 +5470,53 @@ EOS } sub post_leapp ($self) { + return unless -e MYSQL_PROFILE_FILE; - if ( -e MYSQL_PROFILE_FILE ) { - my $original_profile = File::Slurper::read_text(MYSQL_PROFILE_FILE) // 'localhost'; - INFO(qq{Reactivating "$original_profile" MySQL profile}); + my $original_profile = File::Slurper::read_text(MYSQL_PROFILE_FILE) // 'localhost'; + INFO(qq{Reactivating "$original_profile" MySQL profile}); - my $output = $self->ssystem_capture_output( '/usr/local/cpanel/scripts/manage_mysql_profiles', '--activate', "$original_profile" ); - my $stdout = join qq{\n}, @{ $output->{'stdout'} }; - - unless ( $stdout =~ m{MySQL profile activation done} ) { - die <<~"EOS"; - Unable to reactivate the original remote MySQL profile "$original_profile": + my $output = $self->ssystem_capture_output( '/usr/local/cpanel/scripts/manage_mysql_profiles', '--activate', "$original_profile" ); + my $stdout = join qq{\n}, @{ $output->{'stdout'} }; - $stdout + unless ( $stdout =~ m{MySQL profile activation done} ) { + die <<~"EOS"; + Unable to reactivate the original remote MySQL profile "$original_profile": - Please resolve the reported problems then run this script again with: + $stdout - /scripts/elevate-cpanel --continue + Please resolve the reported problems then run this script again with: - EOS - } + /scripts/elevate-cpanel --continue - unlink MYSQL_PROFILE_FILE; + EOS } + unlink MYSQL_PROFILE_FILE or WARN( "Could not delete " . MYSQL_PROFILE_FILE . ": $!" ); + return; } sub _ensure_localhost_mysql_profile_is_active ( $self, $should_create_localhost_profile ) { - - if ( Cpanel::MysqlUtils::MyCnf::Basic::is_local_mysql() ) { - return; - } + return if Cpanel::MysqlUtils::MyCnf::Basic::is_local_mysql(); my $profile_manager = Cpanel::MysqlUtils::RemoteMySQL::ProfileManager->new(); + my $profile = $profile_manager->get_active_profile('dont_die') || 'localhost'; + if ($should_create_localhost_profile) { + INFO( "Saving the currently active MySQL Profile ($profile) to " . MYSQL_PROFILE_FILE ); + File::Slurper::write_text( MYSQL_PROFILE_FILE, $profile ); + } + try { $profile_manager->validate_profile('localhost'); $self->_activate_localhost_profile($profile_manager); } catch { - if ($should_create_localhost_profile) { - INFO("Attempting to create new localhost MySQL profile..."); - $self->_create_new_localhost_profile($profile_manager); - $self->_ensure_localhost_mysql_profile_is_active(0); - } - else { - die "Unable to generate/enable localhost MySQL profile: $_\n"; - } + die "Unable to generate/enable localhost MySQL profile: $_\n" unless $should_create_localhost_profile; + INFO("Attempting to create new localhost MySQL profile..."); + $self->_create_new_localhost_profile($profile_manager); + $self->_ensure_localhost_mysql_profile_is_active(0); }; return; @@ -5526,9 +5524,6 @@ EOS sub _create_new_localhost_profile ( $self, $profile_manager ) { - my $active_profile = $profile_manager->get_active_profile('dont_die'); - File::Slurper::write_text( MYSQL_PROFILE_FILE, $active_profile ); - my $password = Cpanel::PasswdStrength::Generate::generate_password( 16, no_othersymbols => 1 ); try { diff --git a/lib/Elevate/Components/DatabaseUpgrade.pm b/lib/Elevate/Components/DatabaseUpgrade.pm index ff014b9c8..2e9a67e9d 100644 --- a/lib/Elevate/Components/DatabaseUpgrade.pm +++ b/lib/Elevate/Components/DatabaseUpgrade.pm @@ -47,41 +47,46 @@ sub pre_leapp ($self) { } sub post_leapp ($self) { + return unless -e MYSQL_PROFILE_FILE; - if ( -e MYSQL_PROFILE_FILE ) { - my $original_profile = File::Slurper::read_text(MYSQL_PROFILE_FILE) // 'localhost'; - INFO(qq{Reactivating "$original_profile" MySQL profile}); + my $original_profile = File::Slurper::read_text(MYSQL_PROFILE_FILE) // 'localhost'; + INFO(qq{Reactivating "$original_profile" MySQL profile}); - my $output = $self->ssystem_capture_output( '/usr/local/cpanel/scripts/manage_mysql_profiles', '--activate', "$original_profile" ); - my $stdout = join qq{\n}, @{ $output->{'stdout'} }; - - unless ( $stdout =~ m{MySQL profile activation done} ) { - die <<~"EOS"; - Unable to reactivate the original remote MySQL profile "$original_profile": + my $output = $self->ssystem_capture_output( '/usr/local/cpanel/scripts/manage_mysql_profiles', '--activate', "$original_profile" ); + my $stdout = join qq{\n}, @{ $output->{'stdout'} }; - $stdout + unless ( $stdout =~ m{MySQL profile activation done} ) { + die <<~"EOS"; + Unable to reactivate the original remote MySQL profile "$original_profile": - Please resolve the reported problems then run this script again with: + $stdout - /scripts/elevate-cpanel --continue + Please resolve the reported problems then run this script again with: - EOS - } + /scripts/elevate-cpanel --continue - unlink MYSQL_PROFILE_FILE; + EOS } + unlink MYSQL_PROFILE_FILE or WARN( "Could not delete " . MYSQL_PROFILE_FILE . ": $!" ); + return; } sub _ensure_localhost_mysql_profile_is_active ( $self, $should_create_localhost_profile ) { - - if ( Cpanel::MysqlUtils::MyCnf::Basic::is_local_mysql() ) { - return; - } + return if Cpanel::MysqlUtils::MyCnf::Basic::is_local_mysql(); my $profile_manager = Cpanel::MysqlUtils::RemoteMySQL::ProfileManager->new(); + # Immediately record the currently active profile, as othrewise you can + # miss it in the try/catch below. Default to localhost, because if there's + # no answer, something's probably wrong in a way we don't *want* to touch. + my $profile = $profile_manager->get_active_profile('dont_die') || 'localhost'; + if ($should_create_localhost_profile) { + INFO( "Saving the currently active MySQL Profile ($profile) to " . MYSQL_PROFILE_FILE ); + File::Slurper::write_text( MYSQL_PROFILE_FILE, $profile ); + } + # Validate that the current “localhost” profile exists, and contains valid settings. try { $profile_manager->validate_profile('localhost'); @@ -90,14 +95,10 @@ sub _ensure_localhost_mysql_profile_is_active ( $self, $should_create_localhost_ # Otherwise attempt to recreate it, overwriting the existing profile. catch { - if ($should_create_localhost_profile) { - INFO("Attempting to create new localhost MySQL profile..."); - $self->_create_new_localhost_profile($profile_manager); - $self->_ensure_localhost_mysql_profile_is_active(0); - } - else { - die "Unable to generate/enable localhost MySQL profile: $_\n"; - } + die "Unable to generate/enable localhost MySQL profile: $_\n" unless $should_create_localhost_profile; + INFO("Attempting to create new localhost MySQL profile..."); + $self->_create_new_localhost_profile($profile_manager); + $self->_ensure_localhost_mysql_profile_is_active(0); }; return; @@ -105,9 +106,6 @@ sub _ensure_localhost_mysql_profile_is_active ( $self, $should_create_localhost_ sub _create_new_localhost_profile ( $self, $profile_manager ) { - my $active_profile = $profile_manager->get_active_profile('dont_die'); - File::Slurper::write_text( MYSQL_PROFILE_FILE, $active_profile ); - my $password = Cpanel::PasswdStrength::Generate::generate_password( 16, no_othersymbols => 1 ); try { diff --git a/t/components-DatabaseUpgrade.t b/t/components-DatabaseUpgrade.t index e9f09d626..30f08d9c2 100644 --- a/t/components-DatabaseUpgrade.t +++ b/t/components-DatabaseUpgrade.t @@ -1,11 +1,12 @@ #!/usr/local/cpanel/3rdparty/bin/perl +package test::cpev::components; # Copyright 2024 WebPros International, LLC # All rights reserved. # copyright@cpanel.net http://cpanel.net # This code is subject to the cPanel license. Unauthorized copying is prohibited. -package test::cpev::components; +use cPstrict; use FindBin; @@ -20,11 +21,6 @@ use Test::MockFile qw/strict/; use lib $FindBin::Bin . "/lib"; use Test::Elevate; -use cPstrict; - -use strict; -use warnings; - use constant PROFILE_FILE => Elevate::Components::DatabaseUpgrade::MYSQL_PROFILE_FILE; my $db_upgrade = bless {}, 'Elevate::Components::DatabaseUpgrade'; @@ -117,7 +113,8 @@ my $db_upgrade = bless {}, 'Elevate::Components::DatabaseUpgrade'; { note('Checking _ensure_localhost_mysql_profile_is_active'); - my $mock_mysqlutils = Test::MockModule->new('Cpanel::MysqlUtils::MyCnf::Basic'); + my $saved_db_file_mock = Test::MockFile->file('/var/cpanel/elevate-mysql-profile'); + my $mock_mysqlutils = Test::MockModule->new('Cpanel::MysqlUtils::MyCnf::Basic'); $mock_mysqlutils->redefine( 'is_local_mysql', 1 ); my $instantiated_pm = 0; @@ -128,7 +125,30 @@ my $db_upgrade = bless {}, 'Elevate::Components::DatabaseUpgrade'; is $instantiated_pm, 0, 'ProfileManager instance not created when is_local_mysql returns true'; $mock_mysqlutils->redefine( 'is_local_mysql', 0 ); - $mock_pm->redefine( 'validate_profile', sub { die 'foo' } ); + $mock_pm->redefine( + 'validate_profile', sub { die 'foo' }, + 'read_profiles', + sub { + return { + 'localhost' => { + 'mysql_port' => '3306', + 'mysql_pass' => 'hunter2', + 'mysql_host' => 'localhost', + 'mysql_user' => 'azurediamond', + 'setup_via' => 'Auto-Migrated active profile', + 'active' => 0, + }, + 'remote' => { + 'mysql_port' => '69420', + 'mysql_pass' => 'NHTSA_R00lz', + 'mysql_host' => 'test.test', + 'mysql_user' => 'crash_test_dummy', + 'setup_via' => "The People's Quality Front", + 'active' => 1 + }, + }; + }, + ); like( dies { $db_upgrade->_ensure_localhost_mysql_profile_is_active(0) }, @@ -146,10 +166,12 @@ my $db_upgrade = bless {}, 'Elevate::Components::DatabaseUpgrade'; 'Died as expected when _activate_localhost_profile fails and should_create_localhost_profile is false' ); + $mock_pm->redefine( 'validate_profile', sub { die "whugg" } ); my $created_new_localhost_profile = 0; $mock_db_upgrade->redefine( '_create_new_localhost_profile', sub { $created_new_localhost_profile = 1 } ); eval { $db_upgrade->_ensure_localhost_mysql_profile_is_active(1) }; is $created_new_localhost_profile, 1, '_create_new_localhost_profile() was called when existing profile failed to validate/activate'; + message_seen( 'INFO', qr{Saving the currently active MySQL Profile \(remote\) to /var/cpanel/elevate-mysql-profile} ); message_seen( 'INFO', qr/Attempting to create new localhost MySQL profile/ ); clear_messages_seen();