Skip to content

Commit

Permalink
feat: always restore the workspace and always delete the agent pool
Browse files Browse the repository at this point in the history
  • Loading branch information
Apollorion committed May 28, 2024
1 parent 2b86c34 commit 42a1bd2
Show file tree
Hide file tree
Showing 2 changed files with 58 additions and 25 deletions.
3 changes: 3 additions & 0 deletions .editorconfig
Original file line number Diff line number Diff line change
Expand Up @@ -9,3 +9,6 @@ insert_final_newline = true
indent_size = 2
indent_style = space
trim_trailing_whitespace = true

[*.py]
indent_size = 4
80 changes: 55 additions & 25 deletions spacemk/exporters/terraform.py
Original file line number Diff line number Diff line change
Expand Up @@ -19,6 +19,12 @@
from spacemk.exporters import BaseExporter


class TerraformExporterPlanFailure(Exception):
def __init__(self, organization_id: str, workspace_id: str):
message = f"Could not trigger a plan for the '{organization_id}/{workspace_id}' workspace"
super().__init__(message)


class TerraformExporter(BaseExporter):
def __init__(self, config: dict):
super().__init__(config)
Expand Down Expand Up @@ -333,11 +339,18 @@ def find_variable(data: dict, variable_id: str) -> dict:
return data

for organization_id, workspaces in organizations.items():
logging.info(f"Start local TFC/TFE agent for organization '{organization_id}'")

agent_pool_id = self._create_agent_pool(organization_id=organization_id)
# Bring the variable scope up, so we can use these in the `finally` block
agent_pool_id = None
agent_container = None
current_workspace_id = None
workspace_data_backup = None
restored_agent = True

try:
logging.info(f"Start local TFC/TFE agent for organization '{organization_id}'")

agent_pool_id = self._create_agent_pool(organization_id=organization_id)

agent_container_name = f"smk-tfc-agent-{organization_id}"
agent_container = self._start_agent_container(
agent_pool_id=agent_pool_id, container_name=agent_container_name
Expand All @@ -347,6 +360,7 @@ def find_variable(data: dict, variable_id: str) -> dict:
agent_container_id = agent_container.id

for workspace_id, workspace_variables in workspaces.items():
current_workspace_id = workspace_id
current_configuration_version_id = find_workspace(data, workspace_id).get(
"relationships.current-configuration-version.data.id"
)
Expand Down Expand Up @@ -383,6 +397,7 @@ def find_variable(data: dict, variable_id: str) -> dict:
}
},
)
restored_agent = False

logging.info(f"Trigger a plan for the '{organization_id}/{workspace_id}' workspace")
run_data = self._extract_data_from_api(
Expand All @@ -402,9 +417,13 @@ def find_variable(data: dict, variable_id: str) -> dict:
"type": "runs",
}
},
)[
0
] # KLUDGE: There should be a way to pull single item from the API instead of a list of items
)

if len(run_data) == 0:
raise TerraformExporterPlanFailure(organization_id, workspace_id)

# KLUDGE: There should be a way to pull single item from the API instead of a list of items
run_data = run_data[0]

logging.info("Retrieve the output for the plan")
plan_id = run_data.get("relationships.plan.data.id")
Expand Down Expand Up @@ -439,23 +458,8 @@ def find_variable(data: dict, variable_id: str) -> dict:
if workspace and not workspace.get("attributes.vcs-repo.branch"):
workspace["attributes.vcs-repo.branch"] = branch_name

logging.info(f"Restoring the '{organization_id}/{workspace_id}' workspace execution mode")
self._extract_data_from_api(
method="PATCH",
path=f"/workspaces/{workspace_id}",
request_data={
"data": {
"attributes": {
"execution-mode": workspace_data_backup.get("attributes.execution-mode"),
"setting-overwrites": workspace_data_backup.get("attributes.setting-overwrites"),
},
"relationships": {
"agent-pool": workspace_data_backup.get("relationships.agent-pool"),
},
"type": "workspaces",
}
},
)
self._restore_workspace_exec_mode(organization_id, workspace_id, workspace_data_backup)
restored_agent = True

if agent_container.exists() and agent_container.state.running:
logging.debug(f"Local TFC/TFE agent Docker container '{agent_container_id}' logs:")
Expand All @@ -467,9 +471,15 @@ def find_variable(data: dict, variable_id: str) -> dict:
)
finally:
logging.info(f"Stop local TFC/TFE agent for organization '{organization_id}'")
self._stop_agent_container(agent_container)

self._delete_agent_pool(id_=agent_pool_id)
if not restored_agent:
self._restore_workspace_exec_mode(organization_id, current_workspace_id, workspace_data_backup)

if agent_container:
self._stop_agent_container(agent_container)

if agent_pool_id:
self._delete_agent_pool(id_=agent_pool_id)

logging.info("Stop enriching workspace variables data")

Expand Down Expand Up @@ -1378,6 +1388,26 @@ def _start_agent_container(self, agent_pool_id: str, container_name: str) -> Con

return container

def _restore_workspace_exec_mode(self, organization_id: str, workspace_id: str,
workspace_data_backup: dict) -> None:
logging.info(f"Restoring the '{organization_id}/{workspace_id}' workspace execution mode")
self._extract_data_from_api(
method="PATCH",
path=f"/workspaces/{workspace_id}",
request_data={
"data": {
"attributes": {
"execution-mode": workspace_data_backup.get("attributes.execution-mode"),
"setting-overwrites": workspace_data_backup.get("attributes.setting-overwrites"),
},
"relationships": {
"agent-pool": workspace_data_backup.get("relationships.agent-pool"),
},
"type": "workspaces",
}
},
)

def _stop_agent_container(self, container: Container):
if not container.exists() or not container.state.running:
logging.warning(f"Local TFC/TFE agent '{container}' is already stopped before trying to stop it. Ignoring.")
Expand Down

0 comments on commit 42a1bd2

Please sign in to comment.