check_ldap.c 15 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508509510511512513514515516517518519520521522523524525526527528529530531532533534535536537538539540541542543544545546547548549550551552553554555556557558559560561562563564565566567568569570571572573574575576577578579580581582583584585586587588589
  1. /*****************************************************************************
  2. *
  3. * Nagios check_ldap plugin
  4. *
  5. * License: GPL
  6. * Copyright (c) 2000-2014 Nagios Plugins Development Team
  7. *
  8. * Description:
  9. *
  10. * This file contains the check_ldap plugin
  11. *
  12. *
  13. * This program is free software: you can redistribute it and/or modify
  14. * it under the terms of the GNU General Public License as published by
  15. * the Free Software Foundation, either version 3 of the License, or
  16. * (at your option) any later version.
  17. *
  18. * This program is distributed in the hope that it will be useful,
  19. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  20. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  21. * GNU General Public License for more details.
  22. *
  23. * You should have received a copy of the GNU General Public License
  24. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  25. *
  26. *
  27. *****************************************************************************/
  28. /* progname may be check_ldaps */
  29. char *progname = "check_ldap";
  30. const char *copyright = "2000-2014";
  31. const char *email = "devel@nagios-plugins.org";
  32. #include "common.h"
  33. #include "netutils.h"
  34. #include "utils.h"
  35. #include <lber.h>
  36. #define LDAP_DEPRECATED 1
  37. #include <ldap.h>
  38. enum {
  39. UNDEFINED = 0,
  40. #ifdef HAVE_LDAP_SET_OPTION
  41. DEFAULT_PROTOCOL = 2,
  42. #endif
  43. DEFAULT_PORT = 389
  44. };
  45. int process_arguments (int, char **);
  46. int validate_arguments (void);
  47. void print_help (void);
  48. void print_usage (void);
  49. char ld_defattr[] = "(objectclass=*)";
  50. char *ld_attr = ld_defattr;
  51. char *ld_uri = NULL;
  52. char *ld_host = NULL;
  53. char *ld_base = NULL;
  54. char *ld_passwd = NULL;
  55. char *ld_binddn = NULL;
  56. int ld_port = -1;
  57. #ifdef HAVE_LDAP_SET_OPTION
  58. int ld_protocol = DEFAULT_PROTOCOL;
  59. #endif
  60. #ifndef LDAP_OPT_SUCCESS
  61. # define LDAP_OPT_SUCCESS LDAP_SUCCESS
  62. #endif
  63. double warn_time = UNDEFINED;
  64. double crit_time = UNDEFINED;
  65. thresholds *entries_thresholds = NULL;
  66. struct timeval tv;
  67. char* warn_entries = NULL;
  68. char* crit_entries = NULL;
  69. int starttls = FALSE;
  70. int ssl_on_connect = FALSE;
  71. int verbose = 0;
  72. int check_cert = FALSE;
  73. int days_till_exp_warn, days_till_exp_crit;
  74. /* for ldap tls */
  75. char *SERVICE = "LDAP";
  76. int
  77. main (int argc, char *argv[])
  78. {
  79. LDAP *ld;
  80. LDAPMessage *result;
  81. /* should be int result = STATE_UNKNOWN; */
  82. int status = STATE_UNKNOWN;
  83. long microsec;
  84. double elapsed_time;
  85. /* for ldap tls */
  86. int tls;
  87. int version=3;
  88. /* for entry counting */
  89. LDAPMessage *next_entry;
  90. int status_entries = STATE_OK;
  91. int num_entries = 0;
  92. setlocale (LC_ALL, "");
  93. bindtextdomain (PACKAGE, LOCALEDIR);
  94. textdomain (PACKAGE);
  95. if (strstr(argv[0],"check_ldaps")) {
  96. xasprintf (&progname, "check_ldaps");
  97. }
  98. /* Parse extra opts if any */
  99. argv=np_extra_opts (&argc, argv, progname);
  100. if (process_arguments (argc, argv) == ERROR)
  101. usage4 (_("Could not parse arguments"));
  102. if (strstr(argv[0],"check_ldaps") && ! starttls && ! ssl_on_connect)
  103. starttls = TRUE;
  104. /* initialize alarm signal handling */
  105. signal (SIGALRM, socket_timeout_alarm_handler);
  106. /* set socket timeout */
  107. alarm (timeout_interval);
  108. /* get the start time */
  109. gettimeofday (&tv, NULL);
  110. /* initialize ldap */
  111. if (ld_uri != NULL)
  112. {
  113. #ifdef HAVE_LDAP_INITIALIZE
  114. int result = ldap_initialize(&ld, ld_uri);
  115. if (result != LDAP_SUCCESS)
  116. {
  117. printf ("Failed to connect to LDAP server at %s: %s\n",
  118. ld_uri, ldap_err2string(result));
  119. return STATE_CRITICAL;
  120. }
  121. #else
  122. printf ("Sorry, this version of %s was compiled without URI support!\n",
  123. argv[0]);
  124. return STATE_CRITICAL;
  125. #endif
  126. }
  127. #ifdef HAVE_LDAP_INIT
  128. else if (!(ld = ldap_init (ld_host, ld_port))) {
  129. printf ("Could not connect to the server at port %i\n", ld_port);
  130. return STATE_CRITICAL;
  131. }
  132. #else
  133. else if (!(ld = ldap_open (ld_host, ld_port))) {
  134. if (verbose)
  135. ldap_perror(ld, "ldap_open");
  136. printf (_("Could not connect to the server at port %i\n"), ld_port);
  137. return STATE_CRITICAL;
  138. }
  139. #endif /* HAVE_LDAP_INIT */
  140. #ifdef HAVE_LDAP_SET_OPTION
  141. /* set ldap options */
  142. if (ldap_set_option (ld, LDAP_OPT_PROTOCOL_VERSION, &ld_protocol) !=
  143. LDAP_OPT_SUCCESS ) {
  144. printf(_("Could not set protocol version %d\n"), ld_protocol);
  145. return STATE_CRITICAL;
  146. }
  147. #endif
  148. if (ld_port == LDAPS_PORT || ssl_on_connect) {
  149. xasprintf (&SERVICE, "LDAPS");
  150. #if defined(HAVE_LDAP_SET_OPTION) && defined(LDAP_OPT_X_TLS)
  151. /* ldaps: set option tls */
  152. tls = LDAP_OPT_X_TLS_HARD;
  153. if (ldap_set_option (ld, LDAP_OPT_X_TLS, &tls) != LDAP_SUCCESS)
  154. {
  155. if (verbose)
  156. ldap_perror(ld, "ldaps_option");
  157. printf (_("Could not init TLS at port %i!\n"), ld_port);
  158. return STATE_CRITICAL;
  159. }
  160. if (check_cert == TRUE)
  161. return ldap_check_cert(ld);
  162. #else
  163. printf (_("TLS not supported by the libraries!\n"));
  164. return STATE_CRITICAL;
  165. #endif /* LDAP_OPT_X_TLS */
  166. } else if (starttls) {
  167. xasprintf (&SERVICE, "LDAP-TLS");
  168. #if defined(HAVE_LDAP_SET_OPTION) && defined(HAVE_LDAP_START_TLS_S)
  169. /* ldap with startTLS: set option version */
  170. if (ldap_get_option(ld,LDAP_OPT_PROTOCOL_VERSION, &version) == LDAP_OPT_SUCCESS )
  171. {
  172. if (version < LDAP_VERSION3)
  173. {
  174. version = LDAP_VERSION3;
  175. ldap_set_option(ld, LDAP_OPT_PROTOCOL_VERSION, &version);
  176. }
  177. }
  178. /* call start_tls */
  179. if (ldap_start_tls_s(ld, NULL, NULL) != LDAP_SUCCESS)
  180. {
  181. if (verbose)
  182. ldap_perror(ld, "ldap_start_tls");
  183. printf (_("Could not init startTLS at port %i!\n"), ld_port);
  184. return STATE_CRITICAL;
  185. }
  186. if (check_cert == TRUE)
  187. return ldap_check_cert(ld);
  188. #else
  189. printf (_("startTLS not supported by the library, needs LDAPv3!\n"));
  190. return STATE_CRITICAL;
  191. #endif /* HAVE_LDAP_START_TLS_S */
  192. }
  193. /* bind to the ldap server */
  194. if (ldap_bind_s (ld, ld_binddn, ld_passwd, LDAP_AUTH_SIMPLE) !=
  195. LDAP_SUCCESS) {
  196. if (verbose)
  197. ldap_perror(ld, "ldap_bind");
  198. printf (_("Could not bind to the LDAP server\n"));
  199. return STATE_CRITICAL;
  200. }
  201. /* do a search of all objectclasses in the base dn */
  202. if (ldap_search_s (ld, ld_base, (crit_entries!=NULL || warn_entries!=NULL) ? LDAP_SCOPE_SUBTREE : LDAP_SCOPE_BASE, ld_attr, NULL, 0, &result)
  203. != LDAP_SUCCESS) {
  204. if (verbose)
  205. ldap_perror(ld, "ldap_search");
  206. printf (_("Could not search/find objectclasses in %s\n"), ld_base);
  207. return STATE_CRITICAL;
  208. } else if (crit_entries!=NULL || warn_entries!=NULL) {
  209. num_entries = ldap_count_entries(ld, result);
  210. }
  211. /* unbind from the ldap server */
  212. ldap_unbind (ld);
  213. /* reset the alarm handler */
  214. alarm (0);
  215. /* calcutate the elapsed time and compare to thresholds */
  216. microsec = deltime (tv);
  217. elapsed_time = (double)microsec / 1.0e6;
  218. if (crit_time!=UNDEFINED && elapsed_time>crit_time)
  219. status = STATE_CRITICAL;
  220. else if (warn_time!=UNDEFINED && elapsed_time>warn_time)
  221. status = STATE_WARNING;
  222. else
  223. status = STATE_OK;
  224. if(entries_thresholds != NULL) {
  225. if (verbose) {
  226. printf ("entries found: %d\n", num_entries);
  227. print_thresholds("entry thresholds", entries_thresholds);
  228. }
  229. status_entries = get_status(num_entries, entries_thresholds);
  230. if (status_entries == STATE_CRITICAL) {
  231. status = STATE_CRITICAL;
  232. } else if (status != STATE_CRITICAL) {
  233. status = status_entries;
  234. }
  235. }
  236. /* print out the result */
  237. if (crit_entries!=NULL || warn_entries!=NULL) {
  238. printf (_("LDAP %s - found %d entries in %.3f seconds|%s %s\n"),
  239. state_text (status),
  240. num_entries,
  241. elapsed_time,
  242. fperfdata ("time", elapsed_time, "s",
  243. (int)warn_time, warn_time,
  244. (int)crit_time, crit_time,
  245. TRUE, 0, FALSE, 0),
  246. sperfdata ("entries", (double)num_entries, "",
  247. warn_entries,
  248. crit_entries,
  249. TRUE, 0.0, FALSE, 0.0));
  250. } else {
  251. printf (_("LDAP %s - %.3f seconds response time|%s\n"),
  252. state_text (status),
  253. elapsed_time,
  254. fperfdata ("time", elapsed_time, "s",
  255. (int)warn_time, warn_time,
  256. (int)crit_time, crit_time,
  257. TRUE, 0, FALSE, 0));
  258. }
  259. return status;
  260. }
  261. /* process command-line arguments */
  262. int
  263. process_arguments (int argc, char **argv)
  264. {
  265. int c;
  266. char *temp;
  267. int option = 0;
  268. /* initialize the long option struct */
  269. static struct option longopts[] = {
  270. {"help", no_argument, 0, 'h'},
  271. {"version", no_argument, 0, 'V'},
  272. {"timeout", required_argument, 0, 't'},
  273. {"hostname", required_argument, 0, 'H'},
  274. {"uri", required_argument, 0, 'U'},
  275. {"base", required_argument, 0, 'b'},
  276. {"attr", required_argument, 0, 'a'},
  277. {"bind", required_argument, 0, 'D'},
  278. {"pass", required_argument, 0, 'P'},
  279. #ifdef HAVE_LDAP_SET_OPTION
  280. {"ver2", no_argument, 0, '2'},
  281. {"ver3", no_argument, 0, '3'},
  282. #endif
  283. {"starttls", no_argument, 0, 'T'},
  284. {"ssl", no_argument, 0, 'S'},
  285. {"age", required_argument, 0, 'A'},
  286. {"use-ipv4", no_argument, 0, '4'},
  287. {"use-ipv6", no_argument, 0, '6'},
  288. {"port", required_argument, 0, 'p'},
  289. {"warn", required_argument, 0, 'w'},
  290. {"crit", required_argument, 0, 'c'},
  291. {"warn-entries", required_argument, 0, 'W'},
  292. {"crit-entries", required_argument, 0, 'C'},
  293. {"verbose", no_argument, 0, 'v'},
  294. {0, 0, 0, 0}
  295. };
  296. if (argc < 2)
  297. return ERROR;
  298. for (c = 1; c < argc; c++) {
  299. if (strcmp ("-to", argv[c]) == 0)
  300. strcpy (argv[c], "-t");
  301. }
  302. while (1) {
  303. c = getopt_long (argc, argv, "hvV234TS6t:c:w:H:b:p:a:D:P:U:C:W:A:", longopts, &option);
  304. if (c == -1 || c == EOF)
  305. break;
  306. switch (c) {
  307. case 'h': /* help */
  308. print_help ();
  309. exit (STATE_OK);
  310. case 'V': /* version */
  311. print_revision (progname, NP_VERSION);
  312. exit (STATE_OK);
  313. case 't': /* timeout period */
  314. timeout_interval = parse_timeout_string(optarg);
  315. break;
  316. case 'U':
  317. ld_uri = optarg;
  318. break;
  319. case 'H':
  320. ld_host = optarg;
  321. break;
  322. case 'b':
  323. ld_base = optarg;
  324. break;
  325. case 'p':
  326. ld_port = atoi (optarg);
  327. break;
  328. case 'a':
  329. ld_attr = optarg;
  330. break;
  331. case 'D':
  332. ld_binddn = optarg;
  333. break;
  334. case 'P':
  335. ld_passwd = optarg;
  336. break;
  337. case 'w':
  338. warn_time = strtod (optarg, NULL);
  339. break;
  340. case 'c':
  341. crit_time = strtod (optarg, NULL);
  342. break;
  343. case 'W':
  344. warn_entries = optarg;
  345. break;
  346. case 'C':
  347. crit_entries = optarg;
  348. break;
  349. #ifdef HAVE_LDAP_SET_OPTION
  350. case '2':
  351. ld_protocol = 2;
  352. break;
  353. case '3':
  354. ld_protocol = 3;
  355. break;
  356. #endif
  357. case '4':
  358. address_family = AF_INET;
  359. break;
  360. case 'v':
  361. verbose++;
  362. break;
  363. case 'T':
  364. if (! ssl_on_connect)
  365. starttls = TRUE;
  366. else
  367. usage_va(_("%s cannot be combined with %s"), "-T/--starttls", "-S/--ssl");
  368. break;
  369. case 'A': /* Check SSL cert validity */
  370. #ifndef HAVE_SSL
  371. usage4 (_("Invalid option - SSL is not available"));
  372. #else
  373. if (starttls || ssl_on_connect || strstr(argv[0],"check_ldaps")) {
  374. if ((temp=strchr(optarg,','))!=NULL) {
  375. *temp = '\0';
  376. if (!is_intnonneg (temp))
  377. usage2 (_("Invalid certificate expiration period"), optarg);
  378. days_till_exp_warn = atoi(optarg);
  379. *temp = ',';
  380. temp++;
  381. if (!is_intnonneg (temp))
  382. usage2 (_("Invalid certificate expiration period"), temp);
  383. days_till_exp_crit = atoi (temp);
  384. } else {
  385. days_till_exp_crit = 0;
  386. if (!is_intnonneg (optarg))
  387. usage2 (_("Invalid certificate expiration period"), optarg);
  388. days_till_exp_warn = atoi (optarg);
  389. }
  390. check_cert = TRUE;
  391. } else {
  392. usage_va(_("%s requires either %s or %s"), "-A/--age", "-S/--ssl", "-T/--starttls");
  393. }
  394. break;
  395. #endif
  396. case 'S':
  397. if (! starttls) {
  398. ssl_on_connect = TRUE;
  399. if (ld_port == -1)
  400. ld_port = LDAPS_PORT;
  401. } else
  402. usage_va(_("%s cannot be combined with %s"), "-S/--ssl", "-T/--starttls");
  403. break;
  404. case '6':
  405. #ifdef USE_IPV6
  406. address_family = AF_INET6;
  407. #else
  408. usage (_("IPv6 support not available\n"));
  409. #endif
  410. break;
  411. default:
  412. usage5 ();
  413. }
  414. }
  415. c = optind;
  416. if (ld_host == NULL && is_host(argv[c]))
  417. ld_host = strdup (argv[c++]);
  418. if (ld_base == NULL && argv[c])
  419. ld_base = strdup (argv[c++]);
  420. if (ld_port == -1)
  421. ld_port = DEFAULT_PORT;
  422. return validate_arguments ();
  423. }
  424. int
  425. validate_arguments ()
  426. {
  427. if ((ld_host==NULL || strlen(ld_host)==0) &&
  428. (ld_uri==NULL || strlen(ld_uri)==0))
  429. usage4 (_("Please specify the host name or LDAP URI\n"));
  430. if (ld_base==NULL)
  431. usage4 (_("Please specify the LDAP base DN\n"));
  432. if (crit_entries!=NULL || warn_entries!=NULL) {
  433. set_thresholds(&entries_thresholds,
  434. warn_entries, crit_entries);
  435. }
  436. return OK;
  437. }
  438. void
  439. print_help (void)
  440. {
  441. char *myport;
  442. xasprintf (&myport, "%d", DEFAULT_PORT);
  443. print_revision (progname, NP_VERSION);
  444. printf ("Copyright (c) 1999 Didi Rieder (adrieder@sbox.tu-graz.ac.at)\n");
  445. printf (COPYRIGHT, copyright, email);
  446. printf ("\n\n");
  447. print_usage ();
  448. printf (UT_HELP_VRSN);
  449. printf (UT_EXTRA_OPTS);
  450. printf (UT_HOST_PORT, 'p', myport);
  451. printf (UT_IPv46);
  452. printf (" %s\n", "-a, --attr=ATTRIBUTE");
  453. printf (" %s\n", _("ldap attribute to search (default: \"(objectclass=*)\""));
  454. printf (" %s\n", "-b, --base=BASE");
  455. printf (" %s\n", _("ldap base (eg. ou=my unit, o=my org, c=at"));
  456. printf (" %s\n", "-D, --bind=DN");
  457. printf (" %s\n", _("ldap bind DN (if required)"));
  458. printf (" %s\n", "-P, --pass=PASSWORD");
  459. printf (" %s\n", _("ldap password (if required)"));
  460. printf (" %s\n", "-T, --starttls");
  461. printf (" %s\n", _("use starttls mechanism introduced in protocol version 3"));
  462. printf (" %s\n", "-S, --ssl");
  463. printf (" %s %i\n", _("use ldaps (ldap v2 ssl method). this also sets the default port to"), LDAPS_PORT);
  464. printf (" %s\n", "-A, --age=INTEGER[,INTEGER]");
  465. printf (" %s\n", _("Minimum number of days a certificate has to be valid"));
  466. #ifdef HAVE_LDAP_SET_OPTION
  467. printf (" %s\n", "-2, --ver2");
  468. printf (" %s\n", _("use ldap protocol version 2"));
  469. printf (" %s\n", "-3, --ver3");
  470. printf (" %s\n", _("use ldap protocol version 3"));
  471. printf (" (%s %d)\n", _("default protocol version:"), DEFAULT_PROTOCOL);
  472. #endif
  473. printf (UT_WARN_CRIT);
  474. printf (" %s\n", "-W, --warn-entries=INTEGER");
  475. printf (" %s\n", _("Number of found entries to result in warning status"));
  476. printf (" %s\n", "-C, --crit-entries=INTEGER");
  477. printf (" %s\n", _("Number of found entries to result in critical status"));
  478. printf (UT_CONN_TIMEOUT, DEFAULT_SOCKET_TIMEOUT);
  479. printf (UT_VERBOSE);
  480. printf ("\n");
  481. printf ("%s\n", _("Notes:"));
  482. printf (" %s\n", _("If this plugin is called via 'check_ldaps', method 'STARTTLS' will be"));
  483. printf (_(" implied (using default port %i) unless --port=636 is specified. In that case\n"), DEFAULT_PORT);
  484. printf (" %s\n", _("'SSL on connect' will be used no matter how the plugin was called."));
  485. printf (" %s\n", _("This detection is deprecated, please use 'check_ldap' with the '--starttls' or '--ssl' flags"));
  486. printf (" %s\n", _("to define the behaviour explicitly instead."));
  487. printf (" %s\n", _("The parameters --warn-entries and --crit-entries are optional."));
  488. printf (UT_SUPPORT);
  489. }
  490. void
  491. print_usage (void)
  492. {
  493. printf ("%s\n", _("Usage:"));
  494. printf (" %s (-H <host>|-U <uri>) -b <base_dn> [-p <port>] [-a <attr>] [-D <binddn>]\n", progname);
  495. printf (" [-P <password>] [-w <warn_time>] [-c <crit_time>] [-t timeout] [-A <age>]%s\n",
  496. #ifdef HAVE_LDAP_SET_OPTION
  497. "\n [-2|-3] [-4|-6]"
  498. #else
  499. ""
  500. #endif
  501. );
  502. }
  503. int ldap_check_cert (LDAP *ld)
  504. {
  505. SSL *ssl;
  506. int rc;
  507. rc = ldap_get_option(ld, LDAP_OPT_X_TLS_SSL_CTX, &ssl);
  508. if (rc == LDAP_OPT_ERROR || ssl == NULL) {
  509. printf ("%s\n",_("CRITICAL - Cannot retrieve ssl session from connection."));
  510. return STATE_CRITICAL;
  511. }
  512. return np_net_ssl_check_cert_real(ssl, days_till_exp_warn, days_till_exp_crit);
  513. }