File manager - Edit - /opt/gsutil/gslib/tests/test_shim_util.py
Back
# -*- coding: utf-8 -*- # Copyright 2021 Google LLC. All Rights Reserved. # # Licensed under the Apache License, Version 2.0 (the "License"); # you may not use this file except in compliance with the License. # You may obtain a copy of the License at # # http://www.apache.org/licenses/LICENSE-2.0 # # Unless required by applicable law or agreed to in writing, software # distributed under the License is distributed on an "AS IS" BASIS, # WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. # See the License for the specific language governing permissions and # limitations under the License. """Tests for shim_util.py.""" from __future__ import absolute_import from __future__ import print_function from __future__ import division from __future__ import unicode_literals import collections from contextlib import contextmanager import os import re import subprocess from unittest import mock from boto import config from gslib import command from gslib import command_argument from gslib import exception from gslib.commands import rsync from gslib.commands import version from gslib.commands import test from gslib.cs_api_map import ApiSelector from gslib.tests import testcase from gslib.utils import boto_util from gslib.utils import constants from gslib.utils import shim_util from gslib.utils import system_util from gslib.tests import util @contextmanager def _mock_boto_config(boto_config_dict): """"Mock boto config replacing any exiting config. The util.SetBotoConfigForTest has a use_existing_config flag that can be set to False, but it does not work if the config has been already loaded, which is the case for all unit tests that do not use RunCommand method. Args: boto_config_dict. A dict with key=<boto section name> and value=<a dict of boto field name and the value for that field>. """ def _config_get_side_effect(section, key, default_value=None): return boto_config_dict.get(section, {}).get(key, default_value) with mock.patch.object(config, 'get', autospec=True) as mock_get: with mock.patch.object(config, 'getbool', autospec=True) as mock_getbool: with mock.patch.object(config, 'items', autospec=True) as mock_items: mock_get.side_effect = _config_get_side_effect mock_getbool.side_effect = _config_get_side_effect mock_items.return_value = boto_config_dict.items() yield class FakeCommandWithGcloudStorageMap(command.Command): """Implementation of a fake gsutil command.""" command_spec = command.Command.CreateCommandSpec('fake_shim', min_args=1, max_args=constants.NO_MAX, supported_sub_args='deilrz:', file_url_ok=True) gcloud_storage_map = shim_util.GcloudStorageMap( gcloud_command=['objects', 'fake'], flag_map={ '-r': shim_util.GcloudStorageFlag(gcloud_flag='-x'), '-z': shim_util.GcloudStorageFlag(gcloud_flag='--zip'), '-l': shim_util.GcloudStorageFlag( gcloud_flag='--ludicrous-list', repeat_type=shim_util.RepeatFlagType.LIST), '-d': shim_util.GcloudStorageFlag( gcloud_flag='--delightful-dict', repeat_type=shim_util.RepeatFlagType.DICT), '-e': shim_util.GcloudStorageFlag(gcloud_flag={ 'on': '--e-on', 'off': '--e-off' }), '-f': None, }) help_spec = command.Command.HelpSpec( help_name='fake_shim', help_name_aliases=[], help_type='command_help', help_one_line_summary='Fake one line summary for the command.', help_text='Help text for fake command.', subcommand_help_text={}, ) def RunCommand(self): print('FakeCommandWithGcloudStorageMap called') class FakeCommandWithSubCommandWithGcloudStorageMap(command.Command): """Implementation of a fake gsutil command.""" command_spec = command.Command.CreateCommandSpec( 'fake_with_sub', min_args=1, max_args=constants.NO_MAX, supported_sub_args='ay:', file_url_ok=True, argparse_arguments={ 'set': [ command_argument.CommandArgument. MakeZeroOrMoreCloudBucketURLsArgument() ], 'get': [ command_argument.CommandArgument.MakeNCloudBucketURLsArgument(1) ], }) gcloud_storage_map = shim_util.GcloudStorageMap(gcloud_command={ 'set': shim_util.GcloudStorageMap( gcloud_command=['buckets', 'update'], flag_map={ '-a': shim_util.GcloudStorageFlag(gcloud_flag='-x'), '-y': shim_util.GcloudStorageFlag(gcloud_flag='--yyy'), }), 'get': shim_util.GcloudStorageMap(gcloud_command=['buckets', 'describe'], flag_map={}) }, flag_map={}) help_spec = command.Command.HelpSpec( help_name='fake_with_sub', help_name_aliases=[], help_type='command_help', help_one_line_summary='Fake one line summary for the command.', help_text='Help text for fake command with sub commands.', subcommand_help_text={}, ) class TestGetGCloudStorageArgs(testcase.GsUtilUnitTestCase): """Test Command.get_gcloud_storage_args method.""" def setUp(self): super().setUp() self._fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-z', 'opt1', '-r', 'arg1', 'arg2'], headers={}, debug=1, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) def test_get_gcloud_storage_args_parses_command_and_flags(self): gcloud_args = self._fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, ['objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2']) def test_get_gcloud_storage_args_parses_custom_command_map(self): gcloud_args = self._fake_command.get_gcloud_storage_args( shim_util.GcloudStorageMap( gcloud_command=['objects', 'custom-fake'], flag_map={ '-z': shim_util.GcloudStorageFlag(gcloud_flag='-a'), '-r': shim_util.GcloudStorageFlag(gcloud_flag='-b'), })) self.assertEqual( gcloud_args, ['objects', 'custom-fake', '-a', 'opt1', '-b', 'arg1', 'arg2']) def test_get_gcloud_storage_args_parses_command_in_list_format(self): self._fake_command.gcloud_command = ['objects', 'fake'] gcloud_args = self._fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, ['objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2']) def test_get_gcloud_storage_args_parses_subcommands(self): fake_with_subcommand = FakeCommandWithSubCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['set', '-y', 'opt1', '-a', 'arg1', 'arg2'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_with_subcommand.get_gcloud_storage_args() self.assertEqual( gcloud_args, ['buckets', 'update', '--yyy', 'opt1', '-x', 'arg1', 'arg2']) def test_get_gcloud_storage_args_with_flags_to_ignore(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['positional_arg', '-f', '-r', 'opt2', '-f'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, ['objects', 'fake', 'positional_arg', '-x', 'opt2']) def test_get_gcloud_storage_args_with_positional_arg_at_beginning(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['positional_arg', '-z', 'opt1', '-r', 'opt2'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual( gcloud_args, ['objects', 'fake', 'positional_arg', '--zip', 'opt1', '-x', 'opt2']) def test_get_gcloud_storage_args_with_positional_arg_in_middle(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-z', 'opt1', 'positional_arg', '-r', 'opt2'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual( gcloud_args, ['objects', 'fake', '--zip', 'opt1', 'positional_arg', '-x', 'opt2']) def test_get_gcloud_storage_args_with_repeat_flag_list(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-l', 'flag_value1', '-l', 'flag_value2', 'positional_arg'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, [ 'objects', 'fake', 'positional_arg', '--ludicrous-list=flag_value1,flag_value2' ]) def test_get_gcloud_storage_args_with_repeat_flag_dict(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=[ '-d', 'flag_key1:flag_value1', '-d', 'flag_key2:flag_value2', 'positional_arg' ], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, [ 'objects', 'fake', 'positional_arg', '--delightful-dict=flag_key1=flag_value1,flag_key2=flag_value2' ]) def test_get_gcloud_storage_args_with_value_translated_to_flag(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-e', 'on', 'positional_arg'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, ['objects', 'fake', '--e-on', 'positional_arg']) fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=[ 'positional_arg', '-e', 'off', ], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) gcloud_args = fake_command.get_gcloud_storage_args() self.assertEqual(gcloud_args, ['objects', 'fake', 'positional_arg', '--e-off']) def test_raises_error_for_invalid_value_translated_to_flag(self): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-e', 'incorrect', 'positional_arg'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) with self.assertRaisesRegex( ValueError, 'Flag value not in translation map for "-e": incorrect'): gcloud_args = fake_command.get_gcloud_storage_args() def test_raises_error_if_gcloud_storage_map_is_missing(self): self._fake_command.gcloud_storage_map = None with self.assertRaisesRegex( exception.GcloudStorageTranslationError, 'Command "fake_shim" cannot be translated to gcloud storage' ' because the translation mapping is missing'): self._fake_command.get_gcloud_storage_args() def test_raises_error_if_gcloud_command_is_of_incorrect_type(self): self._fake_command.gcloud_storage_map = shim_util.GcloudStorageMap( gcloud_command='some fake command as a string', flag_map={}) with self.assertRaisesRegex( ValueError, 'Incorrect mapping found for "fake_shim" command'): self._fake_command.get_gcloud_storage_args() def test_raises_error_if_command_option_mapping_is_missing(self): self._fake_command.gcloud_storage_map = shim_util.GcloudStorageMap( gcloud_command=['fake'], flag_map={ '-z': shim_util.GcloudStorageFlag('-a') # Mapping for -r is missing. }) with self.assertRaisesRegex( exception.GcloudStorageTranslationError, 'Command option "-r" cannot be translated to gcloud storage'): self._fake_command.get_gcloud_storage_args() def test_raises_error_if_sub_command_mapping_is_missing(self): fake_with_subcommand = FakeCommandWithSubCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['set', '-y', 'opt1', '-a', 'arg1', 'arg2'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) fake_with_subcommand.gcloud_storage_map = shim_util.GcloudStorageMap( gcloud_command={}, # Missing mapping for set. flag_map={}) with self.assertRaisesRegex( exception.GcloudStorageTranslationError, 'Command "fake_with_sub" cannot be translated to gcloud storage' ' because the translation mapping is missing.'): fake_with_subcommand.get_gcloud_storage_args() def test_raises_error_if_flags_mapping_at_top_level_for_subcommand(self): fake_with_subcommand = FakeCommandWithSubCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['set', '-y', 'opt1', '-a', 'arg1', 'arg2'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) fake_with_subcommand.gcloud_storage_map.flag_map = {'a': 'b'} with self.assertRaisesRegex( ValueError, 'Flags mapping should not be present at the top-level command' ' if a sub-command is used. Command: fake_with_sub'): fake_with_subcommand.get_gcloud_storage_args() class TestTranslateToGcloudStorageIfRequested(testcase.ShimUnitTestBase): """Test Command.translate_to_gcloud_storage_if_requested method.""" def setUp(self): super().setUp() self._fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-z', 'opt1', '-r', 'arg1', 'arg2'], headers={}, debug=0, trace_token=None, parallel_operations=True, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) def test_gets_gcloud_binary_path_on_non_windows(self): with mock.patch.object(system_util, 'IS_WINDOWS', new=False): self.assertEqual(shim_util._get_gcloud_binary_path('fake_root'), os.path.join('fake_root', 'bin', 'gcloud')) def test_gets_gcloud_binary_path_on_windows(self): with mock.patch.object(system_util, 'IS_WINDOWS', new=True): self.assertEqual(shim_util._get_gcloud_binary_path('fake_root'), os.path.join('fake_root', 'bin', 'gcloud.cmd')) def test_returns_false_with_use_gcloud_storage_never(self): """Should not attempt translation.""" with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'False')]): with mock.patch.object(self._fake_command, 'get_gcloud_storage_args', autospec=True) as mock_get_gcloud_storage_args: self.assertFalse( self._fake_command.translate_to_gcloud_storage_if_requested()) self.assertFalse(mock_get_gcloud_storage_args.called) def test_returns_true_with_valid_gcloud_storage_map(self): """Should return True and perform the translation.""" with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self.assertTrue( self._fake_command.translate_to_gcloud_storage_if_requested()) # Verify translation. expected_gcloud_path = shim_util._get_gcloud_binary_path('fake_dir') self.assertEqual(self._fake_command._translated_gcloud_storage_command, [ expected_gcloud_path, 'objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2' ]) def test_with_cloudsdk_root_dir_unset_and_gcloud_binary_path_set(self): """Should return True and perform the translation.""" gcloud_path = shim_util._get_gcloud_binary_path('fake_dir') with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': None, 'GCLOUD_BINARY_PATH': gcloud_path, }): self.assertTrue( self._fake_command.translate_to_gcloud_storage_if_requested()) # Verify translation. self.assertEqual(self._fake_command._translated_gcloud_storage_command, [ gcloud_path, 'objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2' ]) def test_returns_false_if_invalid_use_gcloud_storage_value(self): with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'invalid') ]): self.assertFalse( self._fake_command.translate_to_gcloud_storage_if_requested()) def test_raises_error_if_invalid_hidden_shim_mode_value(self): with util.SetBotoConfigForTest([('GSUtil', 'hidden_shim_mode', 'invalid')]): with self.assertRaisesRegex( exception.CommandException, 'CommandException: Invalid option specified for' ' GSUtil:use_gcloud_storage config setting. Should be one of:' ' no_fallback | dry_run | none'): self._fake_command.translate_to_gcloud_storage_if_requested() def test_raises_error_if_cloudsdk_root_dir_is_none(self): with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_ROOT_DIR': None, }): with self.assertRaisesRegex( exception.CommandException, 'CommandException: Requested to use "gcloud storage" but the ' 'gcloud binary path cannot be found. This might happen if you' ' attempt to use gsutil that was not installed via Cloud SDK.' ' You can manually set the `CLOUDSDK_ROOT_DIR` environment variable' ' to point to the google-cloud-sdk installation directory to' ' resolve the issue. Alternatively, you can set' ' `use_gcloud_storage=False` to disable running the command' ' using gcloud storage.'): self._fake_command.translate_to_gcloud_storage_if_requested() def test_raises_error_if_pass_credentials_to_gsutil_is_missing(self): error_regex = ( r'CommandException: Requested to use "gcloud storage" but gsutil' r' is not using the same credentials as' r' gcloud. You can make gsutil use the same credentials' r' by running:\n' r'{} config set pass_credentials_to_gsutil True').format( re.escape(shim_util._get_gcloud_binary_path('fake_dir'))) with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_ROOT_DIR': 'fake_dir', 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': None }): with self.assertRaisesRegex(exception.CommandException, error_regex): self._fake_command.translate_to_gcloud_storage_if_requested() @mock.patch.object(boto_util, 'UsingGsHmac', return_value=True) def test_raises_error_if_using_gs_hmac_without_xml_support(self, _): with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_ROOT_DIR': 'fake_dir', 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', }): self._fake_command.command_spec = command.Command.CreateCommandSpec( 'fake_shim', gs_api_support=[ApiSelector.JSON]) with self.assertRaisesRegex( exception.CommandException, 'CommandException: Requested to use "gcloud storage" with Cloud' ' Storage XML API HMAC credentials but the "fake_shim" command can' ' only be used with the Cloud Storage JSON API.'): self._fake_command.translate_to_gcloud_storage_if_requested() def test_raises_error_if_gcloud_storage_map_missing(self): self._fake_command.gcloud_storage_map = None with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): with self.assertRaisesRegex( exception.CommandException, 'CommandException: Command "fake_shim" cannot be translated to' ' gcloud storage because the translation mapping is missing.'): self._fake_command.translate_to_gcloud_storage_if_requested() def test_use_gcloud_storage_true_with_hidden_shim_mode_not_set(self): """Should not raise error.""" with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', None)]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): # return_stderr does not work here. Probably because we have # defined the FakeCommand in the same module. stdout, mock_log_handler = self.RunCommand('fake_shim', args=['-i', 'arg1'], return_stdout=True, return_log_handler=True) self.assertIn( 'Cannot translate gsutil command to gcloud storage.' ' Going to run gsutil command. Error: Command option "-i"' ' cannot be translated to gcloud storage', mock_log_handler.messages['error']) self.assertIn('FakeCommandWithGcloudStorageMap called', stdout) def test_dry_run_mode_prints_translated_command(self): """Should print the gcloud command and run gsutil.""" with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'True', 'hidden_shim_mode': 'dry_run' } }): with util.SetEnvironmentForTest({'CLOUDSDK_ROOT_DIR': 'fake_dir'}): stdout, mock_log_handler = self.RunCommand('fake_shim', args=['arg1'], return_stdout=True, return_log_handler=True) self.assertIn( 'Gcloud Storage Command: {} objects fake arg1'.format( shim_util._get_gcloud_binary_path('fake_dir')), mock_log_handler.messages['info']) self.assertIn( 'FakeCommandWithGcloudStorageMap called'.format( shim_util._get_gcloud_binary_path('fake_dir')), stdout) def test_non_dry_mode_logs_translated_command_to_debug_logs(self): with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): with mock.patch.object(self._fake_command, 'logger', autospec=True) as mock_logger: self._fake_command.translate_to_gcloud_storage_if_requested() # Verify translation. mock_logger.debug.assert_has_calls([ mock.call('Gcloud Storage Command: {} objects' ' fake --zip opt1 -x arg1 arg2'.format( shim_util._get_gcloud_binary_path('fake_dir'))), mock.call('Environment variables for Gcloud Storage:'), mock.call('%s=%s', 'CLOUDSDK_METRICS_ENVIRONMENT', 'gsutil_shim'), mock.call('%s=%s', 'CLOUDSDK_STORAGE_RUN_BY_GSUTIL_SHIM', 'True') ], any_order=True) def test_print_gcloud_storage_env_vars_in_dry_run_mode(self): """Should log the command and env vars to logger.info""" with mock.patch.object(self._fake_command, 'logger', autospec=True) as mock_logger: self._fake_command._print_gcloud_storage_command_info( ['fake', 'gcloud', 'command'], {'fake_env_var': 'val'}, dry_run=True) expected_calls = [ mock.call('Gcloud Storage Command: fake gcloud command'), mock.call('Environment variables for Gcloud Storage:'), mock.call('%s=%s', 'fake_env_var', 'val'), ] self.assertEqual(mock_logger.info.mock_calls, expected_calls) def test_top_level_flags_get_translated(self): """Should return True and perform the translation.""" boto_config = { 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } } with _mock_boto_config(boto_config): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['arg1', 'arg2'], headers={}, # Headers will be tested separately. debug=3, # -D option trace_token='fake_trace_token', user_project='fake_user_project', parallel_operations=False, # Without the -m option. bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) self.assertTrue(fake_command.translate_to_gcloud_storage_if_requested()) # Verify translation. expected_gcloud_path = shim_util._get_gcloud_binary_path('fake_dir') self.assertEqual(fake_command._translated_gcloud_storage_command, [ expected_gcloud_path, 'objects', 'fake', 'arg1', 'arg2', '--verbosity', 'debug', '--billing-project=fake_user_project', '--trace-token=fake_trace_token' ]) self.assertCountEqual( fake_command._translated_env_variables, { 'CLOUDSDK_STORAGE_PROCESS_COUNT': '1', 'CLOUDSDK_STORAGE_THREAD_COUNT': '1', 'CLOUDSDK_METRICS_ENVIRONMENT': 'gsutil_shim', 'CLOUDSDK_STORAGE_RUN_BY_GSUTIL_SHIM': 'True', }) def test_anonymous_credentials_sets_disable_credentials_env_variable(self): self._mock_subprocess_run.return_value.stdout = '' boto_config = { 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } } with _mock_boto_config(boto_config): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self.assertTrue( self._fake_command.translate_to_gcloud_storage_if_requested()) # Verify translation. expected_gcloud_path = shim_util._get_gcloud_binary_path('fake_dir') self.assertCountEqual( self._fake_command._translated_env_variables, { 'CLOUDSDK_METRICS_ENVIRONMENT': 'gsutil_shim', 'CLOUDSDK_STORAGE_RUN_BY_GSUTIL_SHIM': 'True', 'CLOUDSDK_AUTH_DISABLE_CREDENTIALS': 'True', }) self._mock_subprocess_run.assert_called_once_with( [expected_gcloud_path, 'config', 'get', 'account'], stdout=-1, stderr=-1, encoding='utf-8') def test_gcloud_config_get_account_nonzero_returncode_raises_error(self): self._mock_subprocess_run.return_value.stdout = b'' self._mock_subprocess_run.return_value.stderr = b'fake error message' self._mock_subprocess_run.return_value.returncode = 1 expected_gcloud_path = shim_util._get_gcloud_binary_path('fake_dir') boto_config = { 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } } with _mock_boto_config(boto_config): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): with self.assertRaisesRegex( exception.CommandException, "Error occurred while trying to retrieve gcloud's active account." " Error: b'fake error message"): self._fake_command.translate_to_gcloud_storage_if_requested() self._mock_subprocess_run.assert_called_once_with( [expected_gcloud_path, 'config', 'get', 'account'], stdout=-1, stderr=-1, encoding='utf-8', ) def test_parallel_operations_true_does_not_add_process_count_env_vars(self): with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self._fake_command.parallel_operations = True self._fake_command.translate_to_gcloud_storage_if_requested() self.assertNotIn('CLOUDSDK_STORAGE_PROCESS_COUNT', self._fake_command._translated_env_variables) self.assertNotIn('CLOUDSDK_STORAGE_THREAD_COUNT', self._fake_command._translated_env_variables) def test_parallel_operations_false_but_parallelism_turned_on_for_rsync(self): command = rsync.RsyncCommand(command_runner=mock.ANY, args=['arg1', 'arg2'], headers={}, debug=0, trace_token=None, parallel_operations=False, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): command.translate_to_gcloud_storage_if_requested() self.assertNotIn('CLOUDSDK_STORAGE_PROCESS_COUNT', command._translated_env_variables) self.assertNotIn('CLOUDSDK_STORAGE_THREAD_COUNT', command._translated_env_variables) def test_parallelism_turned_on_for_rsync_unless_boto_set_sequential(self): with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback'), ('GSUtil', 'parallel_process_count', '1'), ('GSUtil', 'thread_process_count', '1')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): command = rsync.RsyncCommand( command_runner=mock.ANY, args=['arg1', 'arg2'], headers={}, debug=0, trace_token=None, parallel_operations=False, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) command.translate_to_gcloud_storage_if_requested() self.assertEqual( command._translated_env_variables['CLOUDSDK_STORAGE_PROCESS_COUNT'], '1') self.assertEqual( command._translated_env_variables['CLOUDSDK_STORAGE_THREAD_COUNT'], '1') def test_debug_value_4_adds_log_http_flag(self): # Debug level 4 represents the -DD option. with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self._fake_command.debug = 4 self._fake_command.translate_to_gcloud_storage_if_requested() self.assertEqual(self._fake_command._translated_gcloud_storage_command, [ shim_util._get_gcloud_binary_path('fake_dir'), 'objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2', '--verbosity', 'debug', '--log-http' ]) @mock.patch.object(constants, 'IMPERSONATE_SERVICE_ACCOUNT', new='fake_service_account') def test_impersonate_service_account_translation(self): """Should add the --impersonate-service-account flag.""" with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self._fake_command.translate_to_gcloud_storage_if_requested() self.assertEqual( self._fake_command._translated_gcloud_storage_command, [ shim_util._get_gcloud_binary_path('fake_dir'), 'objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2', '--impersonate-service-account=fake_service_account' ]) def test_quiet_mode_translation_adds_no_user_output_enabled_flag(self): with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self._fake_command.quiet_mode = True self._fake_command.translate_to_gcloud_storage_if_requested() self.assertEqual(self._fake_command._translated_gcloud_storage_command, [ shim_util._get_gcloud_binary_path('fake_dir'), 'objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2', '--no-user-output-enabled' ]) def test_returns_false_for_version_command(self): command = version.VersionCommand( command_runner=mock.ANY, args=[], headers={}, debug=0, trace_token=None, parallel_operations=True, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with mock.patch.object(command, 'get_gcloud_storage_args', autospec=True) as mock_get_gcloud_storage_args: self.assertFalse(command.translate_to_gcloud_storage_if_requested()) self.assertFalse(mock_get_gcloud_storage_args.called) def test_returns_false_for_test_command(self): command = test.TestCommand(command_runner=mock.ANY, args=[], headers={}, debug=0, trace_token=None, parallel_operations=True, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with mock.patch.object(command, 'get_gcloud_storage_args', autospec=True) as mock_get_gcloud_storage_args: self.assertFalse(command.translate_to_gcloud_storage_if_requested()) self.assertFalse(mock_get_gcloud_storage_args.called) class TestHeaderTranslation(testcase.GsUtilUnitTestCase): """Test gsutil header translation.""" def setUp(self): super().setUp() self._fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-z', 'opt1', '-r', 'arg1', 'arg2'], headers={}, debug=0, trace_token=None, parallel_operations=True, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) @mock.patch.object(subprocess, 'run', autospec=True) def test_translated_headers_get_added_to_final_command( self, mock_subprocess_run): mock_subprocess_run.return_value.returncode = 0 with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'always', 'hidden_shim_mode': 'no_fallback' } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['arg1', 'arg2'], headers={'Content-Type': 'fake_val'}, debug=1, trace_token=None, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) self.assertTrue(fake_command.translate_to_gcloud_storage_if_requested()) self.assertEqual(fake_command._translated_gcloud_storage_command, [ shim_util._get_gcloud_binary_path('fake_dir'), 'objects', 'fake', 'arg1', 'arg2', '--content-type=fake_val' ]) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) def test_translate_headers_returns_correct_flags_for_data_transfer_command( self): self._fake_command.headers = { # Data tranfer related headers. 'Cache-Control': 'fake_Cache_Control', 'Content-Disposition': 'fake_Content-Disposition', 'Content-Encoding': 'fake_Content-Encoding', 'Content-Language': 'fake_Content-Language', 'Content-Type': 'fake_Content-Type', 'Content-MD5': 'fake_Content-MD5', 'custom-time': 'fake_time', # Precondition headers. 'x-goog-if-generation-match': 'fake_gen_match', 'x-goog-if-metageneration-match': 'fake_metagen_match', # Custom metadata. 'x-goog-meta-cAsE': 'sEnSeTiVe', 'x-goog-meta-gfoo': 'fake_goog_meta', 'x-amz-meta-afoo': 'fake_amz_meta', 'x-amz-afoo': 'fake_amz_custom_header', } flags = self._fake_command._translate_headers() self.assertCountEqual(flags, [ '--cache-control=fake_Cache_Control', '--content-disposition=fake_Content-Disposition', '--content-encoding=fake_Content-Encoding', '--content-language=fake_Content-Language', '--content-type=fake_Content-Type', '--content-md5=fake_Content-MD5', '--custom-time=fake_time', '--if-generation-match=fake_gen_match', '--if-metageneration-match=fake_metagen_match', '--update-custom-metadata=cAsE=sEnSeTiVe', '--update-custom-metadata=gfoo=fake_goog_meta', '--update-custom-metadata=afoo=fake_amz_meta', '--additional-headers=x-amz-afoo=fake_amz_custom_header', ]) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) def test_translate_custom_headers_returns_correct_flags(self): flags = self._fake_command._translate_headers( {'Cache-Control': 'fake_Cache_Control'}) self.assertCountEqual(flags, ['--cache-control=fake_Cache_Control']) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) def test_translate_custom_headers_handles_multiple_additional_headers(self): flags = self._fake_command._translate_headers( collections.OrderedDict([('header1', 'value1'), ('header2', 'value2')])) self.assertCountEqual( flags, ['--additional-headers=header1=value1,header2=value2']) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) def test_translate_clear_headers_returns_correct_flags(self): flags = self._fake_command._translate_headers( {'Cache-Control': 'fake_Cache_Control'}, unset=True) self.assertCountEqual(flags, ['--clear-cache-control']) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) def test_translate_headers_for_data_transfer_command_with_additional_header( self): """Should log a warning.""" self._fake_command.headers = {'additional': 'header'} with mock.patch.object(self._fake_command.logger, 'warn', autospec=True) as mock_warning: self.assertEqual(self._fake_command._translate_headers(), ['--additional-headers=additional=header']) mock_warning.assert_called_once_with( 'Header additional:header cannot be translated to a gcloud' ' storage equivalent flag. It is being treated as an arbitrary' ' request header.') @mock.patch.object(shim_util, 'PRECONDITONS_ONLY_SUPPORTED_COMMANDS', new={'fake_shim'}) def test_translate_valid_headers_for_precondition_supported_command(self): self._fake_command.headers = { 'x-goog-if-generation-match': 'fake_gen_match', 'x-goog-if-metageneration-match': 'fake_metagen_match', # Custom metadata. These should be ignored. 'x-goog-meta-foo': 'fake_goog_meta', } flags = self._fake_command._translate_headers() self.assertCountEqual(flags, [ '--if-generation-match=fake_gen_match', '--if-metageneration-match=fake_metagen_match', ]) @mock.patch.object(shim_util, 'PRECONDITONS_ONLY_SUPPORTED_COMMANDS', new={'fake_shim'}) def test_translate_short_headers_for_precondition_supported_command(self): self._fake_command.headers = { 'x-goog-generation-match': 'fake_gen_match', 'x-goog-metageneration-match': 'fake_metagen_match', # Custom metadata. These should be ignored. 'x-goog-meta-foo': 'fake_goog_meta', } flags = self._fake_command._translate_headers() self.assertCountEqual(flags, [ '--if-generation-match=fake_gen_match', '--if-metageneration-match=fake_metagen_match', ]) @mock.patch.object(shim_util, 'PRECONDITONS_ONLY_SUPPORTED_COMMANDS', new={'fake_shim'}) def test_translate_headers_for_precondition_supported_command_with_additional_header( self): """Should be ignored and not raise any error.""" self._fake_command.headers = {'additional': 'header'} with mock.patch.object(self._fake_command.logger, 'warn', autospec=True) as mock_warning: self.assertEqual(self._fake_command._translate_headers(), ['--additional-headers=additional=header']) mock_warning.assert_called_once_with( 'Header additional:header cannot be translated to a gcloud' ' storage equivalent flag. It is being treated as an arbitrary' ' request header.') def test_translate_headers_only_uses_additional_headers_for_commands_not_in_allowlist( self): # Allowlist is defined by the shim_util.COMMANDS_SUPPORTING_ALL_HEADERS and # the shim_util.PRECONDITONS_ONLY_SUPPORTED_COMMANDS list. # The fake_shim command defined by self._fake_command is not part of # either of the list, and hence the headers must be ignored. self._fake_command.headers = { # Header from the DATA_TRANSFER_HEADERS list. 'Cache-Control': 'fake_Cache_Control', # Header from the PRECONDITIONS_HEADERS list. 'x-goog-if-generation-match': 'fake_gen_match', # Custom metadata. These should be ignored. 'x-goog-meta-foo': 'fake_goog_meta', # Additional header. Should be added. 'additional': 'header' } self.assertEqual(self._fake_command._translate_headers(), ['--additional-headers=additional=header']) @mock.patch.object(shim_util, 'PRECONDITONS_ONLY_SUPPORTED_COMMANDS', new={'fake_shim'}) def test_translate_headers_ignores_x_goog_api_version_header(self): self._fake_command.headers = { 'x-goog-if-generation-match': 'fake_gen_match', 'x-goog-api-version': '2', # Should be ignored. } self.assertEqual(self._fake_command._translate_headers(), ['--if-generation-match=fake_gen_match']) class TestGetFlagFromHeader(testcase.GsUtilUnitTestCase): """Test Command.get_flag_from_header function. We only test the unset functionality because rest of the workflows have been already tested indirectly in TestHeaderTranslation. """ def test_get_flag_from_header_with_unset_true_for_data_transfer_headers(self): # Ideally we should use parameterized, but avoiding adding a new dependency # just for the shim. headers_to_expected_flag_map = { 'Cache-Control': '--clear-cache-control', 'Content-Disposition': '--clear-content-disposition', 'Content-Encoding': '--clear-content-encoding', 'Content-Language': '--clear-content-language', 'Content-Type': '--clear-content-type', 'custom-time': '--clear-custom-time', } for header, expected_flag in headers_to_expected_flag_map.items(): result = shim_util.get_flag_from_header(header, 'fake_val', unset=True) self.assertEqual(result, expected_flag) def test_get_flag_from_header_with_unset_true_for_precondition_headers(self): """Should return None.""" # Ideally we should use parameterized, but avoiding adding a new dependency # just for the shim. for header in [ 'x-goog-if-generation-match', 'x-goog-if-metageneration-match', ]: result = shim_util.get_flag_from_header(header, 'fake_val', unset=True) self.assertIsNone(result) def test_get_flag_from_header_with_unset_true_for_content_md5(self): """Should return None.""" result = shim_util.get_flag_from_header('Content-MD5', 'fake_val', unset=True) self.assertIsNone(result) def test_get_flag_from_header_with_unset_true_for_invalid_header(self): """Should return None.""" result = shim_util.get_flag_from_header('invalid_header', 'fake_val', unset=True) self.assertIsNone(result) def test_get_flag_from_header_with_unset_true_for_metadata_headers(self): """Should return --remove-custom-metadata flag.""" headers_to_expected_flag_map = { 'x-goog-meta-foo': '--remove-custom-metadata=foo', 'x-amz-meta-foo': '--remove-custom-metadata=foo', } for header, expected_flag in headers_to_expected_flag_map.items(): result = shim_util.get_flag_from_header(header, 'fake_val', unset=True) self.assertEqual(result, expected_flag) class TestFormatFlagUtils(testcase.GsUtilUnitTestCase): """Test utils used for generating gcloud --format flags.""" def test_gets_format_flag_caret_on_non_windows(self): with mock.patch.object(system_util, 'IS_WINDOWS', new=False): self.assertEqual(shim_util.get_format_flag_caret(), '^') def test_gets_format_flag_escaped_caret_on_windows(self): with mock.patch.object(system_util, 'IS_WINDOWS', new=True): self.assertEqual(shim_util.get_format_flag_caret(), '^^^^') def test_gets_format_flag_newline_on_non_windows(self): with mock.patch.object(system_util, 'IS_WINDOWS', new=False): self.assertEqual(shim_util.get_format_flag_newline(), '\n') def test_gets_format_flag_escaped_newline_on_windows(self): with mock.patch.object(system_util, 'IS_WINDOWS', new=True): self.assertEqual(shim_util.get_format_flag_newline(), r'^\^n') class TestBotoTranslation(testcase.GsUtilUnitTestCase): """Test gsutil header translation.""" def setUp(self): super().setUp() self._fake_command = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-z', 'opt1', '-r', 'arg1', 'arg2'], headers={}, debug=0, trace_token=None, parallel_operations=True, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) @mock.patch.object(shim_util, 'COMMANDS_SUPPORTING_ALL_HEADERS', new={'fake_shim'}) @mock.patch.object(subprocess, 'run', autospec=True) def test_translated_boto_config_gets_added(self, mock_subprocess_run): """Should add translated env vars as well flags.""" mock_subprocess_run.return_value.returncode = 0 with _mock_boto_config({ 'GSUtil': { 'use_gcloud_storage': 'True', 'hidden_shim_mode': 'no_fallback', 'content_language': 'foo', 'default_project_id': 'fake_project', } }): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): self.assertTrue( self._fake_command.translate_to_gcloud_storage_if_requested()) # Verify translation. expected_gcloud_path = shim_util._get_gcloud_binary_path('fake_dir') self.assertEqual( self._fake_command._translated_gcloud_storage_command, [ expected_gcloud_path, 'objects', 'fake', '--zip', 'opt1', '-x', 'arg1', 'arg2', '--content-language=foo' ]) self.assertEqual( self._fake_command._translated_env_variables, { 'CLOUDSDK_CORE_PROJECT': 'fake_project', 'CLOUDSDK_METRICS_ENVIRONMENT': 'gsutil_shim', 'CLOUDSDK_STORAGE_RUN_BY_GSUTIL_SHIM': 'True' }) def test_gcs_json_endpoint_translation(self): with _mock_boto_config({ 'Credentials': { 'gs_json_host': 'foo_host', 'gs_json_port': '1234', 'json_api_version': 'v2', } }): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual( env_vars, { 'CLOUDSDK_API_ENDPOINT_OVERRIDES_STORAGE': 'https://foo_host:1234/storage/v2', }) def test_gcs_json_endpoint_translation_with_missing_port(self): with _mock_boto_config({ 'Credentials': { 'gs_json_host': 'foo_host', 'json_api_version': 'v2', } }): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual(env_vars, { 'CLOUDSDK_API_ENDPOINT_OVERRIDES_STORAGE': 'https://foo_host/storage/v2', }) def test_gcs_json_endpoint_translation_usees_default_version_v1(self): with _mock_boto_config( {'Credentials': { 'gs_json_host': 'foo_host', 'gs_json_port': '1234', }}): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual( env_vars, { 'CLOUDSDK_API_ENDPOINT_OVERRIDES_STORAGE': 'https://foo_host:1234/storage/v1' }) def test_s3_endpoint_translation(self): with _mock_boto_config( {'Credentials': { 's3_host': 's3_host', 's3_port': '1234', }}): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual( env_vars, {'CLOUDSDK_STORAGE_S3_ENDPOINT_URL': 'https://s3_host:1234'}) def test_s3_endpoint_translation_with_missing_port(self): with _mock_boto_config({'Credentials': {'s3_host': 's3_host',}}): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual(env_vars, {'CLOUDSDK_STORAGE_S3_ENDPOINT_URL': 'https://s3_host'}) @mock.patch.object(shim_util, 'ENCRYPTION_SUPPORTED_COMMANDS', new={'fake_shim'}) def test_encryption_key_gets_converted_to_flag(self): with _mock_boto_config({'GSUtil': {'encryption_key': 'fake_key'}}): flags, _ = self._fake_command._translate_boto_config() self.assertEqual(flags, ['--encryption-key=fake_key']) @mock.patch.object(shim_util, 'ENCRYPTION_SUPPORTED_COMMANDS', new={'fake_shim'}) def test_decryption_key_gets_converted_to_flag(self): with _mock_boto_config({ 'GSUtil': collections.OrderedDict([ ('decryption_key1', 'key1'), ('decryption_key12', 'key12'), ('decryption_key100', 'key100'), ]) }): flags, _ = self._fake_command._translate_boto_config() self.assertEqual(flags, ['--decryption-keys=key1,key12,key100']) @mock.patch.object(boto_util, 'UsingGsHmac', return_value=False) def test_gs_hmac_auth_env_when_not_using_gs_hmac(self, mock_using_gs_hmac): with _mock_boto_config({ 'Credentials': { 'gs_access_key_id': 'foo', 'gs_secret_access_key': 'bar', } }): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(mock_using_gs_hmac.call_count, 2) self.assertEqual(flags, []) self.assertEqual(env_vars, {}) @mock.patch.object(boto_util, 'UsingGsHmac', return_value=True) def test_boto_config_translation_for_supported_fields(self, _): with _mock_boto_config({ 'Credentials': { 'aws_access_key_id': 'AWS_ACCESS_KEY_ID_value', 'aws_secret_access_key': 'AWS_SECRET_ACCESS_KEY_value', 'gs_access_key_id': 'CLOUDSDK_STORAGE_GS_XML_ACCESS_KEY_ID_value', 'gs_secret_access_key': 'CLOUDSDK_STORAGE_GS_XML_SECRET_ACCESS_KEY_value', 'use_client_certificate': True, }, 'Boto': { 'proxy': 'CLOUDSDK_PROXY_ADDRESS_value', 'proxy_type': 'CLOUDSDK_PROXY_TYPE_value', 'proxy_port': 'CLOUDSDK_PROXY_PORT_value', 'proxy_user': 'CLOUDSDK_PROXY_USERNAME_value', 'proxy_pass': 'CLOUDSDK_PROXY_PASSWORD_value', 'proxy_rdns': 'CLOUDSDK_PROXY_RDNS_value', 'http_socket_timeout': 'HTTP_TIMEOUT_value', 'ca_certificates_file': 'CA_CERTS_FILE_value', 'https_validate_certificates': False, 'max_retry_delay': 'BASE_RETRY_DELAY_value', 'num_retries': 'MAX_RETRIES_value', }, 'GSUtil': { 'check_hashes': 'CHECK_HASHES_value', 'default_project_id': 'CLOUDSDK_CORE_PROJECT_value', 'disable_analytics_prompt': 'USAGE_REPORTING_value', 'use_magicfile': 'USE_MAGICFILE_value', 'parallel_composite_upload_threshold': '100M', 'resumable_threshold': '256K', 'rsync_buffer_lines': '32000', }, 'OAuth2': { 'client_id': 'CLOUDSDK_AUTH_CLIENT_ID_value', 'client_secret': 'AUTH_CLIENT_SECRET_value', 'provider_authorization_uri': 'CLOUDSDK_AUTH_AUTH_HOST_value', 'provider_token_uri': 'CLOUDSDK_AUTH_TOKEN_HOST_value', }, }): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.maxDiff = None self.assertDictEqual( env_vars, { 'AWS_ACCESS_KEY_ID': 'AWS_ACCESS_KEY_ID_value', 'AWS_SECRET_ACCESS_KEY': 'AWS_SECRET_ACCESS_KEY_value', 'CLOUDSDK_CONTEXT_AWARE_USE_CLIENT_CERTIFICATE': True, 'CLOUDSDK_STORAGE_GS_XML_ACCESS_KEY_ID': 'CLOUDSDK_STORAGE_GS_XML_ACCESS_KEY_ID_value', 'CLOUDSDK_STORAGE_GS_XML_SECRET_ACCESS_KEY': 'CLOUDSDK_STORAGE_GS_XML_SECRET_ACCESS_KEY_value', 'CLOUDSDK_PROXY_ADDRESS': 'CLOUDSDK_PROXY_ADDRESS_value', 'CLOUDSDK_PROXY_ADDRESS': 'CLOUDSDK_PROXY_ADDRESS_value', 'CLOUDSDK_PROXY_TYPE': 'CLOUDSDK_PROXY_TYPE_value', 'CLOUDSDK_PROXY_PORT': 'CLOUDSDK_PROXY_PORT_value', 'CLOUDSDK_PROXY_USERNAME': 'CLOUDSDK_PROXY_USERNAME_value', 'CLOUDSDK_PROXY_PASSWORD': 'CLOUDSDK_PROXY_PASSWORD_value', 'CLOUDSDK_PROXY_RDNS': 'CLOUDSDK_PROXY_RDNS_value', 'CLOUDSDK_CORE_HTTP_TIMEOUT': 'HTTP_TIMEOUT_value', 'CLOUDSDK_CORE_CUSTOM_CA_CERTS_FILE': 'CA_CERTS_FILE_value', 'CLOUDSDK_AUTH_DISABLE_SSL_VALIDATION': True, 'CLOUDSDK_STORAGE_BASE_RETRY_DELAY': 'BASE_RETRY_DELAY_value', 'CLOUDSDK_STORAGE_MAX_RETRIES': 'MAX_RETRIES_value', 'CLOUDSDK_STORAGE_CHECK_HASHES': 'CHECK_HASHES_value', 'CLOUDSDK_CORE_PROJECT': 'CLOUDSDK_CORE_PROJECT_value', 'CLOUDSDK_CORE_DISABLE_USAGE_REPORTING': 'USAGE_REPORTING_value', 'CLOUDSDK_STORAGE_USE_MAGICFILE': 'USE_MAGICFILE_value', 'CLOUDSDK_STORAGE_PARALLEL_COMPOSITE_UPLOAD_THRESHOLD': '100M', 'CLOUDSDK_STORAGE_RESUMABLE_THRESHOLD': '256K', 'CLOUDSDK_STORAGE_RSYNC_LIST_CHUNK_SIZE': '32000', 'CLOUDSDK_AUTH_CLIENT_ID': 'CLOUDSDK_AUTH_CLIENT_ID_value', 'CLOUDSDK_AUTH_CLIENT_SECRET': 'AUTH_CLIENT_SECRET_value', 'CLOUDSDK_AUTH_AUTH_HOST': 'CLOUDSDK_AUTH_AUTH_HOST_value', 'CLOUDSDK_AUTH_TOKEN_HOST': 'CLOUDSDK_AUTH_TOKEN_HOST_value', }) def test_missing_mappging_gets_ignored(self): with _mock_boto_config({'GSUtil': {'unsupported_field': 'foo'}}): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual(env_vars, {}) def test_truthy_https_validate_certificates(self): """Should not set CLOUDSDK_AUTH_DISABLE_SSL_VALIDATION.""" with _mock_boto_config({'GSUtil': {'https_validate_certificates': True}}): flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual(env_vars, {}) def test_missing_required_fields_log_error(self): with _mock_boto_config({ 'GSUtil': { 'stet_binary_path': 'foo_bin', 'stet_config_path': 'foo_config' } }): with mock.patch.object(self._fake_command.logger, 'error', autospec=True) as mock_logger: flags, env_vars = self._fake_command._translate_boto_config() self.assertEqual(flags, []) self.assertEqual(env_vars, {}) self.assertCountEqual(mock_logger.mock_calls, [ mock.call('The boto config field GSUtil:stet_binary_path cannot be' ' translated to gcloud storage equivalent.'), mock.call('The boto config field GSUtil:stet_config_path cannot be' ' translated to gcloud storage equivalent.') ]) class TestRunGcloudStorage(testcase.GsUtilUnitTestCase): """Test Command.run_gcloud_storage method.""" @mock.patch.object(os.environ, 'copy', return_value={'old_key': 'old_value'}) @mock.patch.object(subprocess, 'run', autospec=True) def test_calls_subprocess_with_translated_command_and_env_vars( self, mock_run, mock_environ_copy): command_instance = FakeCommandWithGcloudStorageMap( command_runner=mock.ANY, args=['-z', 'opt1', '-r', 'arg1', 'arg2'], headers={}, debug=mock.ANY, trace_token=mock.ANY, parallel_operations=mock.ANY, bucket_storage_uri_class=mock.ANY, gsutil_api_class_map_factory=mock.MagicMock()) with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': 'fake_dir', }): command_instance._translated_env_variables = { 'new_key': 'new_value', } command_instance._translated_gcloud_storage_command = ['gcloud', 'foo'] actual_return_code = command_instance.run_gcloud_storage() mock_run.assert_called_once_with(['gcloud', 'foo'], env={ 'old_key': 'old_value', 'new_key': 'new_value' }) mock_environ_copy.assert_called_once_with() self.assertEqual(actual_return_code, mock_run.return_value.returncode) class TestShimE2E(testcase.GsUtilIntegrationTestCase): def test_runs_gcloud_storage_if_use_gcloud_storage_true(self): with util.SetBotoConfigForTest([('GSUtil', 'use_gcloud_storage', 'True'), ('GSUtil', 'hidden_shim_mode', 'no_fallback')]): with util.SetEnvironmentForTest({ 'CLOUDSDK_CORE_PASS_CREDENTIALS_TO_GSUTIL': 'True', 'CLOUDSDK_ROOT_DIR': None, }): stderr = self.RunGsUtil(['-D', 'ls'], return_stderr=True, expected_status=1) # This is a proxy to ensure that the test attempted to call # gcloud binary. self.assertIn('gcloud binary path cannot be found', stderr)
| ver. 1.4 |
Github
|
.
| PHP 8.2.28 | Generation time: 0.02 |
proxy
|
phpinfo
|
Settings