Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 1 | |
| 2 | #include <ngx_config.h> |
Igor Sysoev | d581fd5 | 2003-05-13 16:02:32 +0000 | [diff] [blame] | 3 | #include <ngx_core.h> |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 4 | #include <ngx_http.h> |
Igor Sysoev | dc479b4 | 2003-03-20 16:09:44 +0000 | [diff] [blame] | 5 | #include <ngx_http_config.h> |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 6 | #include <ngx_http_core_module.h> |
Igor Sysoev | 7300977 | 2003-02-06 17:21:13 +0000 | [diff] [blame] | 7 | #include <ngx_http_output_filter.h> |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 8 | |
Igor Sysoev | dc479b4 | 2003-03-20 16:09:44 +0000 | [diff] [blame] | 9 | |
Igor Sysoev | 2b54238 | 2002-08-20 14:48:28 +0000 | [diff] [blame] | 10 | |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 11 | int ngx_http_static_handler(ngx_http_request_t *r) |
| 12 | { |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 13 | int rc, key, i; |
| 14 | ngx_log_e level; |
| 15 | ngx_err_t err; |
| 16 | ngx_hunk_t *h; |
| 17 | ngx_http_type_t *type; |
| 18 | ngx_http_log_ctx_t *ctx; |
| 19 | ngx_http_core_loc_conf_t *clcf; |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 20 | |
Igor Sysoev | 6ddfbf0 | 2003-05-15 15:42:53 +0000 | [diff] [blame] | 21 | rc = ngx_http_discard_body(r); |
| 22 | |
| 23 | if (rc != NGX_OK) { |
| 24 | return rc; |
| 25 | } |
| 26 | |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 27 | if (r->method != NGX_HTTP_GET && r->method != NGX_HTTP_HEAD) { |
| 28 | return NGX_HTTP_NOT_ALLOWED; |
| 29 | } |
| 30 | |
Igor Sysoev | a58e3ca | 2002-09-02 14:48:24 +0000 | [diff] [blame] | 31 | ctx = r->connection->log->data; |
| 32 | ctx->action = "sending response"; |
| 33 | |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 34 | if (r->file.fd == NGX_INVALID_FILE) { |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 35 | r->file.fd = ngx_open_file(r->file.name.data, NGX_FILE_RDONLY); |
Igor Sysoev | b086905 | 2002-12-10 18:05:12 +0000 | [diff] [blame] | 36 | |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 37 | if (r->file.fd == NGX_INVALID_FILE) { |
| 38 | err = ngx_errno; |
Igor Sysoev | e0268b9 | 2002-09-11 15:18:33 +0000 | [diff] [blame] | 39 | |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 40 | if (err == NGX_ENOENT || err == NGX_ENOTDIR) { |
| 41 | level = NGX_LOG_ERR; |
| 42 | rc = NGX_HTTP_NOT_FOUND; |
Igor Sysoev | fd67586 | 2003-04-11 16:01:14 +0000 | [diff] [blame] | 43 | |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 44 | } else { |
| 45 | level = NGX_LOG_CRIT; |
| 46 | rc = NGX_HTTP_INTERNAL_SERVER_ERROR; |
| 47 | } |
| 48 | |
| 49 | ngx_log_error(level, r->connection->log, ngx_errno, |
| 50 | ngx_open_file_n " %s failed", r->file.name.data); |
| 51 | return rc; |
Igor Sysoev | fd67586 | 2003-04-11 16:01:14 +0000 | [diff] [blame] | 52 | } |
Igor Sysoev | 2b54238 | 2002-08-20 14:48:28 +0000 | [diff] [blame] | 53 | } |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 54 | |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 55 | if (!r->file.info_valid) { |
| 56 | if (ngx_stat_fd(r->file.fd, &r->file.info) == NGX_FILE_ERROR) { |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 57 | ngx_log_error(NGX_LOG_CRIT, r->connection->log, ngx_errno, |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 58 | ngx_stat_fd_n " %s failed", r->file.name.data); |
| 59 | |
| 60 | if (ngx_close_file(r->file.fd) == NGX_FILE_ERROR) |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 61 | ngx_log_error(NGX_LOG_ALERT, r->connection->log, ngx_errno, |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 62 | ngx_close_file_n " %s failed", r->file.name.data); |
| 63 | |
| 64 | return NGX_HTTP_INTERNAL_SERVER_ERROR; |
| 65 | } |
| 66 | |
| 67 | r->file.info_valid = 1; |
| 68 | } |
| 69 | |
Igor Sysoev | 6ddfbf0 | 2003-05-15 15:42:53 +0000 | [diff] [blame] | 70 | #if !(WIN32) /* the not regular files are probably Unix specific */ |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 71 | |
| 72 | if (!ngx_is_file(r->file.info)) { |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 73 | ngx_log_error(NGX_LOG_CRIT, r->connection->log, ngx_errno, |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 74 | "%s is not regular file", r->file.name.data); |
Igor Sysoev | e0268b9 | 2002-09-11 15:18:33 +0000 | [diff] [blame] | 75 | |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 76 | if (ngx_close_file(r->file.fd) == NGX_FILE_ERROR) |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 77 | ngx_log_error(NGX_LOG_ALERT, r->connection->log, ngx_errno, |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 78 | ngx_close_file_n " %s failed", r->file.name.data); |
| 79 | |
| 80 | return NGX_HTTP_NOT_FOUND; |
Igor Sysoev | 2b54238 | 2002-08-20 14:48:28 +0000 | [diff] [blame] | 81 | } |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 82 | |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 83 | #endif |
| 84 | |
Igor Sysoev | a0bb31f | 2002-12-02 16:09:40 +0000 | [diff] [blame] | 85 | r->headers_out.status = NGX_HTTP_OK; |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 86 | r->headers_out.content_length = ngx_file_size(r->file.info); |
| 87 | r->headers_out.last_modified_time = ngx_file_mtime(r->file.info); |
Igor Sysoev | 2b54238 | 2002-08-20 14:48:28 +0000 | [diff] [blame] | 88 | |
Igor Sysoev | b086905 | 2002-12-10 18:05:12 +0000 | [diff] [blame] | 89 | ngx_test_null(r->headers_out.content_type, |
| 90 | ngx_push_table(r->headers_out.headers), |
| 91 | NGX_HTTP_INTERNAL_SERVER_ERROR); |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 92 | |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 93 | r->headers_out.content_type->key.len = 0; |
| 94 | r->headers_out.content_type->key.data = NULL; |
| 95 | r->headers_out.content_type->value.len = 0; |
| 96 | r->headers_out.content_type->value.data = NULL; |
| 97 | |
| 98 | clcf = ngx_http_get_module_loc_conf(r, ngx_http_core_module); |
Igor Sysoev | b086905 | 2002-12-10 18:05:12 +0000 | [diff] [blame] | 99 | |
Igor Sysoev | b086905 | 2002-12-10 18:05:12 +0000 | [diff] [blame] | 100 | if (r->exten.len) { |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 101 | ngx_http_types_hash_key(key, r->exten); |
Igor Sysoev | 682bf8e | 2002-09-16 15:01:44 +0000 | [diff] [blame] | 102 | |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 103 | type = (ngx_http_type_t *) clcf->types[key].elts; |
| 104 | for (i = 0; i < clcf->types[key].nelts; i++) { |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 105 | if (r->exten.len != type[i].exten.len) { |
| 106 | continue; |
| 107 | } |
| 108 | |
| 109 | if (ngx_strcasecmp(r->exten.data, type[i].exten.data) == 0) { |
| 110 | r->headers_out.content_type->value.len = type[i].type.len; |
| 111 | r->headers_out.content_type->value.data = type[i].type.data; |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 112 | |
| 113 | break; |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 114 | } |
| 115 | } |
| 116 | } |
| 117 | |
| 118 | if (r->headers_out.content_type->value.len == 0) { |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 119 | r->headers_out.content_type->value.len = clcf->default_type.len; |
| 120 | r->headers_out.content_type->value.data = clcf->default_type.data; |
Igor Sysoev | 682bf8e | 2002-09-16 15:01:44 +0000 | [diff] [blame] | 121 | } |
Igor Sysoev | 2b54238 | 2002-08-20 14:48:28 +0000 | [diff] [blame] | 122 | |
Igor Sysoev | 79a8048 | 2003-05-14 17:13:13 +0000 | [diff] [blame] | 123 | /* we need to allocate all before the header would be sent */ |
Igor Sysoev | a58e3ca | 2002-09-02 14:48:24 +0000 | [diff] [blame] | 124 | ngx_test_null(h, ngx_pcalloc(r->pool, sizeof(ngx_hunk_t)), |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 125 | NGX_HTTP_INTERNAL_SERVER_ERROR); |
Igor Sysoev | a58e3ca | 2002-09-02 14:48:24 +0000 | [diff] [blame] | 126 | |
Igor Sysoev | a58e3ca | 2002-09-02 14:48:24 +0000 | [diff] [blame] | 127 | ngx_test_null(h->file, ngx_pcalloc(r->pool, sizeof(ngx_file_t)), |
| 128 | NGX_HTTP_INTERNAL_SERVER_ERROR); |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 129 | |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 130 | |
| 131 | rc = ngx_http_send_header(r); |
| 132 | |
| 133 | if (r->header_only) { |
| 134 | if (rc == NGX_AGAIN) { |
| 135 | ngx_http_set_write_handler(r); |
| 136 | |
| 137 | } else { |
| 138 | ngx_http_finalize_request(r, 0); |
| 139 | } |
| 140 | |
Igor Sysoev | d581fd5 | 2003-05-13 16:02:32 +0000 | [diff] [blame] | 141 | return NGX_OK; |
Igor Sysoev | 6253ca1 | 2003-05-27 12:18:54 +0000 | [diff] [blame^] | 142 | } |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 143 | |
Igor Sysoev | 0a9d145 | 2003-02-11 16:42:23 +0000 | [diff] [blame] | 144 | |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 145 | h->type = NGX_HUNK_FILE|NGX_HUNK_LAST; |
Igor Sysoev | b738757 | 2003-03-11 20:38:13 +0000 | [diff] [blame] | 146 | h->file_pos = 0; |
| 147 | h->file_last = ngx_file_size(r->file.info); |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 148 | |
| 149 | h->file->fd = r->file.fd; |
Igor Sysoev | a58e3ca | 2002-09-02 14:48:24 +0000 | [diff] [blame] | 150 | h->file->log = r->connection->log; |
| 151 | |
| 152 | rc = ngx_http_output_filter(r, h); |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 153 | |
Igor Sysoev | d581fd5 | 2003-05-13 16:02:32 +0000 | [diff] [blame] | 154 | if (r->main == NULL) { |
| 155 | if (rc == NGX_AGAIN) { |
| 156 | ngx_http_set_write_handler(r); |
Igor Sysoev | 42feecb | 2002-12-15 06:25:09 +0000 | [diff] [blame] | 157 | |
Igor Sysoev | d581fd5 | 2003-05-13 16:02:32 +0000 | [diff] [blame] | 158 | } else { |
| 159 | ngx_http_finalize_request(r, 0); |
Igor Sysoev | 0a9d145 | 2003-02-11 16:42:23 +0000 | [diff] [blame] | 160 | } |
Igor Sysoev | 0a9d145 | 2003-02-11 16:42:23 +0000 | [diff] [blame] | 161 | } |
| 162 | |
Igor Sysoev | d581fd5 | 2003-05-13 16:02:32 +0000 | [diff] [blame] | 163 | return NGX_OK; |
Igor Sysoev | 0c331d9 | 2002-08-15 17:20:26 +0000 | [diff] [blame] | 164 | } |