395 lines
13 KiB
Python
395 lines
13 KiB
Python
import uuid
|
|
|
|
import boto3
|
|
import pytest
|
|
from botocore.exceptions import ClientError
|
|
|
|
from moto import mock_databrew
|
|
from moto.core import DEFAULT_ACCOUNT_ID as ACCOUNT_ID
|
|
|
|
|
|
def _create_databrew_client():
|
|
client = boto3.client("databrew", region_name="us-west-1")
|
|
return client
|
|
|
|
|
|
def _create_test_profile_job(
|
|
client,
|
|
dataset_name=None,
|
|
job_name=None,
|
|
output_location=None,
|
|
role_arn=None,
|
|
tags=None,
|
|
):
|
|
kwargs = {}
|
|
kwargs["Name"] = job_name or str(uuid.uuid4())
|
|
kwargs["RoleArn"] = role_arn or str(uuid.uuid4())
|
|
kwargs["DatasetName"] = dataset_name or str(uuid.uuid4())
|
|
kwargs["OutputLocation"] = output_location or {"Bucket": str(uuid.uuid4())}
|
|
if tags is not None:
|
|
kwargs["Tags"] = tags
|
|
|
|
return client.create_profile_job(**kwargs)
|
|
|
|
|
|
def _create_test_recipe_job(
|
|
client,
|
|
job_name=None,
|
|
role_arn=None,
|
|
tags=None,
|
|
encryption_mode=None,
|
|
log_subscription=None,
|
|
dataset_name=None,
|
|
project_name=None,
|
|
):
|
|
kwargs = {}
|
|
kwargs["Name"] = job_name or str(uuid.uuid4())
|
|
kwargs["RoleArn"] = role_arn or str(uuid.uuid4())
|
|
if tags is not None:
|
|
kwargs["Tags"] = tags
|
|
if encryption_mode is not None:
|
|
kwargs["EncryptionMode"] = encryption_mode
|
|
if log_subscription is not None:
|
|
kwargs["LogSubscription"] = log_subscription
|
|
if dataset_name is not None:
|
|
kwargs["DatasetName"] = dataset_name
|
|
if project_name is not None:
|
|
kwargs["ProjectName"] = project_name
|
|
|
|
return client.create_recipe_job(**kwargs)
|
|
|
|
|
|
def _create_test_recipe_jobs(client, count, **kwargs):
|
|
for _ in range(count):
|
|
_create_test_recipe_job(client, **kwargs)
|
|
|
|
|
|
def _create_test_profile_jobs(client, count, **kwargs):
|
|
for _ in range(count):
|
|
_create_test_profile_job(client, **kwargs)
|
|
|
|
|
|
@mock_databrew
|
|
def test_create_profile_job_that_already_exists():
|
|
client = _create_databrew_client()
|
|
|
|
response = _create_test_profile_job(client)
|
|
job_name = response["Name"]
|
|
with pytest.raises(ClientError) as exc:
|
|
_create_test_profile_job(client, job_name=response["Name"])
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ConflictException"
|
|
assert err["Message"] == f"The job {job_name} profile job already exists."
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 409
|
|
|
|
|
|
@mock_databrew
|
|
def test_create_recipe_job_that_already_exists():
|
|
client = _create_databrew_client()
|
|
|
|
response = _create_test_recipe_job(client)
|
|
job_name = response["Name"]
|
|
with pytest.raises(ClientError) as exc:
|
|
_create_test_recipe_job(client, job_name=response["Name"])
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ConflictException"
|
|
assert err["Message"] == f"The job {job_name} recipe job already exists."
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 409
|
|
|
|
|
|
@mock_databrew
|
|
def test_create_recipe_job_with_invalid_encryption_mode():
|
|
client = _create_databrew_client()
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
_create_test_recipe_job(client, encryption_mode="INVALID")
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ValidationException"
|
|
assert (
|
|
err["Message"]
|
|
== "1 validation error detected: Value 'INVALID' at 'encryptionMode' failed to satisfy constraint: Member must satisfy enum value set: [SSE-S3, SSE-KMS]"
|
|
)
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
|
|
@mock_databrew
|
|
def test_create_recipe_job_with_invalid_log_subscription_value():
|
|
client = _create_databrew_client()
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
_create_test_recipe_job(client, log_subscription="INVALID")
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ValidationException"
|
|
assert (
|
|
err["Message"]
|
|
== "1 validation error detected: Value 'INVALID' at 'logSubscription' failed to satisfy constraint: Member must satisfy enum value set: [ENABLE, DISABLE]"
|
|
)
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
|
|
@mock_databrew
|
|
def test_create_recipe_job_with_same_name_as_profile_job():
|
|
client = _create_databrew_client()
|
|
|
|
response = _create_test_profile_job(client)
|
|
job_name = response["Name"]
|
|
with pytest.raises(ClientError) as exc:
|
|
_create_test_recipe_job(client, job_name=response["Name"])
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ConflictException"
|
|
assert err["Message"] == f"The job {job_name} profile job already exists."
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 409
|
|
|
|
|
|
@mock_databrew
|
|
def test_describe_recipe_job():
|
|
client = _create_databrew_client()
|
|
|
|
response = _create_test_recipe_job(client)
|
|
job_name = response["Name"]
|
|
job = client.describe_job(Name=job_name)
|
|
assert job["Name"] == response["Name"]
|
|
assert job["Type"] == "RECIPE"
|
|
assert (
|
|
job["ResourceArn"] == f"arn:aws:databrew:us-west-1:{ACCOUNT_ID}:job/{job_name}"
|
|
)
|
|
assert job["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_describe_job_that_does_not_exist():
|
|
client = _create_databrew_client()
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
client.describe_job(Name="DoesNotExist")
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert err["Message"] == "Job DoesNotExist wasn't found."
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 404
|
|
|
|
|
|
@mock_databrew
|
|
def test_describe_job_with_long_name():
|
|
client = _create_databrew_client()
|
|
name = "a" * 241
|
|
with pytest.raises(ClientError) as exc:
|
|
client.describe_job(Name=name)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ValidationException"
|
|
assert (
|
|
err["Message"]
|
|
== f"1 validation error detected: Value '{name}' at 'name' failed to satisfy constraint: Member must have length less than or equal to 240"
|
|
)
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
|
|
@mock_databrew
|
|
@pytest.mark.parametrize("job_name", ["name", "name with space"])
|
|
def test_update_profile_job(job_name):
|
|
client = _create_databrew_client()
|
|
|
|
# Create the job
|
|
response = _create_test_profile_job(client, job_name=job_name)
|
|
|
|
# Update the job by changing RoleArn
|
|
update_response = client.update_profile_job(
|
|
Name=job_name, RoleArn="a" * 20, OutputLocation={"Bucket": "b" * 20}
|
|
)
|
|
assert update_response["Name"] == job_name
|
|
assert update_response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
# Describe the job to check that RoleArn was updated
|
|
job = client.describe_job(Name=job_name)
|
|
assert job["Name"] == response["Name"]
|
|
assert job["RoleArn"] == "a" * 20
|
|
|
|
|
|
@mock_databrew
|
|
@pytest.mark.parametrize("job_name", ["name", "name with space"])
|
|
def test_update_recipe_job(job_name):
|
|
client = _create_databrew_client()
|
|
|
|
# Create the job
|
|
response = _create_test_recipe_job(client, job_name=job_name)
|
|
|
|
# Update the job by changing RoleArn
|
|
update_response = client.update_recipe_job(Name=job_name, RoleArn="a" * 20)
|
|
assert update_response["Name"] == job_name
|
|
assert update_response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
# Describe the job to check that RoleArn was updated
|
|
job = client.describe_job(Name=job_name)
|
|
assert job["Name"] == response["Name"]
|
|
assert job["RoleArn"] == "a" * 20
|
|
|
|
|
|
@mock_databrew
|
|
def test_update_profile_job_does_not_exist():
|
|
client = _create_databrew_client()
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
client.update_profile_job(
|
|
Name="DoesNotExist", RoleArn="a" * 20, OutputLocation={"Bucket": "b" * 20}
|
|
)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert err["Message"] == "The job DoesNotExist wasn't found"
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 404
|
|
|
|
|
|
@mock_databrew
|
|
def test_update_recipe_job_does_not_exist():
|
|
client = _create_databrew_client()
|
|
|
|
with pytest.raises(ClientError) as exc:
|
|
client.update_recipe_job(Name="DoesNotExist", RoleArn="a" * 20)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert err["Message"] == "The job DoesNotExist wasn't found"
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 404
|
|
|
|
|
|
@mock_databrew
|
|
@pytest.mark.parametrize("job_name", ["name", "name with space"])
|
|
def test_delete_job(job_name):
|
|
client = _create_databrew_client()
|
|
|
|
# Create the job
|
|
_create_test_recipe_job(client, job_name=job_name)
|
|
|
|
# Delete the job
|
|
response = client.delete_job(Name=job_name)
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
assert response["Name"] == job_name
|
|
|
|
# Check the job does not exist anymore
|
|
with pytest.raises(ClientError) as exc:
|
|
client.describe_job(Name=job_name)
|
|
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert err["Message"] == f"Job {job_name} wasn't found."
|
|
|
|
|
|
@mock_databrew
|
|
def test_delete_job_does_not_exist():
|
|
client = _create_databrew_client()
|
|
|
|
# Delete the job
|
|
with pytest.raises(ClientError) as exc:
|
|
client.delete_job(Name="DoesNotExist")
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 404
|
|
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ResourceNotFoundException"
|
|
assert err["Message"] == "The job DoesNotExist wasn't found."
|
|
|
|
|
|
@mock_databrew
|
|
def test_delete_job_with_long_name():
|
|
client = _create_databrew_client()
|
|
name = "a" * 241
|
|
with pytest.raises(ClientError) as exc:
|
|
client.delete_job(Name=name)
|
|
err = exc.value.response["Error"]
|
|
assert err["Code"] == "ValidationException"
|
|
assert (
|
|
err["Message"]
|
|
== f"1 validation error detected: Value '{name}' at 'name' failed to satisfy constraint: Member must have length less than or equal to 240"
|
|
)
|
|
assert exc.value.response["ResponseMetadata"]["HTTPStatusCode"] == 400
|
|
|
|
|
|
@mock_databrew
|
|
def test_job_list_when_empty():
|
|
client = _create_databrew_client()
|
|
|
|
response = client.list_jobs()
|
|
assert len(response["Jobs"]) == 0
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_with_max_results():
|
|
client = _create_databrew_client()
|
|
|
|
_create_test_recipe_jobs(client, 4)
|
|
response = client.list_jobs(MaxResults=2)
|
|
assert len(response["Jobs"]) == 2
|
|
assert "NextToken" in response
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_from_next_token():
|
|
client = _create_databrew_client()
|
|
_create_test_recipe_jobs(client, 10)
|
|
first_response = client.list_jobs(MaxResults=3)
|
|
response = client.list_jobs(NextToken=first_response["NextToken"])
|
|
assert len(response["Jobs"]) == 7
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_with_max_results_greater_than_actual_results():
|
|
client = _create_databrew_client()
|
|
_create_test_recipe_jobs(client, 4)
|
|
response = client.list_jobs(MaxResults=10)
|
|
assert len(response["Jobs"]) == 4
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_recipe_and_profile():
|
|
client = _create_databrew_client()
|
|
|
|
_create_test_recipe_jobs(client, 4)
|
|
_create_test_profile_jobs(client, 2)
|
|
response = client.list_jobs()
|
|
assert len(response["Jobs"]) == 6
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_dataset_name_filter():
|
|
client = _create_databrew_client()
|
|
|
|
_create_test_recipe_jobs(client, 3, dataset_name="TEST")
|
|
_create_test_recipe_jobs(client, 1)
|
|
_create_test_profile_jobs(client, 4, dataset_name="TEST")
|
|
_create_test_profile_jobs(client, 1)
|
|
|
|
response = client.list_jobs(DatasetName="TEST")
|
|
assert len(response["Jobs"]) == 7
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_project_name_filter():
|
|
client = _create_databrew_client()
|
|
|
|
_create_test_recipe_jobs(client, 3, project_name="TEST_PROJECT")
|
|
_create_test_recipe_jobs(client, 1)
|
|
_create_test_profile_jobs(client, 1)
|
|
|
|
response = client.list_jobs(ProjectName="TEST_PROJECT")
|
|
assert len(response["Jobs"]) == 3
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|
|
|
|
|
|
@mock_databrew
|
|
def test_list_jobs_dataset_name_and_project_name_filter():
|
|
client = _create_databrew_client()
|
|
|
|
_create_test_recipe_jobs(client, 1, dataset_name="TEST")
|
|
_create_test_recipe_jobs(client, 1, project_name="TEST_PROJECT")
|
|
_create_test_recipe_jobs(
|
|
client, 10, dataset_name="TEST", project_name="TEST_PROJECT"
|
|
)
|
|
_create_test_recipe_jobs(client, 1)
|
|
_create_test_profile_jobs(client, 1)
|
|
|
|
response = client.list_jobs(DatasetName="TEST", ProjectName="TEST_PROJECT")
|
|
assert len(response["Jobs"]) == 10
|
|
assert response["ResponseMetadata"]["HTTPStatusCode"] == 200
|