/* * Copyright (c) 2008 Allied Telesis Labs NZ * * All rights reserved. * * Author: Angus Salkeld * * This software licensed under BSD license, the text of which follows: * * Redistribution and use in source and binary forms, with or without * modification, are permitted provided that the following conditions are met: * * - Redistributions of source code must retain the above copyright notice, * this list of conditions and the following disclaimer. * - Redistributions in binary form must reproduce the above copyright notice, * this list of conditions and the following disclaimer in the documentation * and/or other materials provided with the distribution. * - Neither the name of the MontaVista Software, Inc. nor the names of its * contributors may be used to endorse or promote products derived from this * software without specific prior written permission. * * THIS SOFTWARE IS PROVIDED BY THE COPYRIGHT HOLDERS AND CONTRIBUTORS "AS IS" * AND ANY EXPRESS OR IMPLIED WARRANTIES, INCLUDING, BUT NOT LIMITED TO, THE * IMPLIED WARRANTIES OF MERCHANTABILITY AND FITNESS FOR A PARTICULAR PURPOSE * ARE DISCLAIMED. IN NO EVENT SHALL THE COPYRIGHT OWNER OR CONTRIBUTORS BE * LIABLE FOR ANY DIRECT, INDIRECT, INCIDENTAL, SPECIAL, EXEMPLARY, OR * CONSEQUENTIAL DAMAGES (INCLUDING, BUT NOT LIMITED TO, PROCUREMENT OF * SUBSTITUTE GOODS OR SERVICES; LOSS OF USE, DATA, OR PROFITS; OR BUSINESS * INTERRUPTION) HOWEVER CAUSED AND ON ANY THEORY OF LIABILITY, WHETHER IN * CONTRACT, STRICT LIABILITY, OR TORT (INCLUDING NEGLIGENCE OR OTHERWISE) * ARISING IN ANY WAY OUT OF THE USE OF THIS SOFTWARE, EVEN IF ADVISED OF * THE POSSIBILITY OF SUCH DAMAGE. */ #include #include #include #include #include #include #include #include #include #include #include "../lib/util.h" #include #include #define SEPERATOR '.' #define SEPERATOR_STR "." #define OBJ_NAME_SIZE 512 typedef enum { ACTION_READ, ACTION_WRITE, ACTION_CREATE, ACTION_CREATE_KEY, ACTION_DELETE, ACTION_PRINT_ALL, ACTION_PRINT_DEFAULT, ACTION_TRACK, } action_types_t; typedef enum { FIND_OBJECT_ONLY, FIND_OBJECT_OR_KEY, FIND_KEY_ONLY } find_object_of_type_t; static void tail_key_changed(confdb_handle_t handle, confdb_change_type_t change_type, hdb_handle_t parent_object_handle, hdb_handle_t object_handle, const void *object_name, size_t object_name_len, const void *key_name, size_t key_name_len, const void *key_value, size_t key_value_len); static void tail_object_created(confdb_handle_t handle, hdb_handle_t parent_object_handle, hdb_handle_t object_handle, const void *name_pt, size_t name_len); static void tail_object_deleted(confdb_handle_t handle, hdb_handle_t parent_object_handle, const void *name_pt, size_t name_len); static void create_object(confdb_handle_t handle, char * name_pt); static void create_object_key(confdb_handle_t handle, char * name_pt); static void write_key(confdb_handle_t handle, char * path_pt); static void get_parent_name(const char * name_pt, char * parent_name); static confdb_callbacks_t callbacks = { .confdb_key_change_notify_fn = tail_key_changed, .confdb_object_create_change_notify_fn = tail_object_created, .confdb_object_delete_change_notify_fn = tail_object_deleted, }; static int debug = 0; static int show_binary = 0; static int action; static void print_binary_key (char *value, size_t value_len) { size_t i; char c; for (i = 0; i < value_len; i++) { c = value[i]; if (c >= ' ' && c < 0x7f && c != '\\') { fputc (c, stdout); } else { if (c == '\\') { printf ("\\\\"); } else { printf ("\\x%02X", c); } } } printf ("\n"); } static void print_key (char *key_name, void *value, size_t value_len, confdb_value_types_t type) { switch (type) { case CONFDB_VALUETYPE_INT16: printf ("%s=%hd\n", key_name, *(int16_t*)value); break; case CONFDB_VALUETYPE_UINT16: printf ("%s=%hu\n", key_name, *(uint16_t*)value); break; case CONFDB_VALUETYPE_INT32: printf ("%s=%d\n", key_name, *(int32_t*)value); break; case CONFDB_VALUETYPE_UINT32: printf ("%s=%u\n", key_name, *(uint32_t*)value); break; case CONFDB_VALUETYPE_INT64: printf ("%s=%"PRIi64"\n", key_name, *(int64_t*)value); break; case CONFDB_VALUETYPE_UINT64: printf ("%s=%"PRIu64"\n", key_name, *(uint64_t*)value); break; case CONFDB_VALUETYPE_FLOAT: printf ("%s=%f\n", key_name, *(float*)value); break; case CONFDB_VALUETYPE_DOUBLE: printf ("%s=%f\n", key_name, *(double*)value); break; case CONFDB_VALUETYPE_STRING: printf ("%s=%s\n", key_name, (char*)value); break; default: case CONFDB_VALUETYPE_ANY: if (!show_binary) { printf ("%s=**binary**(%d)\n", key_name, type); } else { printf ("%s=", key_name); print_binary_key ((char *)value, value_len); } break; } } /* Recursively dump the object tree */ static void print_config_tree(confdb_handle_t handle, hdb_handle_t parent_object_handle, char * parent_name) { hdb_handle_t object_handle; char object_name[OBJ_NAME_SIZE]; size_t object_name_len; char key_name[OBJ_NAME_SIZE]; char key_value[OBJ_NAME_SIZE]; size_t key_value_len; cs_error_t res; int children_printed; confdb_value_types_t type; /* Show the keys */ res = confdb_key_iter_start(handle, parent_object_handle); if (res != CS_OK) { fprintf(stderr, "error resetting key iterator for object "HDB_X_FORMAT" %s\n", parent_object_handle, cs_strerror(res)); exit(EXIT_FAILURE); } children_printed = 0; while ( (res = confdb_key_iter_typed(handle, parent_object_handle, key_name, key_value, &key_value_len, &type)) == CS_OK) { key_value[key_value_len] = '\0'; if (parent_name != NULL) printf("%s%c", parent_name, SEPERATOR); print_key(key_name, key_value, key_value_len, type); children_printed++; } /* Show sub-objects */ res = confdb_object_iter_start(handle, parent_object_handle); if (res != CS_OK) { fprintf(stderr, "error resetting object iterator for "HDB_X_FORMAT" %s\n", parent_object_handle, cs_strerror(res)); exit(EXIT_FAILURE); } while ( (res = confdb_object_iter(handle, parent_object_handle, &object_handle, object_name, &object_name_len)) == CS_OK) { object_name[object_name_len] = '\0'; if (parent_name != NULL) { snprintf(key_value, OBJ_NAME_SIZE, "%s%c%s", parent_name, SEPERATOR, object_name); } else { if ((action == ACTION_PRINT_DEFAULT) && strcmp(object_name, "internal_configuration") == 0) continue; snprintf(key_value, OBJ_NAME_SIZE, "%s", object_name); } print_config_tree(handle, object_handle, key_value); children_printed++; } if (children_printed == 0 && parent_name != NULL) { printf("%s\n", parent_name); } } static int read_in_config_file (char * filename) { confdb_handle_t handle; int result; int ignore; int c; FILE* fh; char buf[1024]; char * line; char * end; char parent_name[OBJ_NAME_SIZE]; if (access (filename, R_OK) != 0) { perror ("Couldn't access file."); return -1; } fh = fopen(filename, "r"); if (fh == NULL) { perror ("Couldn't open file."); return -1; } result = confdb_initialize (&handle, &callbacks); if (result != CONFDB_OK) { fprintf (stderr, "Could not initialize objdb library. Error %d\n", result); fclose (fh); return -1; } while (fgets (buf, 1024, fh) != NULL) { /* find the first real character, if it is * a '#' then ignore this line. * else process. * if no real characters then also ignore. */ ignore = 1; for (c = 0; c < 1024; c++) { if (isblank (buf[c])) continue; if (buf[c] == '#' || buf[c] == '\n') { ignore = 1; break; } ignore = 0; line = &buf[c]; break; } if (ignore == 1) continue; /* kill the \n */ end = strchr (line, '\n'); if (end != NULL) *end = '\0'; if (debug == 2) printf ("%d: %s\n", __LINE__, line); /* find the parent object */ get_parent_name(line, parent_name); if (debug == 2) printf ("%d: %s\n", __LINE__, parent_name); /* create the object */ create_object (handle, parent_name); /* write the attribute */ write_key (handle, line); } confdb_finalize (handle); fclose (fh); return 0; } static int print_all(void) { confdb_handle_t handle; int result; result = confdb_initialize (&handle, &callbacks); if (result != CS_OK) { fprintf (stderr, "Could not initialize objdb library: %s\n", cs_strerror(result)); return 1; } print_config_tree(handle, OBJECT_PARENT_HANDLE, NULL); result = confdb_finalize (handle); return 0; } static int print_help(void) { printf("\n"); printf ("usage: corosync-objctl [-b] object%ckey ... Print an object\n", SEPERATOR); printf (" corosync-objctl -c object%cchild_obj ... Create Object\n", SEPERATOR); printf (" corosync-objctl -d object%cchild_obj ... Delete object\n", SEPERATOR); printf (" corosync-objctl -w object%cchild_obj.key=value ... Create a key\n", SEPERATOR); printf (" corosync-objctl -n object%cchild_obj.key=value ... Create a new object with the key\n", SEPERATOR); printf (" corosync-objctl -t object%cchild_obj ... Track changes\n", SEPERATOR); printf (" corosync-objctl [-b] -a Print all objects\n"); printf (" corosync-objctl -p Load in config from the specified file.\n"); printf("\n"); return 0; } static cs_error_t validate_name(char * obj_name_pt) { if ((strchr (obj_name_pt, SEPERATOR) == NULL) && (strchr (obj_name_pt, '=') == NULL)) return CS_OK; else return CS_ERR_INVALID_PARAM; } static void get_parent_name(const char * name_pt, char * parent_name) { char * tmp; strcpy(parent_name, name_pt); /* first remove the value (it could be a file path */ tmp = strchr(parent_name, '='); if (tmp != NULL) { *tmp = '\0'; tmp--; while (isblank (*tmp)) { *tmp = '\0'; tmp--; } } /* then truncate the child name */ tmp = strrchr(parent_name, SEPERATOR); if (tmp != NULL) *tmp = '\0'; } static void get_key(const char * name_pt, char * key_name, char * key_value) { char * tmp = (char*)name_pt; char str_copy[OBJ_NAME_SIZE]; char * copy_tmp = str_copy; int equals_seen = 0; int in_quotes = 0; /* strip out spaces when not in quotes */ while (*tmp != '\0') { if (*tmp == '=') equals_seen = 1; if (equals_seen && *tmp == '"') { if (in_quotes) in_quotes = 0; else in_quotes = 1; } if (*tmp != ' ' || in_quotes) { *copy_tmp = *tmp; copy_tmp++; } tmp++; } *copy_tmp = '\0'; /* first remove the value (it could have a SEPERATOR in it */ tmp = strchr(str_copy, '='); if (tmp != NULL && strlen(tmp) > 0) { strcpy(key_value, tmp+1); *tmp = '\0'; } else { key_value[0] = '\0'; } /* then remove the name */ tmp = strrchr(str_copy, SEPERATOR); if (tmp == NULL) tmp = str_copy; strcpy(key_name, tmp+1); } static cs_error_t find_object (confdb_handle_t handle, char * name_pt, find_object_of_type_t type, hdb_handle_t * out_handle) { char * obj_name_pt; char * save_pt; hdb_handle_t obj_handle; confdb_handle_t parent_object_handle = OBJECT_PARENT_HANDLE; char tmp_name[OBJ_NAME_SIZE]; cs_error_t res = CS_OK; strncpy (tmp_name, name_pt, sizeof (tmp_name)); tmp_name[sizeof (tmp_name) - 1] = '\0'; obj_name_pt = strtok_r(tmp_name, SEPERATOR_STR, &save_pt); while (obj_name_pt != NULL) { res = confdb_object_find_start(handle, parent_object_handle); if (res != CS_OK) { fprintf (stderr, "Could not start object_find %s\n", cs_strerror(res)); exit (EXIT_FAILURE); } res = confdb_object_find(handle, parent_object_handle, obj_name_pt, strlen (obj_name_pt), &obj_handle); if (res != CS_OK) { return res; } parent_object_handle = obj_handle; obj_name_pt = strtok_r (NULL, SEPERATOR_STR, &save_pt); } *out_handle = parent_object_handle; return res; } static void read_object(confdb_handle_t handle, char * name_pt) { char parent_name[OBJ_NAME_SIZE]; hdb_handle_t obj_handle; cs_error_t res; get_parent_name(name_pt, parent_name); res = find_object (handle, name_pt, FIND_OBJECT_OR_KEY, &obj_handle); if (res == CS_OK) { print_config_tree(handle, obj_handle, parent_name); } } static void write_key(confdb_handle_t handle, char * path_pt) { hdb_handle_t obj_handle; char parent_name[OBJ_NAME_SIZE]; char key_name[OBJ_NAME_SIZE]; char key_value[OBJ_NAME_SIZE]; char old_key_value[OBJ_NAME_SIZE]; size_t old_key_value_len; cs_error_t res; confdb_value_types_t type; /* find the parent object */ get_parent_name(path_pt, parent_name); get_key(path_pt, key_name, key_value); if (debug == 1) printf ("%d: key:\"%s\", value:\"%s\"\n", __LINE__, key_name, key_value); if (validate_name(key_name) != CS_OK) { fprintf(stderr, "Incorrect key name, can not have \"=\" or \"%c\"\n", SEPERATOR); exit(EXIT_FAILURE); } res = find_object (handle, parent_name, FIND_OBJECT_ONLY, &obj_handle); if (res != CS_OK) { fprintf(stderr, "Can't find parent object of \"%s\"\n", path_pt); exit(EXIT_FAILURE); } /* get the current key */ res = confdb_key_get_typed (handle, obj_handle, key_name, old_key_value, &old_key_value_len, &type); if (res == CS_OK) { /* replace the current value */ res = confdb_key_replace (handle, obj_handle, key_name, strlen(key_name), old_key_value, old_key_value_len, key_value, strlen(key_value)); if (res != CS_OK) fprintf(stderr, "Failed to replace the key %s=%s. Error %s\n", key_name, key_value, cs_strerror(res)); } else { /* not there, create a new key */ res = confdb_key_create_typed (handle, obj_handle, key_name, key_value, strlen(key_value), CONFDB_VALUETYPE_STRING); if (res != CS_OK) fprintf(stderr, "Failed to create the key %s=%s. Error %s\n", key_name, key_value, cs_strerror(res)); } } static void create_object(confdb_handle_t handle, char * name_pt) { char * obj_name_pt; char * save_pt; hdb_handle_t obj_handle; hdb_handle_t parent_object_handle = OBJECT_PARENT_HANDLE; char tmp_name[OBJ_NAME_SIZE]; cs_error_t res; strncpy (tmp_name, name_pt, sizeof (tmp_name)); tmp_name[sizeof (tmp_name) - 1] = '\0'; obj_name_pt = strtok_r(tmp_name, SEPERATOR_STR, &save_pt); while (obj_name_pt != NULL) { res = confdb_object_find_start(handle, parent_object_handle); if (res != CS_OK) { fprintf (stderr, "Could not start object_find %s\n", cs_strerror(res)); exit (EXIT_FAILURE); } res = confdb_object_find(handle, parent_object_handle, obj_name_pt, strlen (obj_name_pt), &obj_handle); if (res != CS_OK) { if (validate_name(obj_name_pt) != CS_OK) { fprintf(stderr, "Incorrect object name \"%s\", \"=\" not allowed.\n", obj_name_pt); exit(EXIT_FAILURE); } if (debug) printf ("%s:%d: %s\n", __func__,__LINE__, obj_name_pt); res = confdb_object_create (handle, parent_object_handle, obj_name_pt, strlen (obj_name_pt), &obj_handle); if (res != CS_OK) fprintf(stderr, "Failed to create object \"%s\". Error %s.\n", obj_name_pt, cs_strerror(res)); } parent_object_handle = obj_handle; obj_name_pt = strtok_r (NULL, SEPERATOR_STR, &save_pt); } } static void create_object_key(confdb_handle_t handle, char *name_pt) { char * obj_name_pt; char * new_obj_name_pt; char * save_pt; hdb_handle_t obj_handle; hdb_handle_t parent_object_handle = OBJECT_PARENT_HANDLE; char tmp_name[OBJ_NAME_SIZE]; cs_error_t res; char parent_name[OBJ_NAME_SIZE]; char key_name[OBJ_NAME_SIZE]; char key_value[OBJ_NAME_SIZE]; get_parent_name(name_pt, parent_name); get_key(name_pt, key_name, key_value); strncpy (tmp_name, parent_name, sizeof (tmp_name)); tmp_name[sizeof (tmp_name) - 1] = '\0'; obj_name_pt = strtok_r(tmp_name, SEPERATOR_STR, &save_pt); /* * Create parent object tree */ while (obj_name_pt != NULL) { res = confdb_object_find_start(handle, parent_object_handle); if (res != CS_OK) { fprintf (stderr, "Could not start object_find %d\n", res); exit (EXIT_FAILURE); } new_obj_name_pt = strtok_r (NULL, SEPERATOR_STR, &save_pt); res = confdb_object_find(handle, parent_object_handle, obj_name_pt, strlen (obj_name_pt), &obj_handle); if (res != CS_OK || new_obj_name_pt == NULL) { if (validate_name(obj_name_pt) != CS_OK) { fprintf(stderr, "Incorrect object name \"%s\", \"=\" not allowed.\n", obj_name_pt); exit(EXIT_FAILURE); } if (debug) printf ("%s:%d: %s\n", __func__,__LINE__, obj_name_pt); res = confdb_object_create (handle, parent_object_handle, obj_name_pt, strlen (obj_name_pt), &obj_handle); if (res != CS_OK) { fprintf(stderr, "Failed to create object \"%s\". Error %d.\n", obj_name_pt, res); } } parent_object_handle = obj_handle; obj_name_pt = new_obj_name_pt; } /* * Create key */ res = confdb_key_create_typed (handle, obj_handle, key_name, key_value, strlen(key_value), CONFDB_VALUETYPE_STRING); if (res != CS_OK) { fprintf(stderr, "Failed to create the key %s=%s. Error %d\n", key_name, key_value, res); } } /* Print "?" in place of any non-printable byte of OBJ. */ static void print_name (FILE *fp, const void *obj, size_t obj_len) { const char *p = obj; size_t i; for (i = 0; i < obj_len; i++) { int c = *p++; if (!isprint (c)) { c = '?'; } fputc (c, fp); } } static void tail_key_changed(confdb_handle_t handle, confdb_change_type_t change_type, hdb_handle_t parent_object_handle, hdb_handle_t object_handle, const void *object_name_pt, size_t object_name_len, const void *key_name_pt, size_t key_name_len, const void *key_value_pt, size_t key_value_len) { /* printf("key_changed> %.*s.%.*s=%.*s\n", */ fputs("key_changed> ", stdout); print_name (stdout, object_name_pt, object_name_len); fputs(".", stdout); print_name (stdout, key_name_pt, key_name_len); fputs("=", stdout); print_name (stdout, key_value_pt, key_value_len); fputs("\n", stdout); } static void tail_object_created(confdb_handle_t handle, hdb_handle_t parent_object_handle, hdb_handle_t object_handle, const void *name_pt, size_t name_len) { fputs("object_created>", stdout); print_name(stdout, name_pt, name_len); fputs("\n", stdout); } static void tail_object_deleted(confdb_handle_t handle, hdb_handle_t parent_object_handle, const void *name_pt, size_t name_len) { fputs("object_deleted>", stdout); print_name(stdout, name_pt, name_len); fputs("\n", stdout); } static void listen_for_object_changes(confdb_handle_t handle) { int result; fd_set read_fds; int select_fd; int quit = CS_FALSE; FD_ZERO (&read_fds); if (confdb_fd_get (handle, &select_fd) != CS_OK) { printf ("can't get the confdb selector object.\n"); return; } printf ("Type \"q\" to finish\n"); do { FD_SET (select_fd, &read_fds); FD_SET (STDIN_FILENO, &read_fds); result = select (select_fd + 1, &read_fds, 0, 0, 0); if (result == -1) { perror ("select\n"); } if (FD_ISSET (STDIN_FILENO, &read_fds)) { char inbuf[3]; if (fgets(inbuf, sizeof(inbuf), stdin) == NULL) quit = CS_TRUE; else if (strncmp(inbuf, "q", 1) == 0) quit = CS_TRUE; } if (FD_ISSET (select_fd, &read_fds)) { if (confdb_dispatch (handle, CONFDB_DISPATCH_ALL) != CS_OK) exit(1); } } while (result && quit == CS_FALSE); (void)confdb_stop_track_changes(handle); } static void track_object(confdb_handle_t handle, char * name_pt) { cs_error_t res; hdb_handle_t obj_handle; res = find_object (handle, name_pt, FIND_OBJECT_ONLY, &obj_handle); if (res != CS_OK) { fprintf (stderr, "Could not find object \"%s\". Error %s\n", name_pt, cs_strerror(res)); return; } res = confdb_track_changes (handle, obj_handle, CONFDB_TRACK_DEPTH_RECURSIVE); if (res != CS_OK) { fprintf (stderr, "Could not enable tracking on object \"%s\". Error %s\n", name_pt, cs_strerror(res)); return; } } static void stop_tracking(confdb_handle_t handle) { cs_error_t res; res = confdb_stop_track_changes (handle); if (res != CS_OK) { fprintf (stderr, "Could not stop tracking. Error %s\n", cs_strerror(res)); return; } } static void delete_object(confdb_handle_t handle, char * name_pt) { cs_error_t res; hdb_handle_t obj_handle; res = find_object (handle, name_pt, FIND_OBJECT_ONLY, &obj_handle); if (res == CS_OK) { res = confdb_object_destroy (handle, obj_handle); if (res != CS_OK) fprintf(stderr, "Failed to find object \"%s\" to delete. \n Error %s\n", name_pt, cs_strerror(res)); } else { char parent_name[OBJ_NAME_SIZE]; char key_name[OBJ_NAME_SIZE]; char key_value[OBJ_NAME_SIZE]; /* find the parent object */ get_parent_name(name_pt, parent_name); get_key(name_pt, key_name, key_value); res = find_object (handle, parent_name, FIND_OBJECT_ONLY, &obj_handle); if (res != CS_OK) { fprintf(stderr, "Failed to find the key's parent object \"%s\". Error %s\n", parent_name, cs_strerror(res)); exit (EXIT_FAILURE); } res = confdb_key_delete (handle, obj_handle, key_name, strlen(key_name), key_value, strlen(key_value)); if (res != CS_OK) fprintf(stderr, "Failed to delete key \"%s=%s\" from object \"%s\".\n Error %s\n", key_name, key_value, parent_name, cs_strerror(res)); } } int main (int argc, char *argv[]) { confdb_handle_t handle; cs_error_t result; int c; action = ACTION_READ; for (;;){ c = getopt (argc,argv,"habwncvdtp:"); if (c==-1) { break; } switch (c) { case 'v': debug++; break; case 'h': return print_help(); break; case 'a': action = ACTION_PRINT_ALL; break; case 'b': show_binary++; break; case 'p': return read_in_config_file (optarg); break; case 'c': action = ACTION_CREATE; break; case 'd': action = ACTION_DELETE; break; case 'w': action = ACTION_WRITE; break; case 'n': action = ACTION_CREATE_KEY; break; case 't': action = ACTION_TRACK; break; default : action = ACTION_READ; break; } } if (argc == 1) { action = ACTION_PRINT_DEFAULT; return print_all(); } else if (action == ACTION_PRINT_ALL) { return print_all(); } else if (optind >= argc) { fprintf(stderr, "Expected an object path after options\n"); exit(EXIT_FAILURE); } result = confdb_initialize (&handle, &callbacks); if (result != CS_OK) { fprintf (stderr, "Failed to initialize the objdb API. Error %s\n", cs_strerror(result)); exit (EXIT_FAILURE); } while (optind < argc) { switch (action) { case ACTION_READ: read_object(handle, argv[optind++]); break; case ACTION_WRITE: write_key(handle, argv[optind++]); break; case ACTION_CREATE: create_object(handle, argv[optind++]); break; case ACTION_CREATE_KEY: create_object_key(handle, argv[optind++]); break; case ACTION_DELETE: delete_object(handle, argv[optind++]); break; case ACTION_TRACK: track_object(handle, argv[optind++]); break; } } if (action == ACTION_TRACK) { listen_for_object_changes(handle); stop_tracking(handle); } result = confdb_finalize (handle); if (result != CS_OK) { fprintf (stderr, "Error finalizing objdb API.\n Error %s\n", cs_strerror(result)); exit(EXIT_FAILURE); } return 0; }