PostgreSQL, SQL, Базы данных

PostgreSQL: Установка пользовательских параметров базы данных и/или сессии

Через запрос show all можно посмотреть текущие значения параметров postgres времени выполнения. Но в добавок к имеющимся системным параметрам можно создать и свои, пользовательские.

Имена пользовательских параметров обязательно должны содержать точку, например, my.version, mydb.parameter и т.д. Установить значения этих параметров можно для сессии и/или для конкретной базы данных. Новое значение для базы данных будет считано при новом подключении к ней. В текущем подключении — будет считано предыдущее значение.

Для установки значения параметра для сессии — команда

set <parameter_name> to <value>;

Для конкретной базы данных —

alter database <db_name> set <parameter_name> = <value>;

Подробнее здесь: https://www.postgresql.org/docs/current/config-setting.html. Важно: установить значение пользовательского параметра в контексте базы данных возможно, если такой параметр уже был когда-то определен для сессии.

Получить значение параметра можно через функцию current_setting(<parameter_name>) или show <parameter_name>.

Пример функции получения версии (дата последнего изменения) базы данных:

[pastacode lang=»sql» manual=»CREATE%20OR%20REPLACE%20FUNCTION%20public.db_version(%0A%09)%0A%20%20%20%20RETURNS%20timestamp%20with%20time%20zone%0A%20%20%20%20LANGUAGE%20’plpgsql’%0A%20%20%20%20COST%20100%0A%20%20%20%20VOLATILE%20PARALLEL%20UNSAFE%0AAS%20%24BODY%24%0Abegin%0A%20%20%20%20return%20(select%20current_setting(‘my.version’)%3A%3Atimestamp%20with%20time%20zone)%3B%0A%20%20%20%20exception%0A%20%20%20%20when%20others%20then%0A%20%20%20%20%20%20%20%20set%20my.version%20to%20’1900-01-01%2000%3A00’%3B%0A%20%20%20%20%20%20%20%20alter%20database%20mydatabasename%20set%20my.version%20from%20current%3B%0A%20%20%20%20return%20’1900-01-01%2000%3A00’%3A%3Atimestamp%20with%20time%20zone%3B%0Aend%3B%0A%24BODY%24%3B» message=»» highlight=»» provider=»manual»/]

Если указанный параметр не известен сессии, то произойдет его инициализация. В противном случае — будет возвращено текущее значение.

[pastacode lang=»sql» manual=»alter%20database%20makro%20set%20my.version%20%3D%20’2022-09-30%2010%3A07’%3B» message=»Установка нового значения параметра my.version» highlight=»» provider=»manual»/]

C++, SQL, SQLite

Аналог UNNEST в SQLite

Такой удобной функции, как UNNEST, в SQLite нет (пссс, об UNNEST в PostgreSQL я писала вот здесь). А что использовать, если нам нужно элементы некоего массива/списка/контейнера впихнуть в один запрос вставки данных, например? Можно воспользоваться WITH и VALUES.

Предположим, у нас есть два списка — список кодов и список соответствующих им дат. Можно коды и даты скомпоновать сразу парами, можно — раздельно. В каких-то конкретных случаях будут разные ситуации.

[pastacode lang=»sql» manual=»with%20cte_code(id%2Ccode)%20as%20(%0Aselect%20rowid%2C%20a.*%20from%20%0A%09(values(‘111’)%2C(‘222’)%2C(‘333’))%20as%20a)%2C%0Acte_dates(id%2Cdate)%20as%20(%0Aselect%20rowid%2C%20b.*%20from%20%0A%09(values(‘2022-08-09’)%2C(‘2022-08-10’)%2C%0A%09%20(‘2022-08-11’))%20as%20b)%0A%0Ainsert%20into%20ones(task_id%2C%20code%2C%20date)%0Aselect%201%2C%20a.code%2C%20b.date%0Afrom%20cte_code%20a%20%0Ainner%20join%20cte_dates%20b%20on%20b.id%20%3D%20a.id» message=»Списки поотдельности» highlight=»1,4,9″ provider=»manual»/]

cte_code — список кодов, cte_dates — список дат. task_id — некая константа.

[pastacode lang=»sql» manual=»with%20cte_pairs(code%2C%20date)%20as%20(%0Aselect%20*%20from%20%0A%09(values(‘111’%2C%20’2022-08-09’)%2C%0A%09%20(‘222’%2C%20’2022-08-10’)%2C%0A%09%20(‘333’%2C%20’2022-08-11’)))%0A%0Ainsert%20into%20ones(task_id%2C%20code%2C%20date)%0Aselect%201%2C%20a.code%2C%20a.date%0Afrom%20cte_pairs%20a» message=»Спарованный список» highlight=»1″ provider=»manual»/]

cte_pairs — оба наших списка в одном.

Строка запроса в Qt C++, учитывая наличие QStringList codes и QStringList dates, может выглядеть так:

[pastacode lang=»cpp» manual=»QStringList%20codes%2C%20dates%3B%0AQString%20qry%20%3D%20QString(%22with%20cte_code(id%2Ccode)%20as%20(%20%22%0A%09%09%09%09%09%20%20%22select%20rowid%2C%20a.*%20from%20%22%0A%09%09%09%09%09%20%20%22(values(‘%252’))%20as%20a)%2C%20%22%0A%09%09%09%09%09%20%20%22cte_dates(id%2Cdate)%20as%20(%20%22%0A%09%09%09%09%09%20%20%22select%20rowid%2C%20b.*%20%22%0A%09%09%09%09%09%20%20%22from%20(values(‘%251′))%20as%20b)%20%22%0A%09%09%09%09%09%20%20%0A%09%09%09%09%09%20%20%22insert%20into%20ones(task_id%2C%20code%2C%20date)%20%22%0A%09%09%09%09%09%20%20%22select%201%2C%20a.code%2C%20b.date%20%22%0A%09%09%09%09%09%20%20%22from%20cte_code%20a%20%22%0A%09%09%09%09%09%20%20%22inner%20join%20cte_dates%20b%20on%20b.id%20%3D%20a.id%22)%0A%09.arg(dates.join(%22’)%2C%20(‘%22))%0A%09.arg(codes.join(%22’)%2C%20(‘%22))%3B» message=»» highlight=»» provider=»manual»/]

Музыка♥, О том о сём

Не подпиской на Яндекс.Музыку единой

Эх, хорошо же мы жили с Яндекс.Музыкой и подпиской Плюс. И сказки нон-стоп для дочери, модная-молодежная музыка и подкасты для сына, старенькая и чуть новенькой музыки для нас с мужем. Аудио-произведения. Мой любимый Кино Поиск на Плюс подписке без рекламы! Яндекс.Такси со скидочкой. Да чего там только нет — очень удобно. И в свете недавних (кстати, уже почти 4 месяца уже), всем известных событий наши карты перестали приниматься платежной системой Яндекс.Музыки. Ни одна из валютных, ни рублевые. Эх. К своему удивлению узнала, что головное юридическое лицо компании находится в Нидерландах. Как я заблуждалась, оказывается, думая, что это чисто российская контора.

Однако без подписки Яндекс.Музыка осталась доступной в качестве радио на ПК.

Небольшой плей-листик песенок, запавших в душу (какие-то из них старые-добрые, какие-то для меня новенькие):

P!nk — U + Ur hand ❤️
Red — Who We Are
Thirty Seconds To Mars — A Modern Myth ❤️❤️❤️
Selena Gomes & The Scene — Who Says
Madilyn Bailey — Red Ribbon ❤️❤️❤️
Christina Perri — Human ❤️
Ashes Remain — Everything Good ❤️
Panic! At The Disco — House Of Memories (вам не кажется, что эта песенка стала «официальным» саундтреком к видосам про суд между Джонни Деппом и Эмбер Хёрд? 😆 Я последние заседания смотрела в прямом эфире в оригинале, прям интересно было. Никогда не следила за знаменитостями и вот опять 😆).
Stone Sour — Hesitate
Apocalyptica & Adam Gontier — I Don’t Care (ностальгия и любовь ❤️‍🔥❤️‍🔥❤️‍🔥❤️‍🔥)
и как же без Papa Roach — Scars ❤️.

PostgreSQL

PostgreSQL: Запрос на получение списка комментариев к колонкам таблицы

В сем посте будет освещен один из способов хранить человекочитаемые наименования колонок таблицы.

Организовать это можно через комментарий к полю. Задать его можно через интерфейс pgAdmin, в свойствах поля:

Или прописать в скрипте:

[pastacode lang=»sql» manual=»COMMENT%20ON%20COLUMN%20public.products.product_id%0A%20%20%20%20IS%20’%D0%9A%D0%BE%D0%BC%D0%BC%D0%B5%D0%BD%D1%82%D0%B0%D1%80%D0%B8%D0%B9’%3B» message=»Добавление комментария к колонке» highlight=»» provider=»manual»/]

Допустим, комментарии ко всем колонкам прописаны. Как их теперь получить запросом, чтоб красивенько вывести таблицу на форме?

[pastacode lang=»sql» manual=»select%20subq.attname%3A%3Atext%2C%20d.description%0A%09FROM%20%0A%09%09(SELECT%20c.relname%2C%20obj_description(c.oid)%20obj_description%2C%20a.attname%2C%20c.oid%2C%20a.attnum%0A%09%09FROM%20pg_class%20c%2C%20pg_attribute%20a%0A%09%09WHERE%20c.oid%20%3D%20a.attrelid%0A%09%09AND%20c.relname%20in%20(‘products’)%0A%09%09AND%20a.attnum%20%3E%200)%20subq%20%0A%09LEFT%20OUTER%20JOIN%20pg_description%20d%20ON%20(d.objsubid%20%3D%20subq.attnum%20AND%20d.objoid%20%3D%20subq.oid)%0A%09where%20subq.attname%20not%20ilike%20’%25pg.dropped%25’%3B» message=»Имена колонок таблицы products и комментарии к ним» highlight=»6″ provider=»manual»/]

В скобках в выделенной строке можно перечислить несколько таблиц через запятую. Условие после WHERE нужно для того, чтобы в результате запроса не фигурировали строки, оставшиеся после удаленных колонок таблицы. Вот такие:

 

Мысли вслух

Ветка сакуры упала на грудь…

Помните такой стишок: «Ветка сирени упала на грудь. Милый мой Коля (имя можно изменить), меня не забудь»? 😂

На новом участке посадим перед домом сакуру — она небольшая ростом, и электрическим проводам мешать не будет. А какая красота по весне…

А рядышком — магнолию — для еще пущей сказочности.

PostgreSQL, Базы данных

PostgreSQL: Результаты запроса в JSON

Мне бы хотелось рассказать очень кратенько про две часто мною используемые функции — json_agg и row_to_json.

Первая используется для получения результата выполнения SQL запроса как массив объектов JSON. Таким образом, одна запись складывается в один объект массива:

[pastacode lang=»sql» manual=»select%20json_agg(t)%20from%0A(select%20*%20from%20products)%20t» message=»json_agg» highlight=»» provider=»manual»/]

[pastacode lang=»sql» manual=»select%20json_agg(t)%20from%0A(select%20product_id%2C%20min_storage_temp%2C%20max_storage_temp%20%0A%20from%20products%20where%20coalesce(min_storage_temp%2C»)%20%3C%3E%20»)%20t» message=»Получить данные указанных полей» highlight=»json_agg» provider=»manual»/]

Результат такого запроса будет выглядеть похожим образом:

[
{
«product_id»:»aaa5f25c-5734-11ec-8cd2-00155d999336″,
«min_storage_temp»:»-4″,
«max_storage_temp»:»+6″
},
{
«product_id»:»1735acd6-fd3f-4146-8efc-718d7c55faaf»,
«min_storage_temp»:»0″,
«max_storage_temp»:»+6″
},
{
«product_id»:»c89cc4d8-c531-11eb-958d-772a230b0b21″,
«min_storage_temp»:»-4″,
«max_storage_temp»:»+6″
}
]

Для получения результата запроса, возвращающего одну строку, в качестве объекта JSON можно воспользоваться функцией row_to_json:

[pastacode lang=»sql» manual=»select%20row_to_json(t)%20FROM%20%0A(select%20*%20from%20products%20where%20product_id%20%3D%20’123′)%20t» message=»row_to_json» highlight=»» provider=»manual»/]

[
{
«product_id»:»c89cc4d8-c531-11eb-958d-772a230b0b21″,
«min_storage_temp»:»-4″,
«max_storage_temp»:»+6″
}
]

"Нет" выгоранию, Музыка♥

Love sux but music doesn’t

Мой недобложик жил без меня почти два года, юху! Какой самостоятельный вырос 😅

Мне и сегодня не особо хочется писать какую-то полезность, а просто решила поделиться парочкой песен, которые в данный момент сопровождают меня с моей работой.

Включила вот себе послушать свеженький альбом Avril Lavigne — Love Sux. Люблю ее еще со школы. Ну а сейчас я уже давно не в школе и даже не в университете 😁

Такие вот песенки меня зацепили:

Avalanche

Deja vu

Dare to love me

Слушала через Яндекс Музыку, да, удобно иметь подписку 😇

PostgreSQL, SQL

Прервать соединения IP с базой данных в PostgreSQL

В сообщении Принудительное удаление базы данных в PostgreSQL я писала о том, как принудительно закрыть все соединения с базой данных. Это удобно, когда есть «висящие» соединения, которые никак не хотят завершаться. Или нужно срочно удалить базу данных и нет времени выяснять, кто забыл от нее отключиться 🙂 Сегодня я хочу показать, как прервать соединения IP с базой данных.

Сначала выведем количество подключений с каждого IP к базе данных: Продолжить чтение «Прервать соединения IP с базой данных в PostgreSQL»

Qt

Динамическое создание виджетов Qt. QGridLayout

В этом сообщении хотелось бы немного отредактировать предыдущий подобный пост Qt. Динамическое создание виджетов на форме.

Задача такая же: Есть объект QGroupBox groupBox, в него надо запихнуть виджеты (labels & lineEdits) в QGridLayout лэйауте. Данные берутся из запроса (он представлен в сообщении по ссылке выше).

Более универсальный способ будет выглядеть так:

[pastacode lang=»cpp» manual=»if(ui-%3EgroupBox-%3Elayout())%20%7B%0A%09QLayoutItem%20*child%3B%0A%20%20%20%20while%20((child%20%3D%20ui-%3EgroupBox-%3Elayout()-%3EtakeAt(0))%20!%3D%200)%20%7B%0A%20%20%20%20%09delete%20child-%3Ewidget()%3B%0A%20%20%20%20%20%20%20%20delete%20child%3B%0A%20%20%20%20%7D%0A%20%20%20%20delete%20ui-%3EgroupBox-%3Elayout()%3B%0A%7D%0AQGridLayout%20*layout%20%3D%20new%20QGridLayout(ui-%3EgroupBox)%3B%0Aui-%3EgroupBox-%3EsetLayout(layout)%3B%0Aint%20pos%20%3D%200%3B%20%2F%2F%D1%83%D1%81%D1%82%D0%B0%D0%BD%D0%B0%D0%B2%D0%BB%D0%B8%D0%B2%D0%B0%D0%B5%D0%BC%20%D0%BF%D0%BE%D0%B7%D0%B8%D1%86%D0%B8%D1%8E%20%D1%80%D0%B0%D1%81%D0%BF%D0%BE%D0%BB%D0%BE%D0%B6%D0%B5%D0%BD%D0%B8%D1%8F%20%D0%B2%D0%B8%D0%B4%D0%B6%D0%B5%D1%82%D0%BE%D0%B2%20%D0%B2%200%0Aint%20colsCount%20%3D%202%3B%20%2F%2F%D0%9A%D0%BE%D0%BB%D0%B8%D1%87%D0%B5%D1%81%D1%82%D0%B2%D0%BE%20%D0%BA%D0%BE%D0%BB%D0%BE%D0%BD%D0%BE%D0%BA%20%D0%B2%20%D0%BB%D1%8D%D0%B9%D0%B0%D1%83%D1%82%D0%B5%0Awhile%20(query.next())%20%7B%20%2F%2F%D0%B1%D0%B5%D1%80%D0%B5%D0%BC%20%D1%80%D0%B5%D0%B7%D1%83%D0%BB%D1%8C%D1%82%D0%B0%D1%82%D1%8B%20%D0%B7%D0%B0%D0%BF%D1%80%D0%BE%D1%81%D0%B0%0A%09QLabel%20*newLabel%20%3D%20new%20QLabel(this)%3B%0A%20%20%20%20newLabel-%3EsetText(query.value(%22caption%22).toString()%2B%22%3A%22)%3B%0A%20%20%20%20newLabel-%3EsetObjectName(%22label_%22%2Bquery.value(%22name_%22).toString())%3B%0A%20%20%20%20layout-%3EaddWidget(newLabel%2Cpos%2FcolsCount%2Cpos%25colsCount)%3B%0A%20%20%20%20pos%2B%2B%3B%20%2F%2F%D1%81%D0%B4%D0%B2%D0%B8%D0%B3%D0%B0%D0%B5%D0%BC%D1%81%D1%8F%0A%0A%20%20%20%20QLineEdit%20*newEdit%20%3D%20new%20QLineEdit(this)%3B%0A%20%20%20%20newEdit-%3EsetObjectName(query.value(%22name_%22).toString())%3B%0A%20%20%20%20layout-%3EaddWidget(newEdit%2Cpos%2FcolsCount%2Cpos%25colsCount)%3B%0A%20%20%20%20lineEdits.push_back(newEdit)%3B%0A%20%20%20%20pos%2B%2B%3B%0A%7D» message=»» highlight=»» provider=»manual»/]

В данном случае мне необходимо было разместить объекты в бокс с двумя колонками. Поэтому colsCount = 2. Если их должно быть больше, то значение переменной нужно поменять. Очищение лэйаута позволяет при изменении результатов запроса все очистить и нарисовать все в соответствии с актуальными данными.