zoukankan      html  css  js  c++  java
  • http 结构初始化

      简要而说:accept 到连接后 根据fd 构建一个connection  由于是 http ; 重新封装为http-connection;同时设置fd的读回调;

    回调函数根据是否是https/http 进行区别

    ngx_http_init_connection(ngx_connection_t *c) 
    //当建立连接后开辟ngx_http_connection_t结构,这里面存储该服务器端ip:port所在server{}上下文配置信息,和server_name信息等,然后让
    //ngx_connection_t->data指向该结构,这样就可以通过ngx_connection_t->data获取到服务器端的serv loc 等配置信息以及该server{}中的server_name信息
    
    {
        ngx_uint_t              i;
        ngx_event_t            *rev;
        struct sockaddr_in     *sin;
        ngx_http_port_t        *port;
        ngx_http_in_addr_t     *addr;
        ngx_http_log_ctx_t     *ctx;
        ngx_http_connection_t  *hc;
    #if (NGX_HAVE_INET6)
        struct sockaddr_in6    *sin6;
        ngx_http_in6_addr_t    *addr6;
    #endif
    
        //注意ngx_connection_t和ngx_http_connection_t的区别,前者是建立连接accept前使用的结构,后者是连接成功后使用的结构
        // 使用分层的思想看待问题   ev_connect  &&& http_connect
        hc = ngx_pcalloc(c->pool, sizeof(ngx_http_connection_t));
        if (hc == NULL) {
            ngx_http_close_connection(c);
            return;
        }
    
        //在服务器端accept客户端连接成功(ngx_event_accept)后,通过ngx_get_connection从连接池获取一个ngx_connection_t结构,
        //每个客户端连接对于一个ngx_connection_t结构,并且为其分配一个ngx_http_connection_t结构,ngx_connection_t->data = ngx_http_connection_t,
        c->data = hc;
    
        /* find the server configuration for the address:port */
    
        port = c->listening->servers;  
    
        if (port->naddrs > 1) {  
        
            /*
             * there are several addresses on this port and one of them
             * is an "*:port" wildcard so getsockname() in ngx_http_server_addr()
             * is required to determine a server address
             */
            //说明listen ip:port存在几条没有bind选项,并且存在通配符配置,如listen *:port,那么就需要通过ngx_connection_local_sockaddr来确定
        //究竟客户端是和那个本地ip地址建立的连接
            if (ngx_connection_local_sockaddr(c, NULL, 0) != NGX_OK) { //
                ngx_http_close_connection(c);
                return;
            }
    
            switch (c->local_sockaddr->sa_family) {
    
    #if (NGX_HAVE_INET6)
            case AF_INET6:
                sin6 = (struct sockaddr_in6 *) c->local_sockaddr;
    
                addr6 = port->addrs;
    
                /* the last address is "*" */
    
                for (i = 0; i < port->naddrs - 1; i++) {
                    if (ngx_memcmp(&addr6[i].addr6, &sin6->sin6_addr, 16) == 0) {
                        break;
                    }
                }
    
                hc->addr_conf = &addr6[i].conf;
    
                break;
    #endif
    
            default: /* AF_INET */
                sin = (struct sockaddr_in *) c->local_sockaddr;
    
                addr = port->addrs; 
    
                /* the last address is "*" */
                //根据上面的ngx_connection_local_sockaddr函数获取到客户端连接到本地,本地IP地址获取到后,遍历ngx_http_port_t找到对应
                //的IP地址和端口,然后赋值给ngx_http_connection_t->addr_conf,这里面存储有server_name配置信息以及该ip:port对应的上下文信息
                for (i = 0; i < port->naddrs - 1; i++) {
                    if (addr[i].addr == sin->sin_addr.s_addr) {
                        break;
                    }
                }
    
              /*
                    这里也体现了在ngx_http_init_connection中获取http{}上下文ctx,如果客户端请求中带有host参数,则会继续在ngx_http_set_virtual_server
                    中重新获取对应的server{}和location{},如果客户端请求不带host头部行,则使用默认的server{},见 ngx_http_init_connection  
                */
                hc->addr_conf = &addr[i].conf;
    
                break;
            }
    
        } else {
    
            switch (c->local_sockaddr->sa_family) {
    
    #if (NGX_HAVE_INET6)
            case AF_INET6:
                addr6 = port->addrs;
                hc->addr_conf = &addr6[0].conf;
                break;
    #endif
    
            default: /* AF_INET */
                addr = port->addrs;
                hc->addr_conf = &addr[0].conf;
                break;
            }
        }
    
        /* the default server configuration for the address:port */
        //listen add:port对于的 server{}配置块的上下文ctx
        hc->conf_ctx = hc->addr_conf->default_server->ctx;
    
        ctx = ngx_palloc(c->pool, sizeof(ngx_http_log_ctx_t));
        if (ctx == NULL) {
            ngx_http_close_connection(c);
            return;
        }
    
        ctx->connection = c;
        ctx->request = NULL;
        ctx->current_request = NULL;
    
        c->log->connection = c->number;
        c->log->handler = ngx_http_log_error;
        c->log->data = ctx;
        c->log->action = "waiting for request";
    
        c->log_error = NGX_ERROR_INFO;
    
        rev = c->read;
        // 设置read-ev 的回调
        rev->handler = ngx_http_wait_request_handler;
        c->write->handler = ngx_http_empty_handler;
    
    #if (NGX_HTTP_V2) 
        /* 这里放在SSL的前面是,如果没有配置SSL,则直接不用进行SSL协商而进行HTTP2处理ngx_http_v2_init */
        if (hc->addr_conf->http2) {
            rev->handler = ngx_http_v2_init;
        }
    #endif
    
    #if (NGX_HTTP_SSL)
        {
        ngx_http_ssl_srv_conf_t  *sscf;
    
        sscf = ngx_http_get_module_srv_conf(hc->conf_ctx, ngx_http_ssl_module);
    
        if (sscf->enable || hc->addr_conf->ssl) {
    
            c->log->action = "SSL handshaking";
    
            if (hc->addr_conf->ssl && sscf->ssl.ctx == NULL) {
                ngx_log_error(NGX_LOG_ERR, c->log, 0,
                              "no "ssl_certificate" is defined "
                              "in server listening on SSL port");
                ngx_http_close_connection(c);
                return;
            }
    
            hc->ssl = 1;//如果是 ssl/tls 后面 会进入 tls 的握手 并且关联sock-fd和 ssl 同时设置read-ev的回调解析tls协商
    
            rev->handler = ngx_http_ssl_handshake;
        }
        }
    #endif
    
        if (hc->addr_conf->proxy_protocol) {
            hc->proxy_protocol = 1;
            c->log->action = "reading PROXY protocol";
        }
    
        /*
         如果新连接的读事件ngx_event_t结构体中的标志位ready为1,实际上表示这个连接对应的套接字缓存上已经有用户发来的数据,
         这时就可调用上面说过的ngx_http_init_request方法处理请求。
         */
        //这里只可能是当listen的时候添加了defered参数并且内核支持,在ngx_event_accept的时候才会置1,才可能执行下面的if里面的内容,否则不会只需if里面的内容
        if (rev->ready) {
            /* the deferred accept(), iocp */
            if (ngx_use_accept_mutex) { //如果是配置了accept_mutex,则把该rev->handler延后处理,
            //实际上执行的地方为ngx_process_events_and_timers中的ngx_event_process_posted
                ngx_post_event(rev, &ngx_posted_events);
                return;
            }
    
            rev->handler(rev); //ngx_http_wait_request_handler
            return;
        }
    
    /*
    在有些情况下,当TCP连接建立成功时同时也出现了可读事件(例如,在套接字listen配置时设置了deferred选项时,内核仅在套接字上确实收到请求时才会通知epoll
    调度事件的回调方法。当然,在大部分情况下,ngx_http_init_request方法和
    ngx_http_init_connection方法都是由两个事件(TCP连接建立成功事件和连接上的可读事件)触发调用的
    */
    
    /*
    调用ngx_add_timer方法把读事件添加到定时器中,设置的超时时间则是nginx.conf中client_header_timeout配置项指定的参数。
    也就是说,如果经过client_header_timeout时间后这个连接上还没有用户数据到达,则会由定时器触发调用读事件的ngx_http_init_request处理方法。
     *///把接收事件添加到定时器中,当post_accept_timeout秒还没有客户端数据到来,就关闭连接-----干掉 idle 连接
        ngx_add_timer(rev, c->listening->post_accept_timeout, NGX_FUNC_LINE); 
        ngx_reusable_connection(c, 1);
        //将fd epoll add 到 ep 监听;当下次有数据从客户端发送过来的时候,会在ngx_epoll_process_events把对应的ready置1。
        if (ngx_handle_read_event(rev, 0, NGX_FUNC_LINE) != NGX_OK) { //
            ngx_http_close_connection(c);
            return;
        }
    }
  • 相关阅读:
    ckplayer的Error #2033:Can not call javascript:ckstyle()解决
    C#中的参数关键字params
    c#中的可选参数和命名参数的使用
    c#中的dynamic类型
    c#中关于变量声明那么点事
    C# 自定义控件的一些文章和博客
    datatable,查询,排序,复制等操作
    HTML5 实现图像模糊算法
    FASTCGI程序,做个备份,以后用
    PHP的一些函数
  • 原文地址:https://www.cnblogs.com/codestack/p/13472285.html
Copyright © 2011-2022 走看看