Lecture#19 Logging Schemes

发布时间 2023-04-20 07:16:25作者: Angelia-Wang

数据库在运行时可能遭遇各种故障,这时可能同时有许多正在运行的事务,如果这些事务执行到一半时故障发生了,就可能导致数据库中的数据出现不一致的现象:

image-20230420055259454

这时就需要故障恢复机制来保证数据库的原子性、一致性、持久性。故障恢复机制包含两部分:

  • 在事务执行过程中采取的行动来确保在出现故障时能够恢复 (本节课)
  • 在故障发生后的恢复机制,确保原子性、一致性和持久性 (下节课)

本节大纲:

  • Failure Classification
  • Buffer Pool Policies
  • Shadow Paging
  • Write-Ahead Log
  • Logging Schemes
  • Checkpoints

1 Failure Classification

世上并不存在能够容忍任何故障的容错机制,即便做了异地多活,一次小行星撞地球也能将你的系统一举歼灭,因此在讨论故障恢复之前,我们必须先为故障分类,明确需要容忍的故障类型。

一般故障类型可以分为 3 种:

  1. 事务故障 (Transaction Failures)
    • 逻辑错误 (Logical Errors):由于一些内部约束,如数据一致性约束,导致事务无法正常完成
    • 内部错误 (Internal State Errors):由于数据库内部调度、并发控制,如死锁,导致事务无法正常提交
  2. 系统故障 (System Failures)
    • 软件故障 (Software Failure):如 DBMS 本身的实现问题 (NPE, Divide-by-zero)
    • 硬件故障 (Hardware Failure):DBMS 所在的宿主机发生崩溃,如断电。且一般假设非易失性的存储数据在宿主机崩溃后不会丢失
  3. 存储介质故障 (Storage Media Failures):通常无法修复,此时数据库只能从归档的备份记录中恢复数据。

2 Buffer Pool Management Policies

修改数据时,DBMS 需要先把对应的数据页从持久化存储中读到内存中,然后在内存中根据写请求修改数据,最后将修改后的数据写回到持久化存储。在整个过程中,DBMS 需要保证两点:

  1. DBMS 告知用户事务已经提交成功前,相应的数据必须已经持久化
  2. 如果事务中止,任何数据修改都不应该持久化

如果真的遇上事务故障或者系统故障,DBMS 有两种基本思路来恢复数据一致性,向用户提供上述两方面保证:

  • Undo:将中止或未完成的事务中已经执行的操作回退
  • Redo:将提交的事务执行的操作重做

DBMS 如何支持 undo/redo 取决于它如何管理 buffer pool。我们可以从两个角度来分析 buffer pool 的管理策略:

  • Steal Policy:DBMS 是否允许一个未提交事务修改持久化存储中的数据?

    • 允许 (Steal),则若该事务回滚,则需把已经持久化的数据读进来,回滚数据,再存回去;但在不发生回滚时 DBMS 的 I/O 效率较高
    • 不允许 (No-Steal):若事务回滚,DBMS 无需做任何额外操作;但在不发生回滚时 DBMS 的 I/O 效率较低

    —— 存在数据恢复时间与 DBMS 处理效率的取舍

  • Force Policy:DBMS 是否强制要求一个提交完毕事务的所有数据改动都反映在持久化存储中?

    • 强制 (Force),每次事务提交都必须将数据落盘,数据一致性可以得到完美保障,但 I/O 效率较低
    • 不强制 (No-Force):DBMS 可以延迟批量地将数据落盘,数据一致性可能存在问题,但 I/O 效率较高

    —— 存在数据一致性与 DBMS 处理效率的取舍

我们有 4 种实现组合:

Steal No-Steal
Force Steal + Force No-Steal + Force
No-Force Steal + No-Force No-Steal + No-Force

在实践中使用的主要是 No-Steal + Force 和 Steal + No-Force。

2.1 Shadow Paging: No-Steal + Force

最容易实现的一种策略组合就是 No-Steal + Force:

  • 事务中止后,无需回滚数据,因为该事务修改的数据不会被别的事务捎带落盘
  • 事务提交后,无需重做数据,因为该事务修改的数据必然会被落盘持久化

当然,这种策略组合无法处理 "写入的数据量超过 buffer pool 大小" 的情况。

shadow paging 是 No-Steal + Force 策略的典型代表,它会维护两份数据库数据:

  1. Master:包含所有已经提交事务的数据
  2. Shadow:在 Master 之上增加未提交事务的数据变动

当然,为了提高效率,DBMS 不会复制整个数据库,只需要复制有变动的部分即可,即 copy-on-write。

通常,实现 shadow paging 需要将数据页组织成树状结构,树有两个副本 master 和 shadow,根结点指向 master copy,事务的更新只应用与 shadow copy。当事务提交时,再原子地把 shadow copy 修改成新的 master。

? 如图所示:

image-20230420062351498

左边是内存中的数据结构,由一个根节点指向对应的 page table,其中 page table 对应着磁盘中的相应的数据页。

只读事务从 master page 中获取数据。

写事务执行时,会生成一个 shadow page table,并复制需要修改的数据页,在其上完成相应操作。提交写事务时,DBMS 将 DB Root 的指针指向 shadow page table,并将对应的数据页全部落盘,删掉原 master。

可见,事务提交前,任何被修改的数据都不会被持久化到数据库;事务提交后,所有被修改的数据都会被持久化到数据库。

在 shadow paging 下回滚、恢复数据都很容易:

  • Undo/Rollback:删除 shadow pages (table),啥都不用做
  • Redo:不需要 redo,因为每次写事务都会将数据落盘

shadow paging 很简单,但它的缺陷也很明显:

  • 复制整个 page table 代价较大,尽管可以通过以下措施来降低这种代价:
    • 需要使用类似 B+ 树的数据结构来组织 page table
    • 无需复制整个树状架构,只需要复制到达有变动的叶子节点的路径即可
  • 事务提交的代价较大
    • 需要将所有发生更新的 data page、page table 以及根节点都落盘
    • 容易产生磁盘碎片,使得原先距离近的数据渐行渐远
    • 需要做垃圾收集
    • 只支持一个写事务或一批写事务一次性持久化

在 2010 年之前,SQLite 采用的就是类似 shadow paging 的策略,当写事务需要修改某页数据时,DBMS 会先将原始数据页保存到一个独立的 journal 文件中,然后再将对应的修改持久化到磁盘中。当 SQLite 重启后,如果发现磁盘中存在 journal 文件,则直接将对应的数据页覆盖到磁盘来 undo 未提交事务的更改。

2.2 Write-Ahead Log (WAL): Steal + No-Force

我们发现shadowing page 效率低的主要原因是:DBMS 在实现 shadow paging 时需要将许多不连续的数据页写到磁盘中,随机写对磁盘来说并不友好。因此,我们需要一种方法让 DBMS 将随机写入转换为顺序写入。—— WAL

WAL 指的是 DBMS 除了维持正常的数据文件外,额外地维护一个日志文件,上面记录着所有事务对 DBMS 数据的完整修改记录,这些记录能够帮助数据库在恢复数据时执行 undo/redo。使用 WAL 时,DBMS 必须先将操作日志持久化到独立的日志文件中,然后才修改其真正的数据页。【假设日志在稳定存储中】

通常实现 WAL 采用的 buffer pool policy 为 Steal + No-Force。

WAL Protocol

  1. DBMS 先将事务的操作记录放在内存中 (backed by buffer pool)
  2. 在将 data page 落盘前,所有的日志记录必须先落盘
  3. 在操作日志落盘后,事务就被认为已经提交成功

事务开始时,需要写入一条 <BEGIN> 记录到日志中;事务结束时,需要写入一条 <COMMIT> 记录到日志中;在事务执行过程中,每条日志记录着数据修改的完整信息,如:

  • Transaction Id (事务 id)
  • Object Id (数据记录 id)
  • Before Value (修改前的值),用于 undo 操作
  • After Value (修改后的值),用于 redo 操作
image-20230420064008236

每次事务提交时,DBMS 都必须将日志记录落盘,由于落盘的操作对于内存操作来说用时较长,因此可以利用 group commit 将多个日志批量刷盘从而均摊落盘成本。

WAL Group Commit

  • 事务 commit 前不告诉它 commit 成功,让它等着,等积累一定的待提交日志后一起 commit 落盘日志。(会出现各事务等待的情况)
  • 若经过一定时间后还没满,也 flush到磁盘。
image-20230420064536629

通过以上的分析,我们可以发现,不同的 buffer pool policies 之间存在着运行时效率与数据恢复效率间的权衡:

image-20230420064834111

运行时效率:No-Steal + Force < Steal + No-Force

数据恢复效率:No-Steal + Force > Steal + No-Force

大部分数据库更看重运行时效率,因此几乎所有 DBMS 使用 No-Force + Steal 的 buffer pool policy。

3 Logging Schemes

在记录操作信息时,通常有两种方案:physical logginglogical logging。前者指的是记录物理数据的变化,类似 git diff 做的事情;后者指的是记录逻辑操作内容,如 UPDATE、DELETE 和 INSERT 语句等等。

logical logging 的特点总结如下:

  • 相比 physical logging,记录的内容精简
  • 若存在并发,恢复时很难确定故障时正在执行的语句已经修改了哪部分数据
  • 恢复时需要回放所有事务,这个过程可能很漫长

还有一种混合策略,称为 physiological logging,这种方案不会像 physical logging 一样记录 xx page xx 偏移量上的数据发生 xx 改动,而是记录 xx page 上的 id 为 xx 的数据发生 xx 改动,前者需要关心 data page 在磁盘上的布局,后者则无需关心,? 举例如下:

image-20230420065719841

physiological logging 也是当下最流行的方案。

3.1 Checkpoints

WAL 将随着新的操作执行而无限增长,这样在故障恢复时,DBMS 必须redo整个日志,这将需要很长时间。为了避免这种情况出现,DBMS 需要周期性地记录 checkpoint (标记 checkpoint 之前的数据都已刷到稳定存储中)。

Blocking Checkpoint 实现:

  • DBMS停止接受新的事务,等待所有 active 事务完成。
  • 将当前驻留在主内存中的所有日志记录和 dirty blocks 都刷到稳定存储中。
  • 在日志中写入一条 <CHECKPOINT> 记录并刷新到稳定存储。

? 举例如下:

image-20230420070617675

当 DBMS 发生崩溃时,所有在最新的 checkpoint 之前提交的事务可以直接忽略,如 T1。T2 和 T3 在 checkpoint 前尚未 commit。其中,T2 需要 redo,因为它在 checkpoint 之后,crash 之前提交了,即已经告诉用户事务提交成功;T3 需要 undo,因为它在 crash 之前尚未 commit,即尚未告诉用户事务提交成功。

实现 checkpoints 需要考虑的问题:

  1. 为保证 checkpoint 的正确性,我们要暂停所有事务
  2. 故障恢复时,扫描数据找到未提交的事务可能需要较长的时间
  3. 如何决定 DBMS 执行 checkpoint 的周期:太频繁将导致运行时性能下降;等太久将使得 checkpoint 的内容更多,更耗时,同时数据恢复也要消耗更长的时间

Conclusion

WAL (No-Force + Steal) 几乎永远是最佳选择。

使用检查点,在数据恢复时,undo未提交的事务,redo提交的事务。

参考链接

slides, notes