view utils/smpp-test2.c @ 256:63252528a56c

sip-out: add CDR log line
author Mychaela Falconia <falcon@freecalypso.org>
date Tue, 15 Aug 2023 10:11:51 -0800
parents e1d7db9d734c
children
line wrap: on
line source

/*
 * This program connects to an SMPP server in the role of a client,
 * sends a bind_transceiver request, and then goes into a receiving loop,
 * dumping everything that comes back from the server.  Received packets
 * are also checked for some command opcodes to which we need to generate
 * a response, and simple auto-responses are generated.
 */

#include <sys/types.h>
#include <sys/socket.h>
#include <netinet/in.h>
#include <arpa/inet.h>
#include <stdio.h>
#include <stdlib.h>
#include <string.h>
#include <strings.h>
#include <unistd.h>

static int tcpsock;
static struct sockaddr_in server_sin;
static u_char bind_req[64];
static unsigned bind_req_len;
static char system_id[16], password[9];
static u_char rx_hdr[16];
static unsigned rx_pkt_len, rx_command_id;

static void
construct_bind_req()
{
	u_char *dp;
	unsigned slen;

	dp = bind_req + 4;	/* length will be filled last */
	/* command_id */
	*dp++ = 0;
	*dp++ = 0;
	*dp++ = 0;
	*dp++ = 0x09;		/* bind_transceiver */
	/* empty command_status */
	*dp++ = 0;
	*dp++ = 0;
	*dp++ = 0;
	*dp++ = 0;
	/* sequence_number */
	*dp++ = 0;
	*dp++ = 0;
	*dp++ = 0;
	*dp++ = 1;
	/* system_id */
	slen = strlen(system_id) + 1;
	bcopy(system_id, dp, slen);
	dp += slen;
	/* password */
	slen = strlen(password) + 1;
	bcopy(password, dp, slen);
	dp += slen;
	/* system_type */
	strcpy(dp, "SMPP");
	dp += 5;
	/* interface_version */
	*dp++ = 0x34;
	/* addr_ton */
	*dp++ = 0;
	/* addr_npi */
	*dp++ = 0;
	/* address_range */
	*dp++ = 0;
	bind_req_len = dp - bind_req;
	bind_req[0] = bind_req_len >> 24;
	bind_req[1] = bind_req_len >> 16;
	bind_req[2] = bind_req_len >> 8;
	bind_req[3] = bind_req_len;
}

static void
print_bind_req()
{
	unsigned off, chunk;
	int i, c;

	printf("Constructed bind request of %u bytes\n", bind_req_len);
	for (off = 0; off < bind_req_len; off += chunk) {
		chunk = bind_req_len - off;
		if (chunk > 16)
			chunk = 16;
		printf("%02X:  ", off);
		for (i = 0; i < 16; i++) {
			if (i < chunk)
				printf("%02X ", bind_req[off + i]);
			else
				fputs("   ", stdout);
			if (i == 7 || i == 15)
				putchar(' ');
		}
		for (i = 0; i < chunk; i++) {
			c = bind_req[off + i];
			if (c < ' ' || c > '~')
				c = '.';
			putchar(c);
		}
		putchar('\n');
	}
}

static void
init_stage()
{
	int rc;

	rc = connect(tcpsock, (struct sockaddr *) &server_sin,
		     sizeof(struct sockaddr_in));
	if (rc < 0) {
		perror("connect");
		exit(1);
	}
	rc = write(tcpsock, bind_req, bind_req_len);
	if (rc != bind_req_len) {
		perror("write");
		exit(1);
	}
}

static void
rx_bytes(buf, need_len)
	u_char *buf;
	unsigned need_len;
{
	int cc;
	unsigned remain;

	for (remain = need_len; remain; remain -= cc) {
		cc = read(tcpsock, buf, remain);
		if (cc <= 0) {
			perror("read");
			exit(1);
		}
	}
}

static void
print_hdr()
{
	int i, j, pos;

	fputs("Got header:", stdout);
	pos = 0;
	for (i = 0; i < 4; i++) {
		putchar(' ');
		for (j = 0; j < 4; j++)
			printf("%02X", rx_hdr[pos++]);
	}
	putchar('\n');
}

static void
preen_rx_hdr()
{
	rx_pkt_len = (rx_hdr[0] << 24) | (rx_hdr[1] << 16) | (rx_hdr[2] << 8) |
			rx_hdr[3];
	printf("Rx packet length: %u bytes\n", rx_pkt_len);
	if (rx_pkt_len < 16) {
		printf("Error: packet length is too short\n");
		exit(1);
	}
	rx_command_id = (rx_hdr[4] << 24) | (rx_hdr[5] << 16) |
			(rx_hdr[6] << 8) | rx_hdr[7];
}

static void
read_and_dump_body()
{
	u_char buf[16];
	unsigned offset, chunk;
	int i, c;

	for (offset = 16; offset < rx_pkt_len; offset += chunk) {
		chunk = rx_pkt_len - offset;
		if (chunk > 16)
			chunk = 16;
		rx_bytes(buf, chunk);
		printf("%08X:  ", offset);
		for (i = 0; i < 16; i++) {
			if (i < chunk)
				printf("%02X ", buf[i]);
			else
				fputs("   ", stdout);
			if (i == 7 || i == 15)
				putchar(' ');
		}
		for (i = 0; i < chunk; i++) {
			c = buf[i];
			if (c < ' ' || c > '~')
				c = '.';
			putchar(c);
		}
		putchar('\n');
	}
}

static void
resp_hdr_only()
{
	u_char resp[16];
	int rc;

	/* command_length */
	resp[0] = 0;
	resp[1] = 0;
	resp[2] = 0;
	resp[3] = 16;
	/* command_id */
	resp[4] = rx_hdr[4] | 0x80;
	resp[5] = rx_hdr[5];
	resp[6] = rx_hdr[6];
	resp[7] = rx_hdr[7];
	/* command_status */
	resp[8] = 0;
	resp[9] = 0;
	resp[10] = 0;
	resp[11] = 0;
	/* sequence_number */
	resp[12] = rx_hdr[12];
	resp[13] = rx_hdr[13];
	resp[14] = rx_hdr[14];
	resp[15] = rx_hdr[15];
	/* good to go */
	rc = write(tcpsock, resp, 16);
	if (rc != 16) {
		perror("write");
		exit(1);
	}
}

static void
resp_dummy_msgid()
{
	u_char resp[17];
	int rc;

	/* command_length */
	resp[0] = 0;
	resp[1] = 0;
	resp[2] = 0;
	resp[3] = 17;
	/* command_id */
	resp[4] = rx_hdr[4] | 0x80;
	resp[5] = rx_hdr[5];
	resp[6] = rx_hdr[6];
	resp[7] = rx_hdr[7];
	/* command_status */
	resp[8] = 0;
	resp[9] = 0;
	resp[10] = 0;
	resp[11] = 0;
	/* sequence_number */
	resp[12] = rx_hdr[12];
	resp[13] = rx_hdr[13];
	resp[14] = rx_hdr[14];
	resp[15] = rx_hdr[15];
	/* empty message_id */
	resp[16] = 0;
	/* good to go */
	rc = write(tcpsock, resp, 17);
	if (rc != 17) {
		perror("write");
		exit(1);
	}
}

static void
auto_resp_logic()
{
	switch (rx_command_id) {
	case 0x005:
		printf("Got deliver_sm, responding with deliver_sm_resp\n");
		resp_dummy_msgid();
		return;
	case 0x103:
		printf("Got data_sm, responding with data_sm_resp\n");
		resp_dummy_msgid();
		return;
	case 0x015:
		printf("Got enquire_link, responding with enquire_link_resp\n");
		resp_hdr_only();
		return;
	}
}

main(argc, argv)
	char **argv;
{
	if (argc < 3 || argc > 4) {
		fprintf(stderr, "usage: %s server-ip system-id [password]\n",
			argv[0]);
		exit(1);
	}
	server_sin.sin_family = AF_INET;
	server_sin.sin_addr.s_addr = inet_addr(argv[1]);
	if (server_sin.sin_addr.s_addr == INADDR_NONE) {
		fprintf(stderr, "error: invalid IP address argument \"%s\"\n",
			argv[1]);
		exit(1);
	}
	server_sin.sin_port = htons(2775);
	if (strlen(argv[2]) > 15) {
		fprintf(stderr, "error: system-id string is too long\n");
		exit(1);
	}
	strcpy(system_id, argv[2]);
	if (argv[3]) {
		if (strlen(argv[3]) > 8) {
			fprintf(stderr, "error: password string is too long\n");
			exit(1);
		}
		strcpy(password, argv[3]);
	}
	construct_bind_req();
	setlinebuf(stdout);
	print_bind_req();
	tcpsock = socket(AF_INET, SOCK_STREAM, 0);
	if (tcpsock < 0) {
		perror("socket");
		exit(1);
	}
	init_stage();
	for (;;) {
		rx_bytes(rx_hdr, 16);
		print_hdr();
		preen_rx_hdr();
		read_and_dump_body();
		auto_resp_logic();
	}
}