HEX
Server: Apache/2.4.65 (Ubuntu)
System: Linux ielts-store-v2 6.8.0-1036-gcp #38~22.04.1-Ubuntu SMP Thu Aug 14 01:19:18 UTC 2025 x86_64
User: root (0)
PHP: 7.2.34-54+ubuntu20.04.1+deb.sury.org+1
Disabled: pcntl_alarm,pcntl_fork,pcntl_waitpid,pcntl_wait,pcntl_wifexited,pcntl_wifstopped,pcntl_wifsignaled,pcntl_wifcontinued,pcntl_wexitstatus,pcntl_wtermsig,pcntl_wstopsig,pcntl_signal,pcntl_signal_get_handler,pcntl_signal_dispatch,pcntl_get_last_error,pcntl_strerror,pcntl_sigprocmask,pcntl_sigwaitinfo,pcntl_sigtimedwait,pcntl_exec,pcntl_getpriority,pcntl_setpriority,pcntl_async_signals,
Upload Files
File: //snap/google-cloud-cli/394/lib/googlecloudsdk/command_lib/edge_cloud/container/resource_args.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.
"""Shared resource flags for edge-cloud container commands."""

from __future__ import absolute_import
from __future__ import division
from __future__ import unicode_literals

from googlecloudsdk.api_lib.edge_cloud.container import util
from googlecloudsdk.api_lib.util import messages as messages_util
from googlecloudsdk.calliope import exceptions
from googlecloudsdk.calliope.concepts import concepts
from googlecloudsdk.calliope.concepts import deps
from googlecloudsdk.command_lib.util.concepts import concept_parsers
from googlecloudsdk.core import properties

GDCE_SYS_ADDONS_CONFIG = 'systemAddonsConfig'
GDCE_EXTERNAL_LB_CONFIG = 'externalLoadBalancerAddressPools'


def ClusterAttributeConfig():
  return concepts.ResourceParameterAttributeConfig(
      name='cluster', help_text='Cluster of the {resource}.')


def LocationAttributeConfig():
  return concepts.ResourceParameterAttributeConfig(
      name='location', help_text='Google Cloud location for the {resource}.')


def GetLocationsListingResourceSpec():
  """Gets the location resource spec for listing resources."""
  fallthroughs = [
      ## if location is not set, use value from
      ## gcloud config get-value edge_container/location
      deps.ArgFallthrough('--location'),
      deps.PropertyFallthrough(properties.VALUES.edge_container.location),
  ]

  # TODO(b/332336702): Before this CL, only listing clusters supports the
  # `edge_container/location` property. All other cluster commands do not use
  #  it. Validate if this is a bug with current behavior, and fix if so.
  config = LocationAttributeConfig()
  config.fallthroughs = fallthroughs

  return concepts.ResourceSpec(
      'edgecontainer.projects.locations',
      resource_name='location',
      locationsId=config,
      projectsId=concepts.DEFAULT_PROJECT_ATTRIBUTE_CONFIG)


def GetClusterResourceSpec():
  return concepts.ResourceSpec(
      'edgecontainer.projects.locations.clusters',
      resource_name='cluster',
      clustersId=ClusterAttributeConfig(),
      locationsId=LocationAttributeConfig(),
      projectsId=concepts.DEFAULT_PROJECT_ATTRIBUTE_CONFIG)


def AddLocationOptionalResourceArgForListing(parser):
  """Adds a resource argument for an Edge Container location.

  Args:
    parser: The argparse parser to add the resource arg to.
  """
  concept_parsers.ConceptParser.ForResource(
      '--location',
      GetLocationsListingResourceSpec(),
      'Edge Container location {}.'.format('to list'),
      required=False).AddToParser(parser)


def AddClusterResourceArg(parser, verb, positional=True):
  """Adds a resource argument for an Edge Container cluster.

  Args:
    parser: The argparse parser to add the resource arg to.
    verb: str, the verb to describe the resource, such as 'to update'.
    positional: bool, whether the argument is positional or not.
  """
  name = 'cluster' if positional else '--cluster'
  concept_parsers.ConceptParser.ForResource(
      name,
      GetClusterResourceSpec(),
      'Edge Container cluster {}.'.format(verb),
      required=True).AddToParser(parser)


def NodePoolAttributeConfig():
  return concepts.ResourceParameterAttributeConfig(
      name='nodePool', help_text='Node pool of the {resource}.')


def GetNodePoolResourceSpec():
  return concepts.ResourceSpec(
      'edgecontainer.projects.locations.clusters.nodePools',
      resource_name='node pool',
      clustersId=ClusterAttributeConfig(),
      nodePoolsId=NodePoolAttributeConfig(),
      locationsId=LocationAttributeConfig(),
      projectsId=concepts.DEFAULT_PROJECT_ATTRIBUTE_CONFIG)


def AddNodePoolResourceArg(parser, verb):
  """Adds a resource argument for an Edge Container node pool.

  Args:
    parser: The argparse parser to add the resource arg to.
    verb: str, the verb to describe the resource, such as 'to update'.
  """
  name = 'node_pool'
  concept_parsers.ConceptParser.ForResource(
      name,
      GetNodePoolResourceSpec(),
      'Edge Container node pool {}.'.format(verb),
      required=True).AddToParser(parser)


def ProcessSystemAddonsConfig(args, req):
  """Processes the cluster.system_addons_config.

  Args:
    args: command line arguments.
    req: API request to be issued
  """

  release_track = args.calliope_command.ReleaseTrack()
  msgs = util.GetMessagesModule(release_track)

  data = args.system_addons_config
  try:
    system_addons_config = messages_util.DictToMessageWithErrorCheck(
        data[GDCE_SYS_ADDONS_CONFIG], msgs.SystemAddonsConfig)
  except (messages_util.DecodeError, AttributeError, KeyError) as err:
    raise exceptions.InvalidArgumentException(
        '--system-addons-config',
        '\'{}\''.format(err.args[0] if err.args else err))
  req.cluster.systemAddonsConfig = system_addons_config


def SetSystemAddonsConfig(args, request):
  """Sets the cluster.system_addons_config if specified.

  Args:
    args: command line arguments.
    request: API request to be issued
  """

  if args.IsKnownAndSpecified('system_addons_config'):
    ProcessSystemAddonsConfig(args, request)


def CheckAddressPoolNameUniqueness(external_lb_address_pools):
  """Checks for unique address pool names in the given list of dictionaries.

  Args:
    external_lb_address_pools: A list of dictionaries representing
    ExternalLoadBalancerPool messages.

  Returns:
    str: An error message if a duplicate address pool name is found,
    otherwise None.
  """

  address_pool_set = set()

  for pool in external_lb_address_pools:
    if 'addressPool' in pool and pool['addressPool']:
      if pool['addressPool'] in address_pool_set:
        return f"Duplicate address pool name: {pool['addressPool']}"
      address_pool_set.add(pool['addressPool'])

  return None


def ProcessExternalLoadBalancerAddressPoolsConfig(args, req):
  """Processes the cluster.externalLoadBalancerAddressPools.

  Args:
    args: command line arguments.
    req: API request to be issued
  """

  release_track = args.calliope_command.ReleaseTrack()
  msgs = util.GetMessagesModule(release_track)

  lbdata = args.external_lb_address_pools
  if not lbdata:
    return

  pools = lbdata.get(GDCE_EXTERNAL_LB_CONFIG)
  if not pools:
    return

  err = CheckAddressPoolNameUniqueness(pools)
  if err:
    raise exceptions.InvalidArgumentException(
        '--external-lb-address-pools',
        f'Duplicate address pool found: {err}'
    )

  mpools = []
  try:
    for pool in pools:
      mpool = messages_util.DictToMessageWithErrorCheck(
          pool, msgs.ExternalLoadBalancerPool)
      mpools.append(mpool)
  except (messages_util.DecodeError, AttributeError, KeyError) as err:
    raise exceptions.InvalidArgumentException(
        '--external-lb-address-pools',
        '\'{}\''.format(err.args[0] if err.args else err))
  if mpools:
    req.cluster.externalLoadBalancerAddressPools = mpools


def SetExternalLoadBalancerAddressPoolsConfig(args, request):
  """Sets the cluster.external_lb_address_pools if specified.

  Args:
    args: command line arguments.
    request: API request to be issued
  """

  if args.IsKnownAndSpecified('external_lb_address_pools'):
    ProcessExternalLoadBalancerAddressPoolsConfig(args, request)