2019-12-15 15:06:08 +00:00
|
|
|
import json
|
2022-07-26 19:21:07 +00:00
|
|
|
from copy import deepcopy
|
2019-12-20 09:54:38 +00:00
|
|
|
from datetime import datetime
|
2019-12-15 15:06:08 +00:00
|
|
|
|
|
|
|
import boto3
|
2020-10-06 05:54:49 +00:00
|
|
|
import pytest
|
2023-11-30 15:55:51 +00:00
|
|
|
from botocore.exceptions import ClientError
|
2019-12-15 15:06:08 +00:00
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
from moto import mock_aws
|
2019-12-15 15:06:08 +00:00
|
|
|
|
2022-07-26 19:21:07 +00:00
|
|
|
expected_pipeline_details = {
|
|
|
|
"name": "test-pipeline",
|
|
|
|
"roleArn": "arn:aws:iam::123456789012:role/test-role",
|
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
"configuration": {
|
|
|
|
"S3Bucket": "test-bucket",
|
|
|
|
"S3ObjectKey": "test-object",
|
|
|
|
},
|
|
|
|
"outputArtifacts": [{"name": "artifact"}],
|
|
|
|
"inputArtifacts": [],
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "Stage-2",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Approval",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "Manual",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
"configuration": {},
|
|
|
|
"outputArtifacts": [],
|
|
|
|
"inputArtifacts": [],
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
"version": 1,
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 15:06:08 +00:00
|
|
|
def test_create_pipeline():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
2019-12-22 10:42:15 +00:00
|
|
|
response = create_basic_codepipeline(client, "test-pipeline")
|
2019-12-15 15:06:08 +00:00
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["pipeline"] == expected_pipeline_details
|
|
|
|
assert response["tags"] == [{"key": "key", "value": "value"}]
|
2019-12-15 15:06:08 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 15:06:08 +00:00
|
|
|
def test_create_pipeline_errors():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
client_iam = boto3.client("iam", region_name="us-east-1")
|
2019-12-22 10:42:15 +00:00
|
|
|
create_basic_codepipeline(client, "test-pipeline")
|
2019-12-15 15:06:08 +00:00
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-22 10:42:15 +00:00
|
|
|
create_basic_codepipeline(client, "test-pipeline")
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "CreatePipeline"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "InvalidStructureException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "A pipeline with the name 'test-pipeline' already exists in account '123456789012'"
|
2019-12-15 15:06:08 +00:00
|
|
|
)
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-15 15:06:08 +00:00
|
|
|
client.create_pipeline(
|
|
|
|
pipeline={
|
|
|
|
"name": "invalid-pipeline",
|
|
|
|
"roleArn": "arn:aws:iam::123456789012:role/not-existing",
|
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "CreatePipeline"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "InvalidStructureException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "CodePipeline is not authorized to perform AssumeRole on role arn:aws:iam::123456789012:role/not-existing"
|
2019-12-15 15:06:08 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
wrong_role_arn = client_iam.create_role(
|
|
|
|
RoleName="wrong-role",
|
|
|
|
AssumeRolePolicyDocument=json.dumps(
|
|
|
|
{
|
|
|
|
"Version": "2012-10-17",
|
|
|
|
"Statement": [
|
|
|
|
{
|
|
|
|
"Effect": "Allow",
|
|
|
|
"Principal": {"Service": "s3.amazonaws.com"},
|
|
|
|
"Action": "sts:AssumeRole",
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
),
|
|
|
|
)["Role"]["Arn"]
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-15 15:06:08 +00:00
|
|
|
client.create_pipeline(
|
|
|
|
pipeline={
|
|
|
|
"name": "invalid-pipeline",
|
|
|
|
"roleArn": wrong_role_arn,
|
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "CreatePipeline"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "InvalidStructureException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "CodePipeline is not authorized to perform AssumeRole on role arn:aws:iam::123456789012:role/wrong-role"
|
2019-12-15 15:06:08 +00:00
|
|
|
)
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-15 15:06:08 +00:00
|
|
|
client.create_pipeline(
|
|
|
|
pipeline={
|
|
|
|
"name": "invalid-pipeline",
|
|
|
|
"roleArn": get_role_arn(),
|
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "CreatePipeline"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "InvalidStructureException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "Pipeline has only 1 stage(s). There should be a minimum of 2 stages in a pipeline"
|
2019-12-15 15:06:08 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 15:54:58 +00:00
|
|
|
def test_get_pipeline():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
2019-12-22 10:42:15 +00:00
|
|
|
create_basic_codepipeline(client, "test-pipeline")
|
2019-12-15 15:54:58 +00:00
|
|
|
|
|
|
|
response = client.get_pipeline(name="test-pipeline")
|
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["pipeline"] == expected_pipeline_details
|
|
|
|
assert (
|
|
|
|
response["metadata"]["pipelineArn"]
|
|
|
|
== "arn:aws:codepipeline:us-east-1:123456789012:test-pipeline"
|
2019-12-15 15:54:58 +00:00
|
|
|
)
|
2023-07-08 10:25:06 +00:00
|
|
|
assert isinstance(response["metadata"]["created"], datetime)
|
|
|
|
assert isinstance(response["metadata"]["updated"], datetime)
|
2019-12-15 15:54:58 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 15:54:58 +00:00
|
|
|
def test_get_pipeline_errors():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-15 16:28:59 +00:00
|
|
|
client.get_pipeline(name="not-existing")
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "GetPipeline"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "PipelineNotFoundException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "Account '123456789012' does not have a pipeline with name 'not-existing'"
|
2019-12-15 15:54:58 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 16:28:59 +00:00
|
|
|
def test_update_pipeline():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
2019-12-22 10:42:15 +00:00
|
|
|
create_basic_codepipeline(client, "test-pipeline")
|
2019-12-19 20:41:32 +00:00
|
|
|
|
|
|
|
response = client.get_pipeline(name="test-pipeline")
|
|
|
|
created_time = response["metadata"]["created"]
|
|
|
|
updated_time = response["metadata"]["updated"]
|
|
|
|
|
|
|
|
response = client.update_pipeline(
|
|
|
|
pipeline={
|
|
|
|
"name": "test-pipeline",
|
2019-12-22 10:42:15 +00:00
|
|
|
"roleArn": get_role_arn(),
|
2019-12-19 20:41:32 +00:00
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
2019-12-15 16:28:59 +00:00
|
|
|
},
|
2019-12-19 20:41:32 +00:00
|
|
|
"configuration": {
|
|
|
|
"S3Bucket": "different-bucket",
|
|
|
|
"S3ObjectKey": "test-object",
|
|
|
|
},
|
2022-03-10 14:39:59 +00:00
|
|
|
"outputArtifacts": [{"name": "artifact"}],
|
2019-12-19 20:41:32 +00:00
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "Stage-2",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Approval",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "Manual",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
2019-12-15 16:28:59 +00:00
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["pipeline"] == {
|
|
|
|
"name": "test-pipeline",
|
|
|
|
"roleArn": "arn:aws:iam::123456789012:role/test-role",
|
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
"configuration": {
|
|
|
|
"S3Bucket": "different-bucket",
|
|
|
|
"S3ObjectKey": "test-object",
|
|
|
|
},
|
|
|
|
"outputArtifacts": [{"name": "artifact"}],
|
|
|
|
"inputArtifacts": [],
|
|
|
|
}
|
|
|
|
],
|
2019-12-15 16:28:59 +00:00
|
|
|
},
|
2023-07-08 10:25:06 +00:00
|
|
|
{
|
|
|
|
"name": "Stage-2",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Approval",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "Manual",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"runOrder": 1,
|
|
|
|
"configuration": {},
|
|
|
|
"outputArtifacts": [],
|
|
|
|
"inputArtifacts": [],
|
|
|
|
}
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
"version": 2,
|
|
|
|
}
|
2019-12-15 16:28:59 +00:00
|
|
|
|
2019-12-19 20:41:32 +00:00
|
|
|
metadata = client.get_pipeline(name="test-pipeline")["metadata"]
|
2023-07-08 10:25:06 +00:00
|
|
|
assert metadata["created"] == created_time
|
|
|
|
assert metadata["updated"] > updated_time
|
2019-12-15 16:28:59 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 16:28:59 +00:00
|
|
|
def test_update_pipeline_errors():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-15 16:28:59 +00:00
|
|
|
client.update_pipeline(
|
|
|
|
pipeline={
|
|
|
|
"name": "not-existing",
|
|
|
|
"roleArn": get_role_arn(),
|
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"configuration": {
|
|
|
|
"S3Bucket": "test-bucket",
|
|
|
|
"S3ObjectKey": "test-object",
|
|
|
|
},
|
2022-03-10 14:39:59 +00:00
|
|
|
"outputArtifacts": [{"name": "artifact"}],
|
2019-12-15 16:28:59 +00:00
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "Stage-2",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Approval",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "Manual",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "UpdatePipeline"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "ResourceNotFoundException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "The account with id '123456789012' does not include a pipeline with the name 'not-existing'"
|
2019-12-15 16:28:59 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 16:44:54 +00:00
|
|
|
def test_list_pipelines():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
2019-12-22 10:42:15 +00:00
|
|
|
name_1 = "test-pipeline-1"
|
|
|
|
create_basic_codepipeline(client, name_1)
|
|
|
|
name_2 = "test-pipeline-2"
|
|
|
|
create_basic_codepipeline(client, name_2)
|
2019-12-15 16:44:54 +00:00
|
|
|
|
|
|
|
response = client.list_pipelines()
|
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert len(response["pipelines"]) == 2
|
|
|
|
assert response["pipelines"][0]["name"] == name_1
|
|
|
|
assert response["pipelines"][0]["version"] == 1
|
|
|
|
assert isinstance(response["pipelines"][0]["created"], datetime)
|
|
|
|
assert isinstance(response["pipelines"][0]["updated"], datetime)
|
|
|
|
assert response["pipelines"][1]["name"] == name_2
|
|
|
|
assert response["pipelines"][1]["version"] == 1
|
|
|
|
assert isinstance(response["pipelines"][1]["created"], datetime)
|
|
|
|
assert isinstance(response["pipelines"][1]["updated"], datetime)
|
2019-12-15 16:44:54 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-15 16:58:38 +00:00
|
|
|
def test_delete_pipeline():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
2019-12-22 10:42:15 +00:00
|
|
|
name = "test-pipeline"
|
|
|
|
create_basic_codepipeline(client, name)
|
2023-07-08 10:25:06 +00:00
|
|
|
assert len(client.list_pipelines()["pipelines"]) == 1
|
2019-12-22 10:42:15 +00:00
|
|
|
|
|
|
|
client.delete_pipeline(name=name)
|
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert len(client.list_pipelines()["pipelines"]) == 0
|
2019-12-22 10:42:15 +00:00
|
|
|
|
|
|
|
# deleting a not existing pipeline, should raise no exception
|
|
|
|
client.delete_pipeline(name=name)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-22 10:42:15 +00:00
|
|
|
def test_list_tags_for_resource():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
name = "test-pipeline"
|
|
|
|
create_basic_codepipeline(client, name)
|
|
|
|
|
|
|
|
response = client.list_tags_for_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}"
|
2019-12-22 10:42:15 +00:00
|
|
|
)
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["tags"] == [{"key": "key", "value": "value"}]
|
2019-12-22 10:42:15 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-22 10:42:15 +00:00
|
|
|
def test_list_tags_for_resource_errors():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-22 10:42:15 +00:00
|
|
|
client.list_tags_for_resource(
|
|
|
|
resourceArn="arn:aws:codepipeline:us-east-1:123456789012:not-existing"
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "ListTagsForResource"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "ResourceNotFoundException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "The account with id '123456789012' does not include a pipeline with the name 'not-existing'"
|
2019-12-22 10:42:15 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-23 18:33:37 +00:00
|
|
|
def test_tag_resource():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
name = "test-pipeline"
|
|
|
|
create_basic_codepipeline(client, name)
|
|
|
|
|
|
|
|
client.tag_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}",
|
2019-12-23 18:33:37 +00:00
|
|
|
tags=[{"key": "key-2", "value": "value-2"}],
|
|
|
|
)
|
|
|
|
|
|
|
|
response = client.list_tags_for_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}"
|
2019-12-23 18:33:37 +00:00
|
|
|
)
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["tags"] == [
|
|
|
|
{"key": "key", "value": "value"},
|
|
|
|
{"key": "key-2", "value": "value-2"},
|
|
|
|
]
|
2019-12-23 18:33:37 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-23 18:33:37 +00:00
|
|
|
def test_tag_resource_errors():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
name = "test-pipeline"
|
|
|
|
create_basic_codepipeline(client, name)
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-23 18:33:37 +00:00
|
|
|
client.tag_resource(
|
|
|
|
resourceArn="arn:aws:codepipeline:us-east-1:123456789012:not-existing",
|
|
|
|
tags=[{"key": "key-2", "value": "value-2"}],
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "TagResource"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "ResourceNotFoundException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "The account with id '123456789012' does not include a pipeline with the name 'not-existing'"
|
2019-12-23 18:33:37 +00:00
|
|
|
)
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-23 18:33:37 +00:00
|
|
|
client.tag_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}",
|
2019-12-23 18:33:37 +00:00
|
|
|
tags=[{"key": "aws:key", "value": "value"}],
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "TagResource"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "InvalidTagsException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "Not allowed to modify system tags. System tags start with 'aws:'. msg=[Caller is an end user and not allowed to mutate system tags]"
|
2019-12-23 18:33:37 +00:00
|
|
|
)
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-23 18:33:37 +00:00
|
|
|
client.tag_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}",
|
|
|
|
tags=[{"key": f"key-{i}", "value": f"value-{i}"} for i in range(50)],
|
2019-12-23 18:33:37 +00:00
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "TagResource"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "TooManyTagsException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== f"Tag limit exceeded for resource [arn:aws:codepipeline:us-east-1:123456789012:{name}]."
|
2019-12-23 18:33:37 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-23 18:50:16 +00:00
|
|
|
def test_untag_resource():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
name = "test-pipeline"
|
|
|
|
create_basic_codepipeline(client, name)
|
|
|
|
|
|
|
|
response = client.list_tags_for_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}"
|
2019-12-23 18:50:16 +00:00
|
|
|
)
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["tags"] == [{"key": "key", "value": "value"}]
|
2019-12-23 18:50:16 +00:00
|
|
|
|
|
|
|
client.untag_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}",
|
2019-12-23 18:50:16 +00:00
|
|
|
tagKeys=["key"],
|
|
|
|
)
|
|
|
|
|
|
|
|
response = client.list_tags_for_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}"
|
2019-12-23 18:50:16 +00:00
|
|
|
)
|
2023-07-08 10:25:06 +00:00
|
|
|
assert len(response["tags"]) == 0
|
2019-12-23 18:50:16 +00:00
|
|
|
|
|
|
|
# removing a not existing tag should raise no exception
|
|
|
|
client.untag_resource(
|
2022-11-17 22:41:08 +00:00
|
|
|
resourceArn=f"arn:aws:codepipeline:us-east-1:123456789012:{name}",
|
2019-12-23 18:50:16 +00:00
|
|
|
tagKeys=["key"],
|
|
|
|
)
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2019-12-23 18:50:16 +00:00
|
|
|
def test_untag_resource_errors():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
2020-10-06 05:54:49 +00:00
|
|
|
with pytest.raises(ClientError) as e:
|
2019-12-23 18:50:16 +00:00
|
|
|
client.untag_resource(
|
|
|
|
resourceArn="arn:aws:codepipeline:us-east-1:123456789012:not-existing",
|
|
|
|
tagKeys=["key"],
|
|
|
|
)
|
2020-10-06 06:04:09 +00:00
|
|
|
ex = e.value
|
2023-07-08 10:25:06 +00:00
|
|
|
assert ex.operation_name == "UntagResource"
|
|
|
|
assert ex.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
assert ex.response["Error"]["Code"] == "ResourceNotFoundException"
|
|
|
|
assert (
|
|
|
|
ex.response["Error"]["Message"]
|
|
|
|
== "The account with id '123456789012' does not include a pipeline with the name 'not-existing'"
|
2019-12-23 18:50:16 +00:00
|
|
|
)
|
|
|
|
|
|
|
|
|
2022-07-26 19:21:07 +00:00
|
|
|
simple_trust_policy = {
|
|
|
|
"Version": "2012-10-17",
|
|
|
|
"Statement": [
|
|
|
|
{
|
|
|
|
"Effect": "Allow",
|
|
|
|
"Principal": {"Service": "codepipeline.amazonaws.com"},
|
|
|
|
"Action": "sts:AssumeRole",
|
|
|
|
}
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
|
|
def get_role_arn(name="test-role", trust_policy=None):
|
2019-12-22 10:42:15 +00:00
|
|
|
client = boto3.client("iam", region_name="us-east-1")
|
|
|
|
try:
|
2022-07-26 19:21:07 +00:00
|
|
|
return client.get_role(RoleName=name)["Role"]["Arn"]
|
2019-12-22 10:42:15 +00:00
|
|
|
except ClientError:
|
2022-07-26 19:21:07 +00:00
|
|
|
if trust_policy is None:
|
|
|
|
trust_policy = simple_trust_policy
|
2019-12-22 10:42:15 +00:00
|
|
|
return client.create_role(
|
2022-07-26 19:21:07 +00:00
|
|
|
RoleName=name,
|
|
|
|
AssumeRolePolicyDocument=json.dumps(trust_policy),
|
2019-12-22 10:42:15 +00:00
|
|
|
)["Role"]["Arn"]
|
|
|
|
|
|
|
|
|
2022-07-26 19:21:07 +00:00
|
|
|
def create_basic_codepipeline(client, name, role_arn=None, tags=None):
|
|
|
|
if role_arn is None:
|
|
|
|
role_arn = get_role_arn()
|
|
|
|
if tags is None:
|
|
|
|
tags = [{"key": "key", "value": "value"}]
|
2019-12-22 10:42:15 +00:00
|
|
|
return client.create_pipeline(
|
2019-12-15 16:58:38 +00:00
|
|
|
pipeline={
|
2019-12-22 10:42:15 +00:00
|
|
|
"name": name,
|
2022-07-26 19:21:07 +00:00
|
|
|
"roleArn": role_arn,
|
2019-12-15 16:58:38 +00:00
|
|
|
"artifactStore": {
|
|
|
|
"type": "S3",
|
|
|
|
"location": "codepipeline-us-east-1-123456789012",
|
|
|
|
},
|
|
|
|
"stages": [
|
|
|
|
{
|
|
|
|
"name": "Stage-1",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Source",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "S3",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
"configuration": {
|
|
|
|
"S3Bucket": "test-bucket",
|
|
|
|
"S3ObjectKey": "test-object",
|
|
|
|
},
|
2022-03-10 14:39:59 +00:00
|
|
|
"outputArtifacts": [{"name": "artifact"}],
|
2019-12-15 16:58:38 +00:00
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"name": "Stage-2",
|
|
|
|
"actions": [
|
|
|
|
{
|
|
|
|
"name": "Action-1",
|
|
|
|
"actionTypeId": {
|
|
|
|
"category": "Approval",
|
|
|
|
"owner": "AWS",
|
|
|
|
"provider": "Manual",
|
|
|
|
"version": "1",
|
|
|
|
},
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
|
|
|
],
|
|
|
|
},
|
2022-07-26 19:21:07 +00:00
|
|
|
tags=tags,
|
2019-12-15 16:58:38 +00:00
|
|
|
)
|
2022-07-26 19:21:07 +00:00
|
|
|
|
|
|
|
|
|
|
|
extended_trust_policy = {
|
|
|
|
"Version": "2012-10-17",
|
|
|
|
"Statement": [
|
|
|
|
{
|
|
|
|
"Effect": "Allow",
|
|
|
|
"Principal": {"Service": "codebuild.amazonaws.com"},
|
|
|
|
"Action": "sts:AssumeRole",
|
|
|
|
},
|
|
|
|
{
|
|
|
|
"Effect": "Allow",
|
|
|
|
"Principal": {"Service": "codepipeline.amazonaws.com"},
|
|
|
|
"Action": "sts:AssumeRole",
|
|
|
|
},
|
|
|
|
],
|
|
|
|
}
|
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2022-07-26 19:21:07 +00:00
|
|
|
def test_create_pipeline_with_extended_trust_policy():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
|
|
|
role_arn = get_role_arn(
|
|
|
|
name="test-role-extended", trust_policy=extended_trust_policy
|
|
|
|
)
|
|
|
|
response = create_basic_codepipeline(client, "test-pipeline", role_arn=role_arn)
|
|
|
|
|
|
|
|
extended_pipeline_details = deepcopy(expected_pipeline_details)
|
|
|
|
extended_pipeline_details["roleArn"] = role_arn
|
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["pipeline"] == extended_pipeline_details
|
|
|
|
assert response["tags"] == [{"key": "key", "value": "value"}]
|
2022-07-26 19:21:07 +00:00
|
|
|
|
|
|
|
|
2024-01-07 12:03:33 +00:00
|
|
|
@mock_aws
|
2022-07-26 19:21:07 +00:00
|
|
|
def test_create_pipeline_without_tags():
|
|
|
|
client = boto3.client("codepipeline", region_name="us-east-1")
|
|
|
|
|
|
|
|
response = create_basic_codepipeline(client, "test-pipeline", tags=[])
|
|
|
|
|
2023-07-08 10:25:06 +00:00
|
|
|
assert response["pipeline"] == expected_pipeline_details
|
|
|
|
assert response["tags"] == []
|