dj-database-url/dj_database_url.py

182 lines
5.4 KiB
Python
Raw Normal View History

2012-04-30 17:37:05 +00:00
import os
import urllib.parse as urlparse
2012-06-19 15:53:51 +00:00
2012-04-30 17:37:05 +00:00
# Register database schemes in URLs.
2022-05-04 08:43:30 +00:00
urlparse.uses_netloc.append("postgres")
urlparse.uses_netloc.append("postgresql")
urlparse.uses_netloc.append("pgsql")
urlparse.uses_netloc.append("postgis")
urlparse.uses_netloc.append("mysql")
urlparse.uses_netloc.append("mysql2")
urlparse.uses_netloc.append("mysqlgis")
urlparse.uses_netloc.append("mysql-connector")
urlparse.uses_netloc.append("mssql")
urlparse.uses_netloc.append("mssqlms")
2022-05-04 08:43:30 +00:00
urlparse.uses_netloc.append("spatialite")
urlparse.uses_netloc.append("sqlite")
urlparse.uses_netloc.append("oracle")
urlparse.uses_netloc.append("oraclegis")
urlparse.uses_netloc.append("redshift")
2022-05-17 21:13:39 +00:00
urlparse.uses_netloc.append("cockroach")
urlparse.uses_netloc.append("timescale")
urlparse.uses_netloc.append("timescalegis")
2022-05-04 08:43:30 +00:00
DEFAULT_ENV = "DATABASE_URL"
2012-04-30 17:37:05 +00:00
2012-06-19 14:57:33 +00:00
SCHEMES = {
"postgres": "django.db.backends.postgresql",
"postgresql": "django.db.backends.postgresql",
"pgsql": "django.db.backends.postgresql",
2022-05-04 08:43:30 +00:00
"postgis": "django.contrib.gis.db.backends.postgis",
"mysql": "django.db.backends.mysql",
"mysql2": "django.db.backends.mysql",
"mysqlgis": "django.contrib.gis.db.backends.mysql",
"mysql-connector": "mysql.connector.django",
"mssql": "sql_server.pyodbc",
"mssqlms": "mssql",
2022-05-04 08:43:30 +00:00
"spatialite": "django.contrib.gis.db.backends.spatialite",
"sqlite": "django.db.backends.sqlite3",
"oracle": "django.db.backends.oracle",
"oraclegis": "django.contrib.gis.db.backends.oracle",
"redshift": "django_redshift_backend",
2022-05-17 21:13:39 +00:00
"cockroach": "django_cockroachdb",
"timescale": "timescale.db.backends.postgresql",
"timescalegis": "timescale.db.backends.postgis",
2012-06-19 14:57:33 +00:00
}
def config(
env=DEFAULT_ENV,
default=None,
engine=None,
conn_max_age=0,
conn_health_checks=False,
ssl_require=False,
test_options=None,
):
2012-04-30 18:55:37 +00:00
"""Returns configured DATABASE dictionary from DATABASE_URL."""
s = os.environ.get(env, default)
2012-05-30 05:23:28 +00:00
if s:
return parse(
s, engine, conn_max_age, conn_health_checks, ssl_require, test_options
)
2012-04-30 18:55:37 +00:00
2022-05-29 09:33:52 +00:00
return {}
2012-04-30 18:55:37 +00:00
def parse(
url,
engine=None,
conn_max_age=0,
conn_health_checks=False,
ssl_require=False,
test_options=None,
):
2012-04-30 18:55:37 +00:00
"""Parses a database URL."""
2022-05-04 08:43:30 +00:00
if url == "sqlite://:memory:":
# this is a special case, because if we pass this URL into
# urlparse, urlparse will choke trying to interpret "memory"
# as a port number
2022-05-04 08:43:30 +00:00
return {"ENGINE": SCHEMES["sqlite"], "NAME": ":memory:"}
2013-01-05 10:28:27 +00:00
# note: no other settings are required for sqlite
# otherwise parse the url as normal
2022-05-29 09:33:52 +00:00
parsed_config = {}
2012-04-30 18:55:37 +00:00
if test_options is None:
test_options = {}
url = urlparse.urlsplit(url)
2012-04-30 18:55:37 +00:00
2016-02-03 00:07:35 +00:00
# Split query strings from path.
path = url.path[1:]
2022-05-04 08:43:30 +00:00
if "?" in path and not url.query:
path, query = path.split("?", 2)
2016-02-03 00:07:35 +00:00
else:
path, query = path, url.query
query = urlparse.parse_qs(query)
2012-06-19 15:21:06 +00:00
# If we are using sqlite and we have no path, then assume we
# want an in-memory database (this is the behaviour of sqlalchemy)
2022-05-04 08:43:30 +00:00
if url.scheme == "sqlite" and path == "":
path = ":memory:"
# Handle postgres percent-encoded paths.
2022-05-04 08:43:30 +00:00
hostname = url.hostname or ""
if "%" in hostname:
2017-01-20 15:06:53 +00:00
# Switch to url.netloc to avoid lower cased paths
hostname = url.netloc
if "@" in hostname:
hostname = hostname.rsplit("@", 1)[1]
if ":" in hostname:
hostname = hostname.split(":", 1)[0]
# Use URL Parse library to decode % encodes
hostname = urlparse.unquote(hostname)
2017-04-08 10:34:30 +00:00
# Lookup specified engine.
2022-12-21 08:19:39 +00:00
if engine is None:
engine = SCHEMES.get(url.scheme)
if engine is None:
raise ValueError(
"No support for '%s'. We support: %s"
% (url.scheme, ", ".join(sorted(SCHEMES.keys())))
)
2017-04-08 10:34:30 +00:00
port = (
str(url.port)
if url.port
and engine in (SCHEMES["oracle"], SCHEMES["mssql"], SCHEMES["mssqlms"])
else url.port
)
2017-04-08 10:34:30 +00:00
2012-04-30 18:55:37 +00:00
# Update with environment configuration.
2022-05-29 09:33:52 +00:00
parsed_config.update(
{
2022-05-04 08:43:30 +00:00
"NAME": urlparse.unquote(path or ""),
"USER": urlparse.unquote(url.username or ""),
"PASSWORD": urlparse.unquote(url.password or ""),
"HOST": hostname,
"PORT": port or "",
"CONN_MAX_AGE": conn_max_age,
"CONN_HEALTH_CHECKS": conn_health_checks,
}
)
if test_options:
parsed_config.update(
{
'TEST': test_options,
}
)
2012-04-30 18:55:37 +00:00
2016-02-03 00:07:35 +00:00
# Pass the query string into OPTIONS.
options = {}
2016-02-03 00:07:35 +00:00
for key, values in query.items():
2022-05-04 08:43:30 +00:00
if url.scheme == "mysql" and key == "ssl-ca":
options["ssl"] = {"ca": values[-1]}
continue
options[key] = values[-1]
2016-02-03 00:07:35 +00:00
if ssl_require:
2022-05-04 08:43:30 +00:00
options["sslmode"] = "require"
2016-02-03 00:07:35 +00:00
# Support for Postgres Schema URLs
2022-05-04 08:43:30 +00:00
if "currentSchema" in options and engine in (
"django.contrib.gis.db.backends.postgis",
"django.db.backends.postgresql_psycopg2",
"django.db.backends.postgresql",
"django_redshift_backend",
"timescale.db.backends.postgresql",
"timescale.db.backends.postgis",
):
2022-05-04 08:43:30 +00:00
options["options"] = "-c search_path={0}".format(options.pop("currentSchema"))
2016-02-03 00:07:35 +00:00
2015-03-23 20:03:20 +00:00
if options:
2022-05-29 09:33:52 +00:00
parsed_config["OPTIONS"] = options
2015-03-23 20:03:20 +00:00
2013-12-15 16:59:18 +00:00
if engine:
2022-05-29 09:33:52 +00:00
parsed_config["ENGINE"] = engine
2012-05-01 16:23:09 +00:00
2022-05-29 09:33:52 +00:00
return parsed_config