如何使用 django-elasticsearch-dsl 将 django-modeltranslation 创建的列索引到 Elastic?
How to index columns created by django-modeltranslation to Elastic, using django-elasticsearch-dsl?
我已经用 django-modeltranslation and implemented search using django-elasticsearch-dsl 翻译了我的模型字段。
问题:
django-modeltranslation 在数据库中创建翻译字段,而不是在我的模型中,并且搜索仅适用于由模型创建的字段。由于 django-elasticsearch-dsl 正在检查模型以重建搜索索引。
当我尝试时:
python3 manage.py search_index --rebuild
我收到错误:
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/__init__.py", line 24, in setup
apps.populate(settings.INSTALLED_APPS)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/apps/registry.py", line 122, in populate
app_config.ready()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/apps.py", line 14, in ready
self.module.autodiscover()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/__init__.py", line 11, in autodiscover
autodiscover_modules('documents')
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/utils/module_loading.py", line 47, in autodiscover_modules
import_module('%s.%s' % (app_config.name, module_to_search))
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/Users/barbos/django-symbolsite/symbolgraph/search/documents.py", line 7, in <module>
class SymbolDocument(Document):
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/registries.py", line 65, in register_document
field_instance = document.to_field(field_name, django_field)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/documents.py", line 142, in to_field
"to an Elasticsearch field!".format(field_name)
django_elasticsearch_dsl.exceptions.ModelFieldNotMappedError: Cannot convert model field name_ru to an Elasticsearch field!
Oleh-MacSymbol-Pro:symbolgraph barbos$ python3 manage.py search_index --rebuild
Traceback (most recent call last):
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/documents.py", line 138, in to_field
model_field.__class__](attr=field_name)
KeyError: <class 'modeltranslation.fields.field_factory.<locals>.TranslationFieldSpecific'>
During handling of the above exception, another exception occurred:
Traceback (most recent call last):
File "manage.py", line 21, in <module>
main()
File "manage.py", line 17, in main
execute_from_command_line(sys.argv)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/core/management/__init__.py", line 401, in execute_from_command_line
utility.execute()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/core/management/__init__.py", line 377, in execute
django.setup()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/__init__.py", line 24, in setup
apps.populate(settings.INSTALLED_APPS)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/apps/registry.py", line 122, in populate
app_config.ready()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/apps.py", line 14, in ready
self.module.autodiscover()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/__init__.py", line 11, in autodiscover
autodiscover_modules('documents')
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/utils/module_loading.py", line 47, in autodiscover_modules
import_module('%s.%s' % (app_config.name, module_to_search))
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/Users/barbos/django-symbolsite/symbolgraph/search/documents.py", line 7, in <module>
class SymbolDocument(Document):
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/registries.py", line 65, in register_document
field_instance = document.to_field(field_name, django_field)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/documents.py", line 142, in to_field
"to an Elasticsearch field!".format(field_name)
django_elasticsearch_dsl.exceptions.ModelFieldNotMappedError: Cannot convert model field name_ru to an Elasticsearch field!
Django 3.0.5
django-modeltranslation & django-elasticsearch-dsl - 最新版本
elasticsearch-7.6.2
db.sqlite3(我要迁移到 PostgreSQL)
相关的 django 文件:
#settings.py:
INSTALLED_APPS = [
'modeltranslation',
'django_elasticsearch_dsl',
'search',
'symbol',
'category',
'homepage',
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
]
ELASTICSEARCH_DSL={
'default': {
'hosts': 'localhost:9200'
},
}
---
#translation.py
from modeltranslation.translator import translator, TranslationOptions
from .models import Symbol
class SymbolTranslationOptions(TranslationOptions):
fields = ('name', 'content', 'meaning', 'tag',)
translator.register(Symbol, SymbolTranslationOptions)
---
#models.py
from django.db import models
from django.db.models.constraints import UniqueConstraint
class Symbol(models.Model):
id = models.IntegerField(primary_key=True)
slug = models.SlugField(unique=True) #uniqe
codepoint = models.CharField(max_length=255, unique=True) #this list should be in separete table
symbol = models.CharField(max_length=50, unique=True)
name = models.CharField(max_length=255, unique=True)
category = models.ForeignKey('Category', on_delete=models.CASCADE)
subcategory = models.ForeignKey('Subcategory', on_delete=models.CASCADE)
show_in_category = models.BooleanField()
popular = models.BooleanField(blank=True, unique=False, default=False)
tag = models.CharField(max_length=255,blank=True) #this list should be in separete table
content = models.TextField(blank=True)
meaning = models.TextField(blank=True)
imgslug = models.CharField(unique=False, max_length=255, blank=True) #uniqe
# name_en = models.CharField(max_length=255, unique=True, null=True)
# name_ru = models.CharField(max_length=255, unique=True, null=True)
# tag_en = models.CharField(max_length=255,blank=True, null=True)
# tag_ru = models.CharField(max_length=255,blank=True, null=True)
# content_en = models.TextField(blank=True, null=True)
# content_ru = models.TextField(blank=True, null=True)
# meaning_en = models.TextField(blank=True, null=True)
# meaning_ru = models.TextField(blank=True, null=True)
def __str__(self):
return self.name
---
#documents.py
from django_elasticsearch_dsl import Document
from django_elasticsearch_dsl.registries import registry
from symbol.models import Symbol
@registry.register_document
class SymbolDocument(Document):
class Index:
# Name of the Elasticsearch index
name = 'symbols'
# See Elasticsearch Indices API reference for available settings
settings = {'number_of_shards': 1,
'number_of_replicas': 0}
class Django:
model = Symbol # The model associated with this Document
# The fields of the model you want to be indexed in Elasticsearch
fields = [
'symbol',
'name',
'tag',
'content',
'meaning',
'shortcode',
'codepoint',
'slug',
'id',
#
# 'name_ru',
# 'name_uk',
# 'tag_ru',
# 'tag_uk',
# 'content_ru',
# 'content_uk',
# 'meaning_ru',
# 'meaning_uk',
# 'also_called_ru',
# 'also_called_uk',
]
我发现了一种使用其他语言进行搜索的糟糕方法:
- 在已安装的应用程序中禁用 django-modeltranslation。
- 在模型中添加由 django-modeltranslation 在数据库中创建的所有字段。
- 将这些字段添加到搜索应用程序中的 documents.py 以创建 Elastic 索引并重新启动 django 服务器
- 检查是否“$ python3 manage.py makemigrations” returns: "No changes detected".
- $ python3 manage.py search_index --rebuild
- 还原步骤 3、2、1 并重新启动 django 服务器
我需要将翻译包含到 Elasticsearch 索引中。
作为开发中的小白,我不知道下一步该做什么。
我有两个想法(更像是一个方向),但不知道如何实现:
- 将 django-elasticsearch-dsl 更改为直接从数据库索引。
- 更改 django-modeltranslation 以便能够将它在数据库中创建的字段添加到模型中。
您可以在 Document
class 中显式添加字段。喜欢关注
from django_elasticsearch_dsl import TextField
@registry.register_document
class SymbolDocument(Document):
name_ru = TextField()
....
我已经用 django-modeltranslation and implemented search using django-elasticsearch-dsl 翻译了我的模型字段。
问题: django-modeltranslation 在数据库中创建翻译字段,而不是在我的模型中,并且搜索仅适用于由模型创建的字段。由于 django-elasticsearch-dsl 正在检查模型以重建搜索索引。
当我尝试时:
python3 manage.py search_index --rebuild
我收到错误:
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/__init__.py", line 24, in setup
apps.populate(settings.INSTALLED_APPS)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/apps/registry.py", line 122, in populate
app_config.ready()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/apps.py", line 14, in ready
self.module.autodiscover()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/__init__.py", line 11, in autodiscover
autodiscover_modules('documents')
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/utils/module_loading.py", line 47, in autodiscover_modules
import_module('%s.%s' % (app_config.name, module_to_search))
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/Users/barbos/django-symbolsite/symbolgraph/search/documents.py", line 7, in <module>
class SymbolDocument(Document):
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/registries.py", line 65, in register_document
field_instance = document.to_field(field_name, django_field)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/documents.py", line 142, in to_field
"to an Elasticsearch field!".format(field_name)
django_elasticsearch_dsl.exceptions.ModelFieldNotMappedError: Cannot convert model field name_ru to an Elasticsearch field!
Oleh-MacSymbol-Pro:symbolgraph barbos$ python3 manage.py search_index --rebuild
Traceback (most recent call last):
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/documents.py", line 138, in to_field
model_field.__class__](attr=field_name)
KeyError: <class 'modeltranslation.fields.field_factory.<locals>.TranslationFieldSpecific'>
During handling of the above exception, another exception occurred:
Traceback (most recent call last):
File "manage.py", line 21, in <module>
main()
File "manage.py", line 17, in main
execute_from_command_line(sys.argv)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/core/management/__init__.py", line 401, in execute_from_command_line
utility.execute()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/core/management/__init__.py", line 377, in execute
django.setup()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/__init__.py", line 24, in setup
apps.populate(settings.INSTALLED_APPS)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/apps/registry.py", line 122, in populate
app_config.ready()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/apps.py", line 14, in ready
self.module.autodiscover()
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/__init__.py", line 11, in autodiscover
autodiscover_modules('documents')
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django/utils/module_loading.py", line 47, in autodiscover_modules
import_module('%s.%s' % (app_config.name, module_to_search))
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/importlib/__init__.py", line 127, in import_module
return _bootstrap._gcd_import(name[level:], package, level)
File "<frozen importlib._bootstrap>", line 1006, in _gcd_import
File "<frozen importlib._bootstrap>", line 983, in _find_and_load
File "<frozen importlib._bootstrap>", line 967, in _find_and_load_unlocked
File "<frozen importlib._bootstrap>", line 677, in _load_unlocked
File "<frozen importlib._bootstrap_external>", line 728, in exec_module
File "<frozen importlib._bootstrap>", line 219, in _call_with_frames_removed
File "/Users/barbos/django-symbolsite/symbolgraph/search/documents.py", line 7, in <module>
class SymbolDocument(Document):
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/registries.py", line 65, in register_document
field_instance = document.to_field(field_name, django_field)
File "/Library/Frameworks/Python.framework/Versions/3.7/lib/python3.7/site-packages/django_elasticsearch_dsl/documents.py", line 142, in to_field
"to an Elasticsearch field!".format(field_name)
django_elasticsearch_dsl.exceptions.ModelFieldNotMappedError: Cannot convert model field name_ru to an Elasticsearch field!
Django 3.0.5 django-modeltranslation & django-elasticsearch-dsl - 最新版本 elasticsearch-7.6.2 db.sqlite3(我要迁移到 PostgreSQL)
相关的 django 文件:
#settings.py:
INSTALLED_APPS = [
'modeltranslation',
'django_elasticsearch_dsl',
'search',
'symbol',
'category',
'homepage',
'django.contrib.admin',
'django.contrib.auth',
'django.contrib.contenttypes',
'django.contrib.sessions',
'django.contrib.messages',
'django.contrib.staticfiles',
]
ELASTICSEARCH_DSL={
'default': {
'hosts': 'localhost:9200'
},
}
---
#translation.py
from modeltranslation.translator import translator, TranslationOptions
from .models import Symbol
class SymbolTranslationOptions(TranslationOptions):
fields = ('name', 'content', 'meaning', 'tag',)
translator.register(Symbol, SymbolTranslationOptions)
---
#models.py
from django.db import models
from django.db.models.constraints import UniqueConstraint
class Symbol(models.Model):
id = models.IntegerField(primary_key=True)
slug = models.SlugField(unique=True) #uniqe
codepoint = models.CharField(max_length=255, unique=True) #this list should be in separete table
symbol = models.CharField(max_length=50, unique=True)
name = models.CharField(max_length=255, unique=True)
category = models.ForeignKey('Category', on_delete=models.CASCADE)
subcategory = models.ForeignKey('Subcategory', on_delete=models.CASCADE)
show_in_category = models.BooleanField()
popular = models.BooleanField(blank=True, unique=False, default=False)
tag = models.CharField(max_length=255,blank=True) #this list should be in separete table
content = models.TextField(blank=True)
meaning = models.TextField(blank=True)
imgslug = models.CharField(unique=False, max_length=255, blank=True) #uniqe
# name_en = models.CharField(max_length=255, unique=True, null=True)
# name_ru = models.CharField(max_length=255, unique=True, null=True)
# tag_en = models.CharField(max_length=255,blank=True, null=True)
# tag_ru = models.CharField(max_length=255,blank=True, null=True)
# content_en = models.TextField(blank=True, null=True)
# content_ru = models.TextField(blank=True, null=True)
# meaning_en = models.TextField(blank=True, null=True)
# meaning_ru = models.TextField(blank=True, null=True)
def __str__(self):
return self.name
---
#documents.py
from django_elasticsearch_dsl import Document
from django_elasticsearch_dsl.registries import registry
from symbol.models import Symbol
@registry.register_document
class SymbolDocument(Document):
class Index:
# Name of the Elasticsearch index
name = 'symbols'
# See Elasticsearch Indices API reference for available settings
settings = {'number_of_shards': 1,
'number_of_replicas': 0}
class Django:
model = Symbol # The model associated with this Document
# The fields of the model you want to be indexed in Elasticsearch
fields = [
'symbol',
'name',
'tag',
'content',
'meaning',
'shortcode',
'codepoint',
'slug',
'id',
#
# 'name_ru',
# 'name_uk',
# 'tag_ru',
# 'tag_uk',
# 'content_ru',
# 'content_uk',
# 'meaning_ru',
# 'meaning_uk',
# 'also_called_ru',
# 'also_called_uk',
]
我发现了一种使用其他语言进行搜索的糟糕方法:
- 在已安装的应用程序中禁用 django-modeltranslation。
- 在模型中添加由 django-modeltranslation 在数据库中创建的所有字段。
- 将这些字段添加到搜索应用程序中的 documents.py 以创建 Elastic 索引并重新启动 django 服务器
- 检查是否“$ python3 manage.py makemigrations” returns: "No changes detected".
- $ python3 manage.py search_index --rebuild
- 还原步骤 3、2、1 并重新启动 django 服务器
我需要将翻译包含到 Elasticsearch 索引中。 作为开发中的小白,我不知道下一步该做什么。
我有两个想法(更像是一个方向),但不知道如何实现:
- 将 django-elasticsearch-dsl 更改为直接从数据库索引。
- 更改 django-modeltranslation 以便能够将它在数据库中创建的字段添加到模型中。
您可以在 Document
class 中显式添加字段。喜欢关注
from django_elasticsearch_dsl import TextField
@registry.register_document
class SymbolDocument(Document):
name_ru = TextField()
....