- 论坛徽章:
- 0
|
- /*
- #-------------------------------------------------------------------------------
- #
- # $Id: netstat-nat.c,v 1.28 2005/01/29 15:24:37 mardan Exp $
- #
- # $Log: netstat-nat.c,v $
- # Revision 1.28 2005/01/29 15:24:37 mardan
- # Some cleanups, bumped to version 1.4.5
- #
- # Revision 1.27 2005/01/23 16:33:09 mardan
- # Added protocol resolving
- #
- # Revision 1.26 2005/01/21 22:54:14 mardan
- # Added some forgotten states
- #
- # Revision 1.25 2005/01/01 17:02:24 mardan
- # Extraction of IPs and ports more dynamicly so it can be used with layer7 and
- # maybe others when layout of ip_conntrack changes
- # Added autoconf
- #
- # Revision 1.24 2003/09/01 20:36:52 mardan
- # Fixed small bug which didn't allow to display hostnames in expanded mode,
- # not enough bytes where allocated.
- #
- # Revision 1.23 2003/08/31 10:59:15 mardan
- # Merged patch from Guomundur D. Haraldsson <gdh@binhex.EU.org> which does a
- # more properly memory alloction and saver copies of variables.
- # Changed versions to v1.4.3. Ready to release if found stable.
- # Changed my e-mail to danny@tweegy.demon.nl
- #
- # Revision 1.22 2003/02/08 17:41:44 mardan
- # made some last minor changes.
- # ready to release v1.4.2
- #
- # Revision 1.21 2003/01/24 21:24:34 mardan
- # Added unknown protocol, display as 'raw'
- # Fixed hussle up in states when sorting connections
- #
- # Revision 1.20 2003/01/02 15:40:48 mardan
- # Merged patch from Marceln, which removes unused variables, more understandable
- # memory allocation error message, check to exit when there are no NAT connections
- # and making netstat-nat compatible with uLibC.
- # Updated files to v1.4.2
- #
- # Revision 1.19 2002/09/22 20:10:19 mardan
- # Added '-v: print version'
- # Added 'uninstall' to Makefile
- # Updated all other files.
- #
- # Revision 1.18 2002/09/22 17:16:08 mardan
- # Rewritten connection_table to allocate memory dynamicly.
- #
- # Revision 1.17 2002/09/12 19:32:12 mardan
- # Added display local connections to NAT box self
- # Updated README
- # Small changes in Makefile
- #
- # Revision 1.16 2002/09/08 20:23:48 mardan
- # Added sort by connection option. (source/destination IP/port)
- # Updated README and man-page.
- #
- # Revision 1.15 2002/08/07 19:25:59 mardan
- # Fixed bug, displayed wrong icmp connection in state REPLIED (dest was gateway).
- #
- # Revision 1.14 2002/08/07 19:02:54 mardan
- # Fixed 'icmp' bug. Segmentation fault occured when displaying NATed icmp connections.
- #
- # Revision 1.13 2002/08/06 19:32:54 mardan
- # Added small feature: no header output.
- # Lots of code cleanup.
- #
- # Revision 1.12 2002/08/03 00:22:22 mardan
- # Added portname resolving based on the listed names in 'services'.
- # Re-arranged the layout.
- # Added a Makefile and a header file.
- # Updated the README.
- #
- # Revision 1.11 2002/07/12 20:05:54 mardan
- # Added argument for extended view of hostnames.
- # Moved display-code into one function.
- # Removed most unnessacery code.
- # Updated README
- #
- # Revision 1.10 2002/07/10 19:58:33 mardan
- # Added filtering by destination-host, re-arranged some code to work properly.
- # Tested DNAT icmp and udp.(pls report if any bugs occur)
- # Fixed a few declaration bugs.
- #
- # Revision 1.9 2002/07/09 20:00:36 mardan
- # Added fully DNAT support (udp & icmp not fully tested yet, but should work),
- # including argument support for (S)(D)NAT selection.
- # Re-arranged layout code, can possible merged into one function.
- # Some few minor changes.
- # Started to work on destination-host selection.
- #
- # Revision 1.8 2002/07/07 20:27:47 mardan
- # Added display by source host/IP.
- # Made a few fixes/changes.
- # Updated the REAMDE.
- #
- # Revision 1.7 2002/06/30 19:55:41 mardan
- # Added README and COPYING (license) FILES.
- #
- # Revision 1.6 2002/06/23 16:27:26 mardan
- # Finished udp.
- # Maybe some layout changes in future? therwise tool is finished.
- #
- # Revision 1.5 2002/06/23 14:07:46 mardan
- # Added protocol arg option.
- # Todo: udp protocol
- #
- # Revision 1.4 2002/06/23 12:57:35 mardan
- # Added ident strings for test :-)
- #
- # Revision 1.3 2002/06/23 12:47:08 mardan
- # Fixed resolved hostname hussle-up/layout
- # Moved all source code into netstat-nat.c
- #
- # Revision 1.2 2002/06/23 11:56:09 mardan
- # Added NAT icmp display.
- # Still need to do udp (more states possible)
- # Really need to fix resolved hostnames display, still hussled up.
- #
- # Revision 1.1.1.1 2002/05/04 01:08:06 mardan
- # Initial import of netstat-nat, the C version.
- # Array pointers really needs to be fixed, still lots of other bugs..
- # So far only TCP displayed.
- # No commandline args for e.g. no_nameresolving, protocol.
- #
- #
- #
- # Copyright (c) 2005 by D.Wijsman (danny@tweegy.demon.nl).
- # All rights reserved.
- #
- # This program is free software; you can redistribute it and/or modify it
- # under the terms of the GNU General Public License as published by the Free
- # Software Foundation; either version 2 of the License, or (at your option)
- # any later version.
- #
- # This program is distributed in the hope that it will be useful, but WITHOUT
- # ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or
- # FITNESS FOR A PARTICULAR PURPOSE. See the GNU General Public License for
- # more details.
- #
- # You should have received a copy of the GNU General Public License
- # along with this program; see the file COPYING. If not, write to
- # the Free Software Foundation, 675 Mass Ave, Cambridge, MA 02139, USA.
- #
- #
- #-------------------------------------------------------------------------------
- */
- #include "netstat-nat.h"
- static char const rcsid[] = "$Id: netstat-nat.c,v 1.28 2005/01/29 15:24:37 mardan Exp $";
- char SRC_IP[50];
- char DST_IP[50];
- int SNAT = 1;
- int DNAT = 1;
- int LOCAL = 0;
- static char PROTOCOL[4];
- int connection_index = 0;
- char ***connection_table;
- int main(int argc, char *argv[])
- {
- const char *args = "hnp:s:d:SDxor:L?v";
- static int SORT_ROW = 1;
- static int EXT_VIEW = 0;
- static int RESOLVE = 1;
- static int no_hdr = 0;
- FILE *f;
- char line[200];
- char src[50];
- char dst[50];
- char buf[100];
- char buf2[100];
- char from[50] = "NATed Address";
- char dest[50] = "Foreign Address";
-
- char ***pa;
- char *store;
- int index, a, b, c, j, r;
-
- connection_table = (char ***) xcalloc((1) * sizeof(char **));
- // check parameters
- while ((c = getopt(argc, argv, args)) != -1) {
- switch (c) {
- case 'h':
- display_help();
- return 1;
- case '?':
- display_help();
- return 1;
- case 'v':
- printf("Version %s\n", VERSION);
- return(0);
- case 'n':
- RESOLVE = 0;
- break;
- case 'p':
- strcopy(PROTOCOL, sizeof(PROTOCOL), optarg);
- break;
- case 's':
- strcopy(SRC_IP, sizeof(SRC_IP), optarg);
- lookup_ip(SRC_IP, sizeof(SRC_IP));
- break;
- case 'd':
- strcopy(DST_IP, sizeof(DST_IP), optarg);
- lookup_ip(DST_IP, sizeof(DST_IP));
- break;
- case 'S':
- DNAT = 0;
- break;
- case 'D':
- SNAT = 0;
- break;
- case 'L':
- SNAT = 0;
- DNAT = 0;
- LOCAL = 1;
- break;
- case 'x':
- EXT_VIEW = 1;
- break;
- case 'o':
- no_hdr = 1;
- break;
- case 'r':
- if (optarg == NULL || optarg == '\0') {
- display_help();
- return 1;
- }
- if (strcmp(optarg, "scr") == 0) SORT_ROW = 1; //default
- if (strcmp(optarg, "dst") == 0) SORT_ROW = 2;
- if (strcmp(optarg, "src-port") == 0) SORT_ROW = 3;
- if (strcmp(optarg, "dst-port") == 0) SORT_ROW = 4;
- if (strcmp(optarg, "state") == 0) SORT_ROW = 5;
- break;
- }
- }
-
- // some checking for IPTables and read file
- if ((f = fopen("/proc/net/ip_conntrack","r")) == NULL) {
- // if ((f = fopen("./conntrack.dump","r")) == NULL) {
- printf("Could not read info about connections from the kernel, make sure netfilter is enabled in kernel or by modules.\n");
- return 1;
- }
-
- // process conntrack table
- if (!no_hdr) {
- if (LOCAL) {
- strcopy(from, sizeof(from), "Source Address");
- strcopy(dest, sizeof(dest), "Destination Address");
- }
- if (!EXT_VIEW) {
- printf("%-6s%-31s%-31s%-6s\n", "Proto", from, dest, "State");
- } else {
- printf("%-6s%-41s%-41s%-6s\n", "Proto", from, dest, "State");
- }
- }
- while (fgets(line, sizeof(line), f) != NULL)
- {
- process_entry(line);
- }
- fclose(f);
-
- // create index of arrays pointed to main connection array
- if (connection_index == 0) {
- // There are no connections at this moment!
- return (0);
- }
-
- pa = (char ***) xcalloc((connection_index) * sizeof(char **));
- for (index = 0; index < connection_index; index++) {
- pa[index] = (char **) xcalloc((ROWS) * sizeof(char *));
- for (j = 0; j < ROWS; j++) {
- pa[index][j] = (char *) xcalloc(2);
- pa[index][j] = &connection_table[index][j][0];
- }
- }
- // sort by protocol and defined row
- for (a = 0; a < connection_index - 1; a++) {
- for (b = a + 1; b < connection_index; b++) {
- r = strcmp(pa[a][0], pa[b][0]);
- if (r > 0) {
- for (j = 0; j < ROWS; j++) {
- store = pa[a][j];
- pa[a][j] = pa[b][j];
- pa[b][j] = store;
- }
- }
- if (r == 0) {
- if (strcmp(pa[a][SORT_ROW], pa[b][SORT_ROW]) > 0) {
- for (j = 0; j < ROWS; j++) {
- store = pa[a][j];
- pa[a][j] = pa[b][j];
- pa[b][j] = store;
- }
- }
- }
- }
- }
- // print connections
- for (index = 0; index < connection_index; index++) {
- if (RESOLVE) {
- lookup_hostname(&pa[index][1]);
- lookup_hostname(&pa[index][2]);
- if (strlen(pa[index][3]) > 0 || strlen(pa[index][4]) > 0) {
- lookup_portname(&pa[index][3], pa[index][0]);
- lookup_portname(&pa[index][4], pa[index][0]);
- }
- }
- if (!EXT_VIEW) {
- strcopy(buf, sizeof(buf), "");
- strncat(buf, pa[index][1], 29 - strlen(pa[index][3]));
- if (!strcmp(pa[index][0], "tcp") || !strcmp(pa[index][0], "udp")) {
- snprintf(buf2, sizeof(buf2), "%s:%s", buf, pa[index][3]);
- }
- else {
- snprintf(buf2, sizeof(buf2), "%s", buf);
- }
- snprintf(src, sizeof(src), "%-31s", buf2);
- strcopy(buf, sizeof(buf), "");
- strncat(buf, pa[index][2], 29 - strlen(pa[index][4]));
- if (!strcmp(pa[index][0], "tcp") || !strcmp(pa[index][0], "udp")) {
- snprintf(buf2, sizeof(buf2), "%s:%s", buf, pa[index][4]);
- }
- else {
- snprintf(buf2, sizeof(buf2), "%s", buf);
- }
- snprintf(dst, sizeof(dst), "%-31s", buf2);
- } else {
- strcopy(buf, sizeof(buf), "");
- strncat(buf, pa[index][1], 39 - strlen(pa[index][3]));
- if (!strcmp(pa[index][0], "tcp") || !strcmp(pa[index][0], "udp")) {
- snprintf(buf2, sizeof(buf2), "%s:%s", buf, pa[index][3]);
- }
- else {
- snprintf(buf2, sizeof(buf2), "%s", buf);
- }
- snprintf(src , sizeof(src), "%-41s", buf2);
- strcopy(buf, sizeof(buf), "");
- strncat(buf, pa[index][2], 39 - strlen(pa[index][4]));
- if (!strcmp(pa[index][0], "tcp") || !strcmp(pa[index][0], "udp")) {
- snprintf(buf2, sizeof(buf2), "%s:%s", buf, pa[index][4]);
- }
- else {
- snprintf(buf2, sizeof(buf2), "%s", buf);
- }
- snprintf(dst, sizeof(dst), "%-41s", buf2);
- }
- printf("%-6s%s%s%-11s\n", pa[index][0], src, dst, pa[index][5]);
- }
- return(0);
- }
- // get protocol
- int get_protocol(char *line, char *protocol)
- {
- int i,j, protocol_nr;
- char protocol_name[10] = "";
- char protocol_raw[5] = "";
-
- if (string_search(line, "tcp")) {
- memcpy(protocol, "tcp", 3);
- }
- else if (string_search(line, "udp")) {
- memcpy(protocol, "udp", 3);
- }
- else if (string_search(line, "icmp")) {
- memcpy(protocol, "icmp", 4);
- }
- else {
- // here we search for protocol number and give it a name (get_protocol_name)
- for (i = 0; i < strlen(line); i++ ) {
- if(!strncmp(&line[i], "unknown ", 9)) {
- i += 9;
- for (j = i; j < strlen(line); j++) {
- if (line[j] == ' ') {
- break;
- }
- strncat(protocol_raw, &line[j], 1);
- }
- protocol_nr = atoi(protocol_raw);
- get_protocol_name(protocol_name, protocol_nr);
- memcpy(protocol, protocol_name, 9);
- break;
- }
- }
- //memcpy(protocol, "raw", 3);
- }
- // printf("PROTO: %s\n", protocol);
- return(0);
- }
- // get connection status
- int get_connection_state(char *line, char *state)
- {
- if (string_search(line, "ESTABLISHED")) {
- memcpy(state, "ESTABLISHED", 11);
- }
- else if (string_search(line, "TIME_WAIT")) {
- memcpy(state, "TIME_WAIT", 9);
- }
- else if (string_search(line, "FIN_WAIT")) {
- memcpy(state, "FIN_WAIT", 8);
- }
- else if (string_search(line, "SYN_RECV")) {
- memcpy(state, "SYN_RECV", 8);
- }
- else if (string_search(line, "SYN_SENT")) {
- memcpy(state, "SYN_SENT", 8);
- }
- else if (string_search(line, "UNREPLIED")) {
- memcpy(state, "UNREPLIED", 9);
- }
- else if (string_search(line, "CLOSE")) {
- memcpy(state, "CLOSE", 5);
- }
- else if (string_search(line, "ASSURED")) {
- memcpy(state, "ASSURED", 7);
- }
- else {
- if (string_search(line, "udp")) {
- memcpy(state, "UNREPLIED", 9);
- }
- else {
- memcpy(state, " ", 1);
- }
- }
- // printf("STATE: %s\n", state);
- return(0);
- }
- void process_entry(char *line)
- {
- int count;
- count = 0;
- char srcip_f[16] = "";
- char dstip_f[16] = "";
- char srcip_s[16] = "";
- char dstip_s[16] = "";
- char srcport[6] = "";
- char dstport[6] = "";
- char protocol[5] = "";
- char state[12] = "";
- search_first_hit("src=", line, srcip_f);
- search_first_hit("dst=", line, dstip_f);
- search_sec_hit("src=", line, srcip_s);
- search_sec_hit("dst=", line, dstip_s);
- search_first_hit("sport=", line, srcport);
- search_first_hit("dport=", line, dstport);
- get_protocol(line, protocol);
- if (strcmp(PROTOCOL, "")) {
- if (strncmp(PROTOCOL, protocol, 3)) {
- // printf("RETURN\n");
- return;
- }
- }
- get_connection_state(line, state);
- if (SNAT) {
- if ((!strcmp(srcip_f, dstip_s) == 0) && (strcmp(dstip_f, srcip_s) == 0)) {
- check_src_dst(protocol, srcip_f, dstip_f, srcport, dstport, state);
- }
- }
- if (DNAT) {
- if ((strcmp(srcip_f, dstip_s) == 0) && (!strcmp(dstip_f, srcip_s) == 0)) {
- check_src_dst(protocol, srcip_f, srcip_s, srcport, dstport, state);
- }
- }
- if (LOCAL) {
- if ((strcmp(srcip_f, dstip_s) == 0) && (strcmp(dstip_f, srcip_s) == 0)) {
- check_src_dst(protocol, srcip_f, srcip_s, srcport, dstport, state);
- }
- }
- // printf("%s %s %s %s %s %s\n", protocol, srcip_f, dstip_f, srcip_s, dstip_s, state);
- }
复制代码 |
|