RBAC Patch 3: Glance tests

This patch chain aims to suggest a set of default policies for user
management on stx-openstack. We suggest the creation of the project_admin
and project_readonly roles and provide some policies to fine tune the
access control over the Openstack services to those roles, as described
on README.md.

Also, we provide a set of tests to ensure the policies and permissions
are all working as expected on site for the cloud administrators.

This commit includes Glance related tests and functions.

Story: 2008910
Task: 42501

Signed-off-by: Heitor Matsui <heitorvieira.matsui@windriver.com>
Signed-off-by: Thiago Brito <thiago.brito@windriver.com>
Co-authored-by: Miriam Yumi Peixoto <miriam.yumipeixoto@windriver.com>
Co-authored-by: Leonardo Zaccarias <leonardo.zaccarias@windriver.com>
Co-authored-by: Rogerio Oliveira Ferraz <rogeriooliveira.ferraz@windriver.com>

Change-Id: I2738b6d99a59fbbd2fd65a96a3ddd31167b9d13f
This commit is contained in:
Heitor Matsui 2021-05-17 18:07:57 -03:00
parent ecb1e24972
commit 107e3242b2
3 changed files with 755 additions and 0 deletions

View File

@ -634,3 +634,247 @@ class OpenStackBasicTesting():
self._get_snapshot(snapshot_name).id
)
self.cc.volume_snapshots.delete_metadata(snapshot, metadata_keys)
# -------------------------------------------------------------------------
# Image methods - Glance
# -------------------------------------------------------------------------
def _create_image(self, image_name, filename=None, admin=False,
disk_format="qcow2", container_format="bare",
visibility="private", wait=True, timeout=3 * 60,
autoclear=True):
"""
# create_image(name, filename=None, container=None, md5=None, sha256=None,
# disk_format=None, container_format=None,
# disable_vendor_agent=True, allow_duplicates=False,
# meta=None, wait=False, timeout=3600, data=None,
# validate_checksum=False, use_import=False, stores=None,
# all_stores=None, all_stores_must_succeed=None, **kwargs)
# Upload an image.
# Parameters
# name (str) Name of the image to create. If it is a pathname of an
# image, the name will be constructed from the extensionless basename of
# the path.
# filename (str) The path to the file to upload, if needed. (optional,
# defaults to None)
# data Image data (string or file-like object). It is mutually exclusive
# with filename
# container (str) Name of the container in swift where images should be
# uploaded for import if the cloud requires such a thing. (optional,
# defaults to images)
# md5 (str) md5 sum of the image file. If not given, an md5 will be
# calculated.
# sha256 (str) sha256 sum of the image file. If not given, an md5 will
# be calculated.
# disk_format (str) The disk format the image is in. (optional, defaults
# to the os-client-config config value for this cloud)
# container_format (str) The container format the image is in. (optional,
# defaults to the os-client-config config value for this cloud)
# disable_vendor_agent (bool) Whether or not to append metadata flags to
# the image to inform the cloud in question to not expect a vendor agent to
# be runing. (optional, defaults to True)
# allow_duplicates If true, skips checks that enforce unique image name.
# (optional, defaults to False)
# meta A dict of key/value pairs to use for metadata that bypasses
# automatic type conversion.
# wait (bool) If true, waits for image to be created. Defaults to true -
# however, be aware that one of the upload methods is always synchronous.
# timeout Seconds to wait for image creation. None is forever.
# validate_checksum (bool) If true and cloud returns checksum, compares
# return value with the one calculated or passed into this call. If value
# does not match - raises exception. Default is false
# use_import (bool) Use the interoperable image import mechanism to
# import the image. This defaults to false because it is harder on the
# target cloud so should only be used when needed, such as when the user
# needs the cloud to transform image format. If the cloud has disabled
# direct uploads, this will default to true.
# stores List of stores to be used when enabled_backends is activated in
# glance. List values can be the id of a store or a Store instance. Implies
# use_import equals True.
# all_stores Upload to all available stores. Mutually exclusive with
# store and stores. Implies use_import equals True.
# all_stores_must_succeed When set to True, if an error occurs during the
# upload in at least one store, the worfklow fails, the data is deleted
# from stores where copying is done (not staging), and the state of the
# image is unchanged. When set to False, the workflow will fail (data
# deleted from stores, …) only if the import fails on all stores specified
# by the user. In case of a partial success, the locations added to the
# image will be the stores where the data has been correctly uploaded.
# Default is True. Implies use_import equals True.
# Additional kwargs will be passed to the image creation as additional
# metadata for the image and will have all values converted to string
# except for min_disk, min_ram, size and virtual_size which will be
# converted to int.
# If you are sure you have all of your data types correct or have an
# advanced need to be explicit, use meta. If you are just a normal
# consumer, using kwargs is likely the right choice.
# If a value is in meta and kwargs, meta wins.
# Returns
# A munch.Munch of the Image object
# Raises
# SDKException if there are problems uploading
"""
os_sdk_conn = self.os_sdk_conn
if admin:
os_sdk_conn = self.os_sdk_admin_conn
image = os_sdk_conn.image.create_image(name=image_name,
filename=filename,
container_format=container_format,
disk_format=disk_format,
visibility=visibility,
wait=wait, timeout=timeout)
if debug1:
print("created image: " + image.name + " id: " + image.id)
if autoclear:
self.images_clearing.append(image.id)
return image
def _delete_image(self, name_or_id, autoclear=True):
"""
# Delete an image
# Parameters
# name_or_id The name or ID of a image.
# ignore_missing (bool) When set to False ResourceNotFound will be
# raised when the image does not exist. When set to True, no
# exception will be set when attempting to delete a nonexistent
# image.
# Returns
# None
"""
image = self._find_image(name_or_id)
if image:
self.os_sdk_conn.image.delete_image(image.id, ignore_missing=False)
if debug1: print(
"created image: " + image.name + " id: " + image.id)
if autoclear:
self.images_clearing.remove(image.id)
def _list_images(self, **query):
"""
# Return a generator of images
# Parameters
# query (kwargs) Optional query parameters to be sent to limit the
# resources being returned.
# Returns
# A generator of image objects
# Return type
# Image
"""
return self.gc.images.list(**query)
def _update_image(self, image, **attrs):
"""
# Update a image
# Parameters
# image Either the ID of a image or a Image instance.
# Attrs kwargs
# The attributes to update on the image represented by value.
# Returns
# The updated image
# Return type
# Image
"""
self.os_sdk_conn.image.update_image(image, **attrs)
return self._get_image(image.id)
def _upload_image(self, image_id, filename):
"""
# upload_image(container_format=None, disk_format=None, data=None, **attrs)
# Create and upload a new image from attributes
# Parameters
# container_format Format of the container. A valid value is ami,
# ari, aki, bare, ovf, ova, or docker.
# disk_format The format of the disk. A valid value is ami, ari,
# aki, vhd, vmdk, raw, qcow2, vdi, or iso.
# data The data to be uploaded as an image.
# attrs (dict) Keyword arguments which will be used to create a
# Image, comprised of the properties on the Image class.
# Returns
# The results of image creation
# Return type
# Image
"""
image = self.gc.images.upload(
image_id,
open(filename, 'rb')
)
return image
def _download_image(self, image, stream=False, output=None,
chunk_size=1024):
"""
# Download an image
# This will download an image to memory when stream=False, or allow
# streaming downloads using an iterator when stream=True. For examples of
# working with streamed responses, see Downloading an Image with
# stream=True.
# Parameters
# image The value can be either the ID of an image or a Image
# instance.
# stream (bool)
# When True, return a requests.Response instance allowing you to
# iterate over the response data stream instead of storing its
# entire contents in memory. See requests.Response.iter_content()
# for more details. NOTE: If you do not consume the entirety of the
# response you must explicitly call requests.Response.close() or
# otherwise risk inefficiencies with the requests librarys
# handling of connections.
# When False, return the entire contents of the response.
# output Either a file object or a path to store data into.
# chunk_size (int) size in bytes to read from the wire and buffer
# at one time. Defaults to 1024
# Returns
# When output is not given - the bytes comprising the given Image
# when stream is False, otherwise a requests.Response instance.
# When output is given - a Image instance.
"""
return self.os_sdk_conn.image.download_image(image, stream=True,
output=output,
chunk_size=chunk_size)
def _find_image(self, image_name_or_id, ignore_missing=True):
"""
# Find a single image
# Parameters
# image_name_or_id The name or ID of a image.
# ignore_missing (bool) When set to False ResourceNotFound will be
# raised when the resource does not exist. When set to True, None
# will be returned when attempting to find a nonexistent resource.
# Returns
# One Image or None
"""
return self.os_sdk_conn.image.find_image(image_name_or_id,
ignore_missing=ignore_missing)
def _get_image(self, image_name_or_id):
"""
# Get a single image
# Parameters
# image_name_or_id The name or ID of a image.
# Returns
# One Image
# Raises
# ResourceNotFound when no resource can be found.
"""
image = self._find_image(image_name_or_id, ignore_missing=False)
return self.os_sdk_conn.image.get_image(image.id)
def _deactivate_image(self, image):
"""
# Deactivate an image
# Parameters
# image Either the ID of a image or a Image instance.
# Returns
# None
"""
self.os_sdk_conn.image.deactivate_image(image.id)
def _reactivate_image(self, image):
"""
# Deactivate an image
# Parameters
# image Either the ID of a image or a Image instance.
# Returns
# None
"""
self.os_sdk_conn.image.reactivate_image(image.id)

View File

@ -0,0 +1,511 @@
#
# Copyright (c) 2021 Wind River Systems, Inc.
#
# SPDX-License-Identifier: Apache-2.0
#
# All Rights Reserved.
#
from tests import rbac_test_base
class TestImages(rbac_test_base.TestClass):
# ---------------------------------------------------------------------
# TC-1
# Description:
# - user11 as project_admin of project1, can create/list/detail/modify/
# delete/upload/download images of project1
# - user11 can deactivate/reactivate images of project1
# - user13 can get list/detail of images of project1
# ---------------------------------------------------------------------
def test_uc_image_1(self):
"""
1. user11 can create/delete an image
2. user11 can upload an image file
3. user11 can detail the image
4. user11 can deactivate the image of project1
5. user11 can NOT download the deactivate image
6. user11 active the image
7. user11 can download the image
8. user12/user13 can list the images of project1"
"""
print("TC-1.1")
# 1.1. user11 can create/delete an image
self.set_connections_for_user(self.user11)
image1 = self._create_image("image1")
self.assertIn(image1.name,
[image.name for image in self._list_images()])
# print(image1.status)
self.assertEqual(image1.status, "queued")
self._delete_image(image1)
self.assertNotIn("image1",
[image.name for image in self._list_images()])
print("TC-1.2")
# 1.2. user11 can upload an image file
image_project1 = self._create_image(
"image-project1",
filename="cirros-0.3.4-x86_64-disk.img"
)
self.assertIn(
image_project1.name,
[image.name for image in self._list_images()]
)
print("TC-1.3")
# 1.3. user11 can detail the image
image_project1_details = self._get_image_by_id(image_project1.id)
self.assertEqual(image_project1.name, image_project1_details.name)
# print(image_project1_details.status)
self.assertEqual(image_project1_details.status, "active")
print("TC-1.4")
# 1.4. user11 can deactivate the image of project1
self._deactivate_image(image_project1)
image_project1 = self._get_image_by_id(image_project1.id)
# print(image_project1.status)
self.assertEqual(image_project1.status, "deactivated")
print("TC-1.5")
# 1.5. user11 can NOT download the deactivate image
download_response = self._download_image(image_project1)
self.assertEqual(download_response.status_code, 403) #HTTP 403 Forbidden
self.assertIn("The requested image has been deactivated. Image data download is forbidden.", download_response.text)
print("TC-1.6")
# 1.6. user11 active the image
self._reactivate_image(image_project1)
image_project1 = self._get_image_by_id(image_project1.id)
self.assertEqual(image_project1.status, "active")
print("TC-1.7")
# 1.7. user11 can download the image
self.assertEqual(self._download_image(image_project1).status_code, 200)
print("TC-1.8")
# 1.8. user12/user13 can list the images of project1"
for user in (self.user12, self.user13):
self.set_connections_for_user(user)
self.assertIn(image_project1.name,
[image.name for image in self._list_images()])
# ---------------------------------------------------------------------
# TC-2
# Description:
# - user12, as member of project1, can create/modify/upload images of
# project1
# - user12 can NOT delete/deactivate/reactivate images of project1
# ---------------------------------------------------------------------
def test_uc_image_2(self):
"""
1. user12 can create/modify an image
2. user12 can upload an image file
3. user12 can detail the image
4. user12 can download the image
5. user12 can NOT delete the image
6. user12 can NOT deactivate/reactivate the image of project1
"""
# For the following test items under this test case
self.set_connections_for_user(self.user12)
# ---------------------------------------------------------------------
# 2.1. user12 can create/modify an image of project1
# ---------------------------------------------------------------------
print("TC-2.1")
# Create
self.set_connections_for_user(self.user12)
image2 = self._create_image("image2")
self.assertIn("image2", [image.name for image in self._list_images()])
# print(image2.status)
self.assertEqual(image2.status, "queued")
# Modify (update)
image2 = self._update_image(image2, name="image2-project1")
self.assertEqual(image2.name, "image2-project1")
self.assertIn(
"image2-project1",
[image.name for image in self._list_images()]
)
self.assertNotIn(
"image2",
[image.name for image in self._list_images()]
)
# ---------------------------------------------------------------------
# 2.2. user12 can upload an image file
# ---------------------------------------------------------------------
print("TC-2.2")
# Upload
self._upload_image(image2.id, "cirros-0.3.4-x86_64-disk.img")
image2 = self._get_image_by_id(image2.id)
# print(image2.status)
self.assertEqual(image2.status, "active")
# ---------------------------------------------------------------------
# 2.3. user12 can detail the image
# ---------------------------------------------------------------------
print("TC-2.3")
# Detail (get)
image2 = self._get_image_by_id(image2.id)
self.assertEqual(image2.name, "image2-project1")
# ---------------------------------------------------------------------
# 2.4. user12 can download the image
# ---------------------------------------------------------------------
print("TC-2.4")
# Download
download_response = self._download_image(image2)
# print(download_response)
self.assertEqual(download_response.status_code, 200) # HTTP 200 OK
# ---------------------------------------------------------------------
# 2.5. user12 can NOT delete the image
# ---------------------------------------------------------------------
print("TC-2.5")
# Delete attempt fails
self.assertRaisesRegex(
Exception,
"You are not authorized to complete delete_image action.",
self._delete_image, image2
)
self.assertIn("image2-project1", [image.name for image in self._list_images()])
# ---------------------------------------------------------------------
# 2.6. user12 can NOT deactivate/reactivate the image of project1
# ---------------------------------------------------------------------
print("TC-2.6")
print("TC-2.6.1")
# Deactivate attempt fails for a project member
self._deactivate_image(image2)
image2 = self._get_image_by_id(image2.id)
# print(image2.status)
self.assertEqual(image2.status, "active")
print("TC-2.6.2")
# Deactivate action succeeds for the project admin
self.set_connections_for_user(self.user11)
self._deactivate_image(image2)
image2 = self._get_image_by_id(image2.id)
# print(image2.status)
self.assertEqual(image2.status, "deactivated")
print("TC-2.6.3")
# Reactivate attempt fails for a project member
self.set_connections_for_user(self.user12)
self._reactivate_image(image2)
image2 = self._get_image_by_id(image2.id)
# print(image2.status)
self.assertEqual(image2.status, "deactivated")
print("TC-2.6.4")
# Reactivate action succeeds for the project admin
self.set_connections_for_user(self.user11)
self._reactivate_image(image2)
image2 = self._get_image_by_id(image2.id)
# print(image2.status)
self.assertEqual(image2.status, "active")
# ---------------------------------------------------------------------
# TC-3
# Description:
# user11/12/13 as member of project1, can list/detail/download images
# of project1 and public image of project2, not private image of
# project2
# ---------------------------------------------------------------------
def test_uc_image_3(self):
"""
1. user21 create a public image in project2
2. user11/user12/user13 can list/detail/download the public image of project2
3. user21 create a private image in project2
4. user11/user12/user13 can NOT find the private image of project2
"""
# ---------------------------------------------------------------------
# 3.1. user21 create a public image in project2
# ---------------------------------------------------------------------
print("TC-3.1")
self.set_connections_for_user(self.user21)
image20 = self._create_image("image-project2-public",
filename="cirros-0.3.4-x86_64-disk.img",
visibility="public")
self.assertIn("image-project2-public",
[image.name for image in self._list_images()])
image20 = self._get_image_by_id(image20.id)
# print(image20.status)
self.assertEqual(image20.status, "active")
# ---------------------------------------------------------------------
# 3.2. user11/user12/user13 can list/detail/download the public image
# of project2
# ---------------------------------------------------------------------
print("TC-3.2")
for user in (self.user11, self.user12, self.user13):
self.set_connections_for_user(user)
# List
self.assertIn(
"image-project2-public",
[image.name for image in self._list_images()]
)
# Detail
image = self._get_image_by_id(image20.id)
self.assertEqual(image.name, "image-project2-public")
# Download
download_response = self._download_image(image20)
# print(download_response.status_code)
self.assertEqual(download_response.status_code, 200) # HTTP 200 OK
# ---------------------------------------------------------------------
# 3.3. user21 create a private image in project2
# ---------------------------------------------------------------------
print("TC-3.3")
self.set_connections_for_user(self.user21)
image21 = self._create_image("image-project2-private",
filename="cirros-0.3.4-x86_64-disk.img",
visibility="private")
self.assertIn("image-project2-private",
[image.name for image in self._list_images()])
image21 = self._get_image_by_id(image21.id)
# print(image21.status)
self.assertEqual(image21.status, "active")
# ---------------------------------------------------------------------
# 3.4. user11/user12/user13 can NOT find the private image of project2
# ---------------------------------------------------------------------
print("TC-3.4")
# Users of project 1 can NOT find the image
for user in (self.user11, self.user12, self.user13):
self.set_connections_for_user(user)
self.assertNotIn("image-project2-private",
[image.name for image in self._list_images()])
# Users of project 2 can find the image
for user in (self.user21, self.user22, self.user23):
self.set_connections_for_user(user)
self.assertIn("image-project2-private",
[image.name for image in self._list_images()])
# ---------------------------------------------------------------------
# 3.5. Project 1 members can NOT delete/modify/deactivate/reactivate
# image of project 2
# ---------------------------------------------------------------------
print("TC-3.5")
print("TC-3.5.1")
# Delete attempt fails for another project's members
for user in (self.user11, self.user12, self.user13):
self.set_connections_for_user(user)
self.assertRaisesRegex(
Exception,
"You are not permitted to delete this image",
self._delete_image, image20
)
self.assertIn("image-project2-public", [image.name for image in self._list_images()])
print("TC-3.5.2")
# Modify (update) attempt fails for another project's members
for user in (self.user11, self.user12, self.user13):
self.set_connections_for_user(user)
self.assertRaisesRegex(
Exception,
"Forbidden",
self._update_image, image20,
name="image-project2-public-newname"
)
self.assertNotIn("image-project2-public-newname", [image.name for image in self._list_images()])
self.assertIn("image-project2-public", [image.name for image in self._list_images()])
print("TC-3.5.3")
# Deactivate attempt fails for another project's admin
self.set_connections_for_user(self.user11)
self._deactivate_image(image20)
image20 = self._get_image_by_id(image20.id)
# print(image20.status)
self.assertEqual(image20.status, "active")
print("TC-3.5.4")
# Deactivate job succeeds for the admin of project 2
self.set_connections_for_user(self.user21)
self._deactivate_image(image20)
image20 = self._get_image_by_id(image20.id)
# print(image20.status)
self.assertEqual(image20.status, "deactivated")
print("TC-3.5.5")
# Reactivate attempt fails for another project's admin
self.set_connections_for_user(self.user11)
self._reactivate_image(image20)
image20 = self._get_image_by_id(image20.id)
# print(image20.status)
self.assertEqual(image20.status, "deactivated")
print("TC-3.5.6")
# Reactivate job succeeds for the admin of project 2
self.set_connections_for_user(self.user21)
self._reactivate_image(image20)
image20 = self._get_image_by_id(image20.id)
# print(image20.status)
self.assertEqual(image20.status, "active")
# ---------------------------------------------------------------------
# TC-4
# Description:
# User 11 can publicize a image of project1, while users 12 and 13
# cannot
# ---------------------------------------------------------------------
def test_uc_image_4(self):
"""
1. user11 tries to create/update a image 'image11' with
'visibility: public', should succeed,
2. user12/13 tries to create/update a image 'image12' with
'visibility: public', should fail,"
"""
# ---------------------------------------------------------------------
# 4.1. user11 tries to create/update a image 'image11' with
# 'visibility: public' should succeed
# ---------------------------------------------------------------------
print("TC-4.1")
self.set_connections_for_user(self.user11)
# Create image to publicize it
image11 = self._create_image("image11P", filename="cirros-0.3.4-x86_64-disk.img", visibility="public")
self.assertIn("image11P", [image.name for image in self._list_images()])
image11 = self._get_image_by_id(image11.id)
# print(image11.status)
self.assertEqual(image11.status, "active")
# Modify (update) image to publicize it
shared_image_11 = self._create_image("shared-image-11", filename="cirros-0.3.4-x86_64-disk.img", visibility="shared")
public_image = self._update_image(shared_image_11, name="public-image", visibility="public")
self.assertEqual(public_image.name, "public-image")
self.assertEqual(public_image.visibility, "public")
self.assertIn("public-image", [image.name for image in self._list_images()])
self.assertNotIn("shared-image-11", [image.name for image in self._list_images()])
# ---------------------------------------------------------------------
# 4.2. user12/13 attempt to create/update an image 'image12' to
# publicize image should fail
# ---------------------------------------------------------------------
print("TC-4.2")
# Create attempt fails for project members
for user in (self.user12, self.user13):
self.set_connections_for_user(user)
self.assertRaisesRegex(
Exception,
"You are not authorized to complete publicize_image action",
self._create_image, "image12P", filename="cirros-0.3.4-x86_64-disk.img", visibility="public"
)
self.assertNotIn("image12P", [image.name for image in self._list_images()])
# Modify (update) attempt fails for project members
self.set_connections_for_user(self.user12)
shared_image_12 = self._create_image("shared-image-12", filename="cirros-0.3.4-x86_64-disk.img", visibility="shared")
for user in (self.user12, self.user13):
self.set_connections_for_user(user)
self.assertRaisesRegex(
Exception,
"Forbidden",
self._update_image, shared_image_12, name="public-image-12", visibility="public"
)
self.assertIn("shared-image-12", [image.name for image in self._list_images()])
self.assertNotIn("public-image-12", [image.name for image in self._list_images()])
# ---------------------------------------------------------------------
# TC-5
# Description:
# User 11 can communitize a image of project1, while users 12 and 13
# cannot
# ---------------------------------------------------------------------
def test_uc_image_5(self):
"""
1. user11 create/update a image 'image11' with 'visibility: community'
2. user12/13 tries to create/update a image 'image12' with 'visibility: community', should fail
"""
# ---------------------------------------------------------------------
# 5.1. user11 can create/update a image 'image11' with
# 'visibility: community'
# ---------------------------------------------------------------------
print("TC-5.1")
self.set_connections_for_user(self.user11)
# Create image to communitize it
image11 = self._create_image("image11C", filename="cirros-0.3.4-x86_64-disk.img", visibility="community")
self.assertIn("image11C", [image.name for image in self._list_images()])
image11 = self._get_image_by_id(image11.id)
# print(image11.status)
self.assertEqual(image11.status, "active")
# Modify (update) image to communitize it
shared_image_11 = self._create_image("shared-image-11", filename="cirros-0.3.4-x86_64-disk.img", visibility="shared")
community_image = self._update_image(shared_image_11, name="community-image", visibility="community")
self.assertEqual(community_image.name, "community-image")
self.assertEqual(community_image.visibility, "community")
self.assertIn("community-image", [image.name for image in self._list_images()])
self.assertNotIn("shared-image-11", [image.name for image in self._list_images()])
# ---------------------------------------------------------------------
# 5.2. user12/13 attempt to create/update an image 'image12' to
# communitize the image should fail
# ---------------------------------------------------------------------
print("TC-5.2")
# Create attempt fails for project members
for user in (self.user12, self.user13):
self.set_connections_for_user(user)
self.assertRaisesRegex(
Exception,
"You are not authorized to complete communitize_image action",
self._create_image, "image12C",
filename="cirros-0.3.4-x86_64-disk.img", visibility="community"
)
self.assertNotIn("image12C", [image.name for image in self._list_images()])
# Modify (update) attempt fails for project members
self.set_connections_for_user(self.user12)
shared_image_12 = self._create_image("shared-image-12", filename="cirros-0.3.4-x86_64-disk.img", visibility="shared")
for user in (self.user12, self.user13):
self.set_connections_for_user(user)
self.assertRaisesRegex(
Exception,
"Forbidden",
self._update_image, shared_image_12, name="community-image-12", visibility="community"
)
self.assertIn("shared-image-12", [image.name for image in self._list_images()])
self.assertNotIn("community-image-12", [image.name for image in self._list_images()])