oceanbase oceanbase数据库认证
本文主要分享针对想压测oceanbase时需要了解的一些技术原理。这些建议可以帮助用户对oceanbase做一些调优,再结合测试程序快速找到适合业务的最佳性能。由于oceanbase自身参数很多、部署形态也比较灵活,这里并没有给出具体步骤。
压测的本质就是对一个会话的逻辑设计很高的并发。首先需要了解单个会话在数据库内部的读写逻辑。比如说,业务会话1对数据库发起一个dml sql,第一次修改某笔记录,数据库会怎么做呢?
为了便于理解ob的行为,我们先看看oracle是怎么做的。后面有对比才可以加深理解。
oracle 读写特点
oracle会话第一次修改一行记录,如果该记录所在块(8k大小)不在内存(buffer cache)里时会先从磁盘文件里读入到内存里。这个称为一次物理读,为了性能考虑,oracle一次会连续读取相邻的多个块。然后就直接在该块上修改,修改之前会先记录redo和undo(包括undo的redo)。然后这个数据块就是脏块(dirty block)。假设事务没有提交,其他会话又来读取这个记录,由于隔离级别是读已提交(read committed),oracle会在内存里克隆当前数据块到新的位置,新块包含了最新的未提交数据。然后oracle在新块上逆向应用undo链表中的记录,将数据块回滚到读需要的那个版本(scn),然后才能读。这个也称为一次一致性读(consistency read),这个新块也称为cr块。
即使是修改一条记录一个字段的几个字节,整个块(8k大小)都会是脏块。随着业务持续写入,大量脏块会消耗数据库内存。所以oracle会有多重机制刷脏块到磁盘数据文件上。在事务日志切换的时候也会触发刷脏块操作。如果业务压力测试oracle,大量的写导致事务日志切换很频繁,对应的刷脏操作可能相对慢了,就会阻塞日志切换,也就阻塞了业务写入。这就是oracle的特点。解决办法就是加大事务日志文件,增加事务日志成员或者用更快的磁盘存放事务日志和数据文件。
oracle里一个表就是一个segment(如果有大对象列还会有独立的segment,这个先忽略),segment由多个不一定连续的extent组成,extent由连续的block(每个大小默认8k)组成,extent缺点是可能会在后期由于频繁删除和插入产生空间碎片。
oceanbase 读写特点
oceanbase会话第一次修改一行记录,如果该记录所在块(64k大小)不在内存(block cache)里时也会先从磁盘文件里读入到内存里。这个称为一次物理读。然后要修改时跟oracle做法不同的是,oceanbase会新申请一小块内存用于存放修改的内容,并且链接到前面block cache里该行记录所在块的那笔记录下。如果修改多次,每次修改都跟前面修改以链表形式关联。同样在修改之前也要先在内存里记录redo。每次修改都会记录一个内部版本号,记录的每个版本就是一个增量。其他会话读取的时候会先从block cache中该记录最早读入的那个版本(称为基线版本)开始读,然后叠加应用后面的增量版本直到合适的版本(类似oracle中scn概念)。(随着版本演进,这里细节逻辑可能会有变化。)
ob的这个读方式简单说就是从最早的版本读起,逐步应用增量(类似redo,但跟redo日志无关)。而oracle一致性读是从最新的版本读起,逐步回滚(应用undo)。在ob里,没有undo。当版本链路很长时,ob的读性能会略下降,所以ob也有个checkpoint线程定时将记录的多个版本合并为少数几个版本。这个合并称为小合并(minor compaction)。此外,ob在内存里针对行记录还有缓存,
从上面过程还可以看出,每次修改几个字节,在内存里的变脏的块只有增量版本所在的块(默认写满才会重新申请内存),基线数据块是一直不变化。所以ob里脏块产生的速度非常小,脏块就可以在内存里保存更久的时间。实际上ob的设计就是脏块默认不刷盘。那如果机器挂了,会不会丢数据呢?
ob跟oracle一样,修改数据块之前会先记录redo,在事务提交的时候,redo要先写到磁盘上(redo同时还会发送往其他两个副本节点,这个先忽略)。有redo在,就不怕丢数据。此外,增量部分每天还是会落盘一次。在落盘之前,内存中的基线数据和相关的增量数据会在内存里进行一次合并(称merge),最终以sstable的格式写回到磁盘。如果说内存里块内部产生碎片,在合并的那一刻,这个碎片空间基本被消弭掉了。所以说ob的数据文件空间碎片很小,不需要做碎片整理。同时ob的这个设计也极大降低了lsm的写放大问题。
当业务压测写ob时,脏块的量也会增长,最终达到增量内存限制,这时候业务就无法写入,需要ob做合并释放内存。ob的合并比较耗io、cpu(有参数可以控制合并力度),并且也不会等到内存用尽才合并,实际会设置一个阈值。同时为了规避合并,设计了一个转储机制。当增量内存使用率超过阈值后,就开启转储。转储就是直接把增量内存写到磁盘上(不合并)。转储对性能的影响很小,可以高峰期发生,并且可以转储多次(参数配置)。
ob增量内存就类似一个水池,业务写是进水管在放水, 转储和大合并是出水管。水位就是当前增量内存使用率。当进水的速度快于出水,池子可能就会满。这时候业务写入就会报内存不足的错误。
这就是ob读写的特点,解决方法就是加大ob内存、或者允许ob自动对业务写入速度限流。
ob 在commit的时候redo落盘会写磁盘。读数据的时候内存未命中的时候会有物理读,转储和大合并的时候落盘会有密集型写io。这些都依赖磁盘读写性能。所以建议磁盘都是ssd盘,并且建议日志盘和数据盘使用独立的文件系统。如果是nvme接口的闪存卡或者大容量ssd盘,那日志盘和数据盘放在一起也可以。不要使用lvm对nvme接口的大容量ssd做划分,那样瓶颈可能会在lvm自身。
ob的增量通常都在内存里,内存不足的时候会有转储,可以转储多次。尽管如此,建议测试机器的内存不要太小,防止频繁的增量转储。通常建议192g内存以上。
ob集群的节点数至少要有三个。如果是功能了解,在单机上起3个ob进程模拟三节点是可以的,但是如果是性能测试,那建议还是使用三台同等规格的物理机比较合适。机器规格不一致时,最小能力的机器可能会制约整个集群的性能。 oceanbase集群的手动部署请参考《oceanbase数据库实践入门——手动搭建oceanbase集群》。在部署好oceanbase之后,建议先简单了解一下oceanbase的使用方法,详情请参考文章《oceanbase数据库实践入门——常用操作sql》。
如果要验证ob的弹性缩容、水平扩展能力,建议至少要6节点(部署形态2-2-2)。并且测试租户(实例)的每个zone里的资源单元数量至少也要为2个,才可以发挥多机能力。这是因为ob是多租户设计,对资源的管理比较类似云数据库思想,所以里面设计有点精妙,详情请参见《揭秘oceanbase的弹性伸缩和负载均衡原理》。
下面是一个租户的测试租户资源初始化建议
create resource unit s1, max_cpu=2, max_memory='10g', min_memory='10g', max_iops=10000, min_iops=1000, max_session_num=1000000, max_disk_size=536870912; create resource unit s2, max_cpu=4, max_memory='20g', min_memory='20g', max_iops=20000, min_iops=5000, max_session_num=1000000, max_disk_size=1073741824; create resource unit s3, max_cpu=8, max_memory='40g', min_memory='40g', max_iops=50000, min_iops=10000, max_session_num=1000000, max_disk_size=2147483648; select * from __all_unit_config; create resource pool pool_demo unit = 's2', unit_num = 2; select * from __all_resource_pool order by resource_pool_id desc ; create tenant t_obdemo resource_pool_list=('pool_demo'); alter tenant t_obdemo set variables ob_tcp_invited_nodes='%';
请注意上面的unit_num=2这个很关键。如果unit_num=1,ob会认为这个租户是个小租户,后面负载均衡处理时会有个默认规则。
create database sbtest; grant all privileges on sbtest.* to sbuser@'%' identified by 'sbtest';
因为测试场景跟业务有关,这里就以常见的sysbench场景举例分析
sysbench工具可以建几个结构相同的表,然后执行纯读、纯写、读写混合。其中读又分根据主键或者二级索引查询,等值查询、in查询或范围查询几种。详细的可以查看官方介绍。
如果用sysbench压测ob,创建很多表是一种方法。另外一种方法就是创建分区表。oceanbase是分布式数据库,数据迁移和高可用的最小粒度是分区,分区是数据表的子集。分区表有多个分区,非分区表只有一个分区。分区表的拆分细节是业务可以定义的。oceanbase可以将多个分区分布到不同节点,也可能不会分布到多个节点。这个取决于ob集群和租户的规划设计。所以对sysbench创建的表(分区),在性能分析时要分析分区具体的位置。
建表准备
下面是sysbench里分区表示例,修改 oltp_common.lua:
query = string.format([[ create table sbtest%d( id %s, k integer default '0' not null, c char(120) default '' not null, pad char(60) default '' not null, %s (id,k) ) partition by hash(k) partitions %s %s %s]], table_num, id_def, id_index_def, part_num, engine_def, extra_table_options)
需要注意用分区表后,主键列和唯一索引列需要包含分区键。分区表的索引有本地(local)索引和全局索引两种。本地索引的存储跟数据是在一起的,全局索引的存储是独立的。全局索引是为了应对查询条件不是分区键的场景,没有全局索引时,会扫描所有分区的本地索引。这两种索引的性能优劣没有定论,以实际业务场景测试为准。此外,传统数据库索引对修改操作会有负面影响,分布式数据库的全局索引对修改操作的影响可能更大,因为一个简单的dml语句都会因为要同步修改全局索引而产生分布式事务。而本地索引就没有分布式事务问题。所以对全局索引的使用场景要谨慎评估。这个特性不是ob特有,只要是分布数据库都会面临这个问题。
数据分布均衡
本节是说明ob数据分布背后的原理和方法。
oceanbase是分布式数据库,它通过每个机器上的observer进程将多个机器的资源能力聚合成一个大的资源池,然后再为每个业务分配不同资源规格的租户(实例)。所以每个业务租户(实例)的资源能力都只是整个集群能力的子集。业务并不一定能使用到全部机器资源。
oceanbase里的数据都有三份,细到每个分区有三个副本,角色上有1个leader副本2个follower副本。默认只有leader副本提供读写服务,leader副本所在的节点才有可能提供服务,有负载。oceanbase调整各个节点负载的方法是通过调整内部leader副本的位置实现的。这个调整可以让oceanbase自动做,也可以手动控制。
oceanbase自动负载均衡是参数enable_rebalance控制,默认值为true。可以查看确认。修改用alter system语句。
alter system set enable_rebalance=true; show parameters like 'enable_rebalance';
查看实际表的分区leader副本位置
查看分区分布 select t5.tenant_id,t5.tenant_name,t3.database_name, t4.tablegroup_name, t2.table_name, t1.partition_id, concat(t1.svr_ip,':',t1.svr_port) observer, t1.role, t1.data_size,t1.row_count from `gv$partition` t1 join `__all_table` t2 on (t1.tenant_id=t2.tenant_id and t1.table_id=t2.table_id) join `__all_database` t3 on (t2.tenant_id=t3.tenant_id and t2.database_id=t3.database_id) left join `__all_tablegroup` t4 on (t1.tenant_id=t4.tenant_id and t1.tablegroup_id=t4.tablegroup_id) join `__all_tenant` t5 on (t1.tenant_id=t5.tenant_id) where t5.tenant_id = 1020 and role=1 and database_name in ('sysbenchtest') order by t5.tenant_id,t3.database_id,t1.tablegroup_id,t1.partition_id, t1.role;
sysbench命令参数
sysbench的机制是压测过程中如果有错误就会报错退出,所以需要针对一些常见的错误进行忽略处理,这样sysbench会话可以重试继续运行。比如说主键或者唯一键冲突、事务被杀等等。
下面的运行命令供参考
./sysbench --test=./oltp_read_only.lua --mysql-host=***.***.82.173 --mysql-port=4001 --mysql-db=test --mysql-user="sbuser" --mysql-password=sbtest --tables=16 --table_size=100000000 --threads=32 --time=300 --report-interval=5 --db-driver=mysql --db-ps-mode=disable --skip-trx=on --mysql-ignore-errors=6002,6004,4012,2013,4016 prepare
./sysbench --test=./oltp_read_only.lua --mysql-host=***.***.82.173 --mysql-port=4001 --mysql-db=test --mysql-user="sbuser" --mysql-password=sbtest --tables=16 --table_size=100000000 --threads=96 --time=600 --db-driver=mysql --db-ps-mode=disable --skip-trx=on --mysql-ignore-errors=6002,6004,4012,2013,4016 --secondary=on run
./sysbench --test=./oltp_write_only.lua --mysql-host=***.***.82.173 --mysql-port=4001 --mysql-db=test --mysql-user="sbuser" --mysql-password=sbtest --tables=16 --table_size=100000000 --threads=32 --time=600 --report-interval=5 --db-driver=mysql --db-ps-mode=disable --skip-trx=on --mysql-ignore-errors=6002,6004,4012,2013,4016,1062 run
./sysbench --test=./oltp_read_write.lua --mysql-host=***.***.82.173 --mysql-port=4001 --mysql-db=test --mysql-user="sbuser" --mysql-password=sbtest --tables=16 --table_size=100000000 --threads=32 --time=600 --report-interval=5 --db-driver=mysql --db-ps-mode=disable --skip-trx=on --mysql-ignore-errors=6002,6004,4012,2013,4016,1062 run
测试观察
在纯写或者读写测试中,注意观察增量增量内存使用进度。如果写入速度比转储和合并速度还快,那会碰到内存不足写入失败错误。这就是ob租户资源相对不足了。观察这个内存使用进度可以通过 dooba脚本。dooba脚本默认在/home/admin/oceanbase/bin/目录下。
使用示例如下:
python dooba -h11..84.84 -uroot@sysobdemo -p2883 -p**
ob有个内部视图gv$sql_audit可以查看执行过所有成功或失败的sql,用来分析具体的sql性能。用法详情参见开云体育官网入口官网(oceanbase.alipay.com) 或 文章《阿里数据库性能诊断的利器——sql全量日志》。
select /* read_consistency(weak) query_timeout(1000000000) */ usec_to_time(request_time) req_time, svr_ip, trace_id, sid, client_ip, tenant_id,tenant_name,user_name,db_name, query_sql, affected_rows,ret_code, event, state, elapsed_time, execute_time, queue_time, decode_time, get_plan_time, block_cache_hit, bloom_filter_cache_hit, block_index_cache_hit, disk_reads,retry_cnt,table_scan, memstore_read_row_count, ssstore_read_row_count, round(request_memory_used/1024/1024) req_mem_mb from gv$sql_audit where tenant_id=1012 and user_name in ('demouser') order by request_time desc limit 100;
经验总结
本节是一些测试场景的经验总结,需要提前了解一些oceanbase的原理特性介绍。
实际测试情形可能会出现由于是三节点部署,所以测试时压力都打到一台服务器上,这个建议用六节点测试。还有个办法就是禁用自动负载均衡手动调整分区位置。调整是ob给业务的手段,业务上有些表会有join,为了性能(避免跨节点请求),业务需要这种干预能力。
还有一种情形是写入压力非常大,跑了一段时间后报内存不足的提示,这个就是租户内存资源相对写入速度和量不足了(ob的转储和合并对内存的回收赶不上写入对内存的消耗),此时需要扩容或者调整测试需求。ob 2.x版本还有自动对应用写入限速功能(自我保护),这个会影响测试报告里性能结果。如果数据库分析sql性能以及分布式调优都做了,那可以认为当前写入的tps就是数据库的写入峰值了。需要注意的是不同的硬件,不同的租户规格,不同的测试场景,这个tps能力都表现不同。
sysbench有个batch insert功能,当表是分区表的时候,默认这个batch insert 很可能会产生分布式事务,性能比单表写入要慢。需要靠提高客户端并发数来提升总的吞吐量。此外这个批量的大小不宜太大。由于ob支持sql执行计划缓存,sql文本过大且并发很高时,会在sql解析环节面临内存不足问题。数据库内存的大部分还是主要用于存取数据。java的addbatch方法也同理,建议批量大小设置为100以内。
在查询验证数据的时候,可能会碰到超时类错误。ob里超时的可能场景有多个:
以上超时时间都可以在租户里根据实际情况修改。
关于ob的错误号解释可以查看开云体育官网入口官网系统错误码 (http://oceanbase.alipay.com/docs/oceanbase/参考类/系统错误码/tslkmg)
关于使用分布式数据库测试,不同的做法可以得到不同的结果。即使是同一个ob租户(实例),不同的人设计的表结构不同,或者sql不同都有可能取得不同的性能数据。这些不同都是可以从原理上给出解释。熟知这些原理的更容易发挥ob的分布式数据库特点。使用越深入会发现这个原理越多且更加有趣。当然环境、场景的不同,还是会遇到一些问题。如果有碰到问题,欢迎公众号留言讨论。
oceanbase数据库实践入门——常用操作sql
http://mp.weixin.qq.com/s?spm=a2c4e.11153940.0.0.2a6d74f5axj6cg&__biz=mzu3otc2mdqxng==&mid=2247483909&idx=1&sn=e67effbf7bd25ed45806c006eaad5479&chksm=fd607a4aca17f35c1d2b0c044e5fe36f230ccf06f4c52aa8a171311aca15490a4e24f45fb471&scene=21wechat_redirect
oceanbase分区表有什么不同?
http://mp.weixin.qq.com/s?spm=a2c4e.11153940.0.0.2a6d74f5axj6cg&__biz=mzu3otc2mdqxng==&mid=2247483854&idx=1&sn=720ecfa2b8438049527c60d0ac9e85a8&chksm=fd607981ca17f097cae06419ccc635fc530da406b7e0da7a68b7381dfdcc6e78d73b75943fc0&scene=21wechat_redirect
阿里数据库性能诊断的利器——sql全量日志
http://mp.weixin.qq.com/s?spm=a2c4e.11153940.0.0.2a6d74f5axj6cg&__biz=mzu3otc2mdqxng==&mid=2247483844&idx=1&sn=bad7e1fc0ac6e3a82778c23636f2463d&chksm=fd60798bca17f09d3d36cec6ac67be36be91e8e92b2520637b7327983f1701a00f63d459406b&scene=21wechat_redirect
揭秘oceanbase的弹性伸缩和负载均衡原理
http://mp.weixin.qq.com/s?spm=a2c4e.11153940.0.0.2a6d74f5axj6cg&__biz=mzu3otc2mdqxng==&mid=2247483956&idx=1&sn=8ddc7604c01dfbcdf31f755b4d5d39bc&chksm=fd607a7bca17f36da0f85d8fab2b2e7e9c4a8a49ff8439bc2af442e2b92a7594e453de594cf8&scene=21wechat_redirect
作者:mq4096
了解更多