Skip to content

Commit

Permalink
Added identity commands + --assign-identity flag to containerapp crea…
Browse files Browse the repository at this point in the history
…te (#8)

* Added identity show and assign.

* Finisheed identity remove.

* Added helps, updated identity remove to work with identity names instead of requiring identity resource ids.

* Moved helper function to utils.

* Require --identities flag when removing identities.

* Added message for assign identity with no specified identity.

* Added --assign-identity flag to containerapp create.

* Moved assign-identity flag to containerapp create.

* Fixed small logic error on remove identities when passing duplicate identities. Added warnings for certain edge cases.

* Updated param definition for identity assign --identity default.

* Added identity examples in help.

* Made sure secrets were not removed when assigning identities. Added tolerance for [system] passed with capital letters.

* Fixed error from merge.

Co-authored-by: Haroon Feisal <[email protected]>
  • Loading branch information
2 people authored and calvinsID committed Mar 22, 2022
1 parent c09fe39 commit d264bbf
Show file tree
Hide file tree
Showing 5 changed files with 261 additions and 3 deletions.
36 changes: 36 additions & 0 deletions src/containerapp/azext_containerapp/_help.py
Original file line number Diff line number Diff line change
Expand Up @@ -220,6 +220,42 @@
az containerapp env list -g MyResourceGroup
"""

# Identity Commands
helps['containerapp identity'] = """
type: group
short-summary: Manage service (managed) identities for a containerapp
"""

helps['containerapp identity assign'] = """
type: command
short-summary: Assign a managed identity to a containerapp
long-summary: Managed identities can be user-assigned or system-assigned
examples:
- name: Assign system identity.
text: |
az containerapp identity assign
- name: Assign system and user identity.
text: |
az containerapp identity assign --identities [system] myAssignedId
"""

helps['containerapp identity remove'] = """
type: command
short-summary: Remove a managed identity from a containerapp
examples:
- name: Remove system identity.
text: |
az containerapp identity remove [system]
- name: Remove system and user identity.
text: |
az containerapp identity remove --identities [system] myAssignedId
"""

helps['containerapp identity show'] = """
type: command
short-summary: Show the containerapp's identity details
"""

# Ingress Commands
helps['containerapp ingress'] = """
type: group
Expand Down
9 changes: 9 additions & 0 deletions src/containerapp/azext_containerapp/_params.py
Original file line number Diff line number Diff line change
Expand Up @@ -66,6 +66,9 @@ def load_arguments(self, _):
c.argument('ingress', validator=validate_ingress, options_list=['--ingress'], default=None, arg_type=get_enum_type(['internal', 'external']), help="The ingress type.")
c.argument('target_port', type=int, validator=validate_target_port, options_list=['--target-port'], help="The application port used for ingress traffic.")
c.argument('transport', arg_type=get_enum_type(['auto', 'http', 'http2']), help="The transport protocol used for ingress traffic.")

with self.argument_context('containerapp create') as c:
c.argument('assign_identity', nargs='+', help="Space-separated identities. Use '[system]' to refer to the system assigned identity.")
c.argument('traffic_weights', nargs='*', options_list=['--traffic-weight'], help="A list of revision weight(s) for the container app. Space-separated values in 'revision_name=weight' format. For latest revision, use 'latest=weight'")

with self.argument_context('containerapp scale') as c:
Expand Down Expand Up @@ -103,6 +106,12 @@ def load_arguments(self, _):
with self.argument_context('containerapp env show') as c:
c.argument('name', name_type, help='Name of the Container Apps Environment.')

with self.argument_context('containerapp identity') as c:
c.argument('identities', nargs='+', help="Space-separated identities. Use '[system]' to refer to the system assigned identity.")

with self.argument_context('containerapp identity assign') as c:
c.argument('identities', nargs='+', help="Space-separated identities. Use '[system]' to refer to the system assigned identity. Default is '[system]'.")

with self.argument_context('containerapp github-action add') as c:
c.argument('repo_url', help='The GitHub repository to which the workflow file will be added. In the format: https://github.com/<owner>/<repository-name>')
c.argument('token', help='A Personal Access Token with write access to the specified repository. For more information: https://help.github.com/en/github/authenticating-to-github/creating-a-personal-access-token-for-the-command-line')
Expand Down
10 changes: 10 additions & 0 deletions src/containerapp/azext_containerapp/_utils.py
Original file line number Diff line number Diff line change
Expand Up @@ -284,6 +284,16 @@ def _get_existing_secrets(cmd, resource_group_name, name, containerapp_def):

containerapp_def["properties"]["configuration"]["secrets"] = secrets["value"]

def _ensure_identity_resource_id(subscription_id, resource_group, resource):
from msrestazure.tools import resource_id, is_valid_resource_id
if is_valid_resource_id(resource):
return resource

return resource_id(subscription=subscription_id,
resource_group=resource_group,
namespace='Microsoft.ManagedIdentity',
type='userAssignedIdentities',
name=resource)

def _add_or_update_secrets(containerapp_def, add_secrets):
if "secrets" not in containerapp_def["properties"]["configuration"]:
Expand Down
7 changes: 7 additions & 0 deletions src/containerapp/azext_containerapp/commands.py
Original file line number Diff line number Diff line change
Expand Up @@ -56,6 +56,13 @@ def load_command_table(self, _):
# g.custom_command('update', 'update_managed_environment', supports_no_wait=True, exception_handler=ex_handler_factory())
g.custom_command('delete', 'delete_managed_environment', supports_no_wait=True, confirmation=True, exception_handler=ex_handler_factory())


with self.command_group('containerapp identity') as g:
g.custom_command('assign', 'assign_managed_identity', supports_no_wait=True, exception_handler=ex_handler_factory())
g.custom_command('remove', 'remove_managed_identity', supports_no_wait=True, exception_handler=ex_handler_factory())
g.custom_command('show', 'show_managed_identity')


with self.command_group('containerapp github-action') as g:
g.custom_command('add', 'create_or_update_github_action', exception_handler=ex_handler_factory())
g.custom_command('show', 'show_github_action', exception_handler=ex_handler_factory())
Expand Down
202 changes: 199 additions & 3 deletions src/containerapp/azext_containerapp/custom.py
Original file line number Diff line number Diff line change
Expand Up @@ -33,13 +33,19 @@
Dapr as DaprModel,
ContainerResources as ContainerResourcesModel,
Scale as ScaleModel,
Container as ContainerModel, GitHubActionConfiguration, RegistryInfo as RegistryInfoModel, AzureCredentials as AzureCredentialsModel, SourceControl as SourceControlModel)
Container as ContainerModel,
GitHubActionConfiguration,
RegistryInfo as RegistryInfoModel,
AzureCredentials as AzureCredentialsModel,
SourceControl as SourceControlModel,
ManagedServiceIdentity as ManagedServiceIdentityModel)
from ._utils import (_validate_subscription_registered, _get_location_from_resource_group, _ensure_location_allowed,
parse_secret_flags, store_as_secret_and_return_secret_ref, parse_list_of_strings, parse_env_var_flags,
_generate_log_analytics_if_not_provided, _get_existing_secrets, _convert_object_from_snake_to_camel_case,
_object_to_dict, _add_or_update_secrets, _remove_additional_attributes, _remove_readonly_attributes,
_add_or_update_env_vars, _add_or_update_tags, update_nested_dictionary, _update_traffic_Weights,
_get_app_from_revision, raise_missing_token_suggestion, _infer_acr_credentials, _remove_registry_secret, _remove_secret)
_get_app_from_revision, raise_missing_token_suggestion, _infer_acr_credentials, _remove_registry_secret, _remove_secret,
_ensure_identity_resource_id)

logger = get_logger(__name__)

Expand Down Expand Up @@ -325,7 +331,8 @@ def create_containerapp(cmd,
startup_command=None,
args=None,
tags=None,
no_wait=False):
no_wait=False,
assign_identity=[]):
_validate_subscription_registered(cmd, "Microsoft.App")

if yaml:
Expand Down Expand Up @@ -403,6 +410,28 @@ def create_containerapp(cmd,
config_def["ingress"] = ingress_def
config_def["registries"] = [registries_def] if registries_def is not None else None

# Identity actions
identity_def = ManagedServiceIdentityModel
identity_def["type"] = "None"

assign_system_identity = '[system]' in assign_identity
assign_user_identities = [x for x in assign_identity if x != '[system]']

if assign_system_identity and assign_user_identities:
identity_def["type"] = "SystemAssigned, UserAssigned"
elif assign_system_identity:
identity_def["type"] = "SystemAssigned"
elif assign_user_identities:
identity_def["type"] = "UserAssigned"

if assign_user_identities:
identity_def["userAssignedIdentities"] = {}
subscription_id = get_subscription_id(cmd.cli_ctx)

for r in assign_user_identities:
r = _ensure_identity_resource_id(subscription_id, resource_group_name, r)
identity_def["userAssignedIdentities"][r] = {}

scale_def = None
if min_replicas is not None or max_replicas is not None:
scale_def = ScaleModel
Expand Down Expand Up @@ -445,6 +474,7 @@ def create_containerapp(cmd,

containerapp_def = ContainerAppModel
containerapp_def["location"] = location
containerapp_def["identity"] = identity_def
containerapp_def["properties"]["managedEnvironmentId"] = managed_env
containerapp_def["properties"]["configuration"] = config_def
containerapp_def["properties"]["template"] = template_def
Expand Down Expand Up @@ -935,6 +965,172 @@ def delete_managed_environment(cmd, name, resource_group_name, no_wait=False):
handle_raw_exception(e)


def assign_managed_identity(cmd, name, resource_group_name, identities=None, no_wait=False):
_validate_subscription_registered(cmd, "Microsoft.App")

# if no identities, then assign system by default
if not identities:
identities = ['[system]']
logger.warning('Identities not specified. Assigning managed system identity.')

identities = [x.lower() for x in identities]
assign_system_identity = '[system]' in identities
assign_user_identities = [x for x in identities if x != '[system]']

containerapp_def = None

# Get containerapp properties of CA we are updating
try:
containerapp_def = ContainerAppClient.show(cmd=cmd, resource_group_name=resource_group_name, name=name)
except:
pass

if not containerapp_def:
raise CLIError("The containerapp '{}' does not exist".format(name))

_get_existing_secrets(cmd, resource_group_name, name, containerapp_def)

# If identity not returned
try:
containerapp_def["identity"]
containerapp_def["identity"]["type"]
except:
containerapp_def["identity"] = {}
containerapp_def["identity"]["type"] = "None"

if assign_system_identity and containerapp_def["identity"]["type"].__contains__("SystemAssigned"):
logger.warning("System identity is already assigned to containerapp")

# Assign correct type
try:
if containerapp_def["identity"]["type"] != "None":
if containerapp_def["identity"]["type"] == "SystemAssigned" and assign_user_identities:
containerapp_def["identity"]["type"] = "SystemAssigned,UserAssigned"
if containerapp_def["identity"]["type"] == "UserAssigned" and assign_system_identity:
containerapp_def["identity"]["type"] = "SystemAssigned,UserAssigned"
else:
if assign_system_identity and assign_user_identities:
containerapp_def["identity"]["type"] = "SystemAssigned,UserAssigned"
elif assign_system_identity:
containerapp_def["identity"]["type"] = "SystemAssigned"
elif assign_user_identities:
containerapp_def["identity"]["type"] = "UserAssigned"
except:
# Always returns "type": "None" when CA has no previous identities
pass

if assign_user_identities:
try:
containerapp_def["identity"]["userAssignedIdentities"]
except:
containerapp_def["identity"]["userAssignedIdentities"] = {}

subscription_id = get_subscription_id(cmd.cli_ctx)

for r in assign_user_identities:
old_id = r
r = _ensure_identity_resource_id(subscription_id, resource_group_name, r).replace("resourceGroup", "resourcegroup")
try:
containerapp_def["identity"]["userAssignedIdentities"][r]
logger.warning("User identity {} is already assigned to containerapp".format(old_id))
except:
containerapp_def["identity"]["userAssignedIdentities"][r] = {}

try:
r = ContainerAppClient.create_or_update(cmd=cmd, resource_group_name=resource_group_name, name=name, container_app_envelope=containerapp_def, no_wait=no_wait)
# If identity is not returned, do nothing
return r["identity"]

except Exception as e:
handle_raw_exception(e)


def remove_managed_identity(cmd, name, resource_group_name, identities, no_wait=False):
_validate_subscription_registered(cmd, "Microsoft.App")

identities = [x.lower() for x in identities]
remove_system_identity = '[system]' in identities
remove_user_identities = [x for x in identities if x != '[system]']
remove_id_size = len(remove_user_identities)

# Remove duplicate identities that are passed and notify
remove_user_identities = list(set(remove_user_identities))
if remove_id_size != len(remove_user_identities):
logger.warning("At least one identity was passed twice.")

containerapp_def = None
# Get containerapp properties of CA we are updating
try:
containerapp_def = ContainerAppClient.show(cmd=cmd, resource_group_name=resource_group_name, name=name)
except:
pass

if not containerapp_def:
raise CLIError("The containerapp '{}' does not exist".format(name))

_get_existing_secrets(cmd, resource_group_name, name, containerapp_def)

# If identity not returned
try:
containerapp_def["identity"]
containerapp_def["identity"]["type"]
except:
containerapp_def["identity"] = {}
containerapp_def["identity"]["type"] = "None"

if containerapp_def["identity"]["type"] == "None":
raise CLIError("The containerapp {} has no system or user assigned identities.".format(name))

if remove_system_identity:
if containerapp_def["identity"]["type"] == "UserAssigned":
raise CLIError("The containerapp {} has no system assigned identities.".format(name))
containerapp_def["identity"]["type"] = ("None" if containerapp_def["identity"]["type"] == "SystemAssigned" else "UserAssigned")

if remove_user_identities:
subscription_id = get_subscription_id(cmd.cli_ctx)
try:
containerapp_def["identity"]["userAssignedIdentities"]
except:
containerapp_def["identity"]["userAssignedIdentities"] = {}
for id in remove_user_identities:
given_id = id
id = _ensure_identity_resource_id(subscription_id, resource_group_name, id)
wasRemoved = False

for old_user_identity in containerapp_def["identity"]["userAssignedIdentities"]:
if old_user_identity.lower() == id.lower():
containerapp_def["identity"]["userAssignedIdentities"].pop(old_user_identity)
wasRemoved = True
break

if not wasRemoved:
raise CLIError("The containerapp does not have specified user identity '{}' assigned, so it cannot be removed.".format(given_id))

if containerapp_def["identity"]["userAssignedIdentities"] == {}:
containerapp_def["identity"]["userAssignedIdentities"] = None
containerapp_def["identity"]["type"] = ("None" if containerapp_def["identity"]["type"] == "UserAssigned" else "SystemAssigned")

try:
r = ContainerAppClient.create_or_update(cmd=cmd, resource_group_name=resource_group_name, name=name, container_app_envelope=containerapp_def, no_wait=no_wait)
return r["identity"]
except Exception as e:
handle_raw_exception(e)


def show_managed_identity(cmd, name, resource_group_name):
_validate_subscription_registered(cmd, "Microsoft.App")

try:
r = ContainerAppClient.show(cmd=cmd, resource_group_name=resource_group_name, name=name)
except CLIError as e:
handle_raw_exception(e)

try:
return r["identity"]
except:
r["identity"] = {}
r["identity"]["type"] = "None"
return r["identity"]
def create_or_update_github_action(cmd,
name,
resource_group_name,
Expand Down

0 comments on commit d264bbf

Please sign in to comment.