-
Notifications
You must be signed in to change notification settings - Fork 0
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
Refactor CI pipelines out of Pipelines view into a seperate view; Fix…
… duplicate declaration of Pipeline in Pipelines view
- Loading branch information
Showing
3 changed files
with
137 additions
and
6 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,130 @@ | ||
from typing import List | ||
from pydantic import ValidationError | ||
from django.db import DatabaseError, IntegrityError, OperationalError | ||
from django.forms import model_to_dict | ||
|
||
from backend.views.RestrictedAPIView import RestrictedAPIView | ||
from backend.views.http.responses.errors import ( | ||
Conflict, | ||
BadRequest, | ||
NotFound, | ||
Forbidden, | ||
ServerError as ServerErrorResp | ||
) | ||
from backend.views.http.responses import ResourceURLResponse | ||
from backend.views.http.requests import CIPipeline, ImageBuildTask | ||
from backend.models import ( | ||
Pipeline as PipelineModel, | ||
Archive, | ||
PipelineArchive, | ||
TASK_TYPE_IMAGE_BUILD | ||
) | ||
from backend.services.TaskService import service as task_service | ||
from backend.services.GroupService import service as group_service | ||
from backend.errors.api import BadRequestError | ||
from backend.helpers import resource_url_builder | ||
|
||
|
||
class CIPipelines(RestrictedAPIView): | ||
def post(self, request, group_id, *_, **__): | ||
"""Pipeline requests with type 'ci' are supported in order to make the | ||
process of setting up a ci/cd pipeline as simple as possible. Rather than | ||
specifying tasks, dependencies, etc, we let user pass most the required | ||
data in the top level of the pipeline request.""" | ||
# Get the group | ||
group = group_service.get(group_id, request.tenant_id) | ||
if group == None: | ||
return NotFound(f"No group found with id '{group_id}'") | ||
|
||
# Check that the user belongs to the group | ||
if not group_service.user_in_group(request.username, group_id, request.tenant_id): | ||
return Forbidden(message="You do not have access to this group") | ||
|
||
# Validate the request body based on the type of pipeline specified | ||
prepared_request = self.prepare(CIPipeline) | ||
|
||
# Return the failure view instance if validation failed | ||
if not prepared_request.is_valid: | ||
return prepared_request.failure_view | ||
|
||
# Get the JSON encoded body from the validation result | ||
body = prepared_request.body | ||
|
||
# Check that the id of the pipeline is unique | ||
if PipelineModel.objects.filter(id=body.id, group=group).exists(): | ||
return Conflict(f"A Pipeline already exists with the id '{body.id}'") | ||
|
||
# Create the pipeline | ||
try: | ||
pipeline = PipelineModel.objects.create( | ||
id=body.id, | ||
group=group, | ||
owner=request.username, | ||
max_exec_time=body.execution_profile.max_exec_time, | ||
invocation_mode=body.execution_profile.invocation_mode, | ||
max_retries=body.execution_profile.max_retries, | ||
retry_policy=body.execution_profile.retry_policy, | ||
duplicate_submission_policy=body.execution_profile.duplicate_submission_policy, | ||
env=body.dict()["env"], | ||
params=body.dict()["params"] | ||
) | ||
except (IntegrityError, OperationalError) as e: | ||
return BadRequest(message=e.__cause__) | ||
except Exception as e: | ||
return ServerErrorResp(f"{e}") | ||
|
||
# Fetch the archives specified in the request then create relations | ||
# between them and the pipline | ||
pipeline_archives = [] | ||
try: | ||
# Prevent duplicate pipeline archives by casting id array to 'set' | ||
for archive_id in set(body.archive_ids): | ||
# Fetch the archive object | ||
archive = Archive.objects.filter(group=group, id=archive_id).first() | ||
|
||
# Return bad request if archive not found | ||
if archive == None: | ||
pipeline.delete() | ||
return BadRequest(message=f"Archive not found with an id of '{archive_id}' and group_id '{group.id}'") | ||
|
||
pipeline_archives.append( | ||
PipelineArchive.objects.create( | ||
pipeline=pipeline, | ||
archive=archive | ||
) | ||
) | ||
except (IntegrityError, OperationalError, DatabaseError) as e: | ||
# Delete the pipeline | ||
pipeline.delete() | ||
|
||
# Delete the pipeline archive relationships that were just created | ||
[pipeline_archive.delete() for pipeline_archive in pipeline_archives] | ||
return BadRequest(message=e.__cause__) | ||
|
||
try: | ||
# Build an task_request from the pipeline request body | ||
task_request = ImageBuildTask( | ||
id="build", | ||
builder=body.builder, | ||
cache=body.cache, | ||
description="Build an image from a repository and push it to an image registry", | ||
destination=body.destination, | ||
context=body.context, | ||
pipeline_id=pipeline.id, | ||
type=TASK_TYPE_IMAGE_BUILD | ||
) | ||
|
||
# Create 'build' task | ||
task_service.create(pipeline, task_request) | ||
except (ValidationError, BadRequestError) as e: | ||
pipeline.delete() | ||
return BadRequest(message=e) | ||
except (IntegrityError, OperationalError, DatabaseError) as e: | ||
pipeline.delete() | ||
return BadRequest(message=e.__cause__) | ||
except Exception as e: | ||
pipeline.delete() | ||
return ServerErrorResp(message=e) | ||
|
||
return ResourceURLResponse( | ||
url=resource_url_builder(request.url.replace("/ci", "/pipelines"), pipeline.id)) |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters