我的本地机器运行Python 2.5和Ubuntu 8.10上的Nginx, Django是根据最新的开发主干构建的。

对于我请求的每个URL,它抛出:

TemplateDoesNotExist at /appname/path appname/template_name.html Django尝试加载这些模板,顺序如下: 使用loader django.template.loaders. filessystem .function: 使用loader django.template.loaders.app_directories.function: TEMPLATE_DIRS (“/ usr / lib / python2.5 /网站/ projectname /模板”,)

在这种情况下,它是否在寻找/usr/lib/python2.5/site-packages/projectname/templates/appname/template_name.html ?奇怪的是这个文件确实存在于磁盘上。为什么Django找不到它?

我在Ubuntu 9.04上使用Python 2.6在远程服务器上运行相同的应用程序,没有出现这样的问题。其他设置相同。

在我的本地机器上是否有任何配置错误的地方,或者是什么可能导致了这样的错误,我应该调查一下?

在我的settings.py中,我指定了:

SETTINGS_PATH = os.path.normpath(os.path.dirname(__file__))
# Find templates in the same folder as settings.py.
TEMPLATE_DIRS = (
    os.path.join(SETTINGS_PATH, 'templates'),
)

它应该寻找以下文件:

/usr/lib/python2.5/site-packages projectname /模板/ appname1 / template1.html /usr/lib/python2.5/site-packages projectname /模板/ appname1 / template2.html /usr/lib/python2.5/site-packages projectname /模板/ appname2 / template3.html ...

以上所有文件都存在于磁盘上。

解决了

在我尝试之后,它现在工作了:

chown -R www-data:www-data /usr/lib/python2.5/site-packages/projectname/*

这是奇怪的。我不需要在远程服务器上执行此操作即可使其工作。


当前回答

2022年5月更新:

只要你正确地遵循本教程,你不需要改变(触摸)“settings.py”中“TEMPLATES”的默认设置,如下所示:

# "core/settings.py"

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",
            ],
        },
    },
]

同样,只要你正确地遵循本教程,“模板”可以在应用程序文件夹下正确读取,如下所示:

root
├── core
│   ├── settings.py
│   ├── urls.py
│   ...
├── myapp1
│   ├── templates
│   │   └── myapp1
│   │       └── myapp1.html
│   ├── urls.py
│   ├── views.py
│   ...
├── myapp2
│   ├── templates
│   │   └── myapp2
│   │       └── myapp2.html
│   ├── urls.py
│   ├── views.py
│   ... 

而“templates”可以在django项目根目录下正确读取,如下所示:

root
├── core
│   ├── settings.py
│   ├── urls.py
│   ...
├── myapp1
│   ├── urls.py
│   ├── views.py
│   ...
├── myapp2
│   ├── urls.py
│   ├── views.py
│   ...
├── templates
│   ├── myapp1
|   |   └── myapp1.html
|   └── myapp2
|       └── myapp2.html   

所以,你需要做的关键事情就是不要改变(触摸)“settings.py”中“TEMPLATES”的默认设置,如下所示:

# "core/settings.py"

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",
            ],
        },
    },
]

然后,将“myapp1”和“myapp2”应用程序设置为“core/settings.py”中的“INSTALLED_APPS”,如下所示:

# "core/settings.py"

INSTALLED_APPS = [
    ...
    "myapp1",
    "myapp2",
]

然后,如下所示,编写“myapp1”和“myapp2”应用程序的每个“views.py”。注意,你需要把每个“模板”的路径写为“myapp1/myapp1.html”和“myapp2.html”,而不是在“render()”中只写每个“模板”的路径“myapp1.html”和“myapp2.html”,如下所示:

# "myapp1/views.py"

from django.shortcuts import render

def myapp1(request):       # Don't write just "myapp1.html"
    return render(request, "myapp1/myapp1.html")
# "myapp2/views.py"

from django.shortcuts import render

def myapp2(request):       # Don't write just "myapp2.html"
    return render(request, "myapp2/myapp2.html")

然后,设置“myapp1”和“myapp2”应用程序的“views.py”路径,如下图所示:

# "myapp1/views.py"

from django.urls import include, path

from . import views

urlpatterns = [
    path("", views.myapp1, name='myapp1'), # Here
]
# "myapp2/views.py"

from django.urls import include, path

from . import views

urlpatterns = [
    path("", views.myapp2, name='myapp2'), # Here
]

然后,将“core/urls.py”中的“myapp1”和“myapp2”应用程序的每个路径设置为“urls.py”,如下所示。最后,“myapp1”和“myapp2”应用程序的模板将被读取,没有任何错误:

# "core/urls.py"

from django.urls import include, path

urlpatterns = [
    ...
    path("myapp1/", include('myapp1.urls')), # Here
    path("myapp2/", include('myapp2.urls')), # Here
]

其他回答

从Django版本3测试开始,你需要将你的新app添加到已安装的app中。对于一个Django app,不需要修改其他代码

INSTALLED_APPS = [
    'django.contrib.admin',
    'django.contrib.auth',
    'django.contrib.contenttypes',
    'django.contrib.sessions',
    'django.contrib.messages',
    'django.contrib.staticfiles',
    'addyourappnamehere'
]

我试过了,现在有用了

chown -R www-data:www-data /usr/lib/python2.5/site-packages/projectname/*

这是奇怪的。我不需要在远程服务器上这样做,使其工作。

此外,我必须在本地机器上运行以下命令,使所有静态文件可访问,但在远程服务器上,它们都是“根:根”。

chown -R www-data:www-data /var/www/projectname/*

本地机器运行Ubuntu 8.04桌面版。远程服务器安装在Ubuntu 9.04服务器版上。

有人知道为什么吗?

检查模板和appname目录的权限,使用ls -l或尝试从django执行绝对路径open()。

Django TemplateDoesNotExist错误仅仅意味着框架找不到模板文件。

要使用模板加载API,您需要告诉框架将模板存储在何处。这样做的地方是在你的设置文件(settings.py)通过TEMPLATE_DIRS设置。默认情况下,它是一个空元组,所以这个设置告诉Django的模板加载机制去哪里寻找模板。

选择一个你想要存储模板的目录,并将其添加到TEMPLATE_DIRS中,例如:

TEMPLATE_DIRS = (
  '/home/django/myproject/templates',
)

第一个解决方案:

这些设置

TEMPLATE_DIRS = (
    os.path.join(SETTINGS_PATH, 'templates'),
)

这意味着Django将查看项目下templates/目录下的模板。

假设你的Django项目位于/usr/lib/python2.5/site-packages/projectname/,那么Django会在/usr/lib/python2.5/site-packages/projectname/templates/下寻找模板

所以在这种情况下,我们想要移动我们的模板的结构是这样的:

/usr/lib/python2.5/site-packages/projectname/templates/template1.html
/usr/lib/python2.5/site-packages/projectname/templates/template2.html
/usr/lib/python2.5/site-packages/projectname/templates/template3.html

第二个解决方案:

如果这仍然不起作用,并且假设你已经在settings.py中配置了应用程序,像这样:

INSTALLED_APPS = (
    'appname1',
    'appname2',
    'appname3',
)

默认情况下,Django会在每个已安装应用的模板/目录下加载模板。所以在你的目录结构中,我们想要移动我们的模板,就像这样:

/usr/lib/python2.5/site-packages/projectname/appname1/templates/template1.html
/usr/lib/python2.5/site-packages/projectname/appname2/templates/template2.html
/usr/lib/python2.5/site-packages/projectname/appname3/templates/template3.html

默认情况下可能没有定义SETTINGS_PATH。在这种情况下,你会想要定义它(在settings.py中):

import os
SETTINGS_PATH = os.path.dirname(os.path.dirname(__file__))