Add APM #184

Merged
luketainton merged 5 commits from feature/apm into next 2024-04-21 17:17:03 +02:00
10 changed files with 124 additions and 26 deletions

View File

@ -1,3 +1,6 @@
APP_LIFECYCLE="dev"
SENTRY_ENABLED="False"
SENTRY_DSN=""
ADMIN_EMAIL="" ADMIN_EMAIL=""
ADMIN_FIRST_NAME="" ADMIN_FIRST_NAME=""
BOT_NAME="" BOT_NAME=""

View File

@ -1,6 +1,7 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
import logging import logging
import sentry_sdk
from webex_bot.models.command import Command from webex_bot.models.command import Command
from webex_bot.models.response import Response, response_from_adaptive_card from webex_bot.models.response import Response, response_from_adaptive_card
@ -113,6 +114,7 @@ class SubmitTaskCommand(Command):
Submit(title="Cancel", data={"command_keyword": "exit"}), Submit(title="Cancel", data={"command_keyword": "exit"}),
], ],
) )
with sentry_sdk.start_transaction(name="submit_task_command"):
return response_from_adaptive_card(card) return response_from_adaptive_card(card)
@ -147,6 +149,7 @@ class SubmitTaskCallback(Command):
) )
def execute(self, message, attachment_actions, activity) -> str: def execute(self, message, attachment_actions, activity) -> str:
with sentry_sdk.start_transaction(name="submit_task_callback"):
return self.msg return self.msg
@ -158,11 +161,13 @@ class MyTasksCallback(Command):
self.msg: str = "" self.msg: str = ""
def pre_execute(self, message, attachment_actions, activity) -> str: def pre_execute(self, message, attachment_actions, activity) -> str:
with sentry_sdk.start_transaction(name="my_tasks_preexec"):
return "Getting your tasks..." return "Getting your tasks..."
def execute(self, message, attachment_actions, activity) -> str | None: def execute(self, message, attachment_actions, activity) -> str | None:
sender: str = attachment_actions.inputs.get("sender") sender: str = attachment_actions.inputs.get("sender")
result: bool = get_tasks(requestor=sender) result: bool = get_tasks(requestor=sender)
with sentry_sdk.start_transaction(name="my_tasks_exec"):
if not result: if not result:
return "Failed to get tasks. Please try again." return "Failed to get tasks. Please try again."
return return

View File

@ -1,5 +1,8 @@
#!/usr/bin/env python3 #!/usr/bin/env python3
import sentry_sdk
from sentry_sdk.integrations.stdlib import StdlibIntegration
from webex_bot.webex_bot import WebexBot from webex_bot.webex_bot import WebexBot
from app.commands.exit import ExitCommand from app.commands.exit import ExitCommand
@ -7,8 +10,18 @@ from app.commands.submit_task import SubmitTaskCommand
from app.utils.config import config from app.utils.config import config
if config.sentry_enabled:
apm = sentry_sdk.init(
dsn=config.sentry_dsn,
enable_tracing=True,
environment=config.environment,
integrations=[StdlibIntegration()],
spotlight=True
)
def create_bot() -> WebexBot: def create_bot() -> WebexBot:
# Create a Bot Object """Create and return a Webex Bot object."""
webex_bot: WebexBot = WebexBot( webex_bot: WebexBot = WebexBot(
bot_name=config.bot_name, bot_name=config.bot_name,
teams_bot_token=config.webex_token, teams_bot_token=config.webex_token,

View File

@ -1,34 +1,65 @@
#!/usr/bin/env python3 """Configuration module."""
import os import os
class Config: class Config:
"""Configuration module."""
def __init__(self) -> None: def __init__(self) -> None:
"""Configuration module."""
self.__environment: str = os.environ.get("APP_LIFECYCLE", "DEV").upper()
self.__bot_name: str = os.environ["BOT_NAME"] self.__bot_name: str = os.environ["BOT_NAME"]
self.__webex_token: str = os.environ["WEBEX_API_KEY"] self.__webex_token: str = os.environ["WEBEX_API_KEY"]
self.__admin_first_name: str = os.environ["ADMIN_FIRST_NAME"] self.__admin_first_name: str = os.environ["ADMIN_FIRST_NAME"]
self.__admin_emails: list = os.environ["ADMIN_EMAIL"].split(",") self.__admin_emails: list = os.environ["ADMIN_EMAIL"].split(",")
self.__n8n_webhook_url: str = os.environ["N8N_WEBHOOK_URL"] self.__n8n_webhook_url: str = os.environ["N8N_WEBHOOK_URL"]
self.__sentry_dsn: str = os.environ.get("SENTRY_DSN", "")
self.__sentry_enabled: bool = bool(
os.environ.get("SENTRY_ENABLED", "False").upper() == "TRUE"
and self.__sentry_dsn != ""
)
@property
def environment(self) -> str:
"""Returns the current app lifecycle."""
return self.__environment
@property
def sentry_enabled(self) -> bool:
"""Returns True if Sentry SDK is enabled, else False."""
return self.__sentry_enabled
@property
def sentry_dsn(self) -> str:
"""Returns the Sentry DSN value if Sentry SDK is enabled AND
Sentry DSN is set, else blank string."""
if not self.__sentry_enabled:
return ""
return self.__sentry_dsn
@property @property
def bot_name(self) -> str: def bot_name(self) -> str:
"""Returns the bot name."""
return self.__bot_name return self.__bot_name
@property @property
def webex_token(self) -> str: def webex_token(self) -> str:
"""Returns the Webex API key."""
return self.__webex_token return self.__webex_token
@property @property
def admin_first_name(self) -> str: def admin_first_name(self) -> str:
"""Returns the first name of the bot admin."""
return self.__admin_first_name return self.__admin_first_name
@property @property
def admin_emails(self) -> list: def admin_emails(self) -> list:
"""Returns a list of admin email addresses."""
return self.__admin_emails return self.__admin_emails
@property @property
def n8n_webhook_url(self) -> str: def n8n_webhook_url(self) -> str:
"""Returns the n8n webhook URL."""
return self.__n8n_webhook_url return self.__n8n_webhook_url

View File

@ -1,7 +1,18 @@
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
#!/usr/bin/env python3 #!/usr/bin/env python3
from zoneinfo import ZoneInfo
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
from datetime import datetime from datetime import datetime
def timestamp_to_date(timestamp: int) -> str: def timestamp_to_date(timestamp: int) -> str:
return datetime.utcfromtimestamp(timestamp).strftime("%Y-%m-%d") """Convert timestamp to date.
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
Args:
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
timestamp (int): Timestamp to convert.
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
Returns:
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
str: Date in the format YYYY-MM-DD.
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
"""
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
return datetime.fromtimestamp(timestamp=timestamp, tz=ZoneInfo("UTC")).strftime("%Y-%m-%d")
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)

github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)
github-advanced-security[bot] commented 2024-04-21 17:06:24 +02:00 (Migrated from github.com)
Review

"zoneinfo" should be preferred to "pytz" when using Python 3.9 and later

Don't use `pytz` module with Python 3.9 and later.

See more on SonarCloud

Show more details

## "zoneinfo" should be preferred to "pytz" when using Python 3.9 and later <!--SONAR_ISSUE_KEY:AY8BMVpXcfZYxM2gvceW-->Don't use `pytz` module with Python 3.9 and later. <p>See more on <a href="https://sonarcloud.io/project/issues?id=luketainton_roboluke-tasks&issues=AY8BMVpXcfZYxM2gvceW&open=AY8BMVpXcfZYxM2gvceW&pullRequest=184">SonarCloud</a></p> [Show more details](https://github.com/luketainton/roboluke-tasks/security/code-scanning/4)

View File

@ -1,9 +1,18 @@
import requests import requests
import sentry_sdk
from app.utils.config import config from app.utils.config import config
def __n8n_post(data: dict) -> bool: def __n8n_post(data: dict) -> bool:
"""Post data to N8N webhook URL.
Args:
data (dict): Data to post to webhook URL.
Returns:
bool: True if successful, else False.
"""
headers: dict = {"Content-Type": "application/json"} headers: dict = {"Content-Type": "application/json"}
resp: requests.Response = requests.post( resp: requests.Response = requests.post(
url=config.n8n_webhook_url, url=config.n8n_webhook_url,
@ -16,16 +25,38 @@ def __n8n_post(data: dict) -> bool:
def submit_task(summary, description, completion_date, requestor) -> bool: def submit_task(summary, description, completion_date, requestor) -> bool:
"""Submit task to N8N webhook URL.
Args:
summary (str): Summary of task.
description (str): Description of task.
completion_date (str): Completion date of task.
requestor (str): Requestor of task.
Returns:
bool: True if successful, else False.
"""
with sentry_sdk.start_transaction(name="submit_task"):
data: dict = { data: dict = {
"requestor": requestor, "requestor": requestor,
"title": summary, "title": summary,
"description": description, "description": description,
"completiondate": completion_date, "completiondate": completion_date,
} }
return __n8n_post(data=data) _data = __n8n_post(data=data)
return _data
def get_tasks(requestor) -> bool: def get_tasks(requestor) -> bool:
"""Get tasks from N8N webhook URL.
Args:
requestor (str): Requestor of tasks.
Returns:
bool: True if successful, else False.
"""
with sentry_sdk.start_transaction(name="get_tasks"):
headers: dict = {"Content-Type": "application/json"} headers: dict = {"Content-Type": "application/json"}
resp: requests.Response = requests.get( resp: requests.Response = requests.get(
url=config.n8n_webhook_url, url=config.n8n_webhook_url,
@ -34,4 +65,5 @@ def get_tasks(requestor) -> bool:
verify=False, verify=False,
params={"requestor": requestor}, params={"requestor": requestor},
) )
return bool(resp.status_code == 200) _data = bool(resp.status_code == 200)
return _data

View File

@ -33,6 +33,7 @@ python-dotenv==1.0.1
PyYAML==6.0.1 PyYAML==6.0.1
requests==2.31.0 requests==2.31.0
requests-toolbelt==1.0.0 requests-toolbelt==1.0.0
sentry-sdk==1.45.0
six==1.16.0 six==1.16.0
toml==0.10.2 toml==0.10.2
tomli==2.0.1 tomli==2.0.1

View File

@ -1,3 +1,3 @@
export $(cat .env | xargs) export $(cat .env | xargs)
python -B -m app.main python -B -m app.main
unset BOT_NAME WEBEX_API_KEY ADMIN_FIRST_NAME ADMIN_EMAIL N8N_WEBHOOK_URL unset APP_LIFECYCLE SENTRY_ENABLED SENTRY_DSN BOT_NAME WEBEX_API_KEY ADMIN_FIRST_NAME ADMIN_EMAIL N8N_WEBHOOK_URL

0
tests/__init__.py Normal file
View File

View File

@ -11,6 +11,8 @@ vars: dict = {
"ADMIN_FIRST_NAME": "Test", "ADMIN_FIRST_NAME": "Test",
"ADMIN_EMAIL": "test@test.com", "ADMIN_EMAIL": "test@test.com",
"N8N_WEBHOOK_URL": "https://n8n.test.com/webhook/abcdefg", "N8N_WEBHOOK_URL": "https://n8n.test.com/webhook/abcdefg",
"SENTRY_ENABLED": "false",
"SENTRY_DSN": "http://localhost"
} }