File: //snap/google-cloud-cli/current/lib/surface/storage/mv.py
# -*- coding: utf-8 -*- #
# Copyright 2022 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 Unix-like mv command for cloud storage providers."""
from __future__ import absolute_import
from __future__ import division
from __future__ import unicode_literals
from googlecloudsdk.calliope import base
from googlecloudsdk.command_lib.storage import cp_command_util
from googlecloudsdk.command_lib.storage import errors
from googlecloudsdk.command_lib.storage import flags
from googlecloudsdk.command_lib.storage import storage_url
_COMMAND_DESCRIPTION = """
The mv command allows you to move data between your local file system and
the cloud, move data within the cloud, and move data between cloud storage
providers.
*Renaming Groups Of Objects*
You can use the mv command to rename all objects with a given prefix to
have a new prefix. For example, the following command renames all objects
under gs://my_bucket/oldprefix to be under gs://my_bucket/newprefix,
otherwise preserving the naming structure:
  $ {command} gs://my_bucket/oldprefix gs://my_bucket/newprefix
Note that when using mv to rename groups of objects with a common prefix,
you cannot specify the source URL using wildcards; you must spell out the
complete name.
If you do a rename as specified above and you want to preserve ACLs.
*Non-Atomic Operation*
Unlike the case with many file systems, the mv command does not perform a
single atomic operation. Rather, it performs a copy from source to
destination followed by removing the source for each object.
A consequence of this is that, in addition to normal network and operation
charges, if you move a Nearline Storage, Coldline Storage, or Archive
Storage object, deletion and data retrieval charges apply.
See the documentation for pricing details.
"""
_GA_EXAMPLES = """
To move all objects from a bucket to a local directory you could use:
  $ {command} gs://my_bucket/* dir
Similarly, to move all objects from a local directory to a bucket you
could use:
  $ {command} ./dir gs://my_bucket
The following command renames all objects under gs://my_bucket/oldprefix
to be under gs://my_bucket/newprefix, otherwise preserving the naming
structure:
  $ {command} gs://my_bucket/oldprefix gs://my_bucket/newprefix
"""
_ALPHA_EXAMPLES = """
The following command would clear all custom contexts from the destination
object while moving the object to the destination bucket.
  $ {command} gs://my-bucket/object gs://destination-bucket/object \
      --clear-custom-contexts
"""
@base.UniverseCompatible
@base.ReleaseTracks(base.ReleaseTrack.GA)
class Mv(base.Command):
  """Moves or renames objects."""
  detailed_help = {
      "DESCRIPTION": _COMMAND_DESCRIPTION,
      "EXAMPLES": _GA_EXAMPLES,
  }
  @classmethod
  def Args(cls, parser):
    cp_command_util.add_cp_and_mv_flags(parser, cls.ReleaseTrack())
    flags.add_per_object_retention_flags(parser)
  def Run(self, args):
    for url_string in args.source:
      url = storage_url.storage_url_from_string(url_string)
      if isinstance(url, storage_url.CloudUrl) and not url.is_object():
        raise errors.InvalidUrlError("Cannot mv buckets.")
      if url.is_stdio:
        raise errors.InvalidUrlError("Cannot mv stdin.")
    # Must copy children of prefixes and folders.
    args.recursive = True
    self.exit_code = cp_command_util.run_cp(args, delete_source=True)
@base.UniverseCompatible
@base.ReleaseTracks(base.ReleaseTrack.ALPHA)
class MvAlpha(Mv):
  """Moves or renames objects."""
  detailed_help = {
      "DESCRIPTION": _COMMAND_DESCRIPTION,
      "EXAMPLES": _GA_EXAMPLES + _ALPHA_EXAMPLES,
  }