Django 管理员在生产中上传图像

Django admin uploaded image in production

无法找到我通过管理界面上传的图像,尽管它存在于 server.It 负载上,但在开发中很好。我使用 Django Image-kit 调整上传图片的大小。

图片没有显示在网站上,如果我关注图片 url,我会得到

<h1>Not Found</h1>
<p>The requested URL /media/CACHE/images/references/<reference-name>/<image-name>/211b0022f8a2baad7cdc527711dcb.jpg was not found on this server.</p>

上传的图片没有添加到我的静态文件中,显示正确。

我的模型是这样的:

def generate_unique_slug(instance, field):
    original = slugify(field)
    unique = original
    numb = 1
    while instance.objects.filter(slug=unique).exists():
        unique = f'{original}-{numb}'
        numb += 1
    return unique

def reference_main_image_path(instance, filename):
    sub_folder = instance.slug
    return f'references/{sub_folder}/{filename}'

class Reference(models.Model):
    title = models.CharField(max_length=60)
    description = models.TextField(max_length=2000)
    slug = models.SlugField(max_length=12, unique=True, blank=True)
    main_image = models.ImageField(
        upload_to=reference_main_image_path, default='default.png')
    main_thumbnail = ImageSpecField(source='main_image',
                                processors=[ResizeToFill(540, 300)],
                                format='JPEG',
                                options={'quality': 80},)
    tags = TaggableManager()

    def __str__(self):
        return self.title

    def save(self, *args, **kwargs):
        if self.slug:
            if slugify(self.title) != self.slug:
                self.slug = generate_unique_slug(Reference, self.title)
        else:
            self.slug = generate_unique_slug(Reference, self.title)
        super().save(*args, **kwargs)

我的view.py:

def reference(request):
    context = {
        'references': Reference.objects.all(),
    }
    return render(request, 'home/references.html', context)

在'html`代码中我'call'图像是这样的:

<img src="{{ reference.main_thumbnail.url }}"> 

我的 settings.py 静态和媒体变量如下所示:

STATIC_ROOT = os.path.join(BASE_DIR, 'static')
STATIC_URL = '/static/'

MEDIA_ROOT = os.path.join(BASE_DIR, 'media')
MEDIA_URL = '/media/'

我运行manage.py collectstatic

我的 upload_to 应该改变吗?或者我如何告诉 Django 在 'collecting static' 时查看文件夹,然后将 html 更改为在静态文件夹中查看?

我觉得 image-kit 的文档应该只关注 'development' 还是我遗漏了什么?

谢谢!

编辑:我在生产中使用 gunicorn 和 Nginx,但如前所述,我的静态文件加载得很好。

编辑 2: Nginx 配置文件:

server {
    server_name <server_names>;

    location = /favicon.ico { access_log off; log_not_found off; }
    location /static/ {
        root /home/<user>/projects/<project>/<project>;
    }

    location / {
        include proxy_params;
        proxy_pass http://unix:/run/gunicorn.sock;
    }
}

我遗漏了一些 certbot SSL 配置。

编辑 3:urls.py

from django.urls import path
from . import views
from django.conf.urls.static import static
from django.conf import settings


urlpatterns = [
    path('', views.home, name='home'),
    path('ydelser/', views.service, name='service'),
    path('referencer/', views.reference, name='reference'),
    path('kontakt/', views.contact, name='contact'),
]
urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

要使用 nginx 提供媒体文件,您应该添加适当的位置,类似于 static:

location /static/ {
    root /home/<user>/projects/<project>/<project>;
}

location /media/ {
    root /home/<user>/projects/<project>/<project>;
}

就像你在 django 中做的那样:

...
STATIC_URL = '/static/'
...
MEDIA_URL = '/media/'

但是,如果您不想使用 nginx(推荐)提供媒体文件,但仍想使用 django(不推荐)提供媒体文件,请确保行

urlpatterns += static(settings.MEDIA_URL, document_root=settings.MEDIA_ROOT)

当 DEBUG=False 时启用。

请注意,您不应将 media 文件视为 static - 它们在逻辑上是不同的。虽然静态文件属于您的 solution/project 并随附,但媒体文件不是您项目的一部分,通常由用户制作。

ps

如果您已将一些文件上传到您本地的网站,那么它们的位置相应地如下:

def reference_main_image_path(instance, filename):
    sub_folder = instance.slug
    return f'references/{sub_folder}/{filename}'

class Reference(models.Model):
    ...
    main_image = models.ImageField(
        upload_to=reference_main_image_path, default='default.png')

如果您没有将它们(位于上面配置的文件夹中的硬盘驱动器上的文件)复制到生产服务器 - 它们将不会显示。如果需要手动复制它们并将目标文件夹与 nginx 位置匹配。