28

我正在尝试使用 django 和 mongoengine 仅通过 GridFS 提供存储后端。我还有一个 MySQL 数据库。

当我从 django 管理员中删除时,我遇到了一个奇怪的(对我来说)错误,我想知道我是否做错了什么。

我的代码如下所示:

# settings.py
from mongoengine import connect
connect("mongo_storage")

# models.py
from mongoengine.django.storage import GridFSStorage
class MyFile(models.Model):
    name = models.CharField(max_length=50)
    content = models.FileField(upload_to="appsfiles", storage=GridFSStorage())
    creation_time = models.DateTimeField(auto_now_add=True)
    last_update_time = models.DateTimeField(auto_now=True)

我可以很好地上传文件,但是当我删除它们时,似乎有些东西坏了,mongo 数据库似乎处于无法使用的状态,直到我手动删除所有 FileDocument.objects。发生这种情况时,我无法从 django 界面上传文件或删除它们。

从堆栈跟踪我有:

/home/projects/vector/src/mongoengine/django/storage.py in _get_doc_with_name
        doc = [d for d in docs if getattr(d, self.field).name == name] ...
▼ Local vars
Variable    Value
_[1]    
[]
d   

docs    
Error in formatting: cannot set options after executing query
name    
u'testfile.pdf'
self    

/home/projects/vector/src/mongoengine/fields.py in __getattr__
        raise AttributeError 

我是否错误地使用了此功能?

更新:

感谢@zeekay 的回答,我能够让一个工作的gridfs 存储插件工作。我最终根本没有使用 mongoengine。我将修改后的解决方案放在github 上。有一个清晰的示例项目展示了如何使用它。我还将项目上传到pypi

另一个更新:

我强烈推荐django-storages项目。它有很多支持存储的选项,并且比我最初提出的解决方案被更多的人使用。

4

1 回答 1

8

我认为你最好不要为此使用 MongoEngine,我也没有太多运气。这是 的 替代品mongoengine.django.storage.GridFSStorage,适用于管理员。

from django.core.files.storage import Storage
from django.conf import settings

from pymongo import Connection
from gridfs import GridFS

class GridFSStorage(Storage):
    def __init__(self, host='localhost', port=27017, collection='fs'):
        for s in ('host', 'port', 'collection'):
            name = 'GRIDFS_' + s.upper()
            if hasattr(settings, name):
                setattr(self, s, getattr(settings, name))
        for s, v in zip(('host', 'port', 'collection'), (host, port, collection)):
            if v:
                setattr(self, s, v)
        self.db = Connection(host=self.host, port=self.port)[self.collection]
        self.fs = GridFS(self.db)

    def _save(self, name, content):
        self.fs.put(content, filename=name)
        return name

    def _open(self, name, *args, **kwars):
        return self.fs.get_last_version(filename=name)

    def delete(self, name):
        oid = fs.get_last_version(filename=name)._id
        self.fs.delete(oid)

    def exists(self, name):
        return self.fs.exists({'filename': name})

    def size(self, name):
        return self.fs.get_last_version(filename=name).length

GRIDFS_HOST,GRIDFS_PORT并且GRIDFS_COLLECTION可以在您的设置中定义或作为host, port,collection关键字参数传递GridFSStorage给模型的FileField.

我参考了 Django 的自定义存储文档,并松散地按照这个答案回答了一个类似的问题。

于 2011-05-31T20:49:30.103 回答