63.3. Расширяемость
Интерфейс BRIN характеризуется высоким уровнем абстракции и таким образом требует от разработчика метода доступа реализовать только смысловое наполнение обрабатываемого типа данных. Уровень BRIN берёт на себя заботу о параллельном доступе, поддержке журнала и поиске в структуре индекса.
Всё, что нужно, чтобы получить работающий метод доступа BRIN — это реализовать несколько пользовательских методов, определяющих поведение сводных значений, хранящихся в индексе, и их взаимоотношения с ключами сканирования. Словом, BRIN сочетает расширяемость с универсальностью, повторным использованием кода и аккуратным интерфейсом.
Класс операторов для BRIN должен предоставлять четыре метода:
BrinOpcInfo *opcInfo(Oid type_oid)
Возвращает внутреннюю информацию о сводных данных индексированных столбцов. Возвращаемое значение должно указывать на
BrinOpcInfo
(в памяти palloc) со следующим определением:typedef struct BrinOpcInfo { /* Число полей, хранящихся в столбце индекса этого класса операторов */ uint16 oi_nstored; /* Непрозрачный указатель для внутреннего использования классом операторов */ void *oi_opaque; /* Элементы кеша типов для сохранённых столбцов */ TypeCacheEntry *oi_typcache[FLEXIBLE_ARRAY_MEMBER]; } BrinOpcInfo;
Поле
BrinOpcInfo
.oi_opaque
могут использовать подпрограммы класса операторов для передачи информации опорным функциям при сканировании индекса.bool consistent(BrinDesc *bdesc, BrinValues *column, ScanKey key)
Показывает, соответствует ли значение ScanKey заданным индексированным значениям некоторой зоны. Номер целевого атрибута передаётся в составе ключа сканирования.
bool addValue(BrinDesc *bdesc, BrinValues *column, Datum newval, bool isnull)
Для заданного кортежа индекса и индексируемого значения изменяет выбранный атрибут кортежа, чтобы он дополнительно охватывал новое значение. Если в кортеж вносятся какие-либо изменения, возвращается
true
.bool unionTuples(BrinDesc *bdesc, BrinValues *a, BrinValues *b)
Консолидирует два кортежа индекса. Получая два кортежа, изменяет выбранный атрибут первого из них, что он охватывал оба кортежа. Второй кортеж не изменяется.
Основной дистрибутив включает поддержку двух типов классов операторов: minmax и inclusion. Определения классов операторов, использующие их, представлены для встроенных типов данных, насколько это уместно. Пользователь может определить дополнительные классы операторов для других типов данных, применяя аналогичные определения, и обойтись таким образом без написания кода; достаточно будет объявить нужные записи в каталоге. Заметьте, что предположения о семантике стратегий операторов зашиты в исходном коде опорных функций.
Также возможно создать классы операторов, воплощающие полностью другую семантику, разработав реализации четырёх основных опорных функций, описанных выше. Заметьте, что обратная совместимость между разными основными версиями не гарантируется: к примеру, в следующих выпусках могут потребоваться дополнительные опорные функции.
При написании класса операторов для типа данных, представляющего полностью упорядоченное множество, можно использовать опорные функции minmax вместе с соответствующими операторами, как показано в Таблице 63.2. Все члены класса операторов (функции и операторы) являются обязательными.
Таблица 63.2. Номера стратегий и опорных функций для классов операторов minmax
Член класса операторов | Объект |
---|---|
Опорная функция 1 | внутренняя функция brin_minmax_opcinfo() |
Опорная функция 2 | внутренняя функция brin_minmax_add_value() |
Опорная функция 3 | внутренняя функция brin_minmax_consistent() |
Опорная функция 4 | внутренняя функция brin_minmax_union() |
Стратегия оператора 1 | оператор меньше |
Стратегия оператора 2 | оператор меньше-или-равно |
Стратегия оператора 3 | оператор равно |
Стратегия оператора 4 | оператор больше-или-равно |
Стратегия оператора 5 | оператор больше |
При написании класса операторов для сложного типа данных, значения которого включаются в другой тип, можно использовать опорные функции inclusion вместе с соответствующими операторами, как показано в Таблице 63.3. Для этого требуется одна дополнительная функция, которую можно написать на любом языке. Для расширенной функциональности можно определить другие функции. Все операторы являются необязательными. Некоторые из них требует наличия других, что показано в таблице как зависимости.
Таблица 63.3. Номера стратегий и опорных функций для классов операторов inclusion
Член класса операторов | Объект | Зависимость |
---|---|---|
Опорная функция 1 | внутренняя функция brin_inclusion_opcinfo() | |
Опорная функция 2 | внутренняя функция brin_inclusion_add_value() | |
Опорная функция 3 | внутренняя функция brin_inclusion_consistent() | |
Опорная функция 4 | внутренняя функция brin_inclusion_union() | |
Опорная функция 11 | функция для слияния двух элементов | |
Опорная функция 12 | необязательная функция для проверки возможности слияния двух элементов | |
Опорная функция 13 | необязательная функция для проверки, содержится ли один элемент в другом | |
Опорная функция 14 | необязательная функция для проверки, является ли элемент пустым | |
Стратегия оператора 1 | оператор левее | Стратегия оператора 4 |
Стратегия оператора 2 | оператор не-простирается-правее | Стратегия оператора 5 |
Стратегия оператора 3 | оператор перекрывается | |
Стратегия оператора 4 | оператор не-простирается-левее | Стратегия оператора 1 |
Стратегия оператора 5 | оператор правее | Стратегия оператора 2 |
Стратегия оператора 6, 18 | оператор то-же-или-равно | Стратегия оператора 7 |
Стратегия оператора 7, 13, 16, 24, 25 | оператор содержит-или-равно | |
Стратегия оператора 8, 14, 26, 27 | оператор содержится-в-или-равно | Стратегия оператора 3 |
Стратегия оператора 9 | оператор не-простирается-выше | Стратегия оператора 11 |
Стратегия оператора 10 | оператор ниже | Стратегия оператора 12 |
Стратегия оператора 11 | оператор выше | Стратегия оператора 9 |
Стратегия оператора 12 | оператор не-простирается-ниже | Стратегия оператора 10 |
Стратегия оператора 20 | оператор меньше | Стратегия оператора 5 |
Стратегия оператора 21 | оператор меньше-или-равно | Стратегия оператора 5 |
Стратегия оператора 22 | оператор больше | Стратегия оператора 1 |
Стратегия оператора 23 | оператор больше-или-равно | Стратегия оператора 1 |
Номера опорных функций 1-10 зарезервированы для внутренних функций BRIN, так что функции уровня SQL начинаются с номера 11. Опорная функция номер 11 является основной, необходимой для построения индекса. Она должна принимать два аргумента того же типа данных, что и целевой тип класса, и возвращать их объединение. Класс операторов inclusion может сохранять значения объединения в различных типах данных, в зависимости от параметра STORAGE
. Возвращаемое функцией объединения значение должно соответствовать типу данных STORAGE
.
Опорные функции под номерами 12 и 14 предоставляются для поддержки нерегулярностей встроенных типов данных. Функция номер 12 применяется для поддержки работы с сетевыми адресами из различных семейств, которые нельзя объединять. Функция номер 14 применяется для поддержки зон с пустыми значениями. Функция номер 13 является необязательной, но рекомендуемой; она проверяет новое значение, прежде чем оно будет передано функции объединения. Инфраструктура BRIN может соптимизировать некоторые операции, когда объединение не меняется, поэтому применение этой функции может способствовать увеличению быстродействия индекса.
Классы операторов minmax и inclusion поддерживают операторы с разными типами, хотя с ними зависимости становятся более сложными. Класс minmax требует, чтобы для двух аргументов одного типа определялся полный набор операторов. Это позволяет поддерживать дополнительные типы данных, определяя дополнительные наборы операторов. Стратегии операторов класса inclusion могут зависеть от других стратегий, как показано в Таблице 63.3, или от своих собственных стратегий. Для них требуется, чтобы был определён необходимый оператор с типом данных STORAGE
для левого аргумента и другим поддерживаемым типом для правого аргумента реализуемого оператора. См. определение float4_minmax_ops
в качестве примера для minmax и box_inclusion_ops
в качестве примера для inclusion.
53.28. pg_init_privs
#
The catalog pg_init_privs
records information about the initial privileges of objects in the system. There is one entry for each object in the database which has a non-default (non-NULL) initial set of privileges.
Objects can have initial privileges either by having those privileges set when the system is initialized (by initdb) or when the object is created during a CREATE EXTENSION
and the extension script sets initial privileges using the GRANT
system. Note that the system will automatically handle recording of the privileges during the extension script and that extension authors need only use the GRANT
and REVOKE
statements in their script to have the privileges recorded. The privtype
column indicates if the initial privilege was set by initdb or during a CREATE EXTENSION
command.
Objects which have initial privileges set by initdb will have entries where privtype
is 'i'
, while objects which have initial privileges set by CREATE EXTENSION
will have entries where privtype
is 'e'
.
Table 53.28. pg_init_privs
Columns
Column Type Description |
---|
The OID of the specific object |
The OID of the system catalog the object is in |
For a table column, this is the column number (the |
A code defining the type of initial privilege of this object; see text |
The initial access privileges; see Section 5.7 for details |