moto/tests/test_databrew/test_databrew_jobs.py

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
def test_update_profile_job():
client = _create_databrew_client()
# Create the job
response = _create_test_profile_job(client)
job_name = response["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
def test_update_recipe_job():
client = _create_databrew_client()
# Create the job
response = _create_test_recipe_job(client)
job_name = response["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
def test_delete_job():
client = _create_databrew_client()
# Create the job
response = _create_test_recipe_job(client)
job_name = response["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