# -*- 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. """Implementation of objects describe command for getting info on objects.""" from __future__ import absolute_import from __future__ import division from __future__ import unicode_literals from googlecloudsdk.api_lib.storage import api_factory from googlecloudsdk.api_lib.storage import cloud_api from googlecloudsdk.api_lib.storage import request_config_factory from googlecloudsdk.calliope import base from googlecloudsdk.command_lib.storage import encryption_util from googlecloudsdk.command_lib.storage import errors from googlecloudsdk.command_lib.storage import errors_util from googlecloudsdk.command_lib.storage import flags from googlecloudsdk.command_lib.storage import storage_url from googlecloudsdk.command_lib.storage import wildcard_iterator from googlecloudsdk.command_lib.storage.resources import contexts_only_formatter from googlecloudsdk.command_lib.storage.resources import full_resource_formatter from googlecloudsdk.command_lib.storage.resources import gsutil_json_printer from googlecloudsdk.command_lib.storage.resources import resource_util _COMMAND_DESCRIPTION = """ Describe a Cloud Storage object. """ _GA_EXAMPLES = """ Describe a Google Cloud Storage object with the url "gs://bucket/my-object": $ {command} gs://bucket/my-object Describe object with JSON formatting, only returning the "name" key: $ {command} gs://bucket/my-object --format="json(name)" """ _ALPHA_EXAMPLES = """ Describe only contexts attached to objects as key value pairs. $ {command} gs://my-bucket/object --format=contextsonly """ @base.UniverseCompatible @base.ReleaseTracks(base.ReleaseTrack.GA) class Describe(base.DescribeCommand): """Describe a Cloud Storage object.""" detailed_help = { 'DESCRIPTION': _COMMAND_DESCRIPTION, 'EXAMPLES': _GA_EXAMPLES, } @classmethod def Args(cls, parser): parser.add_argument('url', help='Specifies URL of object to describe.') flags.add_additional_headers_flag(parser) flags.add_encryption_flags(parser, command_only_reads_data=True) flags.add_fetch_encrypted_object_hashes_flag(parser, is_list=False) flags.add_raw_display_flag(parser) flags.add_soft_deleted_flag(parser) gsutil_json_printer.GsutilJsonPrinter.Register() if cls.ReleaseTrack() == base.ReleaseTrack.ALPHA: contexts_only_formatter.ContextsOnlyPrinter.Register() def Run(self, args): encryption_util.initialize_key_store(args) if wildcard_iterator.contains_wildcard(args.url): raise errors.InvalidUrlError( 'Describe does not accept wildcards because it returns a single' ' resource. Please use the `ls` or `objects list` command for' ' retrieving multiple resources.') url = storage_url.storage_url_from_string(args.url) errors_util.raise_error_if_not_cloud_object(args.command_path, url) client = api_factory.get_api(url.scheme) resource = client.get_object_metadata( url.bucket_name, url.resource_name, generation=url.generation, fields_scope=cloud_api.FieldsScope.FULL, soft_deleted=args.soft_deleted, ) if (args.fetch_encrypted_object_hashes and cloud_api.Capability.ENCRYPTION in client.capabilities and not (resource.md5_hash and resource.crc32c_hash) and resource.decryption_key_hash_sha256): request_config = request_config_factory.get_request_config( resource.storage_url, decryption_key_hash_sha256=resource.decryption_key_hash_sha256, error_on_missing_key=True) final_resource = client.get_object_metadata( resource.bucket, resource.name, fields_scope=cloud_api.FieldsScope.FULL, generation=resource.generation, request_config=request_config, soft_deleted=args.soft_deleted, ) else: final_resource = resource if args.format == contexts_only_formatter.CONTEXT_ONLY_PRINTER_FORMAT: return final_resource return resource_util.get_display_dict_for_resource( final_resource, full_resource_formatter.ObjectDisplayTitlesAndDefaults, display_raw_keys=args.raw, ) @base.UniverseCompatible @base.ReleaseTracks(base.ReleaseTrack.ALPHA) class DescribeAlpha(Describe): """Describe a Cloud Storage object.""" detailed_help = { 'DESCRIPTION': _COMMAND_DESCRIPTION, 'EXAMPLES': _GA_EXAMPLES + _ALPHA_EXAMPLES, }