目录

Redis源码-4 日志和时间

目标

在平时的开发过程中,调试代码是很重要的。对于调试,打印日志又是最常见,最快速的方式,本文研究redis中的日志模块。研究其中的使用方式。

整体分析

redis中的日志模块写在了server.c中,没有独立成一个文件。本文以最小可运行的方式,抽离出日志模块,以及日志依赖的时间模块。 对于日志,redis封装了serverLog, 该方法第一个参数是日志等级,剩余参数会拼接成字符串。保持了和高级语言一样的方式。可以把日志输出到stdout, 也可以输出到指定文件。 日志中会包含时间,redis又封装了不同精度的时间。 主要关注unix时间戳。它支持原子操作。

源代码

源码

准备工作:从redis源码中拷贝代码

1
2
cp /home/vagrant/github/server_installer/servers/redis/redis-6.2/src/localtime.c .
cp /home/vagrant/github/server_installer/servers/redis/redis-6.2/src/atomicvar.h .

新建server.c

  1
  2
  3
  4
  5
  6
  7
  8
  9
 10
 11
 12
 13
 14
 15
 16
 17
 18
 19
 20
 21
 22
 23
 24
 25
 26
 27
 28
 29
 30
 31
 32
 33
 34
 35
 36
 37
 38
 39
 40
 41
 42
 43
 44
 45
 46
 47
 48
 49
 50
 51
 52
 53
 54
 55
 56
 57
 58
 59
 60
 61
 62
 63
 64
 65
 66
 67
 68
 69
 70
 71
 72
 73
 74
 75
 76
 77
 78
 79
 80
 81
 82
 83
 84
 85
 86
 87
 88
 89
 90
 91
 92
 93
 94
 95
 96
 97
 98
 99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
155
156
157
#include "stdio.h"
#include "server.h"
#include <sys/time.h>
#include <syslog.h>
#include <stdarg.h>


#define NET_IP_STR_LEN 46 /* INET6_ADDRSTRLEN is 46, but we need to be sure */
struct redisServer  server;
/*
 * Gets the proper timezone in a more portable fashion
 * i.e timezone variables are linux specific.
 */
long getTimeZone(void) {
#if defined(__linux__) || defined(__sun)
    return timezone;
#else
    struct timeval tv;
    struct timezone tz;

    gettimeofday(&tv, &tz);

    return tz.tz_minuteswest * 60L;
#endif
}


/* We take a cached value of the unix time in the global state because with
 * virtual memory and aging there is to store the current time in objects at
 * every object access, and accuracy is not needed. To access a global var is
 * a lot faster than calling time(NULL).
 *
 * This function should be fast because it is called at every command execution
 * in call(), so it is possible to decide if to update the daylight saving
 * info or not using the 'update_daylight_info' argument. Normally we update
 * such info only when calling this function from serverCron() but not when
 * calling it from call(). */

void updateCachedTime(int update_daylight_info) {
    server.ustime = ustime();
    server.mstime = server.ustime / 1000;
    time_t unixtime = server.mstime / 1000;
    atomicSet(server.unixtime, unixtime);

    /* To get information about daylight saving time, we need to call
     * localtime_r and cache the result. However calling localtime_r in this
     * context is safe since we will never fork() while here, in the main
     * thread. The logging function will call a thread safe version of
     * localtime that has no locks. */
    if (update_daylight_info) {
        struct tm tm;
        time_t ut = server.unixtime;
        localtime_r(&ut,&tm);
        server.daylight_active = tm.tm_isdst;
    }
}


/* Return the UNIX time in microseconds */
long long ustime(void) {
    struct timeval tv;
    long long ust;

    gettimeofday(&tv, NULL);
    ust = ((long long)tv.tv_sec)*1000000;
    ust += tv.tv_usec;
    return ust;
}


void initServerConfig(void) {
    updateCachedTime(1);

    server.timezone = getTimeZone();
}

int main() {
    tzset();
    initServerConfig();
    server.sentinel_mode = 0;
    server.masterhost = NULL;
    //server.logfile = "/tmp/server.log";
    server.logfile = "";
    server.verbosity = 0;
    server.syslog_enabled = 1;
    serverLog(LL_WARNING, " Redis is starting unixtime is: %ld", server.unixtime);
}





/* Like serverLogRaw() but with printf-alike support. This is the function that
 * is used across the code. The raw version is only used in order to dump
 * the INFO output on crash. */
void _serverLog(int level, const char *fmt, ...) {
    va_list ap;
    char msg[LOG_MAX_LEN];

    va_start(ap, fmt);
    vsnprintf(msg, sizeof(msg), fmt, ap);
    va_end(ap);

    serverLogRaw(level,msg);
}

/*============================ Utility functions ============================ */

/* We use a private localtime implementation which is fork-safe. The logging
 * function of Redis may be called from other threads. */
void nolocks_localtime(struct tm *tmp, time_t t, time_t tz, int dst);


/* Low level logging. To use only for very big messages, otherwise
 * serverLog() is to prefer. */
void serverLogRaw(int level, const char *msg) {
    const int syslogLevelMap[] = { LOG_DEBUG, LOG_INFO, LOG_NOTICE, LOG_WARNING };
    const char *c = ".-*#";
    FILE *fp;
    char buf[64];
    int rawmode = (level & LL_RAW);
    int log_to_stdout = server.logfile[0] == '\0';

    level &= 0xff; /* clear flags */
    if (level < server.verbosity) return;

    fp = log_to_stdout ? stdout : fopen(server.logfile,"a");
    if (!fp) return;

    if (rawmode) {
        fprintf(fp,"%s",msg);
    } else {
        int off;
        struct timeval tv;
        int role_char;
        pid_t pid = getpid();

        gettimeofday(&tv,NULL);
        struct tm tm;
        nolocks_localtime(&tm,tv.tv_sec,server.timezone,server.daylight_active);
        off = strftime(buf,sizeof(buf),"%d %b %Y %H:%M:%S.",&tm);
        snprintf(buf+off,sizeof(buf)-off,"%03d",(int)tv.tv_usec/1000);
        if (server.sentinel_mode) {
            role_char = 'X'; /* Sentinel. */
        } else if (pid != server.pid) {
            role_char = 'C'; /* RDB / AOF writing child. */
        } else {
            role_char = (server.masterhost ? 'S':'M'); /* Slave or Master. */
        }
        fprintf(fp,"%d:%c %s %c %s\n",
            (int)getpid(),role_char, buf,c[level],msg);
    }
    fflush(fp);

    if (!log_to_stdout) fclose(fp);
    if (server.syslog_enabled) syslog(syslogLevelMap[level], "%s", msg);
}

新建server.h

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
#include <fcntl.h>
#include <time.h>
#include "atomicvar.h"
#include <unistd.h>
#define LOG_MAX_LEN    1024 /* Default maximum length of syslog messages.*/

typedef long long ustime_t; /* microsecond time type. */
typedef long long mstime_t; /* millisecond time type. */

long long ustime(void);


/* Log levels */
#define LL_DEBUG 0
#define LL_VERBOSE 1
#define LL_NOTICE 2
#define LL_WARNING 3
#define LL_RAW (1<<10) /* Modifier to log without timestamp */


/* Use macro for checking log level to avoid evaluating arguments in cases log
 * should be ignored due to low level. */
#define serverLog(level, ...) do {\
        if (((level)&0xff) < server.verbosity) break;\
        _serverLog(level, __VA_ARGS__);\
    } while(0)


#ifdef __GNUC__
void _serverLog(int level, const char *fmt, ...)
    __attribute__((format(printf, 2, 3)));
#else
void _serverLog(int level, const char *fmt, ...);
#endif
void serverLogRaw(int level, const char *msg);


struct redisServer {
    pid_t pid;                  /* Main process pid. */
    char *masterhost;               /* Hostname of master */
    int sentinel_mode;          /* True if this instance is a Sentinel. */

    char *logfile;                  /* Path of log file */
    int syslog_enabled;             /* Is syslog enabled? */
    int verbosity;                  /* Loglevel in redis.conf */

    time_t timezone;            /* Cached timezone. As set by tzset(). */
    int daylight_active;        /* Currently in daylight saving time. */
    mstime_t mstime;            /* 'unixtime' in milliseconds. */
    ustime_t ustime;            /* 'unixtime' in microseconds. */
    redisAtomic time_t unixtime; /* Unix time sampled every cron cycle. */
};

在redisServer中,因为在log中使用,所有需要加了一些看似没有用属性。这一点redis做的不是很好,业务和库耦合了。redisServer可以理解为有状态的容器,需要全局有效,共享的,都可以加到这上面来。在函数中直接访问属性就可以了。

新建 Makefile

 1
 2
 3
 4
 5
 6
 7
 8
 9
10
11
all: server
	@echo "anet demo"

server :  server.o localtime.o
	$(CC)   -o $@   $^


.PHONY: clean
clean:
	rm -rf *.o *.d server client

输出

执行make后生成server可执行文件, 执行server后输出

1
279214:C 08 Nov 2021 15:19:27.074 #  Redis is starting unixtime is: 1636355967

使用到的api

  • serverLog(level, …)

文章思路

  • 动手实践是最快的学习方式。用能理解的方式,轻松的学习Redis,借鉴Redis, 并应用。
  • 不过分关注细节,从Api入手

码字不易,感谢点赞