Добавьте BrinIndex в существующую базу данных PostgreSQL, используя Django - PullRequest
0 голосов
/ 11 июня 2018

Я управляю веб-сайтом с использованием Django, база данных которого работает на PostgreSQL.Чтобы повысить производительность, я хочу добавить BrinIndex к модели, которая естественно упорядочена на диске.

Я добавил индекс к определению моей модели следующим образом.

from django.contrib.postgres.indexes import BrinIndex

class Measurement(models.Model):
    class Meta:
        indexes = (
            BrinIndex(fields=['time'])
        )

    time = models.DateTimeField(
        'Time of measurement',
        null=True
    )

Но работает

python3 manage.py makemigrations

Возвращает ошибку

Traceback (most recent call last):
  File "C:\Program Files\JetBrains\PyCharm 2018.1.1\helpers\pycharm\django_manage.py", line 52, in <module>
    run_command()
  File "C:\Program Files\JetBrains\PyCharm 2018.1.1\helpers\pycharm\django_manage.py", line 46, in run_command
    run_module(manage_file, None, '__main__', True)
  File "C:\Python36\lib\runpy.py", line 205, in run_module
    return _run_module_code(code, init_globals, run_name, mod_spec)
  File "C:\Python36\lib\runpy.py", line 96, in _run_module_code
    mod_name, mod_spec, pkg_name, script_name)
  File "C:\Python36\lib\runpy.py", line 85, in _run_code
    exec(code, run_globals)
  File "C:/Users/*USERNAME*/*PROJECT_NAME*\manage.py", line 22, in <module>
    execute_from_command_line(sys.argv)
  File "C:\Users\*USERNAME*\*PROJECT_NAME*\venv\lib\site-packages\django\core\management\__init__.py", line 371, in execute_from_command_line
    utility.execute()
  File "C:\Users\*USERNAME*\*PROJECT_NAME*\venv\lib\site-packages\django\core\management\__init__.py", line 347, in execute
    django.setup()
  File "C:\Users\*USERNAME*\*PROJECT_NAME*\venv\lib\site-packages\django\__init__.py", line 24, in setup
    apps.populate(settings.INSTALLED_APPS)
  File "C:\Users\*USERNAME*\*PROJECT_NAME*\venv\lib\site-packages\django\apps\registry.py", line 112, in populate
    app_config.import_models()
  File "C:\Users\*USERNAME*\*PROJECT_NAME*\venv\lib\site-packages\django\apps\config.py", line 198, in import_models
    self.models_module = import_module(models_module_name)
  File "C:\Python36\lib\importlib\__init__.py", line 126, in import_module
    return _bootstrap._gcd_import(name[level:], package, level)
  File "<frozen importlib._bootstrap>", line 978, in _gcd_import
  File "<frozen importlib._bootstrap>", line 961, in _find_and_load
  File "<frozen importlib._bootstrap>", line 950, in _find_and_load_unlocked
  File "<frozen importlib._bootstrap>", line 655, in _load_unlocked
  File "<frozen importlib._bootstrap_external>", line 678, in exec_module
  File "<frozen importlib._bootstrap>", line 205, in _call_with_frames_removed
  File "C:/Users/*USERNAME*/*PROJECT_NAME*\*APP_NAME*\models.py", line 300, in <module>
    class Measurement(models.Model):
  File "C:\Users\*USERNAME*\*PROJECT_NAME*\venv\lib\site-packages\django\db\models\base.py", line 298, in __new__
    new_class._meta.indexes = [copy.deepcopy(idx) for idx in new_class._meta.indexes]
TypeError: 'BrinIndex' object is not iterable

Я делаю что-то неправильно?Может ли этот индекс не быть добавлен впоследствии?

Ответы [ 2 ]

0 голосов
/ 28 июля 2019

здесь проблема вашего кода при определении кортежей с одним значением, которое вы должны поставить "," в конце значения, иначе python не распознает его как кортеж, например:

a = ('value1') #is a string not a tuple
a = ('value1',) #is a tuple 

вот как должен выглядеть ваш код: из django.contrib.postgres.indexes import BrinIndex

class Measurement(models.Model):
    class Meta:
        indexes = (
            BrinIndex(fields=['time']),  # "," is added here
        )

    time = models.DateTimeField(
        'Time of measurement',
        null=True
    )
0 голосов
/ 11 июня 2018

Несколько вещей, которые нужно попробовать:

  1. Вы не можете всегда иметь индекс для поля NULL .Я не знаю, как работает BrinIndex, но вы можете проверить это.

  2. Определение вашей модели должно включать атрибут db_index:

from django.contrib.postgres.indexes import BrinIndex

class Measurement(models.Model):

    time = models.DateTimeField(
        'Time of measurement',
        db_index=True,
        null=True
    )

    class Meta:
        indexes = (
            BrinIndex(fields=['time'])
        )
Добро пожаловать на сайт PullRequest, где вы можете задавать вопросы и получать ответы от других членов сообщества.
...