-
Notifications
You must be signed in to change notification settings - Fork 7
/
Copy pathrun_checks.c
160 lines (134 loc) · 5.16 KB
/
run_checks.c
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
158
159
160
/*
Copyright 2014-2017 Thumbtack, Inc.
Licensed under the Apache License, Version 2.0 (the "License");
you may not use this file except in compliance with the License.
You may obtain a copy of the License at
http://www.apache.org/licenses/LICENSE-2.0
Unless required by applicable law or agreed to in writing, software
distributed under the License is distributed on an "AS IS" BASIS,
WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
See the License for the specific language governing permissions and
limitations under the License.
*/
#include "run_checks.h"
#include <string.h>
#include <stdlib.h>
#include <libpq-fe.h>
#include "logger.h"
#include "strconst.h"
/* write the error string to `result` and terminate the connection
* gracefully */
static void pg_fail(PGconn *pg_conn, char *result, size_t size)
{
char *errormsg = PQerrorMessage(pg_conn);
/* save the error message from PG to `result` */
if (errormsg)
snprintf(result, size, "%s", errormsg);
}
static int run_custom_check(PGconn *pg_conn, custom_check_t check,
char *result, size_t size)
{
PGresult *pg_result;
char *query_result;
int success = 1;
pg_result = PQexec(pg_conn, CUSTOM_CHECK_QUERY(check));
/* without a result set there's no point on proceeding */
if (PQresultStatus(pg_result) != PGRES_TUPLES_OK) {
pg_fail(pg_conn, result, size);
PQclear(pg_result);
return 0;
}
/* SELECT command that happens to retrieve zero rows still shows
PGRES_TUPLES_OK */
if (PQntuples(pg_result) != 1) {
pg_fail(pg_conn, result, size);
PQclear(pg_result);
return 0;
}
/* we only expect one field with one result */
query_result = PQgetvalue(pg_result, 0, 0);
/* an empty string means that there is nothing to compare to, so
* the check fails */
if (strlen(query_result) == 0) {
snprintf(result, size, STR_QUERY_FAILED_FMT, CUSTOM_CHECK_QUERY(check));
PQclear(pg_result);
return 0;
}
/* compare the query's result with the expected value */
/* = compares as a string */
if (strcmp(CUSTOM_CHECK_OPERATOR(check), "=") == 0)
success = (strcmp(query_result, CUSTOM_CHECK_RESULT(check)) == 0);
/* < and > compare as floating point values */
if (strcmp(CUSTOM_CHECK_OPERATOR(check), "<") == 0)
success = (atof(query_result) < atof(CUSTOM_CHECK_RESULT(check)));
if (strcmp(CUSTOM_CHECK_OPERATOR(check), ">") == 0)
success = (atof(query_result) > atof(CUSTOM_CHECK_RESULT(check)));
/* make sure the error message is written to the result string */
if (! success)
snprintf(result, size, STR_HEALTH_CHECK_ERROR_DETAIL_FMT,
CUSTOM_CHECK_QUERY(check),
CUSTOM_CHECK_OPERATOR(check),
CUSTOM_CHECK_RESULT(check),
query_result);
PQclear(pg_result);
return success;
}
static int run_all_checks(PGconn *pg_conn, checks_list_t checks_list,
char *result, size_t size)
{
int success = 1;
custom_check_t check;
while (checks_list) {
check = CHECKS_LIST_CHECK(checks_list);
if (run_custom_check(pg_conn, check, result, size)) {
logger_write(LOG_INFO, STR_HEALTH_CHECK_SUCCESS_FMT,
CUSTOM_CHECK_QUERY(check),
CUSTOM_CHECK_OPERATOR(check),
CUSTOM_CHECK_RESULT(check));
} else {
success = 0;
logger_write(LOG_ERR, STR_HEALTH_CHECK_ERROR_FMT,
CUSTOM_CHECK_QUERY(check),
CUSTOM_CHECK_OPERATOR(check),
CUSTOM_CHECK_RESULT(check));
}
checks_list = CHECKS_LIST_NEXT(checks_list);
}
return success;
}
/* top level health check: connects to the DB and runs all checks */
/* returns 0 on pg_failure and 1 on success */
/* a string with the description of the error (or "OK") is copied to
* `result` */
extern int run_health_checks(config_t config, char *result, size_t size)
{
/* connection strings: format and actual string to be created */
char pg_conn_str[MAX_STR];
PGconn *pg_conn;
/* create the connection string from the set of parameters */
snprintf(pg_conn_str,
sizeof(pg_conn_str),
STR_PG_CONN_INFO_FMT,
CFG_PG_HOST(config),
CFG_PG_PORT(config),
CFG_PG_DATABASE(config),
CFG_PG_USER(config),
CFG_PG_PASSWORD(config),
CFG_PG_TIMEOUT(config));
/* check successful connection or return error right now */
pg_conn = PQconnectdb(pg_conn_str);
if (PQstatus(pg_conn) != CONNECTION_OK) {
logger_write(LOG_CRIT, STR_DB_CONNECTION_ERROR);
pg_fail(pg_conn, result, size);
PQfinish(pg_conn);
return 0;
}
/* run the custom checks */
if (! run_all_checks(pg_conn, CFG_CUSTOM_CHECKS(config), result, size)) {
PQfinish(pg_conn);
return 0;
}
snprintf(result, size, "%s", STR_ALL_CHECKS_SUCCESSFUL);
PQfinish(pg_conn);
return 1;
}