|
| 1 | +import logging |
1 | 2 | import os |
2 | 3 | from datetime import datetime |
| 4 | +from urllib.parse import urlparse |
| 5 | + |
3 | 6 | from flask import Flask, render_template, g, request, url_for |
4 | 7 | from sqlalchemy.orm import subqueryload |
5 | 8 |
|
|
15 | 18 | from rcdb.web.modules import conditions_module |
16 | 19 | from rcdb.web.modules import select_values_module |
17 | 20 |
|
| 21 | +logger = logging.getLogger(__name__) |
| 22 | + |
18 | 23 | DEBUG = True |
19 | 24 | SECRET_KEY = 'development key' |
20 | 25 | USERNAME = 'admin' |
21 | 26 | PASSWORD = 'default' |
22 | 27 | SQL_CONNECTION_STRING = "mysql+pymysql://rcdb@127.0.0.1/rcdb" |
| 28 | +AVAILABLE_DATABASES = {} # dict of {"name": "connection_string", ...} |
| 29 | +DEFAULT_DATABASE = "" # connection string used as default |
| 30 | + |
| 31 | + |
| 32 | +def _connection_hint(conn_str): |
| 33 | + """Extract a short hint from a SQLAlchemy connection string. |
| 34 | +
|
| 35 | + Examples: |
| 36 | + "mysql+pymysql://rcdb@127.0.0.1/rcdb" -> "rcdb@127.0.0.1" |
| 37 | + "sqlite:///path/to/file.db" -> "file.db" |
| 38 | + """ |
| 39 | + try: |
| 40 | + parsed = urlparse(conn_str) |
| 41 | + if parsed.scheme.startswith("sqlite"): |
| 42 | + # For sqlite, show just the filename |
| 43 | + path = parsed.path.lstrip("/") |
| 44 | + return os.path.basename(path) if path else conn_str |
| 45 | + # For mysql/postgres etc, show user@host |
| 46 | + host = parsed.hostname or "" |
| 47 | + user = parsed.username or "" |
| 48 | + if user: |
| 49 | + return f"{user}@{host}" |
| 50 | + return host or conn_str |
| 51 | + except Exception: |
| 52 | + return conn_str |
23 | 53 |
|
24 | 54 |
|
25 | 55 | # Get the current directory |
|
33 | 63 |
|
34 | 64 | @app.before_request |
35 | 65 | def before_request(): |
| 66 | + available_dbs = app.config.get("AVAILABLE_DATABASES", {}) |
| 67 | + |
| 68 | + if available_dbs: |
| 69 | + # Determine which database to connect to |
| 70 | + cookie_db = request.cookies.get("rcdb_database", "") |
| 71 | + default_conn = app.config.get("DEFAULT_DATABASE", "") |
| 72 | + |
| 73 | + if cookie_db and cookie_db in available_dbs: |
| 74 | + # Cookie points to a valid database |
| 75 | + active_name = cookie_db |
| 76 | + elif default_conn: |
| 77 | + # Find name for the default connection string |
| 78 | + active_name = None |
| 79 | + for name, conn in available_dbs.items(): |
| 80 | + if conn == default_conn: |
| 81 | + active_name = name |
| 82 | + break |
| 83 | + if active_name is None: |
| 84 | + # DEFAULT_DATABASE not in AVAILABLE_DATABASES |
| 85 | + active_name = next(iter(available_dbs)) |
| 86 | + logger.warning( |
| 87 | + "DEFAULT_DATABASE '%s' is not in AVAILABLE_DATABASES, " |
| 88 | + "using '%s' instead.", default_conn, active_name |
| 89 | + ) |
| 90 | + else: |
| 91 | + # No default set, use first available |
| 92 | + active_name = next(iter(available_dbs)) |
| 93 | + |
| 94 | + connection_string = available_dbs[active_name] |
| 95 | + g.active_db_name = active_name |
| 96 | + g.available_databases = available_dbs |
| 97 | + else: |
| 98 | + # Original single-database behavior |
| 99 | + connection_string = app.config["SQL_CONNECTION_STRING"] |
| 100 | + g.active_db_name = None |
| 101 | + g.available_databases = {} |
| 102 | + |
36 | 103 | g.tdb = rcdb.ConfigurationProvider() |
37 | | - g.tdb.connect(app.config["SQL_CONNECTION_STRING"]) |
| 104 | + g.tdb.connect(connection_string) |
38 | 105 | app.jinja_env.globals['datetime_now'] = datetime.now |
39 | 106 |
|
40 | 107 |
|
@@ -101,6 +168,7 @@ def url_for_other_page(page): |
101 | 168 |
|
102 | 169 | app.jinja_env.globals['url_for_other_page'] = url_for_other_page |
103 | 170 | app.jinja_env.globals['rcdb_default_alias'] = rcdb.alias.default_aliases |
| 171 | +app.jinja_env.globals['connection_hint'] = _connection_hint |
104 | 172 |
|
105 | 173 | app.register_blueprint(runs_module) |
106 | 174 | app.register_blueprint(logs_module) |
|
0 commit comments