You can not select more than 25 topics Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.

postsrsd.c 13KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431
  1. /* PostSRSd - Sender Rewriting Scheme daemon for Postfix
  2. * Copyright (c) 2012 Timo Röhling <timo.roehling@gmx.de>
  3. *
  4. * This program is free software: you can redistribute it and/or modify
  5. * it under the terms of the GNU General Public License as published by
  6. * the Free Software Foundation, either version 2 of the License, or
  7. * (at your option) any later version.
  8. *
  9. * This program is distributed in the hope that it will be useful,
  10. * but WITHOUT ANY WARRANTY; without even the implied warranty of
  11. * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
  12. * GNU General Public License for more details.
  13. *
  14. * You should have received a copy of the GNU General Public License
  15. * along with this program. If not, see <http://www.gnu.org/licenses/>.
  16. */
  17. /* This program uses the libsrs2 library. The relevant source
  18. * files have been added to this distribution. */
  19. #include "srs2.h"
  20. #include <sys/types.h>
  21. #include <sys/socket.h>
  22. #include <netdb.h>
  23. #include <errno.h>
  24. #include <unistd.h>
  25. #include <fcntl.h>
  26. #include <pwd.h>
  27. #include <string.h>
  28. #include <poll.h>
  29. #include <wait.h>
  30. #include <syslog.h>
  31. #ifndef VERSION
  32. #define VERSION "1.0"
  33. #endif
  34. static char *self = NULL;
  35. static int bind_service (const char *service, int family)
  36. {
  37. struct addrinfo *addr, *it;
  38. struct addrinfo hints;
  39. int err, sock, flags;
  40. static const int one = 1;
  41. memset (&hints, 0, sizeof(hints));
  42. hints.ai_family = family;
  43. hints.ai_socktype = SOCK_STREAM;
  44. hints.ai_flags = AI_ADDRCONFIG | AI_V4MAPPED;
  45. err = getaddrinfo(NULL, service, &hints, &addr);
  46. if (err != 0) {
  47. fprintf(stderr, "%s: bind_service(%s): %s\n", self, service, gai_strerror(err));
  48. return -1;
  49. }
  50. sock = -1;
  51. for (it = addr; it; it = it->ai_next) {
  52. sock = socket(it->ai_family, it->ai_socktype, it->ai_protocol);
  53. if (sock < 0) goto fail;
  54. if (setsockopt(sock, SOL_SOCKET, SO_REUSEADDR, &one, sizeof(one)) < 0) goto fail;
  55. if (bind(sock, it->ai_addr, it->ai_addrlen) < 0) goto fail;
  56. if (listen(sock, 10) < 0) goto fail;
  57. flags = fcntl (sock, F_GETFL, 0);
  58. if (flags < 0) goto fail;
  59. if (fcntl(sock, F_SETFL, flags | O_NONBLOCK) < 0) goto fail;
  60. break;
  61. fail:
  62. err = errno;
  63. if (sock >= 0) close (sock);
  64. sock = -1;
  65. }
  66. freeaddrinfo (addr);
  67. if (sock < 0)
  68. fprintf (stderr, "%s: bind_service(%s): %s\n", self, service, strerror(err));
  69. return sock;
  70. }
  71. static int is_hexdigit (char c)
  72. {
  73. return (c >= '0' && c <= '9') || (c >= 'a' && c <= 'f') || (c >= 'A' && c <= 'F');
  74. }
  75. static char hex2num (char c)
  76. {
  77. if (c >= '0' && c <= '9') return c - '0';
  78. if (c >= 'a' && c <= 'f') return c - 'a' + 10;
  79. if (c >= 'A' && c <= 'F') return c - 'A' + 10;
  80. return 0;
  81. }
  82. static char num2hex (char c)
  83. {
  84. if (c < 10) return '0' + c;
  85. return 'a' + c - 10;
  86. }
  87. static char hex2char (const char *s)
  88. {
  89. return 16 * hex2num(s[0]) + hex2num(s[1]);
  90. }
  91. static void char2hex (char c, char *buf)
  92. {
  93. buf[0] = num2hex((c >> 4) & 0xf);
  94. buf[1] = num2hex((c ) & 0xf);
  95. }
  96. static char* url_decode (char *buf, size_t len, const char *input)
  97. {
  98. char *output = buf;
  99. if (!input || !output || len == 0) return NULL;
  100. while (*input && --len) {
  101. if (*input == '%' && is_hexdigit(input[1]) && is_hexdigit(input[2])) {
  102. *output++ = hex2char(++input);
  103. input += 2;
  104. } else {
  105. *output++ = *input++;
  106. }
  107. }
  108. *output = 0;
  109. return buf;
  110. }
  111. static char* url_encode (char* buf, size_t len, const char *input)
  112. {
  113. char *output = buf;
  114. if (!input || !output || len == 0) return NULL;
  115. while (*input && --len) {
  116. if (!isascii(*input) || !isgraph(*input) || *input == '%') {
  117. if (len <= 2) break;
  118. *output++ = '%';
  119. char2hex(*input++, output);
  120. output += 2; len -= 2;
  121. } else {
  122. *output++ = *input++;
  123. }
  124. }
  125. *output = 0;
  126. return buf;
  127. }
  128. static void handle_forward (srs_t *srs, FILE *fp, const char *address, const char *domain)
  129. {
  130. int result;
  131. char value[1024];
  132. char outputbuf[1024], *output;
  133. result = srs_forward(srs, value, sizeof(value), address, domain);
  134. if (result == SRS_SUCCESS) {
  135. output = url_encode(outputbuf, sizeof(outputbuf), value);
  136. fprintf (fp, "200 %s\n", output);
  137. if (strcmp(address, value) != 0) syslog (LOG_MAIL | LOG_INFO, "srs_forward: <%s> rewritten as <%s>", address, value);
  138. } else {
  139. fprintf (fp, "500 %s\n", srs_strerror(result));
  140. if (result != SRS_ENOTREWRITTEN)
  141. syslog (LOG_MAIL | LOG_INFO, "srs_forward: <%s> not rewritten: %s", address, srs_strerror(result));
  142. }
  143. fflush (fp);
  144. }
  145. static void handle_reverse (srs_t *srs, FILE *fp, const char *address, const char *domain)
  146. {
  147. int result;
  148. char value[1024];
  149. char outputbuf[1024], *output;
  150. result = srs_reverse(srs, value, sizeof(value), address);
  151. if (result == SRS_SUCCESS) {
  152. output = url_encode(outputbuf, sizeof(outputbuf), value);
  153. fprintf (fp, "200 %s\n", output);
  154. syslog (LOG_MAIL | LOG_INFO, "srs_reverse: <%s> rewritten as <%s>", address, value);
  155. } else {
  156. fprintf (fp, "500 %s\n", srs_strerror(result));
  157. if (result != SRS_ENOTREWRITTEN && result != SRS_ENOTSRSADDRESS)
  158. syslog (LOG_MAIL | LOG_INFO, "srs_reverse: <%s> not rewritten: %s", address, srs_strerror(result));
  159. }
  160. fflush (fp);
  161. }
  162. static void show_help ()
  163. {
  164. fprintf (stdout,
  165. "Sender Rewriting Scheme implementation for Postfix.\n\n"
  166. "Implements two TCP lookup tables to rewrite mail addresses\n"
  167. "as needed. The forward SRS is for sender envelope addresses\n"
  168. "to prevent SPF-related bounces. The reverse SRS is for\n"
  169. "recipient envelope addresses so that bounced mails can be\n"
  170. "routed back to their original sender.\n"
  171. "\n"
  172. "Usage: %s -s<file> -d<domain> [other options]\n"
  173. "Options:\n"
  174. " -s<file> read secrets from file (required)\n"
  175. " -d<domain> set domain name for rewrite (required)\n"
  176. " -f<port> set port for the forward SRS lookup (default: 10001)\n"
  177. " -r<port> set port for the reverse SRS lookup (default: 10002)\n"
  178. " -p<pidfile> write process ID to pidfile (default: none)\n"
  179. " -c<dir> chroot to <dir> (default: none)\n"
  180. " -u<user> switch user id after port bind (default: none)\n"
  181. " -t<seconds> timeout for idle client connections (default: 1800)\n"
  182. " -D fork into background\n"
  183. " -4 force IPv4 socket (default: any)\n"
  184. " -6 force IPv6 socket (default: any)\n"
  185. " -h show this help\n"
  186. " -v show version\n"
  187. "\n",
  188. self
  189. );
  190. }
  191. typedef void(*handle_t)(srs_t*, FILE*, const char*, const char*);
  192. int main (int argc, char **argv)
  193. {
  194. int opt, timeout = 1800, family = AF_UNSPEC;
  195. int daemonize = FALSE;
  196. char *forward_service = NULL, *reverse_service = NULL,
  197. *user = NULL, *domain = NULL, *chroot_dir = NULL;
  198. int forward_sock, reverse_sock;
  199. char *secret_file = NULL, *pid_file = NULL;
  200. FILE *pf = NULL, *sf = NULL;
  201. struct passwd *pwd = NULL;
  202. char secretbuf[1024], *secret = NULL;
  203. char *tmp;
  204. srs_t *srs;
  205. struct pollfd fds[3];
  206. handle_t handler[2] = { handle_forward, handle_reverse };
  207. tmp = strrchr(argv[0], '/');
  208. if (tmp) self = strdup(tmp + 1); else self = strdup(argv[0]);
  209. while ((opt = getopt(argc, argv, "46d:f:r:s:u:t:p:c:Dhv")) != -1) {
  210. switch (opt) {
  211. case '?':
  212. return EXIT_FAILURE;
  213. case '4':
  214. family = AF_INET;
  215. break;
  216. case '6':
  217. family = AF_INET6;
  218. break;
  219. case 'd':
  220. domain = strdup(optarg);
  221. break;
  222. case 'f':
  223. forward_service = strdup(optarg);
  224. break;
  225. case 'r':
  226. reverse_service = strdup(optarg);
  227. break;
  228. case 't':
  229. timeout = atoi(optarg);
  230. break;
  231. case 's':
  232. secret_file = strdup(optarg);
  233. break;
  234. case 'p':
  235. pid_file = strdup(optarg);
  236. break;
  237. case 'u':
  238. user = strdup(optarg);
  239. break;
  240. case 'c':
  241. chroot_dir = strdup(optarg);
  242. break;
  243. case 'D':
  244. daemonize = TRUE;
  245. break;
  246. case 'h':
  247. show_help();
  248. return EXIT_SUCCESS;
  249. case 'v':
  250. fprintf (stdout, "%s\n", VERSION);
  251. return EXIT_SUCCESS;
  252. }
  253. }
  254. if (domain == NULL) {
  255. fprintf (stderr, "%s: You must set a home domain (-d)\n", self);
  256. show_help();
  257. return EXIT_FAILURE;
  258. }
  259. /* The stuff we do first may not be possible from within chroot or without privileges */
  260. /* Open pid file for writing (the actual process ID is filled in later) */
  261. if (pid_file) {
  262. pf = fopen (pid_file, "w");
  263. if (pf == NULL) {
  264. fprintf (stderr, "%s: Cannot write PID: %s\n\n", self, pid_file);
  265. return EXIT_FAILURE;
  266. }
  267. }
  268. /* Read secret. The default installation makes this root accessible only. */
  269. if (secret_file != NULL) {
  270. size_t len;
  271. sf = fopen(secret_file, "rb");
  272. if (sf == NULL) {
  273. fprintf (stderr, "%s: Cannot open file with secret: %s\n", self, secret_file);
  274. return EXIT_FAILURE;
  275. }
  276. } else {
  277. fprintf (stderr, "%s: You must set a secret (-s)\n", self);
  278. show_help();
  279. return EXIT_FAILURE;
  280. }
  281. /* Bind ports. May require privileges if the config specifies ports below 1024 */
  282. if (forward_service != NULL) {
  283. forward_sock = bind_service(forward_service, family);
  284. free (forward_service);
  285. } else {
  286. forward_sock = bind_service("10001", family);
  287. }
  288. if (forward_sock < 0) return EXIT_FAILURE;
  289. if (reverse_service != NULL) {
  290. reverse_sock = bind_service(reverse_service, family);
  291. free (reverse_service);
  292. } else {
  293. reverse_sock = bind_service("10002", family);
  294. }
  295. if (reverse_sock < 0) return EXIT_FAILURE;
  296. /* Open syslog now (NDELAY), because it may no longer reachable from chroot */
  297. openlog (self, LOG_PID | LOG_NDELAY, LOG_MAIL);
  298. /* We also have to lookup the uid of the unprivileged user for the same reason. */
  299. if (user) {
  300. errno = 0;
  301. pwd = getpwnam(user);
  302. if (pwd == NULL) {
  303. if (errno != 0)
  304. fprintf (stderr, "%s: Failed to lookup user: %s\n", self, strerror(errno));
  305. else
  306. fprintf (stderr, "%s: No such user: %s\n", self, user);
  307. return EXIT_FAILURE;
  308. }
  309. }
  310. /* Now we can chroot, which again requires root privileges */
  311. if (chroot_dir) {
  312. if (chdir(chroot_dir) < 0) {
  313. fprintf (stderr, "%s: Cannot change to chroot: %s\n", self, strerror(errno));
  314. return EXIT_FAILURE;
  315. }
  316. if (chroot(chroot_dir) < 0) {
  317. fprintf (stderr, "%s: Failed to enable chroot: %s\n", self, strerror(errno));
  318. return EXIT_FAILURE;
  319. }
  320. }
  321. /* Finally, we revert to the unprivileged user */
  322. if (pwd) {
  323. if (setuid(pwd->pw_uid) < 0) {
  324. fprintf (stderr, "%s: Failed to switch user id: %s\n", self, strerror(errno));
  325. return EXIT_FAILURE;
  326. }
  327. }
  328. /* Standard double fork technique to disavow all knowledge about the controlling terminal */
  329. if (daemonize) {
  330. close(0); close(1); close(2);
  331. if (fork() != 0) return EXIT_SUCCESS;
  332. setsid();
  333. if (fork() != 0) return EXIT_SUCCESS;
  334. }
  335. /* Make note of our actual process ID */
  336. if (pf) {
  337. fprintf (pf, "%d", (int)getpid());
  338. fclose (pf);
  339. }
  340. srs = srs_new();
  341. while ((secret = fgets(secretbuf, sizeof(secretbuf), sf))) {
  342. secret = strtok(secret, "\r\n");
  343. if (secret)
  344. srs_add_secret (srs, secret);
  345. }
  346. fclose (sf);
  347. srs_set_separator (srs, '+');
  348. fds[0].fd = forward_sock;
  349. fds[0].events = POLLIN;
  350. fds[1].fd = reverse_sock;
  351. fds[1].events = POLLIN;
  352. while(TRUE) {
  353. int i, conn;
  354. FILE *fp;
  355. char linebuf[1024], *line;
  356. char keybuf[1024], *key;
  357. if (poll(fds, 2, 1000) < 0) {
  358. if (daemonize)
  359. syslog (LOG_MAIL | LOG_ERR, "Poll failure: %s", strerror(errno));
  360. else
  361. fprintf (stderr, "%s: Poll failure: %s\n", self, strerror(errno));
  362. return EXIT_FAILURE;
  363. }
  364. for (i = 0; i < 2; ++i) {
  365. if (fds[i].revents) {
  366. conn = accept(fds[i].fd, NULL, NULL);
  367. if (conn < 0) continue;
  368. if (fork() == 0) {
  369. fp = fdopen(conn, "r+");
  370. if (fp == NULL) exit(EXIT_FAILURE);
  371. fds[2].fd = conn;
  372. fds[2].events = POLLIN;
  373. if (poll(&fds[2], 1, timeout * 1000) <= 0) return EXIT_FAILURE;
  374. line = fgets(linebuf, sizeof(linebuf), fp);
  375. while (line) {
  376. char* token;
  377. token = strtok(line, " \r\n");
  378. if (token == NULL || strcmp(token, "get") != 0) {
  379. fprintf (fp, "500 Invalid request\n");
  380. return EXIT_FAILURE;
  381. }
  382. token = strtok(NULL, "\r\n");
  383. if (!token) {
  384. fprintf (fp, "500 Invalid request\n");
  385. return EXIT_FAILURE;
  386. }
  387. key = url_decode(keybuf, sizeof(keybuf), token);
  388. if (!key) break;
  389. handler[i](srs, fp, key, domain);
  390. if (poll(&fds[2], 1, timeout * 1000) <= 0) break;
  391. line = fgets(linebuf, sizeof(linebuf), fp);
  392. }
  393. fclose (fp);
  394. return EXIT_SUCCESS;
  395. }
  396. close (conn);
  397. }
  398. }
  399. waitpid(-1, NULL, WNOHANG);
  400. }
  401. return EXIT_SUCCESS;
  402. }