blob: 9ebac94e6f90d80c70f6e4dd3bcd7e1acb18cbd8 [file] [log] [blame]
Igor Sysoev6de5c2c2002-08-06 16:39:45 +00001
2/*
3 TODO: log pid and tid
4*/
5
6/*
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +00007 "[time as ctime()] [alert] 412#3 (32)Broken pipe: anything"
Igor Sysoev6de5c2c2002-08-06 16:39:45 +00008
9 "[time as ctime()] [alert] (32)Broken pipe: anything"
10 "[time as ctime()] [alert] anything"
11*/
12
13#include <ngx_config.h>
Igor Sysoev1c13c662003-05-20 15:37:55 +000014#include <ngx_core.h>
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000015
16
Igor Sysoev1c104622003-06-03 15:42:58 +000017static char *ngx_set_error_log(ngx_conf_t *cf, ngx_command_t *cmd, void *conf);
18
19
20static ngx_str_t errlog_name = ngx_string("errlog");
21
22static ngx_command_t ngx_errlog_commands[] = {
23
24 {ngx_string("error_log"),
25 NGX_MAIN_CONF|NGX_CONF_TAKE1,
26 ngx_set_error_log,
27 0,
28 0,
29 NULL},
30
31 ngx_null_command
32};
33
34
35ngx_module_t ngx_errlog_module = {
36 NGX_MODULE,
37 &errlog_name, /* module context */
38 ngx_errlog_commands, /* module directives */
39 NGX_CORE_MODULE, /* module type */
Igor Sysoeva7f7fa82003-07-11 04:50:59 +000040 NULL, /* init module */
41 NULL /* init child */
Igor Sysoev1c104622003-06-03 15:42:58 +000042};
43
44
Igor Sysoevf5e97c52003-06-26 15:35:36 +000045static ngx_open_file_t ngx_stderr;
46static ngx_log_t ngx_log;
Igor Sysoev1c104622003-06-03 15:42:58 +000047
48
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000049static const char *err_levels[] = {
Igor Sysoev1c13c662003-05-20 15:37:55 +000050 "stderr", "emerg", "alert", "crit", "error",
51 "warn", "notice", "info", "debug"
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000052};
53
54#if (HAVE_VARIADIC_MACROS)
55void ngx_log_error_core(int level, ngx_log_t *log, ngx_err_t err,
56 const char *fmt, ...)
57#else
58void ngx_log_error_core(int level, ngx_log_t *log, ngx_err_t err,
59 const char *fmt, va_list args)
60#endif
61{
Igor Sysoev1c13c662003-05-20 15:37:55 +000062 char errstr[MAX_ERROR_STR];
63 ngx_tm_t tm;
64 size_t len;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000065#if (HAVE_VARIADIC_MACROS)
Igor Sysoev1c13c662003-05-20 15:37:55 +000066 va_list args;
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000067#endif
Igor Sysoev1c104622003-06-03 15:42:58 +000068#if (WIN32)
69 int written;
70#endif
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000071
72 ngx_localtime(&tm);
Igor Sysoev0ad17c02002-08-26 15:18:19 +000073 len = ngx_snprintf(errstr, sizeof(errstr), "%4d/%02d/%02d %02d:%02d:%02d",
Igor Sysoev5eef6182002-12-15 21:08:04 +000074 tm.ngx_tm_year, tm.ngx_tm_mon, tm.ngx_tm_mday,
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000075 tm.ngx_tm_hour, tm.ngx_tm_min, tm.ngx_tm_sec);
76
Igor Sysoev0ad17c02002-08-26 15:18:19 +000077 len += ngx_snprintf(errstr + len, sizeof(errstr) - len - 1,
78 " [%s] ", err_levels[level]);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000079
Igor Sysoeva6717c42002-12-23 06:29:22 +000080 /* pid#tid */
Igor Sysoev0ad17c02002-08-26 15:18:19 +000081 len += ngx_snprintf(errstr + len, sizeof(errstr) - len - 1,
Igor Sysoev239baac2003-06-11 15:28:34 +000082 PID_FMT "#%d: ", ngx_getpid(), 0);
Igor Sysoev6de5c2c2002-08-06 16:39:45 +000083
84#if (HAVE_VARIADIC_MACROS)
85 va_start(args, fmt);
86 len += ngx_vsnprintf(errstr + len, sizeof(errstr) - len - 1, fmt, args);
87 va_end(args);
88#else
89 len += ngx_vsnprintf(errstr + len, sizeof(errstr) - len - 1, fmt, args);
90#endif
91
Igor Sysoev0ad17c02002-08-26 15:18:19 +000092 if (err) {
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +000093
Igor Sysoeva6717c42002-12-23 06:29:22 +000094#if (WIN32)
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +000095 if ((unsigned) err >= 0x80000000) {
Igor Sysoev0ad17c02002-08-26 15:18:19 +000096 len += ngx_snprintf(errstr + len, sizeof(errstr) - len - 1,
Igor Sysoeva6717c42002-12-23 06:29:22 +000097 " (%X: ", err);
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +000098 } else {
Igor Sysoev0ad17c02002-08-26 15:18:19 +000099 len += ngx_snprintf(errstr + len, sizeof(errstr) - len - 1,
Igor Sysoeva6717c42002-12-23 06:29:22 +0000100 " (%d: ", err);
Igor Sysoev1d8d9ee2003-04-28 15:06:39 +0000101 }
102#else
103 len += ngx_snprintf(errstr + len, sizeof(errstr) - len - 1,
104 " (%d: ", err);
105#endif
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000106
107 len += ngx_strerror_r(err, errstr + len, sizeof(errstr) - len - 1);
108 if (len < sizeof(errstr) - 2) {
109 errstr[len++] = ')';
110 } else {
111 len = sizeof(errstr) - 2;
112 }
113 }
114
Igor Sysoeva6717c42002-12-23 06:29:22 +0000115 if (level != NGX_LOG_DEBUG && log->handler) {
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000116 len += log->handler(log->data, errstr + len, sizeof(errstr) - len - 1);
Igor Sysoeva6717c42002-12-23 06:29:22 +0000117 }
Igor Sysoev0ad17c02002-08-26 15:18:19 +0000118
Igor Sysoeva6717c42002-12-23 06:29:22 +0000119 if (len > sizeof(errstr) - 2) {
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000120 len = sizeof(errstr) - 2;
Igor Sysoeva6717c42002-12-23 06:29:22 +0000121 }
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000122
Igor Sysoeva6717c42002-12-23 06:29:22 +0000123#if (WIN32)
124 errstr[len++] = '\r';
Igor Sysoeva6717c42002-12-23 06:29:22 +0000125 errstr[len++] = '\n';
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000126 if (log->file->fd) {
127 WriteFile(log->file->fd, errstr, len, &written, NULL);
Igor Sysoev1c104622003-06-03 15:42:58 +0000128 }
129#else
130 errstr[len++] = '\n';
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000131 write(log->file->fd, errstr, len);
Igor Sysoev1c104622003-06-03 15:42:58 +0000132#endif
133
Igor Sysoeva6717c42002-12-23 06:29:22 +0000134
135#if 0
136 errstr[len] = '\0';
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000137 fputs(errstr, stderr);
Igor Sysoev96f83772002-09-07 10:14:25 +0000138 fflush(stderr);
Igor Sysoeva6717c42002-12-23 06:29:22 +0000139#endif
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000140}
141
Igor Sysoev1c13c662003-05-20 15:37:55 +0000142
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000143#if !(HAVE_VARIADIC_MACROS)
144
145void ngx_log_error(int level, ngx_log_t *log, ngx_err_t err,
146 const char *fmt, ...)
147{
148 va_list args;
149
150 if (log->log_level >= level) {
151 va_start(args, fmt);
152 ngx_log_error_core(level, log, err, fmt, args);
153 va_end(args);
154 }
155}
156
Igor Sysoev1c13c662003-05-20 15:37:55 +0000157
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000158void ngx_log_debug_core(ngx_log_t *log, const char *fmt, ...)
159{
160 va_list args;
161
162 va_start(args, fmt);
163 ngx_log_error_core(NGX_LOG_DEBUG, log, 0, fmt, args);
164 va_end(args);
165}
166
Igor Sysoev1c13c662003-05-20 15:37:55 +0000167
Igor Sysoev6de5c2c2002-08-06 16:39:45 +0000168void ngx_assert_core(ngx_log_t *log, const char *fmt, ...)
169{
170 va_list args;
171
172 va_start(args, fmt);
173 ngx_log_error_core(NGX_LOG_ALERT, log, 0, fmt, args);
174 va_end(args);
175}
176
177#endif
Igor Sysoev1c13c662003-05-20 15:37:55 +0000178
179
Igor Sysoev1c104622003-06-03 15:42:58 +0000180#if 0
181
Igor Sysoev1c13c662003-05-20 15:37:55 +0000182void ngx_log_stderr(ngx_event_t *ev)
183{
184 char errstr[MAX_ERROR_STR];
185 ssize_t n;
186 ngx_err_t err;
187
188 for ( ;; ) {
189 n = read((ngx_fd_t) ev->data, errstr, sizeof(errstr - 1));
190
191 if (n == -1) {
Igor Sysoevfa73aac2003-05-21 13:28:21 +0000192 err = ngx_errno;
Igor Sysoev1c13c662003-05-20 15:37:55 +0000193 if (err == NGX_EAGAIN) {
194 return;
195 }
196
197 ngx_log_error(NGX_LOG_ALERT, &ngx_log, err, "read() failed");
198 return;
199 }
200
201 if (n == 0) {
202 ngx_log_error(NGX_LOG_ALERT, &ngx_log, 0, "stderr clolsed");
203 return;
204 }
205
206 errstr[n] = '\0';
207 ngx_log_error(NGX_LOG_STDERR, &ngx_log, 0, "%s", errstr);
208 }
209}
Igor Sysoev1c104622003-06-03 15:42:58 +0000210
211#endif
212
213
Igor Sysoev1c104622003-06-03 15:42:58 +0000214
215ngx_log_t *ngx_log_init_errlog()
216{
217#if (WIN32)
Igor Sysoevbe2cfc32003-06-15 18:32:13 +0000218
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000219 ngx_stderr.fd = GetStdHandle(STD_ERROR_HANDLE);
Igor Sysoev1c104622003-06-03 15:42:58 +0000220
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000221 if (ngx_stderr.fd == NGX_INVALID_FILE) {
Igor Sysoev1c104622003-06-03 15:42:58 +0000222 /* TODO: where we can log error ? */
223 return NULL;
224
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000225 } else if (ngx_stderr.fd == NULL) {
Igor Sysoev1c104622003-06-03 15:42:58 +0000226 /* there are no associated standard handles */
227 /* TODO: where we can log possible errors ? */
228 }
229
230#else
Igor Sysoevbe2cfc32003-06-15 18:32:13 +0000231
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000232 ngx_stderr.fd = STDERR_FILENO;
Igor Sysoevbe2cfc32003-06-15 18:32:13 +0000233
Igor Sysoev1c104622003-06-03 15:42:58 +0000234#endif
235
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000236 ngx_log.file = &ngx_stderr;
Igor Sysoev1c104622003-06-03 15:42:58 +0000237 ngx_log.log_level = NGX_LOG_INFO;
238 /* STUB */ ngx_log.log_level = NGX_LOG_DEBUG;
239
240 return &ngx_log;
241}
242
243
Igor Sysoevb8c367c2003-07-10 16:26:57 +0000244ngx_log_t *ngx_log_create_errlog(ngx_cycle_t *cycle, ngx_str_t *name)
Igor Sysoev96c56c92003-07-02 14:41:17 +0000245{
246 ngx_log_t *log;
247
248 ngx_test_null(log, ngx_pcalloc(cycle->pool, sizeof(ngx_log_t)), NULL);
249 ngx_test_null(log->file, ngx_push_array(&cycle->open_files), NULL);
Igor Sysoev340b03b2003-07-04 15:10:33 +0000250 log->file->fd = NGX_INVALID_FILE;
Igor Sysoevb8c367c2003-07-10 16:26:57 +0000251 if (name) {
252 log->file->name = *name;
253 }
Igor Sysoev96c56c92003-07-02 14:41:17 +0000254
255 return log;
256}
257
258
Igor Sysoev340b03b2003-07-04 15:10:33 +0000259static char *ngx_set_error_log(ngx_conf_t *cf, ngx_command_t *cmd, void *conf)
260{
261 ngx_str_t *value;
262
263 value = cf->args->elts;
264
265 if (value[1].len == 6 && ngx_strcmp(value[1].data, "stderr") == 0) {
266 cf->cycle->log->file = &ngx_stderr;
267
268 } else {
269 cf->cycle->log->file->name = value[1];
270 }
271
272 return NGX_CONF_OK;
273}
274
275
276#if 0
277
Igor Sysoev1c104622003-06-03 15:42:58 +0000278char *ngx_log_set_errlog(ngx_conf_t *cf, ngx_command_t *cmd, ngx_log_t *log)
279{
280 int len;
281 ngx_err_t err;
282 ngx_str_t *value;
283
284 value = cf->args->elts;
285
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000286 log->file->fd = ngx_open_file(value[1].data,
Igor Sysoev1c104622003-06-03 15:42:58 +0000287 NGX_FILE_RDWR,
288 NGX_FILE_CREATE_OR_OPEN|NGX_FILE_APPEND);
289
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000290 if (log->file->fd == NGX_INVALID_FILE) {
Igor Sysoev1c104622003-06-03 15:42:58 +0000291 err = ngx_errno;
292 len = ngx_snprintf(ngx_conf_errstr, sizeof(ngx_conf_errstr) - 1,
293 ngx_open_file_n " \"%s\" failed (%d: ",
294 value[1].data, err);
295 len += ngx_strerror_r(err, ngx_conf_errstr + len,
296 sizeof(ngx_conf_errstr) - len - 1);
297 ngx_conf_errstr[len++] = ')';
298 ngx_conf_errstr[len++] = '\0';
299 return ngx_conf_errstr;
300 }
301
302#if (WIN32)
Igor Sysoevf5e97c52003-06-26 15:35:36 +0000303 if (ngx_file_append_mode(log->file->fd) == NGX_ERROR) {
Igor Sysoev1c104622003-06-03 15:42:58 +0000304 err = ngx_errno;
305 len = ngx_snprintf(ngx_conf_errstr, sizeof(ngx_conf_errstr) - 1,
306 ngx_file_append_mode_n " \"%s\" failed (%d: ",
307 value[1].data, err);
308 len += ngx_strerror_r(err, ngx_conf_errstr + len,
309 sizeof(ngx_conf_errstr) - len - 1);
310 ngx_conf_errstr[len++] = ')';
311 ngx_conf_errstr[len++] = '\0';
312 return ngx_conf_errstr;
313 }
314#endif
315
316 return NGX_CONF_OK;
317}
Igor Sysoev340b03b2003-07-04 15:10:33 +0000318
319#endif