diff --git a/.github/workflows/deploy-dev.yaml b/.github/workflows/deploy-dev.yaml index cd5056a8..08eb590f 100644 --- a/.github/workflows/deploy-dev.yaml +++ b/.github/workflows/deploy-dev.yaml @@ -119,6 +119,8 @@ jobs: value: "${{ secrets.STRIPE_TEST_SECRET_KEY }}" - name: "DJSTRIPE_WEBHOOK_SECRET" value: "${{ secrets.DJSTRIPE_WEBHOOK_SECRET }}" + - name: "BETTERSTACK_BEARER_TOKEN" + value: "${{ secrets.BETTERSTACK_BEARER_TOKEN }}" - name: "DISCORD_BACKEND_WEBHOOK_URL" value: "${{ secrets.DISCORD_BACKEND_WEBHOOK_URL }}" envFrom: diff --git a/.github/workflows/deploy-prod.yaml b/.github/workflows/deploy-prod.yaml index 0857443d..a2e78928 100644 --- a/.github/workflows/deploy-prod.yaml +++ b/.github/workflows/deploy-prod.yaml @@ -88,6 +88,8 @@ jobs: value: "${{ secrets.STRIPE_TEST_SECRET_KEY }}" - name: "DJSTRIPE_WEBHOOK_SECRET" value: "${{ secrets.DJSTRIPE_WEBHOOK_SECRET }}" + - name: "BETTERSTACK_BEARER_TOKEN" + value: "${{ secrets.BETTERSTACK_BEARER_TOKEN }}" - name: "DISCORD_BACKEND_WEBHOOK_URL" value: "${{ secrets.DISCORD_BACKEND_WEBHOOK_URL }}" envFrom: diff --git a/.github/workflows/deploy-staging.yaml b/.github/workflows/deploy-staging.yaml index e1122446..5b778084 100644 --- a/.github/workflows/deploy-staging.yaml +++ b/.github/workflows/deploy-staging.yaml @@ -88,6 +88,8 @@ jobs: value: "${{ secrets.STRIPE_TEST_SECRET_KEY }}" - name: "DJSTRIPE_WEBHOOK_SECRET" value: "${{ secrets.DJSTRIPE_WEBHOOK_SECRET }}" + - name: "BETTERSTACK_BEARER_TOKEN" + value: "${{ secrets.BETTERSTACK_BEARER_TOKEN }}" - name: "DISCORD_BACKEND_WEBHOOK_URL" value: "${{ secrets.DISCORD_BACKEND_WEBHOOK_URL }}" envFrom: diff --git a/bd_api/apps/core/models.py b/bd_api/apps/core/models.py index 65ed5831..6b625a31 100644 --- a/bd_api/apps/core/models.py +++ b/bd_api/apps/core/models.py @@ -7,6 +7,9 @@ class Metadata(BaseModel): + created_at = models.DateTimeField(auto_now_add=True) + updated_at = models.DateTimeField(auto_now=True) + id = models.UUIDField(primary_key=True, default=uuid4) key = models.JSONField(default=dict, blank=False, null=False) value = models.JSONField(default=dict, blank=False, null=False) diff --git a/bd_api/apps/core/tasks.py b/bd_api/apps/core/tasks.py new file mode 100644 index 00000000..8ffc84d3 --- /dev/null +++ b/bd_api/apps/core/tasks.py @@ -0,0 +1,36 @@ +# -*- coding: utf-8 -*- +from datetime import datetime, timedelta + +from huey import crontab +from huey.contrib.djhuey import periodic_task + +from bd_api.apps.core.models import Metadata +from bd_api.custom.client import BetterStackClient +from bd_api.utils import production_task + + +@periodic_task(crontab(day="1", hour="3", minute="0")) +@production_task +def get_monitor_availability(): + """Get availability metric and save as metadata""" + + def get_period(): + """Get last month period as tuple of strings""" + + today = datetime.today() + since = datetime(today.year, today.month - 1, 1) + until = datetime(today.year, today.month, 1) - timedelta(days=1) + return since.strftime("%Y-%m-%d"), until.strftime("%Y-%m-%d") + + since, until = get_period() + client = BetterStackClient() + for monitor in client.get_monitors(): + Metadata.objects.create( + key={ + "since": since, + "until": until, + "monitor_id": monitor["id"], + "monitor_url": monitor["attributes"]["url"], + }, + value=client.get_monitor_summary(monitor["id"], since, until), + ) diff --git a/bd_api/custom/client.py b/bd_api/custom/client.py index c15f171d..7a84d5b0 100644 --- a/bd_api/custom/client.py +++ b/bd_api/custom/client.py @@ -8,7 +8,7 @@ from google.cloud.storage import Client as GCSClient from google.oauth2.service_account import Credentials from loguru import logger -from requests import post +from requests import Session, post from bd_api.custom.model import BaseModel @@ -72,3 +72,25 @@ def send(self): """Send message if it has body text""" if self._message.count("\n"): send_discord_message(self._message) + + +class BetterStackClient: + def __init__(self) -> None: + self.session = Session() + self.session.headers.update( + {"Authorization": f"Bearer {settings.BETTERSTACK_BEARER_TOKEN}"} + ) + self.url = "https://uptime.betterstack.com" + + def get_monitors(self): + response = self.session.get(f"{self.url}/api/v2/monitors") + response.raise_for_status() + return response.json() + + def get_monitor_summary(self, monitor_id: str, since: str, until: str): + response = self.session.get( + f"{self.url}/api/v2/{monitor_id}/sla", + params={"from": since, "to": until}, + ) + response.raise_for_status() + return response.json() diff --git a/bd_api/settings/base.py b/bd_api/settings/base.py index c56ccd12..78780d33 100644 --- a/bd_api/settings/base.py +++ b/bd_api/settings/base.py @@ -299,3 +299,6 @@ # Discord DISCORD_BACKEND_WEBHOOK_URL = getenv("DISCORD_BACKEND_WEBHOOK_URL") + +# BetterStack +BETTERSTACK_BEARER_TOKEN = getenv("BETTERSTACK_BEARER_TOKEN")