diff --git a/docs/deployment/getting-started.mdx b/docs/deployment/getting-started.mdx deleted file mode 100644 index b09c18527..000000000 --- a/docs/deployment/getting-started.mdx +++ /dev/null @@ -1,112 +0,0 @@ ---- -title: "" -sidebarTitle: "Getting Started" ---- - -# Specifications and Stress Testing of Keep -If you are using Keep and have performance issues, we will be more than happy to help you. Just join our [slack](https://slack.keepqh.dev) and shoot a message on the **#help** channel. - -## Overview - -Spec and stress testing are crucial to ensuring the robust performance and scalability of Keep. -This documentation outlines the key areas of focus for testing Keep under different load conditions, considering both the simplicity of setup for smaller environments and the scalability mechanisms for larger deployments. - -Keep was initially designed to be user-friendly for setups handling less than 10,000 alerts. However, as alert volumes increase, users can leverage advanced features such as Elasticsearch for document storage and Redis + ARQ for queue-based alert ingestion. While these advanced configurations are not fully documented here, they are supported and can be discussed further in our Slack community. - -## How To Reproduce - -To reproduce the stress testing scenarios mentioned above, please refer to the [STRESS.md](https://github.com/keephq/keep/blob/main/STRESS.md) file in Keep's repository. This document provides step-by-step instructions on how to set up, run, and measure the performance of Keep under different load conditions. - -## Performance Testing - -### Factors Affecting Specifications - -The primary parameters that affect the specification requirements for Keep are: -1. **Alerts Volume**: The rate at which alerts are ingested into the system. -2. **Total Alerts**: The cumulative number of alerts stored in the system. -3. **Number of Workflows**: How many automation run as a result of alert. - -### Main Components: -- **Keep Backend** - API and business logic. A container that serves FastAPI on top of gunicorn. -- **Keep Frontend** - Web app. A container that serves the react app. -- **Database** - Stores the alerts and any other operational data. -- **Elasticsearch** (opt out by default) - Stores alerts as document for better search performance. -- **Redis** (opt out by default) - Used, together with ARQ, as an alerts queue. - -### Testing Scenarios: - -- **Low Volume (< 10,000 total alerts, hundreds of alerts per day)**: - - **Setup**: Use a standard relational database (e.g., MySQL, PostgreSQL) with default configurations. - - **Expectations**: Keep should handle queries and alert ingestion with minimal resource usage. - -- **Medium Volume (10,000 - 100,000 total alerts, thousands of alerts per day)**: - - **Setup**: Scale the database to larger instances or clusters. Adjust best practices to the DB (e.g. increasing innodb_buffer_pool_size) - - **Expectations**: CPU and RAM usage should increase proportionally but remain within acceptable limits. - -3. **High Volume (100,000 - 1,000,000 total alerts, >five thousands of alerts per day)**: - - **Setup**: Deploy Keep with Elasticsearch for storing alerts as documents. - - **Expectations**: The system should maintain performance levels despite the large alert volume, with increased resource usage managed through scaling strategies. -4. **Very High Volume (> 1,000,000 total alerts, tens of thousands of alerts per day)**: - - **Setup**: Deploy Keep with Elasticsearch for storing alerts as documents. - - **Setup #2**: Deploy Keep with Redis and with ARQ to use Redis as a queue. - -## Recommended Specifications by Alert Volume - -| **Number of Alerts** | **Keep Backend** | **Keep Database** | **Redis** | **Elasticsearch** | -|------------------------|------------------------------------------------|-------------------------------------------------|------------------------------------------------|------------------------------------------------| -| **< 10,000** | 1 vCPUs, 2GB RAM | 2 vCPUs, 8GB RAM | Not required | Not required | -| **10,000 - 100,000** | 4 vCPUs, 8GB RAM | 8 vCPUs, 32GB RAM, optimized indexing | Not required | Not required | -| **100,000 - 500,000** | 8 vCPUs, 16GB RAM | 8 vCPUs, 32GB RAM, advanced indexing | 4 vCPUs, 8GB RAM | 8 vCPUs, 32GB RAM, 2-3 nodes | -| **> 500,000** | 8 vCPUs, 16GB RAM | 8 vCPUs, 32GB RAM, advanced indexing, sharding| 4 vCPUs, 8GB RAM | 8 vCPUs, 32GB RAM, 2-3 nodes | - -## Performance by Operation Type, Load, and Specification - -| **Operation Type** | **Load** | **Specification** | **Execution Time** | -|-----------------------|----------------------------|------------------------------|-----------------------------------| -| Digest Alert | 100 alerts per minute | 4 vCPUs, 8GB RAM | ~0.5 seconds | -| Digest Alert | 500 alerts per minute | 8 vCPUs, 16GB RAM | ~1 second | -| Digest Alert | 1,000 alerts per minute | 16 vCPUs, 32GB RAM | ~1.5 seconds | -| Run Workflow | 10 workflows per minute | 4 vCPUs, 8GB RAM | ~1 second | -| Run Workflow | 50 workflows per minute | 8 vCPUs, 16GB RAM | ~2 seconds | -| Run Workflow | 100 workflows per minute | 16 vCPUs, 32GB RAM | ~3 seconds | -| Ingest via Queue | 100 alerts per minute | 4 vCPUs, 8GB RAM, Redis | ~0.3 seconds | -| Ingest via Queue | 500 alerts per minute | 8 vCPUs, 16GB RAM, Redis | ~0.8 seconds | -| Ingest via Queue | 1,000 alerts per minute | 16 vCPUs, 32GB RAM, Redis | ~1.2 seconds | - -### Table Explanation: -- **Operation Type**: The specific operation being tested (e.g., digesting alerts, running workflows). -- **Load**: The number of operations per minute being processed (e.g., number of alerts per minute). -- **Specification**: The CPU, RAM, and additional services used for the operation. -- **Execution Time**: Approximate time taken to complete the operation under the given load and specification. - - -## Fine Tuning - -As any deployment has its own characteristics, such as the balance between volume vs. total count of alerts or volume vs. number of workflows, Keep can be fine-tuned with the following parameters: - -1. **Number of Workers**: Adjust the number of Gunicorn workers to handle API requests more efficiently. You can also start additional API servers to distribute the load. -2. **Distinguish Between API Server Workers and Digesting Alerts Workers**: Separate the workers dedicated to handling API requests from those responsible for digesting alerts, ensuring that each set of tasks is optimized according to its specific needs. -3. **Add More RAM to the Database**: Increasing the RAM allocated to your database can help manage larger datasets and improve query performance, particularly when dealing with high volumes of alerts. -4. **Optimize Database Configuration**: Keep was mainly tested on MySQL and PostgreSQL. Different database may have different fine tuning mechanisms. -5. **Horizontal Scaling**: Consider deploying additional instances of the API and database services to distribute the load more effectively. - - - -## FAQ - -### 1. How do I estimate the spec I need for Keep? -To estimate the specifications required for Keep, consider both the number of alerts per minute and the total number of alerts you expect to handle. Refer to the **Recommended Specifications by Alert Volume** table above to match your expected load with the appropriate resources. - -### 2. How do I know if I need Elasticsearch? -Elasticsearch is typically needed when you are dealing with more than 50,000 total alerts or if you require advanced search and query capabilities that are not efficiently handled by a traditional relational database. If your system’s performance degrades significantly as alert volume increases, it may be time to consider Elasticsearch. - -### 3. How do I know if I need Redis? -Redis is recommended when your alert ingestion rate exceeds 1,000 alerts per minute or when you notice that the API is becoming a bottleneck due to high ingestion rates. Redis, combined with ARQ (Asynchronous Redis Queue), can help manage and distribute the load more effectively. - -### 4. What should I do if Keep's performance is still inadequate? -If you have scaled according to the recommendations and are still facing performance issues, consider: -- **Optimizing your database configuration**: Indexing, sharding, and query optimization can make a significant difference. -- **Horizontal scaling**: Distribute the load across multiple instances of the API and database services. -- **Reach out to our Slack community**: For personalized support, reach out to us on Slack, and we’ll help you troubleshoot and optimize your Keep deployment. - -For any additional questions or tailored advice, feel free to join our Slack community where our team and other users are available to assist you. diff --git a/docs/mint.json b/docs/mint.json index c01b99902..373398997 100644 --- a/docs/mint.json +++ b/docs/mint.json @@ -211,7 +211,6 @@ { "group": "Deployment", "pages": [ - "deployment/getting-started", "deployment/configuration", "deployment/monitoring", { diff --git a/keep/api/core/db.py b/keep/api/core/db.py index 4258400dd..d17817421 100644 --- a/keep/api/core/db.py +++ b/keep/api/core/db.py @@ -1270,6 +1270,8 @@ def get_last_alerts( select(Alert, LastAlert.first_timestamp.label("startedAt")) .select_from(LastAlert) .join(Alert, LastAlert.alert_id == Alert.id) + .where(LastAlert.tenant_id == tenant_id) + .where(Alert.tenant_id == tenant_id) ) if timeframe: @@ -1296,9 +1298,7 @@ def get_last_alerts( stmt = stmt.where(*filter_conditions) # Main query for alerts - stmt = stmt.where(Alert.tenant_id == tenant_id).options( - subqueryload(Alert.alert_enrichment) - ) + stmt = stmt.options(subqueryload(Alert.alert_enrichment)) if with_incidents: if dialect_name == "sqlite": diff --git a/keep/api/middlewares.py b/keep/api/middlewares.py index af4e71d1d..0e5567a61 100644 --- a/keep/api/middlewares.py +++ b/keep/api/middlewares.py @@ -1,9 +1,9 @@ +import logging import os -import jwt import time -import logging from importlib import metadata +import jwt from fastapi import Request from starlette.middleware.base import BaseHTTPMiddleware @@ -56,5 +56,9 @@ async def dispatch(self, request: Request, call_next): end_time = time.time() logger.info( f"Request finished: {request.method} {request.url.path} {response.status_code} in {end_time - start_time:.2f}s", + extra={ + "tenant_id": identity, + "status_code": response.status_code, + }, ) return response diff --git a/keep/api/models/db/alert.py b/keep/api/models/db/alert.py index 1d9bbeb52..4e2e5af3a 100644 --- a/keep/api/models/db/alert.py +++ b/keep/api/models/db/alert.py @@ -81,6 +81,20 @@ class LastAlert(SQLModel, table=True): first_timestamp: datetime = Field(nullable=False, index=True) alert_hash: str | None = Field(nullable=True, index=True) + __table_args__ = ( + # Original indexes from MySQL + Index("idx_lastalert_tenant_timestamp", "tenant_id", "first_timestamp"), + Index("idx_lastalert_tenant_timestamp_new", "tenant_id", "timestamp"), + Index( + "idx_lastalert_tenant_ordering", + "tenant_id", + "first_timestamp", + "alert_id", + "fingerprint", + ), + {}, + ) + class LastAlertToIncident(SQLModel, table=True): tenant_id: str = Field(foreign_key="tenant.id", nullable=False, primary_key=True) @@ -109,6 +123,15 @@ class LastAlertToIncident(SQLModel, table=True): ["tenant_id", "fingerprint"], ["lastalert.tenant_id", "lastalert.fingerprint"], ), + Index( + "idx_lastalerttoincident_tenant_fingerprint", + "tenant_id", + "fingerprint", + "deleted_at", + ), + Index( + "idx_tenant_deleted_fingerprint", "tenant_id", "deleted_at", "fingerprint" + ), {}, ) @@ -254,6 +277,13 @@ class Alert(SQLModel, table=True): "fingerprint", "timestamp", ), + Index("idx_fingerprint_timestamp", "fingerprint", "timestamp"), + Index( + "idx_alert_tenant_timestamp_fingerprint", + "tenant_id", + "timestamp", + "fingerprint", + ), ) class Config: diff --git a/keep/api/models/db/migrations/versions/2025-01-01-09-59_dcb7f88a04da.py b/keep/api/models/db/migrations/versions/2025-01-01-09-59_dcb7f88a04da.py new file mode 100644 index 000000000..f99d1998b --- /dev/null +++ b/keep/api/models/db/migrations/versions/2025-01-01-09-59_dcb7f88a04da.py @@ -0,0 +1,85 @@ +"""Few more indexes + +Revision ID: dcb7f88a04da +Revises: 7297ae99cd21 +Create Date: 2025-01-01 09:59:13.393588 + +""" + +import sqlalchemy as sa +from alembic import op + +# revision identifiers, used by Alembic. +revision = "dcb7f88a04da" +down_revision = "7297ae99cd21" +branch_labels = None +depends_on = None + + +def upgrade() -> None: + # ### commands auto generated by Alembic - please adjust! ### + with op.batch_alter_table("alert", schema=None) as batch_op: + batch_op.create_index( + "idx_alert_tenant_timestamp_fingerprint", + ["tenant_id", "timestamp", "fingerprint"], + unique=False, + ) + batch_op.create_index( + "idx_fingerprint_timestamp", ["fingerprint", "timestamp"], unique=False + ) + + with op.batch_alter_table("lastalert", schema=None) as batch_op: + batch_op.alter_column( + "first_timestamp", existing_type=sa.DATETIME(), nullable=False + ) + batch_op.create_index( + "idx_lastalert_tenant_ordering", + ["tenant_id", "first_timestamp", "alert_id", "fingerprint"], + unique=False, + ) + batch_op.create_index( + "idx_lastalert_tenant_timestamp", + ["tenant_id", "first_timestamp"], + unique=False, + ) + batch_op.create_index( + "idx_lastalert_tenant_timestamp_new", + ["tenant_id", "timestamp"], + unique=False, + ) + + with op.batch_alter_table("lastalerttoincident", schema=None) as batch_op: + batch_op.create_index( + "idx_lastalerttoincident_tenant_fingerprint", + ["tenant_id", "fingerprint", "deleted_at"], + unique=False, + ) + batch_op.create_index( + "idx_tenant_deleted_fingerprint", + ["tenant_id", "deleted_at", "fingerprint"], + unique=False, + ) + + # ### end Alembic commands ### + + +def downgrade() -> None: + # ### commands auto generated by Alembic - please adjust! ### + + with op.batch_alter_table("lastalerttoincident", schema=None) as batch_op: + batch_op.drop_index("idx_tenant_deleted_fingerprint") + batch_op.drop_index("idx_lastalerttoincident_tenant_fingerprint") + + with op.batch_alter_table("lastalert", schema=None) as batch_op: + batch_op.drop_index("idx_lastalert_tenant_timestamp_new") + batch_op.drop_index("idx_lastalert_tenant_timestamp") + batch_op.drop_index("idx_lastalert_tenant_ordering") + batch_op.alter_column( + "first_timestamp", existing_type=sa.DATETIME(), nullable=True + ) + + with op.batch_alter_table("alert", schema=None) as batch_op: + batch_op.drop_index("idx_fingerprint_timestamp") + batch_op.drop_index("idx_alert_tenant_timestamp_fingerprint") + + # ### end Alembic commands ###