Release 2024-04-21.1 (#185)
* chore(pip-prod)(deps): bump pyparsing from 3.1.1 to 3.1.2 Bumps [pyparsing](https://github.com/pyparsing/pyparsing) from 3.1.1 to 3.1.2. - [Release notes](https://github.com/pyparsing/pyparsing/releases) - [Changelog](https://github.com/pyparsing/pyparsing/blob/master/CHANGES) - [Commits](https://github.com/pyparsing/pyparsing/compare/3.1.1...pyparsing_3.1.2) --- updated-dependencies: - dependency-name: pyparsing dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump packaging from 23.2 to 24.0 Bumps [packaging](https://github.com/pypa/packaging) from 23.2 to 24.0. - [Release notes](https://github.com/pypa/packaging/releases) - [Changelog](https://github.com/pypa/packaging/blob/main/CHANGELOG.rst) - [Commits](https://github.com/pypa/packaging/compare/23.2...24.0) --- updated-dependencies: - dependency-name: packaging dependency-type: direct:production update-type: version-update:semver-major ... Signed-off-by: dependabot[bot] <support@github.com> * chore(actions)(deps): bump mathieudutour/github-tag-action Bumps [mathieudutour/github-tag-action](https://github.com/mathieudutour/github-tag-action) from 6.1 to 6.2. - [Release notes](https://github.com/mathieudutour/github-tag-action/releases) - [Commits](https://github.com/mathieudutour/github-tag-action/compare/v6.1...v6.2) --- updated-dependencies: - dependency-name: mathieudutour/github-tag-action dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump autopep8 from 2.0.4 to 2.1.0 Bumps [autopep8](https://github.com/hhatto/autopep8) from 2.0.4 to 2.1.0. - [Release notes](https://github.com/hhatto/autopep8/releases) - [Commits](https://github.com/hhatto/autopep8/compare/v2.0.4...v2.1.0) --- updated-dependencies: - dependency-name: autopep8 dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump filelock from 3.13.1 to 3.13.2 Bumps [filelock](https://github.com/tox-dev/py-filelock) from 3.13.1 to 3.13.2. - [Release notes](https://github.com/tox-dev/py-filelock/releases) - [Changelog](https://github.com/tox-dev/filelock/blob/main/docs/changelog.rst) - [Commits](https://github.com/tox-dev/py-filelock/compare/3.13.1...3.13.2) --- updated-dependencies: - dependency-name: filelock dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump filelock from 3.13.2 to 3.13.3 Bumps [filelock](https://github.com/tox-dev/py-filelock) from 3.13.2 to 3.13.3. - [Release notes](https://github.com/tox-dev/py-filelock/releases) - [Changelog](https://github.com/tox-dev/filelock/blob/main/docs/changelog.rst) - [Commits](https://github.com/tox-dev/py-filelock/compare/3.13.2...3.13.3) --- updated-dependencies: - dependency-name: filelock dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump filelock from 3.13.3 to 3.13.4 Bumps [filelock](https://github.com/tox-dev/py-filelock) from 3.13.3 to 3.13.4. - [Release notes](https://github.com/tox-dev/py-filelock/releases) - [Changelog](https://github.com/tox-dev/filelock/blob/main/docs/changelog.rst) - [Commits](https://github.com/tox-dev/py-filelock/compare/3.13.3...3.13.4) --- updated-dependencies: - dependency-name: filelock dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump idna from 3.6 to 3.7 Bumps [idna](https://github.com/kjd/idna) from 3.6 to 3.7. - [Release notes](https://github.com/kjd/idna/releases) - [Changelog](https://github.com/kjd/idna/blob/master/HISTORY.rst) - [Commits](https://github.com/kjd/idna/compare/v3.6...v3.7) --- updated-dependencies: - dependency-name: idna dependency-type: direct:production update-type: version-update:semver-minor ... Signed-off-by: dependabot[bot] <support@github.com> * chore(pip-prod)(deps): bump virtualenv from 20.25.1 to 20.25.3 Bumps [virtualenv](https://github.com/pypa/virtualenv) from 20.25.1 to 20.25.3. - [Release notes](https://github.com/pypa/virtualenv/releases) - [Changelog](https://github.com/pypa/virtualenv/blob/main/docs/changelog.rst) - [Commits](https://github.com/pypa/virtualenv/compare/20.25.1...20.25.3) --- updated-dependencies: - dependency-name: virtualenv dependency-type: direct:production update-type: version-update:semver-patch ... Signed-off-by: dependabot[bot] <support@github.com> * Add APM (#184) * Add APM * Fix unit tests * Remove apm.py * Add SonarCloud recommendations * SonarCloud python:S6890 --------- Signed-off-by: dependabot[bot] <support@github.com> Co-authored-by: dependabot[bot] <49699333+dependabot[bot]@users.noreply.github.com>
This commit is contained in:
parent
99d8b862c6
commit
9830c5cc3a
@ -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=""
|
||||||
|
2
.github/workflows/release.yml
vendored
2
.github/workflows/release.yml
vendored
@ -13,7 +13,7 @@ jobs:
|
|||||||
- uses: actions/checkout@v4
|
- uses: actions/checkout@v4
|
||||||
- name: Bump version and push tag
|
- name: Bump version and push tag
|
||||||
id: tag_version
|
id: tag_version
|
||||||
uses: mathieudutour/github-tag-action@v6.1
|
uses: mathieudutour/github-tag-action@v6.2
|
||||||
with:
|
with:
|
||||||
github_token: ${{ secrets.GITHUB_TOKEN }}
|
github_token: ${{ secrets.GITHUB_TOKEN }}
|
||||||
default_bump: minor
|
default_bump: minor
|
||||||
|
@ -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,7 +114,8 @@ class SubmitTaskCommand(Command):
|
|||||||
Submit(title="Cancel", data={"command_keyword": "exit"}),
|
Submit(title="Cancel", data={"command_keyword": "exit"}),
|
||||||
],
|
],
|
||||||
)
|
)
|
||||||
return response_from_adaptive_card(card)
|
with sentry_sdk.start_transaction(name="submit_task_command"):
|
||||||
|
return response_from_adaptive_card(card)
|
||||||
|
|
||||||
|
|
||||||
class SubmitTaskCallback(Command):
|
class SubmitTaskCallback(Command):
|
||||||
@ -147,7 +149,8 @@ class SubmitTaskCallback(Command):
|
|||||||
)
|
)
|
||||||
|
|
||||||
def execute(self, message, attachment_actions, activity) -> str:
|
def execute(self, message, attachment_actions, activity) -> str:
|
||||||
return self.msg
|
with sentry_sdk.start_transaction(name="submit_task_callback"):
|
||||||
|
return self.msg
|
||||||
|
|
||||||
|
|
||||||
class MyTasksCallback(Command):
|
class MyTasksCallback(Command):
|
||||||
@ -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:
|
||||||
return "Getting your tasks..."
|
with sentry_sdk.start_transaction(name="my_tasks_preexec"):
|
||||||
|
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)
|
||||||
if not result:
|
with sentry_sdk.start_transaction(name="my_tasks_exec"):
|
||||||
return "Failed to get tasks. Please try again."
|
if not result:
|
||||||
return
|
return "Failed to get tasks. Please try again."
|
||||||
|
return
|
||||||
|
15
app/main.py
15
app/main.py
@ -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,
|
||||||
|
@ -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
|
||||||
|
|
||||||
|
|
||||||
|
@ -1,7 +1,18 @@
|
|||||||
#!/usr/bin/env python3
|
#!/usr/bin/env python3
|
||||||
|
|
||||||
|
from zoneinfo import ZoneInfo
|
||||||
|
|
||||||
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.
|
||||||
|
|
||||||
|
Args:
|
||||||
|
timestamp (int): Timestamp to convert.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
str: Date in the format YYYY-MM-DD.
|
||||||
|
"""
|
||||||
|
return datetime.fromtimestamp(timestamp=timestamp, tz=ZoneInfo("UTC")).strftime("%Y-%m-%d")
|
||||||
|
|
||||||
|
@ -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,22 +25,45 @@ def __n8n_post(data: dict) -> bool:
|
|||||||
|
|
||||||
|
|
||||||
def submit_task(summary, description, completion_date, requestor) -> bool:
|
def submit_task(summary, description, completion_date, requestor) -> bool:
|
||||||
data: dict = {
|
"""Submit task to N8N webhook URL.
|
||||||
"requestor": requestor,
|
|
||||||
"title": summary,
|
Args:
|
||||||
"description": description,
|
summary (str): Summary of task.
|
||||||
"completiondate": completion_date,
|
description (str): Description of task.
|
||||||
}
|
completion_date (str): Completion date of task.
|
||||||
return __n8n_post(data=data)
|
requestor (str): Requestor of task.
|
||||||
|
|
||||||
|
Returns:
|
||||||
|
bool: True if successful, else False.
|
||||||
|
"""
|
||||||
|
with sentry_sdk.start_transaction(name="submit_task"):
|
||||||
|
data: dict = {
|
||||||
|
"requestor": requestor,
|
||||||
|
"title": summary,
|
||||||
|
"description": description,
|
||||||
|
"completiondate": completion_date,
|
||||||
|
}
|
||||||
|
_data = __n8n_post(data=data)
|
||||||
|
return _data
|
||||||
|
|
||||||
|
|
||||||
def get_tasks(requestor) -> bool:
|
def get_tasks(requestor) -> bool:
|
||||||
headers: dict = {"Content-Type": "application/json"}
|
"""Get tasks from N8N webhook URL.
|
||||||
resp: requests.Response = requests.get(
|
|
||||||
url=config.n8n_webhook_url,
|
Args:
|
||||||
headers=headers,
|
requestor (str): Requestor of tasks.
|
||||||
timeout=10,
|
|
||||||
verify=False,
|
Returns:
|
||||||
params={"requestor": requestor},
|
bool: True if successful, else False.
|
||||||
)
|
"""
|
||||||
return bool(resp.status_code == 200)
|
with sentry_sdk.start_transaction(name="get_tasks"):
|
||||||
|
headers: dict = {"Content-Type": "application/json"}
|
||||||
|
resp: requests.Response = requests.get(
|
||||||
|
url=config.n8n_webhook_url,
|
||||||
|
headers=headers,
|
||||||
|
timeout=10,
|
||||||
|
verify=False,
|
||||||
|
params={"requestor": requestor},
|
||||||
|
)
|
||||||
|
_data = bool(resp.status_code == 200)
|
||||||
|
return _data
|
||||||
|
@ -1,7 +1,7 @@
|
|||||||
appdirs==1.4.4
|
appdirs==1.4.4
|
||||||
astroid==3.1.0
|
astroid==3.1.0
|
||||||
attrs==23.2.0
|
attrs==23.2.0
|
||||||
autopep8==2.0.4
|
autopep8==2.1.0
|
||||||
backoff==2.2.1
|
backoff==2.2.1
|
||||||
certifi==2024.2.2
|
certifi==2024.2.2
|
||||||
cfgv==3.4.0
|
cfgv==3.4.0
|
||||||
@ -10,35 +10,36 @@ click==8.1.7
|
|||||||
coloredlogs==15.0.1
|
coloredlogs==15.0.1
|
||||||
dill==0.3.8
|
dill==0.3.8
|
||||||
distlib==0.3.8
|
distlib==0.3.8
|
||||||
filelock==3.13.1
|
filelock==3.13.4
|
||||||
future==1.0.0
|
future==1.0.0
|
||||||
humanfriendly==10.0
|
humanfriendly==10.0
|
||||||
identify==2.5.35
|
identify==2.5.35
|
||||||
idna==3.6
|
idna==3.7
|
||||||
iniconfig==2.0.0
|
iniconfig==2.0.0
|
||||||
lazy-object-proxy==1.10.0
|
lazy-object-proxy==1.10.0
|
||||||
mccabe==0.7.0
|
mccabe==0.7.0
|
||||||
mypy-extensions==1.0.0
|
mypy-extensions==1.0.0
|
||||||
nodeenv==1.8.0
|
nodeenv==1.8.0
|
||||||
packaging==23.2
|
packaging==24.0
|
||||||
pathspec==0.12.1
|
pathspec==0.12.1
|
||||||
platformdirs==4.2.0
|
platformdirs==4.2.0
|
||||||
pluggy==1.4.0
|
pluggy==1.4.0
|
||||||
py==1.11.0
|
py==1.11.0
|
||||||
pycodestyle==2.11.1
|
pycodestyle==2.11.1
|
||||||
PyJWT==2.8.0
|
PyJWT==2.8.0
|
||||||
pyparsing==3.1.1
|
pyparsing==3.1.2
|
||||||
python-dateutil==2.9.0.post0
|
python-dateutil==2.9.0.post0
|
||||||
python-dotenv==1.0.1
|
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
|
||||||
tomlkit==0.12.4
|
tomlkit==0.12.4
|
||||||
urllib3==2.2.1
|
urllib3==2.2.1
|
||||||
virtualenv==20.25.1
|
virtualenv==20.25.3
|
||||||
webex-bot==0.4.1
|
webex-bot==0.4.1
|
||||||
webexteamssdk==1.6.1
|
webexteamssdk==1.6.1
|
||||||
websockets==10.2
|
websockets==10.2
|
||||||
|
2
test.sh
2
test.sh
@ -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
0
tests/__init__.py
Normal 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"
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
||||||
|
Loading…
Reference in New Issue
Block a user