Как программы на Си взаимодействуют с сервером БД PostgreSQL::Журнал СА 10.2005
www.samag.ru
     
Поиск   
              
 www.samag.ru    Web  0 товаров , сумма 0 руб.
E-mail
Пароль  
 Запомнить меня
Регистрация | Забыли пароль?
Журнал "Системный администратор"
Журнал «БИТ»
Подписка
Архив номеров
Где купить
Наука и технологии
Авторам
Рекламодателям
Контакты
   

  Опросы
  Статьи

Событие  

В банке рассола ждет сисадмина с полей фрактал-кукумбер

Читайте впечатления о слете ДСА 2024, рассказанные волонтером и участником слета

 Читать далее...

Организация бесперебойной работы  

Бесперебойная работа ИТ-инфраструктуры в режиме 24/7 Как обеспечить ее в нынешних условиях?

Год назад ИТ-компания «Крок» провела исследование «Ключевые тренды сервисного рынка 2023». Результаты

 Читать далее...

Книжная полка  

Читайте и познавайте мир технологий!

Издательство «БХВ» продолжает радовать выпуском интересных и полезных, к тому же прекрасно

 Читать далее...

СУБД PostgreSQL  

СУБД Postgres Pro

Сертификация по новым требованиям ФСТЭК и роль администратора без доступа к данным

 Читать далее...

Критическая инфраструктура  

КИИ для оператора связи. Готовы ли компании к повышению уровня кибербезопасности?

Похоже, что провайдеры и операторы связи начали забывать о требованиях законодательства

 Читать далее...

Архитектура ПО  

Архитектурные метрики. Качество архитектуры и способность системы к эволюционированию

Обычно соответствие программного продукта требованиям мы проверяем через скоуп вполне себе понятных

 Читать далее...

Как хорошо вы это знаете  

Что вам известно о разработках компании ARinteg?

Компания ARinteg (ООО «АРинтег») – системный интегратор на российском рынке ИБ –

 Читать далее...

Графические редакторы  

Рисование абстрактных гор в стиле Paper Cut

Векторный графический редактор Inkscape – яркий представитель той прослойки open source, с

 Читать далее...

День сисадмина  

Учите матчасть! Или как стать системным администратором

Лето – время не только отпусков, но и хорошая возможность определиться с профессией

 Читать далее...

День сисадмина  

Живой айтишник – это всегда движение. Остановка смерти подобна

Наши авторы рассказывают о своем опыте и дают советы начинающим системным администраторам.

 Читать далее...

Виртуализация  

Рынок решений для виртуализации

По данным «Обзора российского рынка инфраструктурного ПО и перспектив его развития», сделанного

 Читать далее...

Книжная полка  

Как стать креативным и востребованным

Издательский дом «Питер» предлагает новинки компьютерной литературы, а также книги по бизнесу

 Читать далее...

Книжная полка  

От создания сайтов до разработки и реализации API

В издательстве «БХВ» недавно вышли книги, которые будут интересны системным администраторам, создателям

 Читать далее...

Разбор полетов  

Ошибок опыт трудный

Как часто мы легко повторяем, что не надо бояться совершать ошибки, мол,

 Читать далее...

1001 и 1 книга  
19.03.2018г.
Просмотров: 6143
Комментарии: 0
Машинное обучение с использованием библиотеки Н2О

 Читать далее...

12.03.2018г.
Просмотров: 6856
Комментарии: 0
Особенности киберпреступлений в России: инструменты нападения и защита информации

 Читать далее...

12.03.2018г.
Просмотров: 4139
Комментарии: 0
Глубокое обучение с точки зрения практика

 Читать далее...

12.03.2018г.
Просмотров: 2978
Комментарии: 0
Изучаем pandas

 Читать далее...

12.03.2018г.
Просмотров: 3781
Комментарии: 0
Программирование на языке Rust (Цветное издание)

 Читать далее...

19.12.2017г.
Просмотров: 3789
Комментарии: 0
Глубокое обучение

 Читать далее...

19.12.2017г.
Просмотров: 6283
Комментарии: 0
Анализ социальных медиа на Python

 Читать далее...

19.12.2017г.
Просмотров: 3134
Комментарии: 0
Основы блокчейна

 Читать далее...

19.12.2017г.
Просмотров: 3434
Комментарии: 0
Java 9. Полный обзор нововведений

 Читать далее...

16.02.2017г.
Просмотров: 7246
Комментарии: 0
Опоздавших не бывает, или книга о стеке

 Читать далее...

17.05.2016г.
Просмотров: 10616
Комментарии: 0
Теория вычислений для программистов

 Читать далее...

30.03.2015г.
Просмотров: 12336
Комментарии: 0
От математики к обобщенному программированию

 Читать далее...

18.02.2014г.
Просмотров: 13969
Комментарии: 0
Рецензия на книгу «Читаем Тьюринга»

 Читать далее...

13.02.2014г.
Просмотров: 9100
Комментарии: 0
Читайте, размышляйте, действуйте

 Читать далее...

12.02.2014г.
Просмотров: 7053
Комментарии: 0
Рисуем наши мысли

 Читать далее...

10.02.2014г.
Просмотров: 5362
Комментарии: 3
Страна в цифрах

 Читать далее...

18.12.2013г.
Просмотров: 4594
Комментарии: 0
Большие данные меняют нашу жизнь

 Читать далее...

18.12.2013г.
Просмотров: 3402
Комментарии: 0
Компьютерные технологии – корень зла для точки роста

 Читать далее...

04.12.2013г.
Просмотров: 3129
Комментарии: 0
Паутина в облаках

 Читать далее...

03.12.2013г.
Просмотров: 3379
Комментарии: 0
Рецензия на книгу «MongoDB в действии»

 Читать далее...

02.12.2013г.
Просмотров: 3000
Комментарии: 0
Не думай о минутах свысока

 Читать далее...

Друзья сайта  

 Как программы на Си взаимодействуют с сервером БД PostgreSQL

Архив номеров / 2005 / Выпуск №10 (35) / Как программы на Си взаимодействуют с сервером БД PostgreSQL

Рубрика: Программирование /  Анализ данных

ВЛАДИМИР МЕШКОВ

Как программы на Си взаимодействуют с сервером БД PostgreSQL

PostgreSQL является эффективным средством для хранения и обработки информации. Разработчики этой СУБД предоставили интерфейсы для многих языков программирования. Поддержка таких языков, как Perl, PHP, Python, обеспечивает широкое применение PostgreSQL в области веб-программирования. Язык системного программирования Cи позволит использовать эту СУБД, когда необходимо добиться от приложения максимального быстродействия.

С егодня мы рассмотрим пример взаимодействия программы на языке Си и сервера баз данных PostgreSQL c использованием библиотеки libpq. В случае отсутствия опыта работы с СУБД PostgreSQL рекомендую начать изучение этой темы со статьи Сергея Супрунова [1].

Обзор библиотеки libpq

Библиотека libpq является программным интерфейсом, обеспечивающим взаимодействие программы, составленной на языке Си, с сервером баз данных PostgreSQL. Эта библиотека содержит набор функций, позволяющих клиентской программе обмениваться информацией с базой данных. Библиотека входит в состав дистрибутива СУБД PostgreSQL.

Для выполнения информационного обмена клиентская программа вначале должна подключиться к базе данных. Для связи с сервером баз данных используется механизм сокетов, при этом если клиент и сервер расположены на одной локальной машине, используется сокет домена AF_UNIX, в случае расположения на удаленных машинах – сокет домена AF_INET. Тип домена указывается в параметрах системного вызова socket.

Для хранения адресной информации сокет домена AF_UNIX использует структурный тип sockaddr:

struct sockaddr {

    sa_family_t sa_family;

    char sa_data[14];

}

Поле sa_family определяет тип домена, к которому принадлежит сокет (AF_UNIX в нашем случае), массив sa_data содержит путь к файлу, который описывает сокет.

Таким образом, сокет домена AF_UNIX представляет собой специальный файл. Сервер PostgreSQL после запуска по умолчанию создает в каталоге /tmp сокет домена AF_UNIX в виде файла .s.PGSQL.5432, посмотреть на который можно при помощи команды ls -la. Среди прочих файлов будет запись следующего вида:

srwxrwxrwt 1 pgsql users 0 Okt 4 10:37 .s.PGSQL.5432

Литера «s» перед правами доступа означает, что данный файл является сокетом.

Команда netstat -a позволяет нам убедиться, что файл /tmp/.s.PGSQL.5432 входит в список активных сокетов домена AF_UNIX. Введем эту команду и увидим запись примерно такого вида:

unix 2 [ ACC ] STREAM LISTENING 20636071 /tmp/.s.PGSQL.5432

Для лучшего понимания рассмотрим тестовый пример взаимодействия процессов через сокет домена AF_UNIX. Ниже представлены два листинга – серверного и клиентского процесса. В целях экономии места обработка ошибок пропущена.

Листинг 1. Серверный процесс

#include <sys/types.h>

#include <sys/socket.h>

int main()

{

    int sock, newsock;

    struct sockaddr saddr;

    char c;

    static char rc = 1;

/* Создаем сокет домена AF_UNIX */

    sock = socket(AF_UNIX, SOCK_STREAM, 0);

/* Заполняем адресную структуру saddr */

    memset((void *)&saddr, 0, sizeof(saddr));

    saddr.sa_family = AF_UNIX; /* тип домена */

    /* путь к файлу */

    memcpy(saddr.sa_data, "/tmp/.sock.new", 14);

    bind(sock, (struct sockaddr *)&saddr, sizeof(struct sockaddr));

    listen(sock, 1);

    for(;;) {

    newsock = accept(sock, NULL, NULL);

    if(fork() == 0) {

        while(recv(newsock, &c, 1, 0) > 0) {

           send(newsock, &i, 1, 0);

           i++;

        }

        close(newsock);

        exit(0);

    }

    close(newsock);

    }

    return 0;

}

Листинг 2. Клиентский процесс

int main()

{

    int sock;

    struct sockaddr saddr;

    char c, rc;

    sock = socket(AF_UNIX, SOCK_STREAM, 0);

    memset((void *)&saddr, 0, sizeof(saddr));

    saddr.sa_family = AF_UNIX;

    memcpy(saddr.sa_data, "/tmp/.sock.new", 14);

    connect(sock, (struct sockaddr *)&saddr, sizeof(struct sockaddr));

    for(;;) {

    c = getchar();

    send(sock, &c, 1, 0);

    if(recv(sock, &rc, 1, 0) > 0) printf("From server: %d\n", rc);

    else {

        close(sock);

        exit(0);

    }

    }

    return 0;

}

Запустим процессы в разных терминалах. Сервер после запуска создаст в каталоге /tmp файл .sock.new. Через этот файл будет осуществляться взаимодействие между клиентом и сервером: клиент будет отправлять серверу символы, вводимые пользователем, а сервер будет возвращать числовые значения, каждый раз увеличивая их на 1. При этом на каждый введенный символ сервер отвечает двумя. Тут все правильно, т.к. серверу передается еще и символ перевода строки « », вот он на него и реагирует.

После остановки сервера сигналом SIGINT (комбинация клавиш ) файл .sock.new останется в каталоге /tmp. Его необходимо удалить вручную, или переопределить обработчик сигнала SIGINT для закрытия сокета и удаления файла .sock.new, иначе при повторном запуске сервера системный вызов bind не сможет привязать адресную структуру к сокету, сообщая нам, что «Address already in use (адрес уже используется)».

Вернемся к рассмотрению темы статьи. Итак, для подключения к серверу баз данных библиотека предоставляет несколько функций, но мы рассмотрим одну – PQsetdbLogin. Прототип этой функции имеет следующий вид:

PGconn *PQsetdbLogin(const char *pghost,

           const char *pgport, const char *pgoptions,

           const char *pgtty, const char *dbName,

           const char *login, const char *pwd);

Эта функция устанавливает новое соединение с базой данных, которое описывается при помощи объекта типа PGconn. Параметрами функции являются:

  • pghost – если сервер и клиент расположены на локальном хосте, этот параметр принимает значение NULL, и взаимодействие с сервером осуществляется через сокет домена AF_UNIX, по умолчанию расположенный в каталоге /tmp. При работе через сеть это поле содержит имя или IP-адрес хоста, на котором находится сервер баз данных;
  • pgport – номер порта (NULL для локального хоста);
  • pgoptions – дополнительные опции, посылаемые серверу для трассировки/отладки соединения;
  • pgtty – терминал или файл для вывода отладочной информации;
  • dbName – имя базы данных;
  • login, pwd – имя пользователя и пароль доступа к базе данных.

Функция PQsetdbLogin всегда возвращает указатель на объект типа PGconn, независимо от того, успешно было установлено соединение или нет. Проверку состояния соединения выполняет функция PQstatus. Объект типа PGconn передается этой функции в качестве параметра, возвращаемое функцией значение характеризует состояние соединения:

  • CONNECTION_BAD – не удалось установить соединение с базой данных;
  • CONNECTION_OK – соединение с базой данных успешно установлено.

Эти значения определены в заголовочном файле libpqfe.h.

После установления соединения клиентская программа может приступить к обмену информацией с базой данных. Для этой цели библиотека libpq предоставляет функцию PQexec, прототип которой имеет следующий вид:

PGresult *PQexec(PGconn *conn, const char *query);

Параметрами функции PQexec являются указатель на объект типа PGconn (результат работы функции PQsetdbLogin) и строка, содержащая запрос к базе данных. Отправив запрос, функция ожидает ответ от базы и сохраняет в структуре типа Pgresult статус запроса и данные, полученные от базы. Для обработки статуса запроса к базе данных используется функция PGresultStatus.

ExecStatusType PQresultStatus(const PGresult *res);

Функция PQresultStatus может возвращать следующие значения, определенные в файле libpq-fe.h:

  • PGRES_EMPTY_QUERY – серверу отправлена пустая строка запроса;
  • PGRES_COMMAND_OK – запрос, не требующий возврата данных из базы, выполнен успешно;
  • PGRES_TUPLES_OK – успешное чтение данных из базы;
  • PGRES_FATAL_ERROR – при обращении к базе данных произошла критическая ошибка.

Если статус запроса равен PGRES_TUPLES_OK, структура PGresult будет содержать данные, полученные от базы. Данные представляют собой последовательность (кортеж) строк таблицы, и каждая строка состоит из нескольких ячеек. Выполнить выборку содержимого определенной ячейки можно при помощи функции PQgetvalue:

char* PQgetvalue(const PGresult *res, int tup_num, int field_num);

Здесь tup_num – это номер строки таблицы, а field_num – номер ячейки в строке, из которой считываются данные. Для определения числа строк, считанных из таблицы, используется функция PQntuples (tuple в переводе с английского означает кортеж, последовательность):

int PQntuples(const PGresult *res);

Функция PQnfields вернет число ячеек в одной строке таблицы:

int PQnfields(const PGresult *res);

По окончании информационного обмена с базой данных клиентская программа должна при помощи функции PQclear освободить структуру PGresult, содержащую результаты запроса, и отключиться от базы, вызвав функцию PQfinish:

void PQclear(PQresult *res);

void PQfinish(PGconn *conn)

Пример использования библиотеки libpq

Рассмотрим простой пример использования библиотеки. Предположим, что у нас имеется каталог, содержащий файлы различных типов (в том числе и специальные). Мы составим две программы на языке Си: первая программа будет выполнять обход указанного ей каталога, считывать и заносить в базу данных имена и размеры всех регулярных файлов из этого каталога и всех вложенных каталогов. Вторая программа будет считывать информацию об этих файлах из базы данных и выводить ее на экран.

Для выполнения этой задачи устанавливаем на локальную машину СУБД PostgreSQL (см. [1]). После инициализации базы данных создаем нового пользователя my_user и новую базу my_database:

createuser -a -d my_user -E -P

createdb -O my_user my_database

Для доступа к базе данных пользователь my_user должен указать пароль. Сам пароль будет храниться в зашифрованном виде, в конфигурационном файле pg_hba.conf меняем значение поля METHOD c trust на md5.

Далее, подключаемся к базе данных my_database и создаем в ней таблицу, состоящую из двух полей: поля fname типа char(100) для хранения имен файлов и поля fsize типа int для хранения размеров файлов.

Заполнение базы данных информацией

Первый этап разработки – программа для заполнения базы данных информацией. Назовем ее insert_data. Входные параметры – имя базы данных, имя таблицы в базе и имя каталога, из которого будут считываться данные о файлах – передаются в параметрах командной строки:

# ./insert_data -d [имя базы данных] -t [имя таблицы] -p [имя каталога]

Определим переменные для хранения имен базы данных, таблицы и каталога для чтения:

unsigned char *dbname = NULL; /* имя базы данных */

unsigned char *table = NULL; /* имя таблицы */

unsigned char *pathname = NULL; /* каталог, из которого считываются данные */

Проверяем число переданных аргументов. Их должно быть 7:

if(argc != 7) usage();

Если количество переданных аргументов не соответствует указанному значению, при помощи функции usage() отобразим формат вызова нашей программы:

void usage()

{

    fprintf(stderr, "Usage: insert_data -d [имя базы данных] -t [имя таблицы] -p [исходный каталог] ");

    exit(0);

}

Считываем параметры командной строки. Разбор командной строки выполним при помощи функции getopt:

while((int c = getopt(argc, argv, "d:t:p:")) != EOF) {

    switch(c) {

        case "d":

           /* имя базы данных */

           dbname = (unsigned char *)optarg;

           break;

        case "t":

           /* имя таблицы */

           table = (unsigned char *)optarg;

           break;

        case "p":

           /* имя каталога */

           pathname = (unsigned char *)optarg;

           break;

        /* ошибка в параметрах */

        case "?":

        default:

           usage();

    }

}

Считываем имя пользователя и пароль для доступа к базе данных:

unsigned char user[80]; /* имя пользователя */

unsigned char pwd[80]; /* пароль доступа к базе данных */

memset(user, 0, sizeof(user));

printf("Login: ");

scanf("%s", user);

memset(pwd, 0, 80);

printf("Password: ");

Перед тем как ввести пароль, из соображений безопасности отключим отображение вводимых символов на экране, изменив настройки управляющего терминала. Для управления свойствами терминала используются функции tcgetattr и tcsetattr:

#include

int tcgetattr(int ttyfd, struct termios *told);

int tcsetattr(int ttyfd, int actions, const struct termios *tnew);

Функция tcgetattr сохраняет текущее состояние терминала в структуре told типа termios. Параметр ttyfd должен быть дескриптором файла, описывающего терминал. Для получения доступа к своему управляющему терминалу процесс может использовать имя файла /dev/tty, которое всегда интерпретируется как текущий управляющий терминал или стандартный вывод с дескриптором 0. Вызов функции tcsetattr установит новое состояние терминала, заданное структурой tnew, а параметр actions определяет, когда и как будут установлены новые атрибуты терминала:

  • TCSNOW – немедленное выполнение изменений;
  • TCSADRAIN – перед установкой новых параметров ожидается опустошение очереди вывода;
  • TCSAFLUSH – ожидается опустошение очереди вывода, затем также очищается очередь ввода.

Для доступа к управляющему терминалу открываем соответствующий файл устройства:

int ttyfd = open("/dev/tty", O_RDWR);

Далее считываем текущее состояние терминала в структуру struct termios t, снимаем флаг отображения символов ECHO в поле c_lflag и устанавливаем новое состояние терминала:

tcgetattr(ttyfd, &t); /* сохраняем настройки терминала */

t.c_lflag &= ~ECHO; /* сбрасываем флаг ECHO */

tcsetattr(ttyfd, TCSANOW, &t); /* устанавливаем новое состояние терминала */

Наличие флага TCSANOW требует немедленного выполнения изменений. Подробности управления терминалом смотрите в man termios.

После этих действий вводим пароль для доступа к базе данных:

scanf("%s", pwd);

Вернем настройки терминала в исходное состояние – включим отображение вводимых символов на экране:

t.c_lflag |= ECHO; /* устанавливаем флаг ECHO */

tcsetattr(ttyfd, TCSANOW, &t);

close(ttyfd);

Подключаемся к базе данных, вызвав функцию PQsetdbLogin. Эта функция вернет указатель на объект типа PGconn, независимо от того, успешно было установлено соединение или нет:

PGconn *conn = PQsetdbLogin(NULL, NULL, NULL, NULL, dbname, user, pwd);

Первые четыре параметра функции PQsetdbLogin установлены в NULL, так как сервер баз данных находится на локальной машине, и дополнительных опций мы ему не передаем. Если сервер расположен на удаленной машине, то вызов функции PQsetdbLogin примет следующий вид:

PQsetdbLogin("192.168.1.1", "5432", NULL, NULL, dbname, user, pwd),

где 192.168.1.1 – IP адрес хоста, на котором установлен сервер баз данных, 5432 – порт, который слушает база.

Анализируем состояние соединения и в случае ошибки завершаем выполнение программы:

if(PQstatus(conn) == CONNECTION_BAD) {

    fprintf(stderr, "Connection to database failed. ");

    fprintf(stderr, "%s", PQerrorMessage(conn));

    exit(1);

}

При успешном установлении соединения с базой данных считываем необходимую нам информацию из указанного каталога. Считывание выполняет рекурсивная функция list_dir(), в параметрах которой мы передаем указатель на объект типа PGconn, имя таблицы в базе данных и имя каталога:

int list_dir(PGconn *conn, unsigned char *table,

    unsigned char *pathname)

{

    struct dirent *d;

    struct stat s;

    DIR *dp;

    PGresult *res; /* результат обращения к базе данных */

    unsigned char full_path[256]; /* абсолютное путевое имя файла */

    unsigned char query[QUERY_LEN]; /* строка запроса к базе данных */

    unsigned char escape_string[80]; /* данные, передаваемые базе */

    /* Открываем каталог */

    if((dp = opendir(pathname)) == NULL) {

           perror("opendir");

           return -1;

    }

    /* Пропускаем родительский и текущий каталоги */

    d = readdir(dp); // "."

    d = readdir(dp); // ".."

    /* Цикл чтения записей каталог */

    while(d = readdir(dp)) {

    /* Формируем абсолютное путевое имя файла и получаем информацию о нем */

           memset(full_path, 0, 256);

           sprintf(full_path, "%s/%s", pathname, d->d_name);

           stat(full_path, &s);

    /* Если это каталог – выполняем рекурсивный вызов функции */

           if(S_ISDIR(s.st_mode)) list_dir(conn, table, full_path);

    /* Добавляем в базу информацию о файле, при этом преобразуем путевое имя файла при помощи */

    /* функции PQescapeString */

           memset(escape_string, 0, 80);

           PQescapeString(escape_string, full_path, 80);

    /* Формируем запрос и отправляем его базе данных */

           memset(query, 0, QUERY_LEN);

           sprintf(query, "INSERT INTO %s values('%s','%u')", table, full_path, s.st_size);

           res = PQexec(conn, query);

    /* Проверяем статус запроса. Он должен быть равен PGRES_COMMAND_OK, т.к. данных от базы мы не получаем */

           if(PQresultStatus(res) !=

                 PGRES_COMMAND_OK) {

                 fprintf(stderr, "INSERT query failed.\n");

                 break;

           }

    }

    closedir(dp);

    PQclear(res);

    return 0;

}

Если функцию PQfinish не вызвать, то в данном случае ничего страшного не произойдет, потому что процесс завершает выполнение. Ядро удаляет процесс из общего списка, уничтожая все служебные структуры, описывающие файлы и сокеты, с которыми процесс работал, а значение дескриптора сокета (так же как и файла) имеет смысл только в контексте процесса, так как по сути это индекс в массиве структур.

Если вместо функции отключения от базы перед выходом из программы организовать бесконечный цикл и ввести в соседнем терминале команду netstat, то можно увидеть, что процесс установил соединение с базой данных через сокет домена AF_UNIX. При остановке процесса сигналом SIGINT (комбинация клавиш Ctrl-C) это соединение исчезает, даже если мы не вызываем функцию PQfinish. Другое дело, если процесс не закрыл соединение и продолжает функционировать (например, если это фоновый процесс). Тогда возможна ситуация несанкционированного использования уже установленного соединения (сокет не закрыт) для доступа к базе данных, и при этом необязательно знать пароль. Поэтому закрывать соединение надо явно.

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

#include

int ftw(const char *path, int(* func)(), int depth);

Первый параметр path определяет имя каталога, с которого должен начаться рекурсивный обход дерева. Параметр depth управляет числом используемых функцией ftw различных дескрипторов файлов. Чем больше значение depth, тем меньше будет случаев повторного открытия каталогов, что сократит общее время обработки вызова. Второй параметр func – это определенная пользователем функция, вызываемая для каждого файла или каталога, найденного в поддереве каталога path. При каждом вызове функции func будут передаваться три аргумента: заканчивающаяся нулевым символом строка с именем объекта, указатель на структуру stat с данными об объекте и целочисленный код. Функция func, следовательно, должна быть построена следующим образом:

int func(const char *name, const struct stat *sptr, int type)

{

    /* Тело функции */

}

Целочисленный аргумент type может принимать одно из нескольких возможных значений, определенных в заголовочном файле и описывающих тип встретившегося объекта:

  • FTW_F – объект является файлом;
  • FTW_D – объект является каталогом;
  • FTW_DNR – объект является каталогом, который нельзя прочесть;
  • FTW_SL – объект является символьной ссылкой;
  • FTW_NS – объект не является символьной ссылкой, для него нельзя успешно выполнить вызов stat.

Работа вызова будет продолжаться до тех пор, пока не будет завершен обход дерева или не возникнет ошибка внутри функции ftw. Обход также закончится, если определенная пользователем функция возвратит ненулевое значение. Тогда функция ftw прекратит работу и вернет значение, возвращенное функцией пользователя. Ошибки внутри функции ftw приведут к возврату значения -1, тогда в переменной errno будет выставлен соответствующий код ошибки.

Вызовем в нашей программе вместо рекурсивной функции list_dir функцию ftw:

ftw(pathname, list_dir1, 1);

Функция list_dir1 передает базе данных информацию о каждом регулярном файле:

int list_dir1(const char *name, const struct stat *s,

    int type)

{

    PGresult *res;

    /* строка запроса к базе данных */

    unsigned char query[QUERY_LEN];

    unsigned char escape_string[80];

    /* Возвращаемся, если вызов stat завершился неудачно */

    if(type == FTW_NS) return 0;

    /* Если объект является регулярным файлом, добавляем информацию о нем в базу */

    if((type == FTW_F) && S_ISREG(s->st_mode)) {

           memset(escape_string, 0, sizeof(escape_string));

           PQescapeString(escape_string, name, sizeof(escape_string));

           memset(query, 0, QUERY_LEN);

           sprintf(query, "INSERT INTO %s values('%s','%u')", table, name, s->st_size);

           res = PQexec(conn, query);

    /* Проверяем статус запроса */

           if(PQresultStatus(res) !=PGRES_COMMAND_OK) {

                 fprintf(stderr, "INSERT query failed.\n");

                 return -1;

           }

           PQclear(res);

    }

    return 0;

}

Для получения исполняемого модуля введем команду:

# gcc -o insert_data insert_data.c -lpq

Чтение информации из базы данных

Второй этап разработки – программа для чтения информации из базы данных. Строится она по такому же принципу, как и предыдущая: в параметрах командной строки передаются имя базы данных и таблицы, выполняется ввод имени и пароля, при этом отображение вводимых символов отключается. После этого подключаемся к базе данных:

# gcc -o insert_data insert_data.c -lpq

conn = PQsetdbLogin(NULL, NULL, NULL, NULL, dbname, user, pwd);

if(PQstatus(conn) == CONNECTION_BAD) {

    fprintf(stderr, "Connection to database failed.\n");

    fprintf(stderr, "%s", PQerrorMessage(conn));

    exit(1);

}

Формируем и отправляем запрос к базе для выборки всех полей из таблицы:

memset(query, 0, QUERY_LEN);

sprintf(query, "SELECT * FROM %s", table);

res = PQexec(conn, query);

В случае успешного чтения данных из базы статус запроса должен быть равен PGRES_TUPLES_OK. Проверяем это:

if(PQresultStatus(res) != PGRES_TUPLES_OK) {

    fprintf(stderr, "SELECT query failed. ");

    goto out;

}

Отображаем результаты чтения:

for(i = 0; i < PQntuples(res); i++) {

    for(n = 0; n < PQnfields(res); n++) printf("%-20s", PQgetvalue(res, i, n));

    printf(" ");

}

Функция PQntuples вернет число прочитанных из таблицы строк, а функция PQnfields – число ячеек в одной строке.

Работоспособность программ была проверена для ОС Linux Slackware 10.2 и FreeBSD 5.2, использовался сервер баз данных PostgreSQL 8.0.3.

Литература:

  1. Супрунов С. PostgreSQL: первые шаги. – Журнал «Системный администратор», №7, 2004 г. – 26-33 с (http://www.samag.ru/cgi-bin/go.pl?q=articles;n=07.2004;a=06).
  2. PostgreSQL 7.3.2 Programmer’s Guide by The PostgreSQL Global Development Group.
  3. Кейт Хэвиленд, Дайна Грей, Бен Салама. Системное программирование в UNIX. Руководство программиста по разработке ПО = Unix System Programming. A programmer’s guide to software development: Пер. с англ. – М., ДМК Пресс, 2000 г. – 368 с., ил.

Комментарии отсутствуют

Добавить комментарий

Комментарии могут оставлять только зарегистрированные пользователи

               Copyright © Системный администратор

Яндекс.Метрика
Tel.: (499) 277-12-45
E-mail: sa@samag.ru