问题描述
我正在关注此tutorial 使用 Google Cloud 部署 Django。在时间戳 (13:39),他在 VS Code 中进入 Django 项目中的 settings.py,并将一些占位符替换为他自己的 Google Cloud 凭据。然而,我的 settings.py 看起来非常不同。块帽中唯一看起来像是占位符的东西是“GOOGLE_CLOUD_PROJECT”和“SETTINGS_NAME”。这是我无法通过的安装部分,因为在下一步尝试执行时
python manage.py makemigrations
我被送到这里:
else:
raise Exception("No local .env or GOOGLE_CLOUD_PROJECT detected. No secrets found.")
希望有人能浏览一下我的 settings.py 并找出我遗漏的内容,为什么 if 块会出现 Exception。
需要注意的一点:顶级导入environ 和google.cloud 未解析。 idk是否与它有关。这就是文件从罐中出来的方式。我不想惹它,以防万一 tshtf :D
settings.py
import io
import os
import environ
from google.cloud import secretmanager
# Build paths inside the project like this: os.path.join(BASE_DIR,...)
BASE_DIR = os.path.dirname(os.path.dirname(os.path.abspath(__file__)))
# [START gaestd_py_django_secret_config]
env = environ.Env(DEBUG=(bool,False))
env_file = os.path.join(BASE_DIR,".env")
if os.path.isfile(env_file):
# Use a local secret file,if provided
env.read_env(env_file)
# [START_EXCLUDE]
elif os.getenv("TRAMPOLINE_CI",None):
# Create local settings if running with CI,for unit testing
placeholder = (
f"SECRET_KEY=a\n"
f"DATABASE_URL=sqlite://{os.path.join(BASE_DIR,'db.sqlite3')}"
)
env.read_env(io.StringIO(placeholder))
# [END_EXCLUDE]
elif os.environ.get("GOOGLE_CLOUD_PROJECT",None):
# Pull secrets from Secret Manager
project_id = os.environ.get("GOOGLE_CLOUD_PROJECT")
client = secretmanager.SecretManagerServiceClient()
settings_name = os.environ.get("SETTINGS_NAME","django_settings")
name = f"projects/{project_id}/secrets/{settings_name}/versions/latest"
payload = client.access_secret_version(name=name).payload.data.decode("UTF-8")
env.read_env(io.StringIO(payload))
else:
raise Exception("No local .env or GOOGLE_CLOUD_PROJECT detected. No secrets found.")
# [END gaestd_py_django_secret_config]
SECRET_KEY = env("SECRET_KEY")
# Security WARNING: don't run with debug turned on in production!
# Change this to "False" when you are ready for production
DEBUG = True
# Security WARNING: App Engine's security features ensure that it is safe to
# have ALLOWED_HOSTS = ['*'] when the app is deployed. If you deploy a Django
# app not on App Engine,make sure to set an appropriate host here.
ALLOWED_HOSTS = ["*"]
# Application deFinition
INSTALLED_APPS = [
"polls.apps.PollsConfig","django.contrib.admin","django.contrib.auth","django.contrib.contenttypes","django.contrib.sessions","django.contrib.messages","django.contrib.staticfiles",]
MIDDLEWARE = [
"django.middleware.security.SecurityMiddleware","django.contrib.sessions.middleware.SessionMiddleware","django.middleware.common.CommonMiddleware","django.middleware.csrf.CsrfViewMiddleware","django.contrib.auth.middleware.AuthenticationMiddleware","django.contrib.messages.middleware.MessageMiddleware","django.middleware.clickjacking.XFrameOptionsMiddleware",]
ROOT_URLconf = "mysite.urls"
TEMPLATES = [
{
"BACKEND": "django.template.backends.django.DjangoTemplates","Dirs": [],"APP_Dirs": True,"OPTIONS": {
"context_processors": [
"django.template.context_processors.debug","django.template.context_processors.request","django.contrib.auth.context_processors.auth","django.contrib.messages.context_processors.messages",],},]
Wsgi_APPLICATION = "mysite.wsgi.application"
# Database
# [START db_setup]
# [START gaestd_py_django_database_config]
# Use django-environ to parse the connection string
DATABASES = {"default": env.db()}
# If the flag as been set,configure to use proxy
if os.getenv("USE_CLOUD_sql_AUTH_PROXY",None):
DATABASES["default"]["HOST"] = "127.0.0.1"
DATABASES["default"]["PORT"] = 5432
# [END gaestd_py_django_database_config]
# [END db_setup]
# Use a in-memory sqlite3 database when testing in CI systems
# Todo(glasnt) CHECK IF THIS IS required because we're setting a val above
if os.getenv("TRAMPOLINE_CI",None):
DATABASES = {
"default": {
"ENGINE": "django.db.backends.sqlite3","NAME": os.path.join(BASE_DIR,"db.sqlite3"),}
}
# Password validation
AUTH_PASSWORD_VALIDATORS = [
{
"NAME": "django.contrib.auth.password_validation.UserAttributeSimilarityValidator",# noqa: 501
},{
"NAME": "django.contrib.auth.password_validation.MinimumLengthValidator",{
"NAME": "django.contrib.auth.password_validation.CommonPasswordValidator",{
"NAME": "django.contrib.auth.password_validation.NumericPasswordValidator",]
# Internationalization
LANGUAGE_CODE = "en-us"
TIME_ZONE = "UTC"
USE_I18N = True
USE_L10N = True
USE_TZ = True
# Static files (CSS,JavaScript,Images)
STATIC_ROOT = "static"
STATIC_URL = "/static/"
STATICFILES_Dirs = []
# Default primary key field type
# https://docs.djangoproject.com/en/3.2/ref/settings/#default-auto-field
DEFAULT_AUTO_FIELD = "django.db.models.BigAutoField"
解决方法
Django 希望您提供一个 SECRET_KEY
。有多种方法可以提供此类密钥,但一种方法是在 .env
文件所在的同一文件夹中创建一个包含 SECRET_KEY
的 manage.py
文件。
可以使用以下 code snippet 生成 SECRET_KEY
(在您认为方便的任何地方运行它,但不要将生成的密钥字符串添加到您的 {{1} } 文件):
settings.py
获取结果输出(它是一个 50 个字符长的字符串)并将其添加到 from django.core.management.utils import get_random_secret_key
get_random_secret_key()
文件中,格式如下:
.env
您可能还需要将 SECRET_KEY="the-50-character-string-here"
文件中的 BASE_DIR
替换为以下内容才能正确加载 settings.py
文件(注意第一行是如何注释掉的)。
.env
在与数据库设置相关的过程中,您可能会遇到更多问题。查看 this 和 this SO 问题。
您还可以考虑查看 Guide,这是您所指的教程所遵循的内容。