Fix SQLite view creation and test domain filter
This commit is contained in:
		
					parent
					
						
							
								b87c76e8c9
							
						
					
				
			
			
				commit
				
					
						fb08fcaa07
					
				
			
		
					 2 changed files with 68 additions and 12 deletions
				
			
		|  | @ -36,6 +36,7 @@ def _get_domains() -> List[str]: | ||||||
|     conn.close() |     conn.close() | ||||||
|     return domains |     return domains | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| def _load_config() -> List[Dict]: | def _load_config() -> List[Dict]: | ||||||
|     if not REPORT_CONFIG.exists(): |     if not REPORT_CONFIG.exists(): | ||||||
|         typer.echo(f"Config file not found: {REPORT_CONFIG}") |         typer.echo(f"Config file not found: {REPORT_CONFIG}") | ||||||
|  | @ -47,10 +48,12 @@ def _load_config() -> List[Dict]: | ||||||
|         raise typer.Exit(1) |         raise typer.Exit(1) | ||||||
|     return data |     return data | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| def _save_json(path: Path, data: List[Dict]) -> None: | def _save_json(path: Path, data: List[Dict]) -> None: | ||||||
|     path.parent.mkdir(parents=True, exist_ok=True) |     path.parent.mkdir(parents=True, exist_ok=True) | ||||||
|     path.write_text(json.dumps(data, indent=2)) |     path.write_text(json.dumps(data, indent=2)) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| def _render_html(interval: str, reports: List[Dict], out_path: Path) -> None: | def _render_html(interval: str, reports: List[Dict], out_path: Path) -> None: | ||||||
|     env = Environment(loader=FileSystemLoader(TEMPLATE_DIR)) |     env = Environment(loader=FileSystemLoader(TEMPLATE_DIR)) | ||||||
|     template = env.get_template("report.html") |     template = env.get_template("report.html") | ||||||
|  | @ -65,6 +68,7 @@ def _bucket_expr(interval: str) -> str: | ||||||
|         raise typer.Exit(1) |         raise typer.Exit(1) | ||||||
|     return f"strftime('{fmt}', datetime(time))" |     return f"strftime('{fmt}', datetime(time))" | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| def _generate_interval(interval: str, domain: Optional[str] = None) -> None: | def _generate_interval(interval: str, domain: Optional[str] = None) -> None: | ||||||
|     cfg = _load_config() |     cfg = _load_config() | ||||||
|     if not cfg: |     if not cfg: | ||||||
|  | @ -79,9 +83,12 @@ def _generate_interval(interval: str, domain: Optional[str] = None) -> None: | ||||||
|     # Create a temporary view so queries can easily be filtered by domain |     # Create a temporary view so queries can easily be filtered by domain | ||||||
|     cur.execute("DROP VIEW IF EXISTS logs_view") |     cur.execute("DROP VIEW IF EXISTS logs_view") | ||||||
|     if domain: |     if domain: | ||||||
|  |         # Parameters are not allowed in CREATE VIEW statements, so we must | ||||||
|  |         # safely interpolate the domain value ourselves. Escape any single | ||||||
|  |         # quotes to prevent malformed queries. | ||||||
|  |         safe_domain = domain.replace("'", "''") | ||||||
|         cur.execute( |         cur.execute( | ||||||
|             "CREATE TEMP VIEW logs_view AS SELECT * FROM logs WHERE host = ?", |             f"CREATE TEMP VIEW logs_view AS SELECT * FROM logs WHERE host = '{safe_domain}'" | ||||||
|             (domain,), |  | ||||||
|         ) |         ) | ||||||
|         out_dir = OUTPUT_DIR / domain / interval |         out_dir = OUTPUT_DIR / domain / interval | ||||||
|     else: |     else: | ||||||
|  | @ -101,12 +108,14 @@ def _generate_interval(interval: str, domain: Optional[str] = None) -> None: | ||||||
|         data = [dict(zip(headers, row)) for row in rows] |         data = [dict(zip(headers, row)) for row in rows] | ||||||
|         json_path = out_dir / f"{name}.json" |         json_path = out_dir / f"{name}.json" | ||||||
|         _save_json(json_path, data) |         _save_json(json_path, data) | ||||||
|         report_list.append({ |         report_list.append( | ||||||
|             "name": name, |             { | ||||||
|             "label": definition.get("label", name.title()), |                 "name": name, | ||||||
|             "chart": definition.get("chart", "line"), |                 "label": definition.get("label", name.title()), | ||||||
|             "json": f"{name}.json", |                 "chart": definition.get("chart", "line"), | ||||||
|         }) |                 "json": f"{name}.json", | ||||||
|  |             } | ||||||
|  |         ) | ||||||
| 
 | 
 | ||||||
|     _save_json(out_dir / "reports.json", report_list) |     _save_json(out_dir / "reports.json", report_list) | ||||||
|     _render_html(interval, report_list, out_dir / "index.html") |     _render_html(interval, report_list, out_dir / "index.html") | ||||||
|  | @ -118,6 +127,7 @@ def _generate_all_domains(interval: str) -> None: | ||||||
|     for domain in _get_domains(): |     for domain in _get_domains(): | ||||||
|         _generate_interval(interval, domain) |         _generate_interval(interval, domain) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| @app.command() | @app.command() | ||||||
| def hourly( | def hourly( | ||||||
|     domain: Optional[str] = typer.Option( |     domain: Optional[str] = typer.Option( | ||||||
|  | @ -133,6 +143,7 @@ def hourly( | ||||||
|     else: |     else: | ||||||
|         _generate_interval("hourly", domain) |         _generate_interval("hourly", domain) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| @app.command() | @app.command() | ||||||
| def daily( | def daily( | ||||||
|     domain: Optional[str] = typer.Option( |     domain: Optional[str] = typer.Option( | ||||||
|  | @ -148,6 +159,7 @@ def daily( | ||||||
|     else: |     else: | ||||||
|         _generate_interval("daily", domain) |         _generate_interval("daily", domain) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| @app.command() | @app.command() | ||||||
| def weekly( | def weekly( | ||||||
|     domain: Optional[str] = typer.Option( |     domain: Optional[str] = typer.Option( | ||||||
|  | @ -163,6 +175,7 @@ def weekly( | ||||||
|     else: |     else: | ||||||
|         _generate_interval("weekly", domain) |         _generate_interval("weekly", domain) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| @app.command() | @app.command() | ||||||
| def monthly( | def monthly( | ||||||
|     domain: Optional[str] = typer.Option( |     domain: Optional[str] = typer.Option( | ||||||
|  | @ -178,5 +191,6 @@ def monthly( | ||||||
|     else: |     else: | ||||||
|         _generate_interval("monthly", domain) |         _generate_interval("monthly", domain) | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
| if __name__ == "__main__": | if __name__ == "__main__": | ||||||
|     app() |     app() | ||||||
|  |  | ||||||
|  | @ -32,11 +32,31 @@ def setup_db(path: Path): | ||||||
|     ) |     ) | ||||||
|     cur.execute( |     cur.execute( | ||||||
|         "INSERT INTO logs (ip, host, time, request, status, bytes_sent, referer, user_agent, cache_status) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)", |         "INSERT INTO logs (ip, host, time, request, status, bytes_sent, referer, user_agent, cache_status) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||||
|         ("127.0.0.1", "example.com", "2024-01-01 10:00:00", "GET / HTTP/1.1", 200, 100, "-", "curl", "MISS"), |         ( | ||||||
|  |             "127.0.0.1", | ||||||
|  |             "example.com", | ||||||
|  |             "2024-01-01 10:00:00", | ||||||
|  |             "GET / HTTP/1.1", | ||||||
|  |             200, | ||||||
|  |             100, | ||||||
|  |             "-", | ||||||
|  |             "curl", | ||||||
|  |             "MISS", | ||||||
|  |         ), | ||||||
|     ) |     ) | ||||||
|     cur.execute( |     cur.execute( | ||||||
|         "INSERT INTO logs (ip, host, time, request, status, bytes_sent, referer, user_agent, cache_status) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)", |         "INSERT INTO logs (ip, host, time, request, status, bytes_sent, referer, user_agent, cache_status) VALUES (?, ?, ?, ?, ?, ?, ?, ?, ?)", | ||||||
|         ("127.0.0.1", "example.com", "2024-01-01 10:05:00", "GET /err HTTP/1.1", 500, 100, "-", "curl", "MISS"), |         ( | ||||||
|  |             "127.0.0.1", | ||||||
|  |             "example.com", | ||||||
|  |             "2024-01-01 10:05:00", | ||||||
|  |             "GET /err HTTP/1.1", | ||||||
|  |             500, | ||||||
|  |             100, | ||||||
|  |             "-", | ||||||
|  |             "curl", | ||||||
|  |             "MISS", | ||||||
|  |         ), | ||||||
|     ) |     ) | ||||||
|     conn.commit() |     conn.commit() | ||||||
|     conn.close() |     conn.close() | ||||||
|  | @ -72,14 +92,36 @@ def test_generate_interval(tmp_path, sample_reports, monkeypatch): | ||||||
|     monkeypatch.setattr(gr, "DB_PATH", db_path) |     monkeypatch.setattr(gr, "DB_PATH", db_path) | ||||||
|     monkeypatch.setattr(gr, "OUTPUT_DIR", tmp_path / "output") |     monkeypatch.setattr(gr, "OUTPUT_DIR", tmp_path / "output") | ||||||
|     monkeypatch.setattr(gr, "REPORT_CONFIG", sample_reports) |     monkeypatch.setattr(gr, "REPORT_CONFIG", sample_reports) | ||||||
|     monkeypatch.setattr(gr, "TEMPLATE_DIR", Path(__file__).resolve().parents[1] / "templates") |     monkeypatch.setattr( | ||||||
|  |         gr, "TEMPLATE_DIR", Path(__file__).resolve().parents[1] / "templates" | ||||||
|  |     ) | ||||||
| 
 | 
 | ||||||
|     gr._generate_interval("hourly") |     gr._generate_interval("hourly") | ||||||
| 
 | 
 | ||||||
|     hits = json.loads((tmp_path / "output" / "hourly" / "hits.json").read_text()) |     hits = json.loads((tmp_path / "output" / "hourly" / "hits.json").read_text()) | ||||||
|     assert hits[0]["value"] == 2 |     assert hits[0]["value"] == 2 | ||||||
|     error_rate = json.loads((tmp_path / "output" / "hourly" / "error_rate.json").read_text()) |     error_rate = json.loads( | ||||||
|  |         (tmp_path / "output" / "hourly" / "error_rate.json").read_text() | ||||||
|  |     ) | ||||||
|     assert error_rate[0]["value"] == pytest.approx(50.0) |     assert error_rate[0]["value"] == pytest.approx(50.0) | ||||||
|     reports = json.loads((tmp_path / "output" / "hourly" / "reports.json").read_text()) |     reports = json.loads((tmp_path / "output" / "hourly" / "reports.json").read_text()) | ||||||
|     assert {r["name"] for r in reports} == {"hits", "error_rate"} |     assert {r["name"] for r in reports} == {"hits", "error_rate"} | ||||||
| 
 | 
 | ||||||
|  | 
 | ||||||
|  | def test_generate_interval_domain_filter(tmp_path, sample_reports, monkeypatch): | ||||||
|  |     db_path = tmp_path / "database" / "ngxstat.db" | ||||||
|  |     setup_db(db_path) | ||||||
|  | 
 | ||||||
|  |     monkeypatch.setattr(gr, "DB_PATH", db_path) | ||||||
|  |     monkeypatch.setattr(gr, "OUTPUT_DIR", tmp_path / "output") | ||||||
|  |     monkeypatch.setattr(gr, "REPORT_CONFIG", sample_reports) | ||||||
|  |     monkeypatch.setattr( | ||||||
|  |         gr, "TEMPLATE_DIR", Path(__file__).resolve().parents[1] / "templates" | ||||||
|  |     ) | ||||||
|  | 
 | ||||||
|  |     gr._generate_interval("hourly", "example.com") | ||||||
|  | 
 | ||||||
|  |     hits = json.loads( | ||||||
|  |         (tmp_path / "output" / "example.com" / "hourly" / "hits.json").read_text() | ||||||
|  |     ) | ||||||
|  |     assert hits[0]["value"] == 2 | ||||||
|  |  | ||||||
		Loading…
	
	Add table
		Add a link
		
	
		Reference in a new issue