diff --git a/app/integrations/google_workspace/google_docs.py b/app/integrations/google_workspace/google_docs.py index 9cd715d5..1de8847b 100644 --- a/app/integrations/google_workspace/google_docs.py +++ b/app/integrations/google_workspace/google_docs.py @@ -33,7 +33,7 @@ def create(title: str) -> str: @handle_google_api_errors -def batch_update(document_id: str, requests: list) -> None: +def batch_update(document_id: str, requests: list) -> dict: """Applies a list of updates to a document in Google Docs. Args: @@ -41,9 +41,9 @@ def batch_update(document_id: str, requests: list) -> None: requests (list): A list of update requests. Returns: - None + dict: The response from the Google Docs API. """ - execute_google_api_call( + return execute_google_api_call( "docs", "v1", "documents", diff --git a/app/integrations/google_workspace/google_drive.py b/app/integrations/google_workspace/google_drive.py index 6e6dbd81..3822dbc8 100644 --- a/app/integrations/google_workspace/google_drive.py +++ b/app/integrations/google_workspace/google_drive.py @@ -14,13 +14,14 @@ @handle_google_api_errors -def add_metadata(file_id, key, value): +def add_metadata(file_id, key, value, delegated_user_email=None): """Add metadata to a file in Google Drive. Args: file_id (str): The file id of the file to add metadata to. key (str): The key of the metadata to add. value (str): The value of the metadata to add. + delegated_user_email (str, optional): The email address of the user to impersonate. Returns: dict: The updated file metadata. @@ -30,6 +31,7 @@ def add_metadata(file_id, key, value): "v3", "files", "update", + delegated_user_email=delegated_user_email, fileId=file_id, body={"appProperties": {key: value}}, fields="name, appProperties", @@ -38,12 +40,13 @@ def add_metadata(file_id, key, value): @handle_google_api_errors -def delete_metadata(file_id, key): +def delete_metadata(file_id, key, delegated_user_email=None): """Delete metadata from a file in Google Drive. Args: file_id (str): The file id of the file to delete metadata from. key (str): The key of the metadata to delete. + delegated_user_email (str, optional): The email address of the user to impersonate. Returns: dict: The updated file metadata. @@ -53,6 +56,7 @@ def delete_metadata(file_id, key): "v3", "files", "update", + delegated_user_email=delegated_user_email, fileId=file_id, body={"appProperties": {key: None}}, fields="name, appProperties", @@ -82,17 +86,19 @@ def list_metadata(file_id): @handle_google_api_errors -def create_folder(name, parent_folder): +def create_folder(name, parent_folder, fields=None): """Create a new folder in Google Drive. Args: name (str): The name of the new folder. parent_folder (str): The id of the parent folder. + fields (str, optional): The fields to include in the response. Returns: - str: The id of the new folder. + dict: A File resource representing the new folder. + (https://developers.google.com/drive/api/reference/rest/v3/files#File) """ - result = execute_google_api_call( + return execute_google_api_call( "drive", "v3", "files", @@ -103,14 +109,12 @@ def create_folder(name, parent_folder): "parents": [parent_folder], }, supportsAllDrives=True, - fields="id", + fields=fields, ) - return result["id"] - @handle_google_api_errors -def create_file_from_template(name, folder, template): +def create_file_from_template(name, folder, template, fields=None): """Create a new file in Google Drive from a template (Docs, Sheets, Slides, Forms, or Sites.) @@ -120,9 +124,9 @@ def create_file_from_template(name, folder, template): template (str): The id of the template to use. Returns: - str: The id of the new file. + dict: A File resource representing the new file with a mask of 'id'. """ - result = execute_google_api_call( + return execute_google_api_call( "drive", "v3", "files", @@ -130,11 +134,9 @@ def create_file_from_template(name, folder, template): fileId=template, body={"name": name, "parents": [folder]}, supportsAllDrives=True, - fields="id", + fields=fields, ) - return result["id"] - @handle_google_api_errors def create_file(name, folder, file_type): @@ -215,15 +217,20 @@ def get_file_by_name(name, folder_id=None): @handle_google_api_errors -def list_folders_in_folder(folder): +def list_folders_in_folder(folder, query=None): """List all folders in a folder in Google Drive. Args: folder (str): The id of the folder to list. + query (str, optional): A query to filter the folders. Returns: list: A list of folders in the folder. """ + base_query = f"parents in '{folder}' and mimeType = 'application/vnd.google-apps.folder' and trashed=false" + if query: + base_query += f" and {query}" + return execute_google_api_call( "drive", "v3", @@ -234,7 +241,7 @@ def list_folders_in_folder(folder): supportsAllDrives=True, includeItemsFromAllDrives=True, corpora="user", - q=f"parents in '{folder}' and mimeType = 'application/vnd.google-apps.folder' and trashed=false", + q=base_query, fields="files(id, name)", ) diff --git a/app/integrations/google_workspace/google_service.py b/app/integrations/google_workspace/google_service.py index 9014ec27..bcc4a22e 100644 --- a/app/integrations/google_workspace/google_service.py +++ b/app/integrations/google_workspace/google_service.py @@ -85,7 +85,7 @@ def wrapper(*args, **kwargs): result, unsupported_params = result if unsupported_params: logging.warning( - f"Unsupported parameters in '{func.__name__}' were filtered out: {', '.join(unsupported_params)}" + f"Unknown parameters in '{func.__name__}' were detected: {', '.join(unsupported_params)}" ) return result except HttpError as e: @@ -152,7 +152,7 @@ def execute_google_api_call( k: v for k, v in formatted_kwargs.items() if k in supported_params } unsupported_params = set(formatted_kwargs.keys()) - set(filtered_params.keys()) - + # filtered_params = kwargs if paginate: all_results = [] request = api_method(**filtered_params) @@ -178,8 +178,9 @@ def get_google_api_command_parameters(resource_obj, method): list: The names of the parameters for the API method, excluding non-parameter documentation. """ api_method = getattr(resource_obj, method) + # Add known parameters not documented in the docstring + parameter_names = ["fields"] - parameter_names = [] if hasattr(api_method, "__doc__") and api_method.__doc__: parsing_parameters = False doc_lines = api_method.__doc__.splitlines() diff --git a/app/integrations/google_workspace/sheets.py b/app/integrations/google_workspace/sheets.py new file mode 100644 index 00000000..555651b8 --- /dev/null +++ b/app/integrations/google_workspace/sheets.py @@ -0,0 +1,63 @@ +"""Google Sheets API calls.""" + +from integrations.google_workspace.google_service import ( + handle_google_api_errors, + execute_google_api_call, +) + + +@handle_google_api_errors +def get_values( + spreadsheetId: str, range: str | None = None, includeGridData=None, fields=None +): + """Gets the values from a Google Sheet. + + Args: + spreadsheetId (str): The id of the Google Sheet. + range (str, optional): The range of the values to retrieve. + includeGridData (bool, optional): Whether to include grid data. + fields (str, optional): The fields to include in the response. + + Returns: + dict: The response from the Google Sheets API. + """ + return execute_google_api_call( + "sheets", + "v4", + "spreadsheets.values", + "get", + spreadsheetId=spreadsheetId, + range=range, + includeGridData=includeGridData, + fields=fields, + ) + + +def batch_update_values( + spreadsheetId: str, + range: str, + values: list, + valueInputOption: str = "USER_ENTERED", +) -> dict: + """Updates values in a Google Sheet. + + Args: + spreadsheetId (str): The id of the Google Sheet. + range (str): The range to update. + values (list): The values to update. + valueInputOption (str, optional): The value input option. + + Returns: + dict: The response from the Google Sheets API. + """ + return execute_google_api_call( + "sheets", + "v4", + "spreadsheets.values", + "batchUpdate", + spreadsheetId=spreadsheetId, + body={ + "valueInputOption": valueInputOption, + "data": [{"range": range, "values": values}], + }, + ) diff --git a/app/modules/incident/incident_helper.py b/app/modules/incident/incident_helper.py index ae0f6695..2b9ce2c6 100644 --- a/app/modules/incident/incident_helper.py +++ b/app/modules/incident/incident_helper.py @@ -1,13 +1,21 @@ import json import logging +import os from slack_sdk import WebClient -from integrations import google_drive -from integrations.google_workspace import google_docs +from slack_bolt import Ack, Respond, App +from integrations.google_workspace import google_docs, google_drive, sheets from integrations.slack import channels as slack_channels from integrations.sentinel import log_to_sentinel from modules.incident import schedule_retro INCIDENT_CHANNELS_PATTERN = r"^incident-\d{4}-" +SRE_DRIVE_ID = os.environ.get("SRE_DRIVE_ID") +SRE_INCIDENT_FOLDER = os.environ.get("SRE_INCIDENT_FOLDER") +INCIDENT_TEMPLATE = os.environ.get("INCIDENT_TEMPLATE") +INCIDENT_LIST = os.environ.get("INCIDENT_LIST") +START_HEADING = "DO NOT REMOVE this line as the SRE bot needs it as a placeholder." +END_HEADING = "Trigger" + help_text = """ \n `/sre incident create-folder ` @@ -34,7 +42,7 @@ """ -def register(bot): +def register(bot: App): bot.action("add_folder_metadata")(add_folder_metadata) bot.action("view_folder_metadata")(view_folder_metadata) bot.view("view_folder_metadata_modal")(list_folders) @@ -46,7 +54,7 @@ def register(bot): bot.action("confirm_click")(confirm_click) -def handle_incident_command(args, client: WebClient, body, respond, ack): +def handle_incident_command(args, client: WebClient, body, respond: Respond, ack: Ack): if len(args) == 0: respond(help_text) return @@ -55,7 +63,11 @@ def handle_incident_command(args, client: WebClient, body, respond, ack): match action: case "create-folder": name = " ".join(args) - respond(google_drive.create_folder(name)) + folder = google_drive.create_folder(name, SRE_INCIDENT_FOLDER) + if folder: + respond(f"Folder `{folder['name']}` created.") + else: + respond(f"Failed to create folder `{name}`.") case "help": respond(help_text) case "list-folders": @@ -63,7 +75,7 @@ def handle_incident_command(args, client: WebClient, body, respond, ack): case "roles": manage_roles(client, body, ack, respond) case "close": - close_incident(client, body, ack) + close_incident(client, body, ack, respond) case "stale": stale_incidents(client, body, ack) case "schedule": @@ -74,7 +86,7 @@ def handle_incident_command(args, client: WebClient, body, respond, ack): ) -def add_folder_metadata(client, body, ack): +def add_folder_metadata(client: WebClient, body, ack): ack() folder_id = body["actions"][0]["value"] blocks = { @@ -126,7 +138,7 @@ def add_folder_metadata(client, body, ack): ) -def archive_channel_action(client, body, ack): +def archive_channel_action(client: WebClient, body, ack, respond): ack() channel_id = body["channel"]["id"] action = body["actions"][0]["value"] @@ -150,7 +162,7 @@ def archive_channel_action(client, body, ack): log_to_sentinel("incident_channel_archive_delayed", body) elif action == "archive": # Call the close_incident function to update the incident document to closed, update the spreadsheet and archive the channel - close_incident(client, channel_info, ack) + close_incident(client, channel_info, ack, respond) # log the event to sentinel log_to_sentinel("incident_channel_archived", body) elif action == "schedule_retro": @@ -160,18 +172,24 @@ def archive_channel_action(client, body, ack): log_to_sentinel("incident_retro_scheduled", body) -def delete_folder_metadata(client, body, ack): +def delete_folder_metadata(client: WebClient, body, ack): ack() folder_id = body["view"]["private_metadata"] key = body["actions"][0]["value"] - google_drive.delete_metadata(folder_id, key) + response = google_drive.delete_metadata(folder_id, key) + if not response: + logging.info(f"Failed to delete metadata `{key}`.\nResponse: {str(response)}") + else: + logging.info(f"Response: {str(response)}") body["actions"] = [{"value": folder_id}] view_folder_metadata(client, body, ack) -def list_folders(client, body, ack): +def list_folders(client: WebClient, body, ack): ack() - folders = google_drive.list_folders() + folders = google_drive.list_folders_in_folder( + SRE_INCIDENT_FOLDER, "not name contains 'Templates'" + ) folders.sort(key=lambda x: x["name"]) blocks = { "type": "modal", @@ -185,13 +203,13 @@ def list_folders(client, body, ack): client.views_open(trigger_id=body["trigger_id"], view=blocks) -def manage_roles(client, body, ack, respond): +def manage_roles(client: WebClient, body, ack, respond): ack() channel_name = body["channel_name"] channel_name = channel_name[ channel_name.startswith("incident-") and len("incident-") : ] - documents = google_drive.get_document_by_channel_name(channel_name) + documents = google_drive.get_file_by_name(channel_name) if len(documents) == 0: respond( @@ -281,7 +299,7 @@ def manage_roles(client, body, ack, respond): client.views_open(trigger_id=body["trigger_id"], view=blocks) -def save_metadata(client, body, ack, view): +def save_metadata(client: WebClient, body, ack, view): ack() folder_id = view["private_metadata"] key = view["state"]["values"]["key"]["key"]["value"] @@ -292,7 +310,7 @@ def save_metadata(client, body, ack, view): view_folder_metadata(client, body, ack) -def save_incident_roles(client, ack, view): +def save_incident_roles(client: WebClient, ack, view): ack() selected_ic = view["state"]["values"]["ic_name"]["ic_select"]["selected_user"] selected_ol = view["state"]["values"]["ol_name"]["ol_select"]["selected_user"] @@ -316,7 +334,7 @@ def save_incident_roles(client, ack, view): ) -def close_incident(client, body, ack): +def close_incident(client: WebClient, body, ack, respond): ack() # get the current chanel id and name channel_id = body["channel_id"] @@ -347,23 +365,31 @@ def close_incident(client, body, ack): response["bookmarks"][item]["link"] ) else: - logging.warning( - "No bookmark link for the incident document found for channel %s", - channel_name, - ) + warning_message = f"No bookmark link for the incident document found for channel {channel_name}" + logging.warning(warning_message) + respond(warning_message) # Update the document status to "Closed" if we can get the document if document_id != "": - google_drive.close_incident_document(document_id) + close_incident_document(document_id) else: - logging.warning( + warning_message = ( "Could not close the incident document - the document was not found." ) + logging.warning(warning_message) + respond(warning_message) # Update the spreadsheet with the current incident with status = closed - google_drive.update_spreadsheet_close_incident(return_channel_name(channel_name)) + update_succeeded = update_spreadsheet_incident_status( + return_channel_name(channel_name), "Closed" + ) + + if not update_succeeded: + warning_message = f"Could not update the incident status in the spreadsheet for channel {channel_name}" + logging.warning(warning_message) + respond(warning_message) - # Need to post the message before the channe is archived so that the message can be delivered. + # Need to post the message before the channel is archived so that the message can be delivered. client.chat_postMessage( channel=channel_id, text=f"<@{user_id}> has archived this channel 👋", @@ -384,6 +410,58 @@ def close_incident(client, body, ack): ) +def close_incident_document(document_id): + # List of possible statuses to be replaced + possible_statuses = ["In Progress", "Open", "Ready to be Reviewed", "Reviewed"] + + # Replace all possible statuses with "Closed" + changes = [ + { + "replaceAllText": { + "containsText": {"text": f"Status: {status}", "matchCase": "false"}, + "replaceText": "Status: Closed", + } + } + for status in possible_statuses + ] + return google_docs.batch_update(document_id, changes) + + +def update_spreadsheet_incident_status(channel_name, status="Closed"): + """Update the status of an incident in the incident list spreadsheet. + + Args: + channel_name (str): The name of the channel to update. + status (str): The status to update the incident to. + + Returns: + bool: True if the status was updated successfully, False otherwise. + """ + valid_statuses = ["Open", "Closed", "In Progress", "Resolved"] + if status not in valid_statuses: + logging.warning("Invalid status %s", status) + return False + sheet_name = "Sheet1" + sheet = sheets.get_values(INCIDENT_LIST, range=sheet_name) + values = sheet.get("values", []) + if len(values) == 0: + logging.warning("No incident found for channel %s", channel_name) + return False + # Find the row with the search value + for i, row in enumerate(values): + if channel_name in row: + # Update the 4th column (index 3) of the found row + update_range = ( + f"{sheet_name}!D{i+1}" # Column D, Rows are 1-indexed in Sheets + ) + updated_sheet = sheets.batch_update_values( + INCIDENT_LIST, update_range, [[status]] + ) + if updated_sheet: + return True + return False + + def stale_incidents(client, body, ack): ack() @@ -678,6 +756,7 @@ def confirm_click(ack, body, client): def view_folder_metadata(client, body, ack): ack() folder_id = body["actions"][0]["value"] + logging.info(f"Viewing metadata for folder {folder_id}") folder = google_drive.list_metadata(folder_id) blocks = { "type": "modal", @@ -804,9 +883,12 @@ def metadata_items(folder): ] -def return_channel_name(input_str): +def return_channel_name(input_str: str): # return the channel name without the incident- prefix and appending a # to the channel name prefix = "incident-" + dev_prefix = prefix + "dev-" + if input_str.startswith(dev_prefix): + return "#" + input_str[len(dev_prefix) :] if input_str.startswith(prefix): return "#" + input_str[len(prefix) :] return input_str diff --git a/app/tests/integrations/google_workspace/test_google_drive.py b/app/tests/integrations/google_workspace/test_google_drive.py index 20a3adf2..155689ae 100644 --- a/app/tests/integrations/google_workspace/test_google_drive.py +++ b/app/tests/integrations/google_workspace/test_google_drive.py @@ -17,6 +17,7 @@ def test_add_metadata_returns_result(execute_google_api_call_mock): "v3", "files", "update", + delegated_user_email=None, fileId="file_id", body={"appProperties": {"key": "value"}}, fields="name, appProperties", @@ -37,6 +38,7 @@ def test_delete_metadata_returns_result(execute_google_api_call_mock): "v3", "files", "update", + delegated_user_email=None, fileId="file_id", body={"appProperties": {"key": None}}, fields="name, appProperties", @@ -68,11 +70,40 @@ def test_list_metadata_returns_result(execute_google_api_call_mock): @patch("integrations.google_workspace.google_drive.execute_google_api_call") -def test_create_folder_returns_folder_id(execute_google_api_call_mock): - execute_google_api_call_mock.return_value = {"id": "test_folder_id"} - assert ( - google_drive.create_folder("test_folder", "parent_folder") == "test_folder_id" +def test_create_folder_returns_folder(execute_google_api_call_mock): + execute_google_api_call_mock.return_value = { + "id": "test_folder_id", + "name": "test_folder", + "appProperties": {}, + } + assert google_drive.create_folder("test_folder", "parent_folder") == { + "id": "test_folder_id", + "name": "test_folder", + "appProperties": {}, + } + execute_google_api_call_mock.assert_called_once_with( + "drive", + "v3", + "files", + "create", + body={ + "name": "test_folder", + "mimeType": "application/vnd.google-apps.folder", + "parents": ["parent_folder"], + }, + supportsAllDrives=True, + fields=None, ) + + +@patch("integrations.google_workspace.google_drive.execute_google_api_call") +def test_create_folder_calls_api_with_fields(execute_google_api_call_mock): + execute_google_api_call_mock.return_value = { + "id": "test_folder_id", + } + assert google_drive.create_folder("test_folder", "parent_folder", "id") == { + "id": "test_folder_id", + } execute_google_api_call_mock.assert_called_once_with( "drive", "v3", @@ -125,12 +156,12 @@ def test_create_file_with_invalid_type_raises_value_error( @patch("integrations.google_workspace.google_drive.execute_google_api_call") -def test_create_file_from_template_returns_file_id(execute_google_api_call_mock): +def test_create_file_from_template_returns_file(execute_google_api_call_mock): execute_google_api_call_mock.return_value = {"id": "test_document_id"} result = google_drive.create_file_from_template( "test_document", "folder_id", "template_id" ) - assert result == "test_document_id" + assert result == {"id": "test_document_id"} execute_google_api_call_mock.assert_called_once_with( "drive", "v3", @@ -139,7 +170,7 @@ def test_create_file_from_template_returns_file_id(execute_google_api_call_mock) fileId="template_id", body={"name": "test_document", "parents": ["folder_id"]}, supportsAllDrives=True, - fields="id", + fields=None, ) diff --git a/app/tests/integrations/google_workspace/test_google_service.py b/app/tests/integrations/google_workspace/test_google_service.py index 7715ec63..58a4fc6e 100644 --- a/app/tests/integrations/google_workspace/test_google_service.py +++ b/app/tests/integrations/google_workspace/test_google_service.py @@ -190,7 +190,7 @@ def test_handle_google_api_errors_processes_unsupported_params( assert result == "test" mock_func.assert_called_once() mocked_logging_warning.assert_called_once_with( - "Unsupported parameters in 'mock_func' were filtered out: unsupported" + "Unknown parameters in 'mock_func' were detected: unsupported" ) @@ -440,4 +440,4 @@ def test_get_google_api_command_parameters_returns_correct_parameters(): result = get_google_api_command_parameters(mock_resource, "method") - assert result == ["arg1", "arg2"] + assert result == ["fields", "arg1", "arg2"] diff --git a/app/tests/integrations/google_workspace/test_sheets.py b/app/tests/integrations/google_workspace/test_sheets.py new file mode 100644 index 00000000..6c593fed --- /dev/null +++ b/app/tests/integrations/google_workspace/test_sheets.py @@ -0,0 +1,88 @@ +from unittest.mock import patch +from integrations.google_workspace import sheets + + +@patch("integrations.google_workspace.sheets.execute_google_api_call") +def test_get_values(mock_execute_google_api_call): + + spreadsheet_id = "1" + range = "A1:B2" + include_grid_data = True + fields = "fields" + + sheets.get_values(spreadsheet_id, range, include_grid_data, fields) + + mock_execute_google_api_call.assert_called_once_with( + "sheets", + "v4", + "spreadsheets.values", + "get", + spreadsheetId=spreadsheet_id, + range=range, + includeGridData=include_grid_data, + fields=fields, + ) + + +@patch("integrations.google_workspace.sheets.execute_google_api_call") +def test_get_values_with_defaults(mock_execute_google_api_call): + + spreadsheet_id = "1" + + sheets.get_values(spreadsheet_id) + + mock_execute_google_api_call.assert_called_once_with( + "sheets", + "v4", + "spreadsheets.values", + "get", + spreadsheetId=spreadsheet_id, + range=None, + includeGridData=None, + fields=None, + ) + + +@patch("integrations.google_workspace.sheets.execute_google_api_call") +def test_batch_update_values(mock_execute_google_api_call): + + spreadsheet_id = "1" + range = "A1:B2" + values = [["a", "b"], ["c", "d"]] + value_input_option = "USER_ENTERED" + + sheets.batch_update_values(spreadsheet_id, range, values, value_input_option) + + mock_execute_google_api_call.assert_called_once_with( + "sheets", + "v4", + "spreadsheets.values", + "batchUpdate", + spreadsheetId=spreadsheet_id, + body={ + "valueInputOption": value_input_option, + "data": [{"range": range, "values": values}], + }, + ) + + +@patch("integrations.google_workspace.sheets.execute_google_api_call") +def test_batch_update_values_with_defaults(mock_execute_google_api_call): + + spreadsheet_id = "1" + range = "A1:B2" + values = [["a", "b"], ["c", "d"]] + + sheets.batch_update_values(spreadsheet_id, range, values) + + mock_execute_google_api_call.assert_called_once_with( + "sheets", + "v4", + "spreadsheets.values", + "batchUpdate", + spreadsheetId=spreadsheet_id, + body={ + "valueInputOption": "USER_ENTERED", + "data": [{"range": range, "values": values}], + }, + ) diff --git a/app/tests/modules/incident/test_incident_helper.py b/app/tests/modules/incident/test_incident_helper.py index d7cd588f..6a6b9132 100644 --- a/app/tests/modules/incident/test_incident_helper.py +++ b/app/tests/modules/incident/test_incident_helper.py @@ -18,13 +18,24 @@ def test_handle_incident_command_with_empty_args(): @patch("modules.incident.incident_helper.google_drive.create_folder") def test_handle_incident_command_with_create_command(create_folder_mock): - create_folder_mock.return_value = "folder created" + create_folder_mock.return_value = {"id": "test_id", "name": "foo bar"} respond = MagicMock() ack = MagicMock() incident_helper.handle_incident_command( ["create-folder", "foo", "bar"], MagicMock(), MagicMock(), respond, ack ) - respond.assert_called_once_with("folder created") + respond.assert_called_once_with("Folder `foo bar` created.") + + +@patch("modules.incident.incident_helper.google_drive.create_folder") +def test_handle_incident_command_with_create_command_error(create_folder_mock): + create_folder_mock.return_value = None + respond = MagicMock() + ack = MagicMock() + incident_helper.handle_incident_command( + ["create-folder", "foo", "bar"], MagicMock(), MagicMock(), respond, ack + ) + respond.assert_called_once_with("Failed to create folder `foo bar`.") def test_handle_incident_command_with_help(): @@ -98,7 +109,8 @@ def test_archive_channel_action_ignore(mock_log_to_sentinel): "user": {"id": "user_id"}, } ack = MagicMock() - incident_helper.archive_channel_action(client, body, ack) + respond = MagicMock() + incident_helper.archive_channel_action(client, body, ack, respond) ack.assert_called_once() client.chat_update( channel="channel_id", @@ -111,10 +123,8 @@ def test_archive_channel_action_ignore(mock_log_to_sentinel): ) -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value="dummy_document_id", @@ -131,7 +141,8 @@ def test_archive_channel_action_archive( "user": {"id": "user_id"}, } ack = MagicMock() - incident_helper.archive_channel_action(client, body, ack) + respond = MagicMock() + incident_helper.archive_channel_action(client, body, ack, respond) assert ack.call_count == 2 mock_log_to_sentinel.assert_called_once_with("incident_channel_archived", body) @@ -166,22 +177,22 @@ def test_delete_folder_metadata(view_folder_metadata_mock, delete_metadata_mock) ) -@patch("modules.incident.incident_helper.google_drive.list_folders") +@patch("modules.incident.incident_helper.google_drive.list_folders_in_folder") @patch("modules.incident.incident_helper.folder_item") -def test_list_folders(folder_item_mock, list_folders_mock): +def test_list_folders(folder_item_mock, list_folders_in_folder_mock): client = MagicMock() body = {"trigger_id": "foo"} ack = MagicMock() - list_folders_mock.return_value = [{"id": "foo", "name": "bar"}] + list_folders_in_folder_mock.return_value = [{"id": "foo", "name": "bar"}] folder_item_mock.return_value = [["folder item"]] incident_helper.list_folders(client, body, ack) - list_folders_mock.assert_called_once() + list_folders_in_folder_mock.assert_called_once() folder_item_mock.assert_called_once_with({"id": "foo", "name": "bar"}) ack.assert_called_once() client.views_open.assert_called_once_with(trigger_id="foo", view=ANY) -@patch("modules.incident.incident_helper.google_drive.get_document_by_channel_name") +@patch("modules.incident.incident_helper.google_drive.get_file_by_name") def test_manage_roles(get_document_by_channel_name_mock): client = MagicMock() body = { @@ -200,7 +211,7 @@ def test_manage_roles(get_document_by_channel_name_mock): client.views_open.assert_called_once_with(trigger_id="trigger_id", view=ANY) -@patch("modules.incident.incident_helper.google_drive.get_document_by_channel_name") +@patch("modules.incident.incident_helper.google_drive.get_file_by_name") def test_manage_roles_with_no_result(get_document_by_channel_name_mock): client = MagicMock() body = { @@ -427,10 +438,8 @@ def test_metadata_items(): ] -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value="dummy_document_id", @@ -438,6 +447,7 @@ def test_metadata_items(): def test_close_incident(mock_extract_id, mock_update_spreadsheet, mock_close_document): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() # Mock the response of client.bookmarks_list mock_client.bookmarks_list.return_value = { @@ -462,6 +472,7 @@ def test_close_incident(mock_extract_id, mock_update_spreadsheet, mock_close_doc "user_id": "U12345", }, mock_ack, + mock_respond, ) # Assert that ack was called @@ -474,16 +485,14 @@ def test_close_incident(mock_extract_id, mock_update_spreadsheet, mock_close_doc # Assert that the Google Drive document and spreadsheet update methods were called mock_close_document.assert_called_once_with("dummy_document_id") - mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test") + mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test", "Closed") # Assert that the Slack client's conversations_archive method was called with the correct channel ID mock_client.conversations_archive.assert_called_once_with(channel="C12345") -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value=None ) @@ -492,6 +501,7 @@ def test_close_incident_no_bookmarks( ): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() # Mock client.bookmarks_list to return no bookmarks mock_client.bookmarks_list.return_value = {"ok": True, "bookmarks": []} @@ -505,18 +515,17 @@ def test_close_incident_no_bookmarks( "user_id": "U12345", }, mock_ack, + mock_respond, ) # Assertions to ensure that document update functions are not called as there are no bookmarks mock_extract_id.assert_not_called() mock_close_document.assert_not_called() - mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test") + mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test", "Closed") -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value=None ) @@ -525,6 +534,7 @@ def test_close_incident_no_bookmarks_error( ): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() # Mock client.bookmarks_list to return no bookmarks mock_client.bookmarks_list.return_value = {"ok": False, "error": "not_in_channel"} @@ -538,19 +548,20 @@ def test_close_incident_no_bookmarks_error( "user_id": "U12345", }, mock_ack, + mock_respond, ) # Assertions to ensure that document update functions are not called as there are no bookmarks mock_extract_id.assert_not_called() mock_close_document.assert_not_called() - mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test") + mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test", "Closed") # Test that the channel that the command is ran in, is not an incident channel. def test_close_incident_not_incident_channel(): mock_client = MagicMock() mock_ack = MagicMock() - + mock_respond = MagicMock() # Mock the response of the private message to have been posted as expected mock_client.chat_postEphemeral.return_value = {"ok": True} @@ -563,6 +574,7 @@ def test_close_incident_not_incident_channel(): "channel_name": "some-other-channel", }, mock_ack, + mock_respond, ) # Assert that ack was called @@ -579,6 +591,7 @@ def test_close_incident_not_incident_channel(): def test_close_incident_cant_send_private_message(caplog): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() # Mock the response of the private message to have been posted as expected mock_client.chat_postEphemeral.return_value = { @@ -601,7 +614,9 @@ def test_close_incident_cant_send_private_message(caplog): # Use the caplog fixture to capture logging with caplog.at_level(logging.ERROR): # Call the function being tested - incident_helper.close_incident(client=mock_client, body=body, ack=mock_ack) + incident_helper.close_incident( + client=mock_client, body=body, ack=mock_ack, respond=mock_respond + ) # Check that the expected error message was logged assert caplog.records # Ensure there is at least one log record @@ -617,10 +632,8 @@ def test_close_incident_cant_send_private_message(caplog): ), "Expected error message not found in log records" -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value="dummy_document_id", @@ -630,6 +643,8 @@ def test_conversations_archive_fail( ): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() + # Mock the response of client.bookmarks_list with a valid bookmark mock_client.bookmarks_list.return_value = { "ok": True, @@ -656,21 +671,20 @@ def test_conversations_archive_fail( "user_id": "U12345", }, mock_ack, + mock_respond, ) # Assertions # Ensure that the Google Drive document update method was called even if archiving fails mock_close_document.assert_called_once_with("dummy_document_id") - mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test") + mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test", "Closed") # Ensure that the client's conversations_archive method was called mock_client.conversations_archive.assert_called_once_with(channel="C12345") -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value="dummy_document_id", @@ -680,6 +694,7 @@ def test_conversations_archive_fail_error_message( ): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() # Mock the response of client.bookmarks_list with a valid bookmark mock_client.bookmarks_list.return_value = { "ok": True, @@ -707,12 +722,13 @@ def test_conversations_archive_fail_error_message( "user_id": "U12345", }, mock_ack, + mock_respond, ) # Assertions # Ensure that the Google Drive document update method was called even if archiving fails mock_close_document.assert_called_once_with("dummy_document_id") - mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test") + mock_update_spreadsheet.assert_called_once_with("#2024-01-12-test", "Closed") # Ensure that the client's conversations_archive method was called mock_client.conversations_archive.assert_called_once_with(channel="C12345") @@ -723,10 +739,8 @@ def test_conversations_archive_fail_error_message( ) -@patch("modules.incident.incident_helper.google_drive.close_incident_document") -@patch( - "modules.incident.incident_helper.google_drive.update_spreadsheet_close_incident" -) +@patch("modules.incident.incident_helper.close_incident_document") +@patch("modules.incident.incident_helper.update_spreadsheet_incident_status") @patch( "integrations.google_workspace.google_docs.extract_google_doc_id", return_value="dummy_document_id", @@ -736,12 +750,13 @@ def test_conversations_archive_succeeds_post_message_who_archived( ): mock_client = MagicMock() mock_ack = MagicMock() + mock_respond = MagicMock() body = { "channel_id": "channel_id", "channel_name": "incident-channel_name", "user_id": "user_id", } - incident_helper.close_incident(mock_client, body, mock_ack) + incident_helper.close_incident(mock_client, body, mock_ack, mock_respond) # Mock the response of client.bookmarks_list with a valid bookmark mock_client.bookmarks_list.return_value = {