pgbench
pgbench — запустить тест производительности Postgres Pro
Синтаксис
pgbench -i [параметр...] [имя_бд]
pgbench [параметр...] [имя_бд]
Описание
pgbench — это простая программа для запуска тестов производительности Postgres Pro. Она многократно выполняет одну последовательность команд, возможно в параллельных сеансах базы данных, а затем вычисляет среднюю скорость транзакций (число транзакций в секунду). По умолчанию pgbench тестирует сценарий, примерно соответствующий TPC-B, который состоит из пяти команд SELECT, UPDATE и INSERT в одной транзакции. Однако вы можете легко протестировать и другие сценарии, написав собственные скрипты транзакций.
Типичный вывод pgbench выглядит так:
transaction type: <builtin: TPC-B (sort of)> scaling factor: 10 query mode: simple number of clients: 10 number of threads: 1 number of transactions per client: 1000 number of transactions actually processed: 10000/10000 tps = 85.184871 (including connections establishing) tps = 85.296346 (excluding connections establishing)
В первых шести строках выводятся значения некоторых самых важных параметров. В следующей строке показывается количество выполненных и запланированных транзакций (это будет произведение числа клиентов и числа транзакций для одного клиента); эти количества будут равны, если только выполнение не завершится досрочно. (В режиме -T выводится только число фактически выполненных транзакций.) В последних двух строках показывается число транзакций в секунду, подсчитанное с учётом и без учёта времени установления подключения к серверу.
Для запускаемого по умолчанию теста типа TPC-B требуется предварительно подготовить определённые таблицы. Чтобы создать и наполнить эти таблицы, следует запустить pgbench с ключом -i (инициализировать). (Если вы применяете нестандартный скрипт, это не требуется, но тем не менее нужно подготовить конфигурацию, нужную вашему тесту.) Запуск инициализации выглядит так:
pgbench -i [другие-параметры]имя_базы
где имя_базы — имя уже существующей базы, в которой будет проводиться тест. (Чтобы указать, как подключиться к серверу баз данных, вы также можете добавить параметры -h, -p и/или -U.)
Внимание
pgbench -i создаёт четыре таблицы pgbench_accounts, pgbench_branches, pgbench_history и pgbench_tellers, предварительно уничтожая существующие таблицы с этими именами. Если вы вдруг используете эти имена в своей базе данных, обязательно переключитесь на другую базу!
С «коэффициентом масштаба», по умолчанию равным 1, эти таблицы изначально содержат такое количество строк:
table # of rows --------------------------------- pgbench_branches 1 pgbench_tellers 10 pgbench_accounts 100000 pgbench_history 0
Эти числа можно (и в большинстве случаев даже нужно) увеличить, воспользовавшись параметром -s (коэффициент масштаба). При этом также может быть полезен ключ -F (фактор заполнения).
Подготовив требуемую конфигурацию, можно запустить тест производительности командой без -i, то есть:
pgbench [параметры]имя_базы
Практически во всех случаях, чтобы получить полезные результаты, необходимо передать какие-либо дополнительные параметры. Наиболее важные параметры: -c (число клиентов), -t (число транзакций), -T (длительность) и -f (файл со скриптом). Полный список параметров приведён ниже.
Параметры
Следующий список разделён на три подраздела: одни параметры используются при инициализации базы данных, другие при проведении тестирования, а третьи в обоих случаях.
Параметры инициализации
pgbench принимает следующие аргументы командной строки для инициализации:
имя_бдУказывает имя базы, в которой будет проводиться тест. Если имя не задано, то используется значение переменной окружения
PGDATABASE. Если и переменная не задана, то в качестве имени базы будет взято имя пользователя, под которым осуществляется подключение.-i--initializeТребуется для вызова режима инициализации.
-Iэтапы_инициализации--init-steps=этапы_инициализацииВыполнять только выбранные из всех обычных подготовительных этапов. В параметре
этапы_инициализацииотдельные символы для каждого этапа выбирают, какие этапы должны выполняться. Все этапы выполняются в определённом порядке. Список этапов по умолчанию:dtgvp. Полный перечень подготовительных этапов:d(Drop, удалить)Удалить все существующие таблицы pgbench.
t(create Tables, создать таблицы)Создать таблицы, используемые стандартным сценарием pgbench, а именно:
pgbench_accounts,pgbench_branches,pgbench_historyиpgbench_tellers.g(Generate data, сгенерировать данные)Сгенерировать данные и загрузить их в стандартные таблицы, заменив все уже существующие данные.
v(Vacuum, очистка)Вызывать
VACUUMдля стандартных таблиц.p(create Primary keys, создать первичные ключи)Создать первичные ключи в стандартных таблицах.
f(create Foreign keys, создать внешние ключи)Создать ограничения внешних ключей между стандартными таблицами. (Заметьте, что это действие по умолчанию не выполняется.)
-Fфактор_заполнения--fillfactor=фактор_заполненияСоздать таблицы
pgbench_accounts,pgbench_tellersиpgbench_branchesс заданным фактором заполнения. Значение по умолчанию — 100.-n--no-vacuumНе выполнять очистку во время инициализации. (Этот параметр выключает этап инициализации
v, даже если он был указан в-I.)-q--quietПереключить вывод в немногословный режим, когда выводится только одно сообщение о прогрессе в 5 секунд. В режиме по умолчанию одно сообщение выводится на каждые 100000 строк, при этом за секунду обычно выводится довольно много строк (особенно на хорошем оборудовании).
-sкоэффициент_масштаба--scale=коэффициент_масштабаУмножить число генерируемых строк на заданный коэффициент. Например, с ключом
-s 100в таблицуpgbench_accountsбудут записаны 10 000 000 строк. Значение по умолчанию — 1. При коэффициенте, равном 20 000 или больше, столбцы, содержащие идентификаторы счетов (столбцыaid), перейдут к большим целым числам (типуbigint), чтобы в них могли уместиться все возможные значения идентификаторов.--foreign-keysСоздать ограничения внешних ключей между стандартными таблицами. (Этот ключ добавляет этап
fк последовательности подготовительных этапов, если он отсутствует.)--index-tablespace=табл_пространство_индексовСоздать индексы в указанном табличном пространстве, а не в пространстве по умолчанию.
--tablespace=табличное_пространствоСоздать таблицы в указанном табличном пространстве, а не в пространстве по умолчанию.
--unlogged-tablesСоздать все таблицы как нежурналируемые, а не как постоянные таблицы.
Параметры тестирования производительности
pgbench принимает следующие аргументы командной строки для тестирования производительности:
-bимя_скрипта[@вес]--builtin=имя_скрипта[@вес]Добавляет в список скриптов, которые будут выполняться, указанный встроенный скрипт. В число встроенных скриптов входят
tpcb-like,simple-updateиselect-only. Также принимаются однозначные начала их имён. Со специальным именемlistпрограмма выводит список встроенных скриптов и немедленно завершается.Дополнительно можно задать целочисленный вес после
@, меняющий вероятность выбора этого скрипта относительно других. По умолчанию вес считается равным 1. Подробности следуют ниже.-cклиенты--client=клиентыЧисло имитируемых клиентов, то есть число одновременных сеансов базы данных. Значение по умолчанию — 1.
-C--connectУстанавливать новое подключение для каждой транзакции вместо одного для каждого клиента. Это полезно для оценивания издержек подключений.
-d--debugВыводить отладочные сообщения.
-Dимя_переменной=значение--define=имя_переменной=значениеОпределить переменную для пользовательского скрипта (см. ниже). Параметр
-Dможет добавляться неоднократно.-fимя_файла[@вес]--file=имя_файла[@вес]Добавить в список выполняемых скриптов скрипт транзакции из файла
имя_файла.Дополнительно можно задать целочисленный вес после
@, меняющий вероятность выбора этого скрипта относительно других. По умолчанию вес считается равным 1. (Если вам нужно передать имя скрипта, содержащее символ@, добавьте к такому имени вес, чтобы исключить неоднозначность прочтения, напримерfilen@me@1.) Подробности следуют ниже.-jпотоки--jobs=потокиЧисло рабочих потоков в pgbench. Использовать нескольких потоков может быть полезно на многопроцессорных компьютерах. Клиенты распределяются по доступным потокам равномерно, насколько это возможно. Значение по умолчанию — 1.
-l--logЗаписать информацию о каждой транзакции в файл протокола. Подробности описаны ниже.
-Lпредел--latency-limit=пределТранзакции, продолжающиеся дольше указанного
предела(в миллисекундах), подсчитываются и отмечаются отдельно, как опаздывающие.В режиме ограничения скорости (
--rate=...) транзакции, которые отстают от графика более чем на заданныйпредел(в мс) и поэтому никак не могут уложиться в отведённый интервал, не передаются серверу вовсе. Они подсчитываются и отмечаются отдельно как пропущенные.-Mрежим_запросов--protocol=режим_запросовПротокол, выбираемый для передачи запросов на сервер:
simple: использовать протокол простых запросов.extended: использовать протокол расширенных запросов.prepared: использовать протокол расширенных запросов с подготовленными операторами.
В режиме
preparedpgbench повторно использует результат разбора запроса, начиная со второй итерации, и поэтому работает быстрее, чем в других режимах.По умолчанию выбирается протокол простых запросов. (За подробностями обратитесь к Главе 50.)
-n--no-vacuumНе производить очистку таблиц перед запуском теста. Этот параметр необходим, если вы применяете собственный сценарий, не затрагивающий стандартные таблицы
pgbench_accounts,pgbench_branches,pgbench_historyиpgbench_tellers.-N--skip-some-updatesЗапустить встроенный упрощённый скрипт simple-update. Краткий вариант записи
-b simple-update.-Pсек--progress=секВыводить отчёт о прогрессе через заданное число секунд (
сек). Выдаваемый отчёт включает время, прошедшее с момента запуска, скорость (в TPS) с момента предыдущего отчёта, а также среднее время ожидания транзакций и стандартное отклонение. В режиме ограничения скорости (-R) время ожидания вычисляется относительно назначенного времени запуска транзакции, а не фактического времени её начала, так что оно включает и среднее время отставания от графика.-r--report-latenciesВыводить по завершении тестировании средняя время ожидания операторов (время выполнения с точки зрения клиента) для каждой команды. Подробности описаны ниже.
-Rскорость передачи--rate=скорость передачиВыполнять транзакции, ориентируясь на заданную скорость, а не максимально быстро (по умолчанию). Скорость задаётся в транзакциях в секунду. Если заданная скорость превышает максимально возможную, это ограничение скорости не повлияет на результаты.
Для получения нужной скорости транзакции запускаются со случайными задержками, имеющими распределение Пуассона. При этом запланированное время запуска отсчитывается от начального времени, а не от завершения предыдущей транзакции. Это означает, что если какие-то транзакции отстанут от изначально рассчитанного времени завершения, всё же возможно, что последующие нагонят график.
В режиме ограничения скорости время ожидания транзакций, выводимое по итогам тестирования, вычисляется, исходя из запланированного времени запуска, так что в него входит время, которое очередная транзакция должна была ждать завершения предыдущей транзакции. Это время называется временем отклонения от графика, и его среднее и максимальное значения выводятся отдельно. Время ожидания транзакций с момента их фактического запуска, то есть время, потраченное на выполнение транзакций в базе данных, можно получить, если вычесть время отклонения от графика из времени ожидания транзакций.
Если ограничение
--latency-limitзадаётся вместе с--rate, транзакция может заведомо не вписываться в отведённое ей время, если предыдущая транзакция завершится слишком поздно, так как ожидаемое время окончания транзакции отсчитывается от времени запуска по графику. Такие транзакции не передаются серверу, а пропускаются и подсчитываются отдельно.Большое значение отклонения от графика свидетельствует о том, что система не успевает выполнять транзакции с заданной скоростью и выбранным числом клиентов и потоков. Когда среднее время ожидания транзакции превышает запланированный интервал между транзакциями, каждая последующая транзакция будет отставать от графика, и чем дольше будет выполняться тестирование, тем больше будет отставание. Когда это наблюдается, нужно уменьшить скорость транзакций.
-sкоэффициент_масштаба--scale=коэффициент_масштабаПоказать заданный коэффициент масштаба в выводе pgbench. Для встроенных тестов это не требуется; корректный коэффициент масштаба будет получен в результате подсчёта строк в таблице
pgbench_branches. Однако при использовании только нестандартных тестов (запускаемых с ключом-f) без этого параметра в качестве коэффициента масштаба будет выводиться 1.-S--select-onlyЗапустить встроенный скрипт select-only (только выборка). Краткий вариант записи
-b select-only.-tтранзакции--transactions=транзакцииЧисло транзакций, которые будут выполняться каждым клиентом (по умолчанию 10).
-Tсекунды--time=секундыВыполнять тест с ограничением по времени (в секундах), а не по числу транзакций для каждого клиента. Параметры
-tи-Tявляются взаимоисключающими.-v--vacuum-allОчищать все четыре стандартные таблицы перед запуском теста. Без параметров
-nи-vpgbench будет очищать от старых записей таблицыpgbench_tellersиpgbench_branches, а также опустошатьpgbench_history.--aggregate-interval=секундыДлительность интервала агрегации (в секундах). Может использоваться только с ключом
-l. С данным параметром в протокол выводится сводка по интервалам, как описано ниже.--log-prefix=префиксЗадать префикс имён файлов для файлов протоколов, создаваемых с ключом
--log. Префикс по умолчанию —pgbench_log.--progress-timestampПри отображении прогресса (с параметром
-P) выводить текущее время (в формате Unix), а не количество секунд от начала запуска. Время задаётся в секундах с точностью до миллисекунд. Это помогает сравнивать журналы, записываемые разными средствами.--random-seed=ЗАТРАВКАУстановить затравку для генератора случайных чисел. Инициализирует генератор случайных чисел, который затем выдаёт последовательность начальных состояний отдельных генераторов для каждого потока.
ЗАТРАВКАможет принимать следующие значения:time(по умолчанию, затравка базируется на текущем времени),rand(задействовать надёжный генератор случайных чисел или выдать ошибку, если он отсутствует) или беззнаковое десятичное число. Генератор случайных чисел может вызываться явно из скрипта pgbench (функциямиrandom...) или неявно (например, для планирования выполнения транзакций с ключом--rate). В случае установки значения явным образом оно выводится в терминале. Любое значение, допустимое в качествеЗАТРАВКИ, можно также задать в переменной окруженияPGBENCH_RANDOM_SEED. Чтобы заданная затравка применялась во всех возможных случаях использования, задайте этот параметр первым или установите переменную окружения.Явное указание определённой затравки позволяет точно воспроизвести выполнение
pgbenchв части использования случайных чисел. Так как случайное состояние поддерживается внутри потока, это означает, что выполнениеpgbenchпри одинаковых запусках повторится в точности, если один поток используется одним клиентом и отсутствуют внешние зависимости или зависимости от данных. Со статистической точки зрения точное воспроизведение выполнения нежелательно, так как это может скрыть вариативность производительности или показать завышенную скорость, например из-за попадания в одни и те же страницы данных. Однако это может быть очень полезно для отладки, например, для повторения редкого сценария, приводящего к ошибке. Используйте данную возможность обдуманно.--sampling-rate=скорость передачиЧастота выборки для записи данных в протокол, изменяя которую можно уменьшить объём протокола. При указании этого параметра в протокол выводится информация только о заданном проценте транзакций. Со значением 1.0 в нём будут отмечаться все транзакции, а с 0.05 только 5%.
Обрабатывая протокол, не забудьте учесть частоту выборки. Например, вычисляя скорость (TPS), вам нужно будет соответственно умножить содержащиеся в нём числа (например, с частотой выборки 0.01 вы получите только 1/100 фактической скорости).
Общие параметры
pgbench принимает следующие общие аргументы командной строки:
-hкомпьютер--host=компьютерАдрес сервера баз данных
-pпорт--port=портНомер порта сервера баз данных
-Uимя_пользователя--username=имя_пользователяИмя пользователя для подключения
-V--versionВывести версию pgbench и завершиться.
-?--helpВывести справку об аргументах командной строки pgbench и завершиться.
Код завершения
В случае успешного выполнения возвращается код 0. Код завершения 1 указывает на статичные проблемы, например, ошибки в параметрах командной строки. При возникновении ошибок во время выполнения, например при обращении к базе данных или выполнении скрипта, выдаётся код завершения 2. В последнем случае pgbench выведет частичные результаты.
Переменные окружения
PGDATABASEPGHOSTPGPORTPGUSERПараметры подключения по умолчанию.
Эта утилита, как и большинство других утилит Postgres Pro, использует переменные среды, поддерживаемые libpq (см. Раздел 31.14).
Замечания
Каково содержание «транзакции», которую выполняет pgbench?
Программа pgbench выполняет тестовые скрипты, выбирая их случайным образом из заданного списка. Это могут быть как встроенные скрипты, задаваемые аргументами -b, так и пользовательские, задаваемые аргументами -f. Для каждого скрипта можно задать относительный вес после @, чтобы скорректировать вероятность его выбора. По умолчанию вес считается равным 1. Скрипты с весом 0 игнорируются.
Стандартный встроенный скрипт (также вызываемый с ключом -b tpcb-like) выдаёт семь команд в транзакции со случайно выбранными aid, tid, bid и delta. Его сценарий написан по мотивам теста производительности TPC-B, но это не собственно TPC-B, потому он называется так.
BEGIN;UPDATE pgbench_accounts SET abalance = abalance + :delta WHERE aid = :aid;SELECT abalance FROM pgbench_accounts WHERE aid = :aid;UPDATE pgbench_tellers SET tbalance = tbalance + :delta WHERE tid = :tid;UPDATE pgbench_branches SET bbalance = bbalance + :delta WHERE bid = :bid;INSERT INTO pgbench_history (tid, bid, aid, delta, mtime) VALUES (:tid, :bid, :aid, :delta, CURRENT_TIMESTAMP);END;
При выборе встроенного скрипта simple-update (или указании -N) шаги 4 и 5 исключаются из транзакции. Это позволяет избежать конкуренции при обращении к этим таблицам, но тест становится ещё менее похожим на TPC-B.
При выборе встроенного теста select-only (или указании -S) выполняется только SELECT.
Пользовательские скрипты
Программа pgbench поддерживает запуск пользовательских сценариев оценки производительности, позволяя заменять стандартный скрипт транзакции (описанный выше) скриптом, считываемым из файла (с параметром -f). В этом случае «транзакцией» считается одно выполнение данного скрипта.
Файл скрипта содержит одну или несколько команд SQL, разделённых точкой с запятой. Пустые строки и строки, начинающиеся с --, игнорируются. В файлах скриптов также могут содержаться «метакоманды», которые обрабатывает сама программа pgbench, как описано ниже.
Примечание
До версии Postgres Pro 9.6, SQL-команды в файлах скриптов завершались символами перевода строки, и поэтому они не могли занимать несколько строк. Теперь для разделения последовательных команд SQL требуется добавлять точку с запятой (хотя без неё можно обойтись в конце SQL-команды, за которой идёт метакоманда). Если вам нужно создать файл скрипта, работающий и со старыми версиями pgbench, записывайте каждую команду SQL в отдельной строке и завершайте её точкой с запятой.
Для файлов скриптов реализован простой механизм подстановки переменных. Имя переменных должно состоять из букв (буквы могут быть не латинскими), подчёркиваний и цифр (но цифра не может быть первым символом). Переменные можно задать в командной строке параметрами -D, описанными выше, или метакомандами, рассматриваемыми ниже. Помимо переменных, которые можно установить параметрами командной строки -D, есть несколько автоматически устанавливаемых переменных; они перечислены в Таблице 259. Если значение этих переменных задаётся в параметре -D, оно переопределяет автоматическое значение. Когда значение переменной определено, его можно вставить в команду SQL, написав :имя_переменной. Каждый клиентский сеанс, если их несколько, получает собственный набор переменных. В одном операторе pgbench поддерживает до 255 ссылок на переменные.
Таблица 259. Автоматические переменные
| Переменная | Описание |
|---|---|
client_id | уникальное число, идентифицирующее клиентский сеанс (начиная с нуля) |
default_seed | затравка, используемая в хеш-функциях по умолчанию |
random_seed | затравка генератора случайных чисел (в отсутствие переопределения с ключом -D) |
scale | текущий коэффициент масштаба |
Метакоманды в скрипте начинаются с обратной косой черты (\) и обычно продолжаются до конца строки, хотя их можно переносить на следующую строку последовательностью символов: обратная косая, возврат каретки. Аргументы метакоманд разделяются пробелами. Поддерживаемые метакоманды представлены ниже:
-
\gset [префикс] Эта команда может применяться для завершения SQL-запросов вместо завершающей точки с запятой (
;).Когда используется эта команда, ожидается, что предыдущий SQL-запрос возвратит одну строку; значения столбцов будут сохранены в переменные с именами столбцов, а если указан
префикс, он будет добавлен к этим именам.В следующем примере окончательный баланс счёта из первого запроса возвращается в переменной
abalance, а целочисленные значения из третьего запроса попадают в переменныеp_twoиp_three. Результат второго запроса отбрасывается.UPDATE pgbench_accounts SET abalance = abalance + :delta WHERE aid = :aid RETURNING abalance \gset -- compound of two queries SELECT 1 \; SELECT 2 AS two, 3 AS three \gset p_
\ifвыражение\elifвыражение\else\endifЭта группа команд реализует вкладываемые условные блоки, подобные
\ifвыражениевpsql. В качестве условных задаются те же выражения, что и в\set, при этом истинным считается любое ненулевое значение.-
\setимя_переменнойвыражение Устанавливает для переменной
имя_переменнойзначение, вычисленное извыражения. Выражение может содержать константуNULL, логические константыTRUEиFALSE, целочисленные константы (например,5432), константы с плавающей точкой (например,3.14159), ссылки на переменные:имя_переменной, операторы с обычными для SQL приоритетами и ассоциативностью, вызовы функций, общие условные SQL-выраженияCASE, а также скобки.Функции и большинство операторов возвращают
NULLдля аргументовNULL.При проверке условия отличные от нуля числовые значения воспринимаются как
TRUE, а числовые нулевые значения иNULL— какFALSE.При переполнениях, вызванных слишком большими числами с плавающей точкой или целыми, а также целочисленными операциями (
+,-,*и/), выдаются ошибки.Если в конструкции
CASEотсутствует заключительноеELSE, значением по умолчанию считаетсяNULL.Примеры:
\set ntellers 10 * :scale \set aid (1021 * random(1, 100000 * :scale)) % \ (100000 * :scale) + 1 \set divx CASE WHEN :x <> 0 THEN :y/:x ELSE NULL END-
\sleepномер[ us | ms | s ] Приостанавливает выполнение скрипта на заданное число микросекунд (
us), миллисекунд (ms) или секунд (s). Когда единицы не указываются, подразумеваются секунды. Здесьчисломожет быть целочисленной константой или ссылкой:имя_переменнойна переменную с целочисленным значением.Пример:
\sleep 10 ms
-
\setshellимя_переменнойкоманда[аргумент... ] Присваивает переменной
имя_переменнойрезультат команды оболочкикомандас указаннымиаргументами. Эта команда должна просто выдать целочисленное значение в стандартный вывод.Здесь
командаи каждыйаргументможет быть либо текстовой константой, либо ссылкой на переменную:имя_переменной. Если вы хотите записатьаргумент, начинающийся с двоеточия, добавьте передаргументомдополнительное двоеточие.Пример:
\setshell назначаемая_переменная команда строковый_аргумент :переменная ::строка_начинающаяся_двоеточием
-
\shellкоманда[аргумент... ] Действует так же, как и
\setshell, но не учитывает результат команды.Пример:
\shell команда строковый_аргумент :переменная ::строка_начинающаяся_двоеточием
Встроенные операторы
Арифметические, битовые и логические операторы, а также операторы сравнения, перечисленные в Таблице 260, встроены в pgbench и могут применяться в выражениях в \set.
Таблица 260. Операторы pgbench, в порядке увеличения приоритета
| Оператор | Описание | Пример | Результат |
|---|---|---|---|
OR | логическое ИЛИ | 5 or 0 | TRUE |
AND | логическое И | 3 and 0 | FALSE |
NOT | логическое НЕ | not false | TRUE |
IS [NOT] (NULL|TRUE|FALSE) | проверки значений | 1 is null | FALSE |
ISNULL|NOTNULL | проверки на NULL | 1 notnull | TRUE |
= | равно | 5 = 4 | FALSE |
<> | не равно | 5 <> 4 | TRUE |
!= | не равно | 5 != 5 | FALSE |
< | меньше | 5 < 4 | FALSE |
<= | меньше или равно | 5 <= 4 | FALSE |
> | больше | 5 > 4 | TRUE |
>= | больше или равно | 5 >= 4 | TRUE |
| | целочисленное битовое ИЛИ | 1 | 2 | 3 |
# | целочисленное битовое исключающее ИЛИ | 1 # 3 | 2 |
& | целочисленное битовое И | 1 & 3 | 1 |
~ | целочисленное битовое НЕ | ~ 1 | -2 |
<< | целочисленный битовый сдвиг влево | 1 << 2 | 4 |
>> | целочисленный битовый сдвиг вправо | 8 >> 2 | 2 |
+ | сложение | 5 + 4 | 9 |
- | вычитание | 3 - 2.0 | 1.0 |
* | умножение | 5 * 4 | 20 |
/ | деление (при целочисленном остаток отбрасывается) | 5 / 3 | 1 |
% | остаток от деления | 3 % 2 | 1 |
- | смена знака | - 2.0 | -2.0 |
Встроенные функции
Функции, перечисленные в Таблице 261, встроены в pgbench и могут применяться в выражениях в метакоманде \set.
Таблица 261. Функции pgbench
| Функция | Тип результата | Описание | Пример | Результат |
|---|---|---|---|---|
| то же, что и a | модуль числа (абсолютное значение) | abs(-17) | 17 |
| то же, что и a | выводит a в stderr и возвращает a | debug(5432.1) | 5432.1 |
| double | приведение к типу с плавающей точкой | double(5432) | 5432.0 |
| double | экспонента | exp(1.0) | 2.718281828459045 |
| double, если любой из аргументов (a) — double, а иначе целое число | наибольшее значение среди аргументов | greatest(5, 4, 3, 2) | 5 |
| integer | псевдоним для hash_murmur2() | hash(10, 5432) | -5817877081768721676 |
| integer | хеш FNV-1a | hash_fnv1a(10, 5432) | -7793829335365542153 |
| integer | хеш MurmurHash2 | hash_murmur2(10, 5432) | -5817877081768721676 |
| integer | приведение к целочисленному типу | int(5.4 + 3.8) | 9 |
| double, если любой из аргументов (a) — double, а иначе целое число | наименьшее значение среди аргументов | least(5, 4, 3, 2.1) | 2.1 |
| double | натуральный логарифм | ln(2.718281828459045) | 1.0 |
| integer | остаток от деления | mod(54, 32) | 22 |
| double | значение константы PI | pi() | 3.14159265358979323846 |
| double | возведение в степень | pow(2.0, 10), power(2.0, 10) | 1024.0 |
| integer | случайное целое число с равномерным распределением в интервале [lb, ub] | random(1, 10) | целое между 1 и 10 |
| integer | случайное целое число с экспоненциальным распределением в интервале [lb, ub], см. ниже | random_exponential(1, 10, 3.0) | целое между 1 и 10 |
| integer | целое число с распределением Гаусса в интервале [lb, ub], см. ниже | random_gaussian(1, 10, 2.5) | целое между 1 и 10 |
| integer | целое число с распределением Ципфа в интервале [lb, ub], см. ниже | random_zipfian(1, 10, 1.5) | целое между 1 и 10 |
| double | квадратный корень | sqrt(2.0) | 1.414213562 |
Функция random выдаёт значения с равномерным распределением, то есть вероятности получения всех чисел в интервале равны. Функции random_exponential, random_gaussian и random_zipfian требуют указания дополнительного параметра типа double, определяющего точную форму распределения.
Для экспоненциального распределения
parameterуправляет распределением, обрезая быстро спадающее экспоненциальное распределение в точкеparameter, а затем это распределение проецируется на целые числа между границами. Точнее говоря, с
f(x) = exp(-parameter * (x - min) / (max - min + 1)) / (1 - exp(-parameter))значение
iмеждуminиmaxвыдаётся с вероятностью:f(i) - f(i + 1).Интуиция подсказывает, что чем больше
parameter, тем чаще будут выдаваться значения, близкие кmin, и тем реже значения, близкие кmax. Чемparameterближе к 0, тем более плоским (более равномерным) будет распределение. В грубом приближении при таком распределении наиболее частый 1% значений в диапазоне рядом сminвыдаётсяparameter% времени. Значениеparameterдолжно быть строго положительным.Для распределения Гаусса по интервалу строится обычное нормальное распределение (классическая кривая Гаусса в форме колокола) и этот интервал обрезается в точке
-parameterслева и+parameterсправа. Вероятнее всего при таком распределении выдаются значения из середины интервала. Точнее говоря, еслиPHI(x)— функция распределения нормальной случайной величины со средним значениемmu, равным(max + min) / 2.0, и
f(x) = PHI(2.0 * parameter * (x - mu) / (max - min + 1)) /
(2.0 * PHI(parameter) - 1)тогда значение
iмеждуminиmaxвключительно выдаётся с вероятностью:f(i + 0.5) - f(i - 0.5). Интуиция подсказывает, что чем большеparameter, тем чаще будут выдаваться значения в середине интервала, и тем реже значения у границminиmax. Около 67% значений будут выдаваться из среднего интервала1.0 / parameter, то есть плюс/минус0.5 / parameterот среднего значения, и 95% из среднего интервала2.0 / parameter, то есть плюс/минус1.0 / parameterот среднего значения; например, еслиparameterравен 4.0, 67% значений выдаются из средней четверти (1.0 / 4.0) интервала (то есть от3.0 / 8.0до5.0 / 8.0) и 95% из средней половины (2.0 / 4.0) интервала (из второй и третьей четвертей). Значениеparameterне может быть меньше 2.0.Функция
random_zipfianгенерирует ограниченное распределение по закону Ципфа.parameterопределяет, насколько неравномерно распределение. Чем большеparameter, тем чаще выдаются значения, близкие к началу интервала. Это распределение таково, что при диапазоне, начинающемся с 1, отношение вероятности получитьkк вероятности полученияk+1равняется((. Например,k+1)/k)**parameterrandom_zipfian(1, ..., 2.5)будет выдавать число1примерно в(2/1)**2.5 = 5.66раза чаще, чем число2, а оно, в свою очередь, будет выдаваться примерно в(3/2)**2.5 = 2.76раза чаще, чем3, и так далее.Это распределение реализовано в pgbench по материалу книги «Non-Uniform Random Variate Generation» («Генерация неравномерно распределённых случайных чисел» Люк Деврой, стр. 550-551, Springer 1986. Вследствие ограничений алгоритма
parameterможет принимать значения только в интервале [1.001, 1000].
Функции хеширования hash, hash_murmur2 и hash_fnv1a принимают на вход аргумент и необязательный параметр с затравкой. Если значение затравки не задаётся, используется значение переменной :default_seed, которая инициализируется случайным числом (если не задаётся явно ключом командной строки -D). Функции хеширования могут использоваться для разброса распределения случайных функций, таких как random_zipfian или random_exponential. Например, следующий скрипт pgbench эмулирует возможную реальную нагрузку, типичную для социальных медиа- и блог-платформ, где несколько пользователей генерируют львиную долю нагрузки:
\set r random_zipfian(0, 100000000, 1.07) \set k abs(hash(:r)) % 1000000
В некоторых случаях требуются другие разнообразные распределения, не коррелирующие друг с другом, и тогда может быть полезно явное указание затравки:
\set k1 abs(hash(:r, :default_seed + 123)) % 1000000 \set k2 abs(hash(:r, :default_seed + 321)) % 1000000
В качестве примера взгляните на встроенное определение транзакции типа TPC-B:
\set aid random(1, 100000 * :scale) \set bid random(1, 1 * :scale) \set tid random(1, 10 * :scale) \set delta random(-5000, 5000) BEGIN; UPDATE pgbench_accounts SET abalance = abalance + :delta WHERE aid = :aid; SELECT abalance FROM pgbench_accounts WHERE aid = :aid; UPDATE pgbench_tellers SET tbalance = tbalance + :delta WHERE tid = :tid; UPDATE pgbench_branches SET bbalance = bbalance + :delta WHERE bid = :bid; INSERT INTO pgbench_history (tid, bid, aid, delta, mtime) VALUES (:tid, :bid, :aid, :delta, CURRENT_TIMESTAMP); END;
С таким скриптом транзакция на каждой итерации будет обращаться к разным, случайно выбираемым строкам. (Этот пример показывает, почему важно, чтобы в каждом клиентском сеансе были собственные переменные — в противном случае они не будут независимо обращаться к разным строкам.)
Протоколирование транзакций
С параметром -l (но без --aggregate-interval), pgbench записывает информацию о каждой транзакции в протокол. Этот файл протокола будет называться , где префикс.nnnпрефикс по умолчанию — pgbench_log, а nnn — PID процесса pgbench. Префикс можно сменить, воспользовавшись ключом --log-prefix. Если параметр -j равен 2 или выше, будет создано несколько рабочих потоков, и каждый будет записывать отдельный протокол. Первый рабочий процесс будет использовать файл с тем же именем, что и в стандартном случае с одним потоком, а файлы остальных потоков будут называться , где префикс.nnn.mmmmmm — последовательный номер рабочего процесса, начиная с 1.
Протокол имеет следующий формат:
код_клиентачисло_транзакцийдлительностьномер_скриптавремя_эпохивремя_мкс[отставание_от_графика]
Здесь код_клиента показывает, в сеансе какого клиента запускалась транзакция, число_транзакций отражает, сколько транзакций выполнялось в этом сеансе, длительность — общее время транзакций (в микросекундах), номер_скрипта показывает, какой файл скрипта использовался (это полезно при указании нескольких скриптов ключами -f и -b), а время_эпохи/время_мкс — отметка времени в формате Unix и смещение в микросекундах (из этих чисел можно получить время стандарта ISO 8601 с дробными секундами), показывающие, когда транзакция была завершена. Поле отставание_от_графика представляет разницу между запланированным временем запуска транзакции и фактическим временем запуска (в микросекундах). Оно выводится, только когда применяется параметр --rate. Когда одновременно применяются параметры --rate и --latency-limit, в поле длительность для пропущенных транзакций будет выводиться skipped.
Фрагмент протокола, полученного при выполнении с одним клиентом:
0 199 2241 0 1175850568 995598 0 200 2465 0 1175850568 998079 0 201 2513 0 1175850569 608 0 202 2038 0 1175850569 2663
Ещё один пример с --rate=100 и --latency-limit=5 (обратите внимание на дополнительный столбец отставание_от_графика):
0 81 4621 0 1412881037 912698 3005 0 82 6173 0 1412881037 914578 4304 0 83 skipped 0 1412881037 914578 5217 0 83 skipped 0 1412881037 914578 5099 0 83 4722 0 1412881037 916203 3108 0 84 4142 0 1412881037 918023 2333 0 85 2465 0 1412881037 919759 740
В этом примере транзакция 82 опоздала, так как её длительность (6.173 мс) превысила ограничение в 5 мс. Следующие две транзакции были пропущены, так как было слишком поздно их начинать.
Когда проводится длительное тестирование с большим количеством транзакций, файлы протоколов могут быть очень объёмными. Чтобы в них записывалась только случайная выборка транзакций, можно запустить команду с параметром --sampling-rate.
Протоколирование с агрегированием
С параметром --aggregate-interval протоколы имеют несколько иной формат:
начало_интервалачисло_транзакцийсумма_длительностисумма_длительности_2мин_длительностьмакс_длительность[сумма_задержкисумма_задержки_2мин_задержкамакс_задержка[пропущено_транзакций] ]
Здесь начало_интервала — начальное время интервала (в формате времени UNIX), число_транзакций — количество транзакций в данном интервале, сумма_длительности — суммарная длительность транзакций, сумма_длительности_2 — сумма квадратов длительностей транзакций в интервале, мин_длительность — минимальная длительность в интервале, а макс_задержка — максимальная. Следующие поля, сумма_задержки, сумма_задержки_2, мин_задержка и макс_задержка, присутствуют, только если применяется параметр --rate. Они отражают время, на которое задержалась каждая транзакция, ожидая завершения предыдущей, то есть разницу между временем фактического запуска и запланированным временем. Самое последнее поле, пропущено_транзакций, тоже присутствует, только если применяется ключ --latency-limit. В нём выводится число транзакций, пропущенных из-за того, что их запуск задержался слишком надолго. Каждая транзакция учитывается в том интервале, в котором она была зафиксирована.
Пример вывода:
1345828501 5601 1542744 483552416 61 2573 1345828503 7884 1979812 565806736 60 1479 1345828505 7208 1979422 567277552 59 1391 1345828507 7685 1980268 569784714 60 1398 1345828509 7073 1979779 573489941 236 1411
Заметьте, что простой протокол (без агрегирования) показывает, какой скрипт использовался для каждой транзакции, в отличие от протокола с агрегированием. Таким образом, если вам нужны подобные сведения, но в разрезе скриптов, вам придётся агрегировать данные самостоятельно.
Время ожидания по операторам
С параметром -r программа pgbench учитывает время выполнения каждого оператора каждым клиентом. Затем по завершении тестирования она выводит среднее по всем значениям как время ожидания каждого оператора.
Со стандартным скриптом вывод будет примерно таким:
starting vacuum...end.
transaction type: <builtin: TPC-B (sort of)>
scaling factor: 1
query mode: simple
number of clients: 10
number of threads: 1
number of transactions per client: 1000
number of transactions actually processed: 10000/10000
latency average = 15.844 ms
latency stddev = 2.715 ms
tps = 618.764555 (including connections establishing)
tps = 622.977698 (excluding connections establishing)
statement latencies in milliseconds:
0.002 \set aid random(1, 100000 * :scale)
0.005 \set bid random(1, 1 * :scale)
0.002 \set tid random(1, 10 * :scale)
0.001 \set delta random(-5000, 5000)
0.326 BEGIN;
0.603 UPDATE pgbench_accounts SET abalance = abalance + :delta WHERE aid = :aid;
0.454 SELECT abalance FROM pgbench_accounts WHERE aid = :aid;
5.528 UPDATE pgbench_tellers SET tbalance = tbalance + :delta WHERE tid = :tid;
7.335 UPDATE pgbench_branches SET bbalance = bbalance + :delta WHERE bid = :bid;
0.371 INSERT INTO pgbench_history (tid, bid, aid, delta, mtime) VALUES (:tid, :bid, :aid, :delta, CURRENT_TIMESTAMP);
1.212 END;
Если задействуется несколько файлов скриптов, средние значения выводятся отдельно для каждого файла.
Учтите, что сбор дополнительных временных показателей влечёт некоторые издержки и приводит к снижению средней скорости и, как результат, падению TPS. На сколько именно снизится скорость, во многом зависит от платформы и оборудования. Хороший способ оценить, каковы эти издержки — сравнить средние значения TPS, получаемые с подсчётом времени операторов и без такого подсчёта.
Полезные советы
Используя pgbench, можно без особого труда получить абсолютно бессмысленные числа. Последуйте приведённым советам, чтобы получить полезные результаты.
Во-первых, никогда не доверяйте тестам, которые выполняются всего несколько секунд. Воспользуйтесь параметром -t и -T и установите время выполнения не меньше нескольких минут, чтобы избавиться от шума в средних значениях. В некоторых случаях для получения воспроизводимых результатов тестирование должно продолжаться несколько часов. Чтобы понять, были ли получены воспроизводимые значения, имеет смысл запустить тестирование несколько раз.
Для стандартного сценария по типу TPC-B начальный коэффициент масштаба (-s) должен быть не меньше числа клиентов, с каким вы намерены проводить тестирование (-c); в противном случае вы, по большому счёту, будете замерять время конкурентных изменений. Таблица pgbench_branches содержит всего -s строк, а каждая транзакция хочет изменить одну из них, так что если значение -c превышает -s, это несомненно приведёт к тому, что многие транзакции будут блокироваться другими.
Стандартный сценарий тестирования также довольно сильно зависит от того, сколько времени прошло с момента инициализации таблиц: накопление неактуальных строк и «мёртвого» пространства в таблицах влияет на результаты. Чтобы правильно оценить результаты, необходимо учитывать, сколько всего изменений было произведено и когда выполнялась очистка. Если же включена автоочистка, это может быть чревато непредсказуемыми изменениями оценок производительности.
Полезность результатов pgbench также может ограничиваться тем, что тестирование с большим числом клиентских сеансов само по себе нагружает систему. Этого можно избежать, запуская pgbench на другом компьютере, не на сервере баз данных, хотя при этом большое значение имеет скорость сети. Иногда, оценивая производительность одного сервера, полезно запускать даже несколько экземпляров pgbench параллельно, на отдельных клиентских компьютерах.
Безопасность
Если к базе данных, которая не приведена в соответствие шаблону безопасного использования схем, имеют доступ недоверенные пользователи, не запускайте pgbench в этой базе. Программа pgbench использует неполные имена и не настраивает для себя путь поиска.