If you want to see the communication process, skip the first headline

I. Server startup process (initial startup or restart) :

1.1 First create oneredisServerInstance variables as server state;

/* Global vars */
struct redisServer server; /* Server global state */
Copy the code

1.2 Then callserver.cIn theinitServerConfigMethod, set with the default valuesredisServerInstance variables;

// Initialize the Unix clock (including the LRU clock) pthread_mutex_init(&server.lruclock_mutex,NULL); pthread_mutex_init(&server.unixtime_mutex,NULL); GetRandomHexChars (server.runid,CONFIG_RUN_ID_SIZE); // Set the server running ID (used as the unique server id in the cluster). // Set the default server port number (#define CONFIG_DEFAULT_SERVER_PORT: 6379 /* TCP port. */)server.port = CONFIG_DEFAULT_SERVER_PORT; Server. arch_bits = (sizeof(long) == 8)? 64-32. // Set the default AOF status to off server.aof_state = AOF_OFF; // Whether to synchronize the AOF buffer to the hard disk each time server.aof_fsync = CONFIG_DEFAULT_AOF_FSYNC; AppendServerSaveParams (60*60,1); // Call BGSVAE to update RDB file with conditional appendServerSaveParams(60*60,1); /* save after 1 hour and 1 change */Copy the code

1.3 Load user configuration options and override the original default configuration;

For example, if I start redis and don’t want to use 6379 as the port for external services, I’ll make a special change in the redis.conf file, such as 12345:

# Accept connections on the specified port, default is 6379 (IANA #815344).
# If port 0 is specified Redis will not listen on a TCP socket.
port 12345
Copy the code

1.4 Initializing the corresponding data structure (Call the initServer function) :

(1) Clients linked list: used to record the client status object redisClient connected to this server:

What information does the redisClient object contain? For example, the name of the client, the socket descriptor of the client, the RESP format of the request data sent by the client over TCP, the parsing information sent by the client (such as whether the request is SET, RPUSH or ZSET, etc.), the number of parameters requested, etc.

RedisClient structure

Requests from clients are stored in the QueryBuf input buffer and are saved through an SDS object.

The server then parses the text according to the RESP protocol, parsing out the actual command and placing it in the argv[] array, with the number of arguments recorded in the argc variable:

PS: There are some other data structures that are not related to CS communication

1.5 Perform some Settings:

(1) Set the process signal processor for the server;

(2) Initialize shared objects: such as 1~10000, OK, ERR and other common objects, so as to avoid repeatedly creating the same object;

(3) Initialize the listening port and associate the corresponding event handler for the socket to be listened on (more on this later in the communication); And create corresponding timing events, such as periodic invocation of serverCron function, which is mainly used to maintain the state of Redis server;

(4) If the AOF function is enabled, open the corresponding AOF file, if not, create it;

(5) Initialize the server background I/O module (BIO) to prepare for future I/O;

The server has been started and the following log is generated:

1.6 RDB or AOF loading:

Considering that the server may have been restarted during an outage, the server will need to be restored to its original state via RDB or AOF files, and the corresponding log will be printed:

Use AOF if there is an AOF file, otherwise use RDB.

1.7 Starting an Event Loop:

Redis has two main event loops, file events (listening to the corresponding socket) and event events (running the scheduled task), and they are processed in a single thread in a time-first fashion so that the Redis server can receive requests from clients.

Two, server side processing request process:

2.1 Client Sends a request:

The client writes the request information to the socket according to the RESP text protocol:

Let’s say we want to send the following command:

127.0.0.1:6379 >set ALI CVNot
Copy the code

So according to the text protocol, it will be modified to

*3\r\n$3\r\n\set\r\n$3\r\nALI\r\nA $5\r\nCVNot\r\n
Copy the code

Transfer over the network

2.2 The server receives requests:

We know that when the client and the server through the three-time handshake to establish a connection, the client will write its own request to send the socket, and the socket file descriptor sent to the server, the server read the request information and write the return information, and finally returned to the client;

If you’re not familiar with the I/O model, you can refer to my previous article: An In-depth Look at the Java IO mechanism;

Redis uses the multiplexed I/O model (event mechanism) to achieve high efficiency even if only one thread is used to process command requests.

2.2.1 Establishing a Connection:

In the Redis is through networking. C/acceptTcpHandler functions as connected to the processor to process the connection request (i.e., the client calls the connect () function to establish the connection with the server through the three-way handshake) :

void acceptTcpHandler(aeEventLoop *el, int fd, void *privdata, int mask) {
    //...
    cfd = anetTcpAccept(server.neterr, fd, cip, sizeof(cip), &cport);
    acceptCommonHandler(cfd,0,cip);
    }
Copy the code

The anetTcpAccept and acceptCommonHandler functions are called. The former is used to establish a connection and returns a socket descriptor if the connection is successful. The latter then associates the socket descriptor with the readQueryFromClient() function as a command request handler to handle requests from the client after the connection is established (e.g. JAVA NIO also registers a read event for the channel in the OP_CONNECT event handler.

static void acceptCommonHandler(int fd, int flags, char *ip) { client *c; // This method registers a command request handler for the socketif(c = createClient(fd)) == NULL) {} // Whether the length of the clients list in the redisServer structure still meets the maximum number of connections required by the server // If not, reject connectionsif (listLength(server.clients) > server.maxclients) {
    }
Copy the code
client *createClient(int fd) {
    client *c = zmalloc(sizeof(client));
    if(fd ! = -1) {// anetNonBlock(NULL,fd); // anetNonBlock(NULL,fd); // anetNonBlock(NULL,fd); // Set the socket to non-blocking anetEnableTcpNoDelay(NULL,fd);ifWhether tcpkeepalive (server) / / maintain the TCP connection anetKeepAlive (NULL, fd, server. Tcpkeepalive); // Re-register an AE_READABLE event for the socket and associate a command request handler with itif (aeCreateFileEvent(server.el,fd,AE_READABLE,
            readQueryFromClient, c) == AE_ERR)
        {
            close(fd);
            zfree(c);
            returnNULL; }}}Copy the code

2.2.2 Sending commands from the Client:

(1) When a client request has been written to the corresponding listened socket file, an AE_READABLE event will be generated. This event will then be placed on the waiting queue. The file event allocator will assign file events to the queue header socket until the thread handling the command finishes processing the current event. Processing using the command request handler registered in the previous phase:

And the process of this processing is:

First, we need to read the RESP text file from the socket file into the input buffer, that is, each client corresponding to the QueryBuf buffer in the redisClient structure on the clients list. Then, the handler will restore the command according to the RESP protocol, and follow the corresponding instructions. Look for the corresponding command implementation function in the command table and point the * CMD pointer in redisClient to this function:

SET
Command request handler

After the processing is completed, the server will be processed and saved in the output buffer of the redisClient of the corresponding request client. The buffer is divided into fixed length buffer and variable buffer. The former is a fixed size buffer, generally 16KB, used to store some OK, ERR and other short strings; Long strings are saved through the redisClient’s Reply list, and you can customize the threshold of the variable buffer implemented by the Reply list, and the client will be closed if the value is greater than that.

Then you wait for the AE_WAITABLE event to write the reply from the cache to the corresponding socket.

2.2.3 Result returned by the server:

When a socket becomes writable (that is, the client calls the read() function), an AE_WRITABLE event is generated, and when the fetch thread processes it, a command is called back to the handler to write the result information from the server to the socket. As with the SET command above, an OK is returned:

In addition, Redis appends the write command to the end of the aOF_buf buffer in protocol format, and calls the flushAppendOnlyFile function each time before ending an event loop. And according to different Settings to decide whether to write and save the contents of the buffer to the AOF file;

Appendfsync options The behavior of the flushAppendOnlyFile
always Writes and synchronizes everything in the buffer to the AOF file
everysec All the contents of the buffer are written to the AOF file. If the event distance from the last synchronization of the AOF file is now more than 1s, the synchronization will take place
no Everything in the buffer is written to the AOF file, but the AOF file is not synchronized. The timing of synchronization is determined by the operating system

Looking at this, I think it’s easy to wonder why writing to AOF files should be written or synchronized at all. In order to improve THE efficiency of I/O, the operating system uses caching technology, that is, the operating system will write data through the write() function into a memory buffer, and then write the data until the specified time or the buffer is full. Whether to synchronize, that is, whether to call fsync or fdatasync to write I/O directly to the hard disk after each write to the buffer.

2.2.4 Client Processing of text Files:

After receiving the information returned by the server, the client will process the corresponding text information and finally display it in the client:

Iii. Summary:

I think the most important thing to remember about database operations is that our interaction with the database is essentially an exchange of commands.

Summarize all of the above with the following diagram: