moto/tests/test_s3/test_s3_logging.py

Ignoring revisions in .git-blame-ignore-revs. Click here to bypass and see the normal blame view.

590 lines
20 KiB
Python
Raw Normal View History

import json
from unittest import SkipTest
from unittest.mock import patch
2023-01-16 23:36:08 +00:00
import boto3
import pytest
from botocore.client import ClientError
2023-01-16 23:36:08 +00:00
from moto import mock_s3, settings
from moto.core import DEFAULT_ACCOUNT_ID
from moto.s3 import s3_backends
from moto.s3.models import FakeBucket
from moto.s3.responses import DEFAULT_REGION_NAME
2023-01-16 23:36:08 +00:00
@mock_s3
def test_put_bucket_logging():
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
2023-01-16 23:36:08 +00:00
bucket_name = "mybucket"
log_bucket = "logbucket"
wrong_region_bucket = "wrongregionlogbucket"
s3_client.create_bucket(Bucket=bucket_name)
# Adding the ACL for log-delivery later...
s3_client.create_bucket(Bucket=log_bucket)
s3_client.create_bucket(
2023-01-16 23:36:08 +00:00
Bucket=wrong_region_bucket,
CreateBucketConfiguration={"LocationConstraint": "us-west-2"},
)
# No logging config:
result = s3_client.get_bucket_logging(Bucket=bucket_name)
2023-01-16 23:36:08 +00:00
assert not result.get("LoggingEnabled")
# A log-bucket that doesn't exist:
with pytest.raises(ClientError) as err:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {"TargetBucket": "IAMNOTREAL", "TargetPrefix": ""}
},
)
assert err.value.response["Error"]["Code"] == "InvalidTargetBucketForLogging"
# A log-bucket that's missing the proper ACLs for LogDelivery:
with pytest.raises(ClientError) as err:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {"TargetBucket": log_bucket, "TargetPrefix": ""}
},
)
assert err.value.response["Error"]["Code"] == "InvalidTargetBucketForLogging"
assert "log-delivery" in err.value.response["Error"]["Message"]
# Add the proper "log-delivery" ACL to the log buckets:
bucket_owner = s3_client.get_bucket_acl(Bucket=log_bucket)["Owner"]
2023-01-16 23:36:08 +00:00
for bucket in [log_bucket, wrong_region_bucket]:
s3_client.put_bucket_acl(
2023-01-16 23:36:08 +00:00
Bucket=bucket,
AccessControlPolicy={
"Grants": [
{
"Grantee": {
"URI": "http://acs.amazonaws.com/groups/s3/LogDelivery",
"Type": "Group",
},
"Permission": "WRITE",
},
{
"Grantee": {
"URI": "http://acs.amazonaws.com/groups/s3/LogDelivery",
"Type": "Group",
},
"Permission": "READ_ACP",
},
{
"Grantee": {"Type": "CanonicalUser", "ID": bucket_owner["ID"]},
"Permission": "FULL_CONTROL",
},
],
"Owner": bucket_owner,
},
)
# A log-bucket that's in the wrong region:
with pytest.raises(ClientError) as err:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": wrong_region_bucket,
"TargetPrefix": "",
}
},
)
assert err.value.response["Error"]["Code"] == "CrossLocationLoggingProhibitted"
# Correct logging:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": log_bucket,
"TargetPrefix": f"{bucket_name}/",
}
},
)
result = s3_client.get_bucket_logging(Bucket=bucket_name)
2023-01-16 23:36:08 +00:00
assert result["LoggingEnabled"]["TargetBucket"] == log_bucket
assert result["LoggingEnabled"]["TargetPrefix"] == f"{bucket_name}/"
assert not result["LoggingEnabled"].get("TargetGrants")
# And disabling:
s3_client.put_bucket_logging(Bucket=bucket_name, BucketLoggingStatus={})
assert not s3_client.get_bucket_logging(Bucket=bucket_name).get("LoggingEnabled")
2023-01-16 23:36:08 +00:00
# And enabling with multiple target grants:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": log_bucket,
"TargetPrefix": f"{bucket_name}/",
"TargetGrants": [
{
"Grantee": {
"ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274",
"Type": "CanonicalUser",
},
"Permission": "READ",
},
{
"Grantee": {
"ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274",
"Type": "CanonicalUser",
},
"Permission": "WRITE",
},
],
}
},
)
result = s3_client.get_bucket_logging(Bucket=bucket_name)
2023-01-16 23:36:08 +00:00
assert len(result["LoggingEnabled"]["TargetGrants"]) == 2
assert (
result["LoggingEnabled"]["TargetGrants"][0]["Grantee"]["ID"]
== "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274"
)
# Test with just 1 grant:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": log_bucket,
"TargetPrefix": f"{bucket_name}/",
"TargetGrants": [
{
"Grantee": {
"ID": "SOMEIDSTRINGHERE9238748923734823917498237489237409123840983274",
"Type": "CanonicalUser",
},
"Permission": "READ",
}
],
}
},
)
result = s3_client.get_bucket_logging(Bucket=bucket_name)
2023-01-16 23:36:08 +00:00
assert len(result["LoggingEnabled"]["TargetGrants"]) == 1
# With an invalid grant:
with pytest.raises(ClientError) as err:
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": log_bucket,
"TargetPrefix": f"{bucket_name}/",
"TargetGrants": [
{
"Grantee": {
"ID": (
"SOMEIDSTRINGHERE9238748923734823917498"
"237489237409123840983274"
),
2023-01-16 23:36:08 +00:00
"Type": "CanonicalUser",
},
"Permission": "NOTAREALPERM",
}
],
}
},
)
assert err.value.response["Error"]["Code"] == "MalformedXML"
@mock_s3
def test_log_file_is_created():
# Create necessary buckets
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
2023-01-16 23:36:08 +00:00
bucket_name = "mybucket"
log_bucket = "logbucket"
s3_client.create_bucket(Bucket=bucket_name)
s3_client.create_bucket(Bucket=log_bucket)
2023-01-16 23:36:08 +00:00
# Enable logging
bucket_owner = s3_client.get_bucket_acl(Bucket=log_bucket)["Owner"]
s3_client.put_bucket_acl(
2023-01-16 23:36:08 +00:00
Bucket=log_bucket,
AccessControlPolicy={
"Grants": [
{
"Grantee": {
"URI": "http://acs.amazonaws.com/groups/s3/LogDelivery",
"Type": "Group",
},
"Permission": "WRITE",
},
{
"Grantee": {
"URI": "http://acs.amazonaws.com/groups/s3/LogDelivery",
"Type": "Group",
},
"Permission": "READ_ACP",
},
{
"Grantee": {"Type": "CanonicalUser", "ID": bucket_owner["ID"]},
"Permission": "FULL_CONTROL",
},
],
"Owner": bucket_owner,
},
)
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": log_bucket,
"TargetPrefix": f"{bucket_name}/",
}
},
)
# Make some requests against the source bucket
s3_client.put_object(Bucket=bucket_name, Key="key1", Body=b"")
s3_client.put_object(Bucket=bucket_name, Key="key2", Body=b"data")
2023-01-16 23:36:08 +00:00
s3_client.put_bucket_logging(
2023-01-16 23:36:08 +00:00
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {"TargetBucket": log_bucket, "TargetPrefix": ""}
},
)
s3_client.list_objects_v2(Bucket=bucket_name)
2023-01-16 23:36:08 +00:00
# Verify files are created in the target (logging) bucket
keys = [k["Key"] for k in s3_client.list_objects_v2(Bucket=log_bucket)["Contents"]]
assert len([k for k in keys if k.startswith("mybucket/")]) == 3
assert len([k for k in keys if not k.startswith("mybucket/")]) == 1
2023-01-16 23:36:08 +00:00
# Verify (roughly) files have the correct content
contents = [
s3_client.get_object(Bucket=log_bucket, Key=key)["Body"].read().decode("utf-8")
2023-01-16 23:36:08 +00:00
for key in keys
]
assert any(c for c in contents if bucket_name in c)
assert any(c for c in contents if "REST.GET.BUCKET" in c)
assert any(c for c in contents if "REST.PUT.BUCKET" in c)
@mock_s3
def test_invalid_bucket_logging_when_permissions_are_false():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
bucket_name = "mybucket"
log_bucket = "logbucket"
s3_client.create_bucket(Bucket=bucket_name)
s3_client.create_bucket(Bucket=log_bucket)
with patch(
"moto.s3.models.FakeBucket._log_permissions_enabled_policy", return_value=False
), patch(
"moto.s3.models.FakeBucket._log_permissions_enabled_acl", return_value=False
):
with pytest.raises(ClientError) as err:
s3_client.put_bucket_logging(
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {"TargetBucket": log_bucket, "TargetPrefix": ""}
},
)
assert err.value.response["Error"]["Code"] == "InvalidTargetBucketForLogging"
assert "log-delivery" in err.value.response["Error"]["Message"]
@mock_s3
def test_valid_bucket_logging_when_permissions_are_true():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
bucket_name = "mybucket"
log_bucket = "logbucket"
s3_client.create_bucket(Bucket=bucket_name)
s3_client.create_bucket(Bucket=log_bucket)
with patch(
"moto.s3.models.FakeBucket._log_permissions_enabled_policy", return_value=True
), patch(
"moto.s3.models.FakeBucket._log_permissions_enabled_acl", return_value=True
):
s3_client.put_bucket_logging(
Bucket=bucket_name,
BucketLoggingStatus={
"LoggingEnabled": {
"TargetBucket": log_bucket,
"TargetPrefix": f"{bucket_name}/",
}
},
)
result = s3_client.get_bucket_logging(Bucket=bucket_name)
assert result["LoggingEnabled"]["TargetBucket"] == log_bucket
assert result["LoggingEnabled"]["TargetPrefix"] == f"{bucket_name}/"
@mock_s3
def test_bucket_policy_not_set():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
s3_backend = s3_backends[DEFAULT_ACCOUNT_ID]["global"]
log_bucket = "log_bucket"
s3_client.create_bucket(Bucket=log_bucket)
log_bucket_obj = s3_backend.get_bucket(log_bucket)
assert (
FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
is False
)
@mock_s3
def test_bucket_policy_principal():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
s3_backend = s3_backends[DEFAULT_ACCOUNT_ID]["global"]
log_bucket = "log_bucket"
s3_client.create_bucket(Bucket=log_bucket)
log_bucket_obj = s3_backend.get_bucket(log_bucket)
invalid_principal_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "not_logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(invalid_principal_policy)
)
assert (
FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
is False
)
s3_client.delete_bucket_policy(Bucket=log_bucket)
valid_principal_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(valid_principal_policy)
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
@mock_s3
def test_bucket_policy_effect():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
s3_backend = s3_backends[DEFAULT_ACCOUNT_ID]["global"]
log_bucket = "log_bucket"
s3_client.create_bucket(Bucket=log_bucket)
log_bucket_obj = s3_backend.get_bucket(log_bucket)
deny_effect_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Deny",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(deny_effect_policy)
)
assert (
FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
is False
)
s3_client.delete_bucket_policy(Bucket=log_bucket)
allow_effect_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(allow_effect_policy)
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
@mock_s3
def test_bucket_policy_action():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
s3_backend = s3_backends[DEFAULT_ACCOUNT_ID]["global"]
log_bucket = "log_bucket"
s3_client.create_bucket(Bucket=log_bucket)
log_bucket_obj = s3_backend.get_bucket(log_bucket)
non_put_object_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:GetObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(non_put_object_policy)
)
assert (
FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
is False
)
s3_client.delete_bucket_policy(Bucket=log_bucket)
put_object_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(Bucket=log_bucket, Policy=json.dumps(put_object_policy))
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
@mock_s3
def test_bucket_policy_resource():
if not settings.TEST_DECORATOR_MODE:
raise SkipTest("Can't patch permission logic in ServerMode")
s3_client = boto3.client("s3", region_name=DEFAULT_REGION_NAME)
s3_backend = s3_backends[DEFAULT_ACCOUNT_ID]["global"]
log_bucket = "log_bucket"
s3_client.create_bucket(Bucket=log_bucket)
log_bucket_obj = s3_backend.get_bucket(log_bucket)
entire_bucket_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(entire_bucket_policy)
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix="prefix"
)
s3_client.delete_bucket_policy(Bucket=log_bucket)
bucket_level_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(bucket_level_policy)
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix="prefix"
)
s3_client.delete_bucket_policy(Bucket=log_bucket)
specfic_prefix_policy = {
"Version": "2012-10-17",
"Statement": [
{
"Sid": "S3ServerAccessLogsPolicy",
"Effect": "Allow",
"Principal": {"Service": "logging.s3.amazonaws.com"},
"Action": ["s3:PutObject"],
"Resource": f"arn:aws:s3:::{log_bucket}/prefix*",
}
],
}
s3_client.put_bucket_policy(
Bucket=log_bucket, Policy=json.dumps(specfic_prefix_policy)
)
assert (
FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix=""
)
is False
)
assert FakeBucket._log_permissions_enabled_policy(
target_bucket=log_bucket_obj, target_prefix="prefix"
)