Detailed explanation of MySQL's Seconds_Behind_Master

Detailed explanation of MySQL's Seconds_Behind_Master

Seconds_Behind_Master

For MySQL master-slave instances, seconds_behind_master is an important parameter to measure the delay between master and slave. The value of seconds_behind_master can be obtained by executing "show slave status;" on the slave.

Original implementation

Definition: The number of seconds that the slave SQL thread is behind processing the master binary log.

Type: time_t (long)

The calculation is as follows:

rpl_slave.cc::show_slave_status_send_data()
if ((mi->get_master_log_pos() == mi->rli->get_group_master_log_pos()) &&
       (!strcmp(mi->get_master_log_name(),
                mi->rli->get_group_master_log_name()))) {
     if (mi->slave_running == MYSQL_SLAVE_RUN_CONNECT)
       protocol->store(0LL);
     else
       protocol->store_null();
   } else {
     long time_diff = ((long)(time(0) - mi->rli->last_master_timestamp) -
                       mi->clock_diff_with_master);
     protocol->store(
         (longlong)(mi->rli->last_master_timestamp ? max(0L, time_diff) : 0));
   }

There are mainly two situations:

  • The SQL thread waits for the IO thread to obtain the host binlog. At this time, seconds_behind_master is 0, indicating that there is no delay between the standby machine and the master machine.
  • The SQL thread processes the relay log. At this time, seconds_behind_master is calculated by (long)(time(0) – mi->rli->last_master_timestamp) – mi->clock_diff_with_master.

last_master_timestamp

definition:

The time of the event in the master database binlog.

type: time_t (long)

Calculation method:

The last_master_timestamp is calculated differently depending on whether the standby server is replicated in parallel.

Non-parallel replication:

rpl_slave.cc:exec_relay_log_event()
if ((!rli->is_parallel_exec() || rli->last_master_timestamp == 0) &&
    !(ev->is_artificial_event() || ev->is_relay_log_event() ||
     (ev->common_header->when.tv_sec == 0) ||
     ev->get_type_code() == binary_log::FORMAT_DESCRIPTION_EVENT ||
     ev->server_id == 0))
{
 rli->last_master_timestamp = ev->common_header->when.tv_sec +
                             (time_t) ev->exec_time;
 DBUG_ASSERT(rli->last_master_timestamp >= 0);
}

In this mode, last_master_timestamp indicates the end time of each event, when.tv_sec indicates the start time of the event, and exec_time indicates the execution time of the transaction. This value is calculated before apply_event, so last_master_timestamp has been updated before event is executed. Since exec_time only exists in Query_log_event, last_master_timestamp changes at different event stages of applying a transaction. Take a transaction containing two insert statements as an example. When this code segment is called, the event type, timestamp, and execution time are printed out.

create table t1(a int PRIMARY KEY AUTO_INCREMENT ,b longblob) engine=innodb;
begin;
insert into t1(b) select repeat('a',104857600);
insert into t1(b) select repeat('a',104857600);
commit;

10T06:41:32.628554Z 11 [Note] [MY-000000] [Repl] event_type: 33 GTID_LOG_EVENT

2020-02-10T06:41:32.628601Z 11 [Note] [MY-000000] [Repl] event_time: 1581316890

2020-02-10T06:41:32.628614Z 11 [Note] [MY-000000] [Repl] event_exec_time: 0

2020-02-10T06:41:32.628692Z 11 [Note] [MY-000000] [Repl] event_type: 2 QUERY_EVENT

2020-02-10T06:41:32.628704Z 11 [Note] [MY-000000] [Repl] event_time: 1581316823

2020-02-10T06:41:32.628713Z 11 [Note] [MY-000000] [Repl] event_exec_time: 35

2020-02-10T06:41:32.629037Z 11 [Note] [MY-000000] [Repl] event_type: 19 TABLE_MAP_EVENT

2020-02-10T06:41:32.629057Z 11 [Note] [MY-000000] [Repl] event_time: 1581316823

2020-02-10T06:41:32.629063Z 11 [Note] [MY-000000] [Repl] event_exec_time: 0

2020-02-10T06:41:33.644111Z 11 [Note] [MY-000000] [Repl] event_type: 30 WRITE_ROWS_EVENT

2020-02-10T06:41:33.644149Z 11 [Note] [MY-000000] [Repl] event_time: 1581316823

2020-02-10T06:41:33.644156Z 11 [Note] [MY-000000] [Repl] event_exec_time: 0

2020-02-10T06:41:43.520272Z 0 [Note] [MY-011953] [InnoDB] Page cleaner took 9185ms to flush 3 and evict 0 pages

2020-02-10T06:42:05.982458Z 11 [Note] [MY-000000] [Repl] event_type: 19 TABLE_MAP_EVENT

2020-02-10T06:42:05.982488Z 11 [Note] [MY-000000] [Repl] event_time: 1581316858

2020-02-10T06:42:05.982495Z 11 [Note] [MY-000000] [Repl] event_exec_time: 0

2020-02-10T06:42:06.569345Z 11 [Note] [MY-000000] [Repl] event_type: 30 WRITE_ROWS_EVENT

2020-02-10T06:42:06.569376Z 11 [Note] [MY-000000] [Repl] event_time: 1581316858

2020-02-10T06:42:06.569384Z 11 [Note] [MY-000000] [Repl] event_exec_time: 0

2020-02-10T06:42:16.506176Z 0 [Note] [MY-011953] [InnoDB] Page cleaner took 9352ms to flush 8 and evict 0 pages

2020-02-10T06:42:37.202507Z 11 [Note] [MY-000000] [Repl] event_type: 16 XID_EVENT

2020-02-10T06:42:37.202539Z 11 [Note] [MY-000000] [Repl] event_time: 1581316890

2020-02-10T06:42:37.202546Z 11 [Note] [MY-000000] [Repl] event_exec_time: 0

Parallel replication:

rpl_slave.cc mts_checkpoint_routine
ts = rli->gaq->empty()
          ? 0
          : reinterpret_cast<Slave_job_group *>(rli->gaq->head_queue())->ts;
 rli->reset_notified_checkpoint(cnt, ts, true);
 /* end-of "Coordinator::"commit_positions" */

In this mode, there is a distribution queue gaq on the standby machine. If gaq is empty, last_commit_timestamp is set to 0; if gaq is not empty, a checkpoint point lwm is maintained at this time, and all transactions before lwm are completed on the standby machine. At this time, last_commit_timestamp is updated to the time after the transaction where lwm is located is completed. The time type is time_t type.

ptr_group->ts = common_header->when.tv_sec +
                   (time_t)exec_time; // Seconds_behind_master related
rli->rli_checkpoint_seqno++;
if (update_timestamp) {
 mysql_mutex_lock(&data_lock);
 last_master_timestamp = new_ts;
 mysql_mutex_unlock(&data_lock);
}

In parallel replication, last_master_timestamp is updated only after event execution is completed, so seconds_behind_master will differ between non-parallel replication and parallel replication.

clock_diff_with_master

definition:

  • The difference in seconds between the clock of the master and the clock of the slave (second - first). It must be signed as it may be <0 or >0. clock_diff_with_master is computed when the I/O thread starts; for this the I/O thread does a SELECT UNIX_TIMESTAMP() on the master.
  • type: long
rpl_slave.cc::get_master_version_and_clock()
if (!mysql_real_query(mysql, STRING_WITH_LEN("SELECT UNIX_TIMESTAMP()")) &&
     (master_res = mysql_store_result(mysql)) &&
     (master_row = mysql_fetch_row(master_res)))
 {
   mysql_mutex_lock(&mi->data_lock);
   mi->clock_diff_with_master=
     (long) (time((time_t*) 0) - strtoul(master_row[0], 0, 10));
   DBUG_EXECUTE_IF("dbug.mts.force_clock_diff_eq_0",
     mi->clock_diff_with_master = 0;);
   mysql_mutex_unlock(&mi->data_lock);
 }

This difference is calculated only once, when the master and slave establish contact.

other

exec_time

definition:

  • the difference from the statement's original start timestamp and the time at which it completed executing.
  • type: unsigned long
struct timeval end_time;
ulonglong micro_end_time = my_micro_time();
my_micro_time_to_timeval(micro_end_time, &end_time);
exec_time = end_time.tv_sec - thd_arg->query_start_in_secs();

Time function

(1) time_t time(time_t timer) time_t is of long type, and the returned value is accurate only to seconds;

(2) int gettimeofday (struct timeval *tv, struct timezone *tz) can obtain the current time in microseconds;

(3) timeval structure

#include <time.h>
stuct timeval {
   time_t tv_sec; /*seconds*/
   suseconds_t tv_usec; ​​/*microseconds*/
}

Summarize

Using seconds_behind_master to measure the master-slave delay can only be accurate to the second level. In some scenarios, seconds_behind_master cannot accurately reflect the delay between the master and the slave. When the master and standby are abnormal, you can combine the seconds_behind_master source code for specific analysis.

The above is the detailed explanation of MySQL Seconds_Behind_Master. For more information about MySQL Seconds_Behind_Master, please pay attention to other related articles on 123WORDPRESS.COM!

You may also be interested in:
  • Why Seconds_Behind_Master is still 0 when MySQL synchronization delay occurs
  • Implementation method of python3 file copy and delayed file copy tasks
  • Sample code for implementing mysql master-slave replication in docker
  • MySQL database Load Data multiple uses
  • MySQL database Shell import_table data import
  • Master-slave synchronization configuration of Mysql database
  • Example code for implementing a simple search engine with MySQL
  • Solution to the problem that MySQL commands cannot be entered in Chinese
  • When the interviewer asked the difference between char and varchar in mysql
  • MySQL slave library Seconds_Behind_Master delay summary

<<:  Implementation of Grid common layout

>>:  Website User Experience Design (UE)

Recommend

JavaScript Regular Expressions Explained

Table of contents 1. Regular expression creation ...

Native JS realizes compound motion of various motions

This article shares with you a compound motion im...

How to implement simple data monitoring with JS

Table of contents Overview first step Step 2 Why ...

Detailed explanation of how to implement secondary cache with MySQL and Redis

Redis Introduction Redis is completely open sourc...

Introduction to Linux File Compression and Packaging

1. Introduction to compression and packaging Comm...

In-depth analysis of MySQL lock blocking

In daily maintenance, threads are often blocked, ...

How to solve the problem that Docker container has no vim command

Find the problem Today, when I tried to modify th...

MySQL 5.6 compressed package installation method

There are two installation methods for MySQL: msi...

XHTML introductory tutorial: Use of list tags

Lists are used to list a series of similar or rela...

Detailed explanation of the use of Vue Smooth DnD, a draggable component of Vue

Table of contents Introduction and Demo API: Cont...

Implementation of drawing audio waveform with wavesurfer.js

1. View the renderings Select forward: Select bac...