nginx模块编程之获取客户ip及端口号

时间:2024-09-02 16:33:20

ngx_request_t结构体中有一个connection定义,该定义指向一个ngx_connection_t的结构体:

结构体定义如下:

 struct ngx_connection_s {
void *data;
ngx_event_t *read;
ngx_event_t *write; ngx_socket_t fd; ngx_recv_pt recv;
ngx_send_pt send;
ngx_recv_chain_pt recv_chain;
ngx_send_chain_pt send_chain; ngx_listening_t *listening; off_t sent; ngx_log_t *log; ngx_pool_t *pool; struct sockaddr *sockaddr;
socklen_t socklen;
ngx_str_t addr_text; ngx_str_t proxy_protocol_addr; #if (NGX_SSL)
ngx_ssl_connection_t *ssl;
#endif struct sockaddr *local_sockaddr;
socklen_t local_socklen; ngx_buf_t *buffer; ngx_queue_t queue; ngx_atomic_uint_t number; ngx_uint_t requests; unsigned buffered:; unsigned log_error:; /* ngx_connection_log_error_e */ unsigned unexpected_eof:;
unsigned timedout:;
unsigned error:;
unsigned destroyed:; unsigned idle:;
unsigned reusable:;
unsigned close:; unsigned sendfile:;
unsigned sndlowat:;
unsigned tcp_nodelay:; /* ngx_connection_tcp_nodelay_e */
unsigned tcp_nopush:; /* ngx_connection_tcp_nopush_e */ unsigned need_last_buf:; #if (NGX_HAVE_IOCP)
unsigned accept_context_updated:;
#endif #if (NGX_HAVE_AIO_SENDFILE)
unsigned busy_count:;
#endif #if (NGX_THREADS)
ngx_thread_task_t *sendfile_task;
#endif
};

结构体中,有两个sockaddr地址结构定义,分别是sockaddr以及local_sockaddr;前者为客户端地址,后者为服务器端地址;下面来看sockaddr的定义:

struct sockaddr {
  unsigned short sa_family; /* address family, AF_xxx */
  char sa_data[]; /* 14 bytes of protocol address */
  };
说明:
sa_family是地址家族,是“AF_xxx”的形式。常设为“AF_INET”,代表Internet(TCP/IP)地址族。
sa_data是协议地址,由sa_family决定。如果sa_family=AF_INET,则sa_data就是sockaddr_in的
sin_addr和sin_port,用于为套接口储存目标地址和端口信息。为了解决struct
sockaddr赋值以及访问,提供了了一个并列的结构struct sockadd_in(“in”代表
“Internet”),换句话说,sockaddr的数据存储大小和结构完全和sockaddr_in一致,可以直接强制转换。
struct sockaddr_in{
short int sin_family; /*地址族信息,通常是AF-xxxx的形式*/
unsigned short int sin_port; /*端口信息*/
struct in_addr sin_addr; /*网络地址*/
unsigned char sin_zero[]; /*补位用的0,to make same size as struct sockaddr*/
}

关于这两个结构体,他们占用的内存大小是一样的,当sockaddr_in.sin_family = AF_INET时,他们的内存布局也一样的!看看sockaddr结构体自身就知道了,它仅仅是个char数组,大小与sockaddr_in等同。

这两个结构体使用转化的例子如下:

  • sockaddr_in结构强制转换成sockaddr结构再传入系统调用函数
/*sockaddr_in强制转化为sockaddr,用于传入系统调用函数*/
int sockfd;
struct sockaddr_in my_addr;
sockfd = socket(AF_INET, SOCK_STREAM, ); my_addr.sin_family = AF_INET; /* 主机字节序 */
my_addr.sin_port = htons(MYPORT); /* short, 网络字节序 */ my_addr.sin_addr.s_addr = inet_addr("192.168.0.1"); bzero(&(my_addr.sin_zero), ); /* zero the rest of the struct */
//memset(&my_addr.sin_zero, 0, 8); bind(sockfd, (struct sockaddr *)&my_addr, sizeof(struct sockaddr));//将sockaddr_in强制转化为sockaddr,用于传入系统调用函数
  • sockaddr结构强制转换成sockaddr_in结构,从而得到ip地址和端口号
struct sockaddr_in *ip = (struct sockaddr_in *) (r->connection->sockaddr);
char* addr = inet_ntoa(ip->sin_addr);//地址转化函数,返回字符串指针
unsigned short port = ntohs(ip->sin_port);//大小端转换