| |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
-> 大数据 -> 一文带你了解「图数据库」Nebula 的存储设计和思考 -> 正文阅读 |
|
[大数据]一文带你了解「图数据库」Nebula 的存储设计和思考 |
在上次的 nebula-storage on nLive 直播中,来自 Nebula 存储团队的负责人王玉珏(四王)同大家分享了 nebula storage 这块的设计思考,也解答了一些来自社区小伙伴的提问。本文整理自该场直播,按照问题涉及的分类进行顺序调整,并非完全按照直播的时间先后排序。 Nebula 的存储架构整个 Storage 主要分三层,最下面是 Store Engine,也 就是 RocksDB,中间是 raft 一致性协议层,最上层 storage service 提供对外的 rpc 接口,比如取点属性,或者边属性,或者是去从某个点去找它的邻居之类的接口。当我们通过语句 Nebula 的存储数据格式这里着重讲述为何 v2.x 会有这些数据格式的改动:在 v1.x 版本中,Nebula VID 主要是 int 类型,所以大家可以看到上图 v1.x 中不管是点还是边,它的 VID 是定长的、占 8 个字节。2.x 版本开始,为了支持 string 类型 VID,VertexID 就变成不定长的 n 个字节。所以大家创建 Space 的时候需要指定 VID 的长度,这个是最主要的改动,其他的话还有一些小的改动,去掉了时间戳。整体来说,目前的存储格式更贴近图的使用场景——从某个点开始找它的邻居,以 v2.x 这样 ?VertexID + EdgeType? 存储格式来保存边的话,可以迅速地找到某个点出边。 同时,v2.x 也做了 key(Nebula 底层是 KV 存储的)编码格式上的改变,简单来说就是把点和边分开。这样的话,取某一个点所有 tag 时通过一次 prefix 就可以直接扫到,避免了像 v1.x 那样扫描点的过程中夹杂多个边的问题。 底层的数据存储针对用户提出的“Nebula 底层如何存储数据”的问题,四王了进行了回复:Nebula 的存储层使用 KV 进行存储点边数据。对于一个点而言,key 里面存储 VID 和它的 tag 类型。点的 value 中,会根据 这个 tag 的 schema,将 schema 中的各个属性进行编码并存在 value 中。比如,player 这个 tag 可能会有一个 age 这样一个整型年龄字段,使用存储的时候会把 age 字段的值,按某种编码保存在 value 中。再来说下边,边的存储 key 会多几个字段,主要是边的起点 ID、边类型、ranking 及终点类型,通过这四元组确定唯一的边。边的 value 和点的 value 类似,根据边的 Schema 字段定义,将各个字段进行编码存储。这里要说一下,Nebula 中存储边是存储两份:Nebula 中的边是有向边,存储层会存储正向边和反向边,这样的好处在于使用 一般来说,图存储分为切边和切点两种方式,像上面说的 Nebula 其实采用了切边方式:一条边存储两份 KV。
切边的话,每一份边存两份,数据总量会比切点大很多,因为图数据边的数量是远大于点的数量,造成边的大量冗余,相对好处是对起点和它的边进行映射时会映射到同一个 partition 上,这样进行一些从单个点触发的 query 时会很快速得到结果。切点的话,由于点可能被分在多个机器上,更新数据时得考虑数据的一致性问题,一般在图计算里面切点的使用会更广泛。 你问我答下面内容收集于之前活动预告的 AMA 环节,以及直播时弹幕中提出的问题。 问题目录
边的 value 存储边属性吗?和上面底层存储里讲的那样,创建 Edge 的 schema 时候会指定边类型上的属性,这些属性会作为底层 RocksDB key 的 value 存储起来,这个 value 的占位是定长的,和下面这个问题类似: 强 Schema 的设计原因
其实本质上原因是用强 Schema 的好处是快,先说下常见的简单数据类型,比如:int 和 double,这样的数据类型长度是固定的,我们会直接在 value 相应的位置进行编码。再说下 string 类型,在 Nebula 中有两种 string :一种是定长 string,长度是固定,和前面的简单数据类型一样,在 value 的固定位置进行编码。另外一种是变长的 string,通常来说大家都会比较倾向于变长 string(灵活),非定长 string 会以指针形式存储。 举个例子,schema 中有个属性是变长 string 类型,我们不会和简单数据类型一样直接编码保存,而是在相应位置保存一个 offset 指针,实际指向 value 中的第 100 个字节,然后在 100 这个位置才保存这个变长 string。所以读取变长 string 的时候,我们需要在 value 中读两次,第一次获取 offset,第二次才能真正把 string 读出来。通过这样的形式,把所有类型的属性都转化成"定长",这样设计的好处是,根据要读取的属性和它前面所有字段的占用字节大小,可以直接计算出要读取的字段在 value 中存储的位置,并把它读出来。读取过程中,不需要读取无关的字段,避免了弱 schema 需要对整个 value 进行解码的问题。 像 Neo4j 这种图数据库,一般是 No Schema,这样写入的时候会比较灵活,但序列化和反序列化时都会消耗一些 CPU,并且读取的时候需要重新解码。
可能 value 长度会不一样,因为本身是变长嘛。
存一份边的设计
其实这是一个比较好的问题,其实在 Nebula? 最早期设计中是只存一份边的属性,这适用于部分业务场景。举个例子,你不需要任何的反向遍历,这种情况下是完全不需要存反向边。目前来说,存反向边最大的意义是方便于我们做反向查询。其实在 Nebula 比较早的版本中,准确说它是只存了反向边的 key,边类型的属性值是没有存,属性值只存在正向边上。它可能带来一些问题,双向遍历或者反向查询时,整个代码逻辑包括处理流程都会比较复杂。 图空间如何做物理隔离大家在用 Nebula 时,首先会建图空间 Meta 如何存储 Schema我们以 这里说下,meta 和 storage 两个 service 底层都是 RocksDB 采用 kv 存储,只不过提供了不一样的接口,比如说,meta 提供的接口,可能就是保存某个 tag,以及 tag 上有哪些属性;或者是机器或者 space 之类的元信息,包括像用户权限、配置信息都是存在 meta 里。storage 也是 kv 存储,不过存储的数据是点边数据,提供的接口是取点、取边、取某个点所有出边之类的图操作。整体上,meta 和 storage 在 kv 存储层代码是一模一样,只不过往上暴露的对外接口是不一样的。 最后,storage 和 meta 是分开存储的,二者不是一个进程且存的目录在启动的时指定的也不一样。
是这样,通常来说 Nebula 建议 meta 以三副本方式部署。这样的话,只挂一台机器是没有问题的。如果单副本部署 meta 挂了的话,是无法对 schema 进行任何操作,包括不能创建 space。因为 storage 和 graph 是不强依赖 meta 的,只有在启动时会从 meta 获取信息,之后都是定期地获取 meta 存储的信息,所以如果你在整个集群跑的过程中,meta 挂了而又不做 schema 修改的话,对 graph 和 storage 是不会有任何影响的。 存储未来规划
性能这块,Nebula 底层采用了 RocksDB,而它的性能主要取决于使用方式,和调参的熟练程度,坦白来说,即便是 Facebook 内部员工来调参也是一门玄学。再者,刚才介绍了 Nebula 的底层 key 存储,比如说 VID 或者是 EdgeType 在底层存储的相对位置某种程度上决定了部分 Query 会有性能影响。从抛开 RocksDB 本身来说,其实还有很多性能上的事情可做:一是写点或者写边时,有些索引需要处理,这会带来额外性能开销。此外,Compaction 和实际业务 workload 也会对性能有很大影响。 稳定性上,Nebula 底层采用 raft 协议,这是保证 Nebula Graph 不丢数据一个非常关键的点。因为只有这层稳定了,再往下面的 RocksDB 写入数据才不会出现数据不一致或者数据丢失的情况发生。此外,Nebula 本身是按照通用型数据库来设计的,会遇到一些通用型数据库共同面临的问题,比如说 DDL 改变;而本身 Nebula 是一款分布式图数据库,也会面临分布式系统所遇到的问题,像网络隔离、网络中断、各种超时或者因为某些原因节点挂了。上面这些问题的话,都需要有应对机制,比如 Nebula 目前支持动态扩缩容,整个流程非常复杂,需要在 meta 上、以及挂掉的节点、剩余“活着”的节点进行数据迁移工作。在这个过程中,中间任何一步失败都要做 Failover 处理。 可用性方面,我们后续会引入主备架构。在有些场景下所涉及的数据量会比较少,不太需要存三副本,单机存储即可。这种全部数据就在单机上的情况,可以减去不必要的 RPC 调用,直接换成本地调用,性能可能会有很大的提升。因为,Nebula 部署一共起 3 个服务:meta、graph 和 storage,如果是单机部署的话,graph + storage 可以放在同一台机器上,原先 graph 需要通过 RPC 调用 storage 接口去获取数据再回到 graph 进行运算。假如你的查询语句是多跳查询,从 graph 发送请求到 storage 这样的调用链路反复执行多次,这会导致网络开销、序列化和反序列化的这些损耗提高。 当 2 个进程(storaged 和 graphd)合在一起,便没有了 RPC 调用,所以性能会有个大提升。此外,这种单机情况下 CPU 利用率会很高,这也是目前 Nebula 存储团队在做的事情,会在下一个大版本同大家见面。 VID 遍历点和边的原理
从上图你可以看到存储了个 Type 类型,在 v1.x 版本中无论点和边 Type 类型都是一样的,所以就会发生上面说到过的扫描点会夹杂多个边的问题。在 v2.x 开始,将点和边的 Type 进行区分,前缀 Type 值就不一样了,给定一个 VID,无论是查所有 tag 还是所有边,都只需要一次前缀查询,且不会扫描额外数据。 数据预校验
是否符合定义的话,大概是这样,创建 Schema 时会要求指定某个字段是 nullable 或者是有默认值,或者既不是 nullable 也不带默认值。当我们插入一条数据的时候,插入语句会要求你“写明”各个字段的值分别是什么。而这条插入 Query 发到存储层后,存储层会检查是不是所有字段值都有设置,或者写入值的字段是否有默认值或者是 nullable。然后程序会去查是不是所有的字段都可以填上值。如果不是的话,系统会报错,告知用户 Query 有问题无法写入。如果没有报错,storage 就会对 value 进行编码,然后通过 raft 最后写到 RocksDB 里,整个流程大概是这样的。 Nebula 监测
这个是非常好的问题,目前答案是不能。这块我们在规划,这个问题的主要原因是 metrics 较少,目前我们支持的 metrics 只有 latency、qps 还有报错的 qps 这三类。每个指标有对应的平均值、最大值、最小值,sum 和 count,以及 p99 之类参数。目前是机器级别的 metrics,后续的话会做两个优化:一个增多 metrics;二是按 space 级别进行统计,对于每个空间来说,我们会提供诸如 fetch、go、lookup 之类语句的 qps。上面是 graph 这边的 metrics,而 storage 这块因为没有强资源隔离能力,还是提供集群或者单个机器级别的 metrics 而不是 space 级别的。 Nebula 的事务
先说下边事务的背景,背景是上面提到的 Nebula 是存了两份边 2 个 kv,这 2 个 kv 可能会存在不同的节点上,这会导致如果有台机器挂了,其中有一条边可能是没有成功写入。所谓边事务或者叫 TOSS,它主要解决的问题就是当我们遇到其中有一台机器宕机时,存储层能够保证这两个边(出边和入边)的最终一致。这个一致性级别是最终一致,没有选择强一致是因为研发过程中碰到一些报错信息以及数据处理流程上的问题,最后选择了最终一致性。 再来说下 TOSS 处理的整体流程,先往第一个要写入数据的机器发正向边信息,在机器上写个标记,看标记有没有写成功,如果成功了进入到下一步,如果失败直接报错。第二步的话,把反向边信息从第一台机器发给第二台机器,能让存正向边的机器向第二台机器发送反向边信息的原因是,Nebula 中正反向边只有起点和终点调换了一个位置,所以存正向边的机器是完全可以拼出反向边。存反向边的机器收到之后,会直接写入边,并将它的写入结果成功与否告诉第一台机器。第一台机器收到这个写入结果之后,假设它是成功的,它就会把之前第一步写的标记删掉,同时换成正常的边,这时整个边的正常写入流程就完成了,这是一个链式的同步机制。 简单说下失败的流程,一开始第一台机器写失败了直接就报错;第一台机器成功之后,第二台机器写失败了,这种情况下机器一会有背景线程,会一直不断尝试修复第二台机器的边,保证和第一台机器一样。当中比较复杂的是,第一台机器会根据第二台机器返回的错误码进行处理。目前来说,所有的流程都会直接把标记删掉,直接换成正常的正向边,同时写些更额外的标记来表示现在需要恢复的失败边,让它们最终保持一致。
是这样,因为点是只存了一份,所以它是不需要事务的。一般来说,问这个问题的人是想强调点和边之间的事务,像插入边时看点是否存在,或者删除点时删除对应边。目前 Nebula 的悬挂点的设计是出于性能上的考虑。如果要解决上面的问题的话,会引入完整的事务,但这样性能会有个数量级的递减。顺便提下,刚说到 TOSS 是链式形式同步信息,上面也提到能这样做的原因是因为第一个节点能完整拼出第二个节点的数据。但链式的话对完整的事务而言,性能下降会更严重,所以未来事务这块的设计不会采纳这种方式。 数据膨胀问题
大家发现如果磁盘占用高,一般来说是 WAL 文件比较多。因为我们导入的数据量一般比较大,这会产生大量的 wal,在 Nebula 中默认的 wal ttl 是 4 个小时,在这 4 个小时中系统的 WAL 日志是完全不会删除的,这就导致占用的磁盘空间会非常大。此外,RocksDB 中也会写入一份数据,相比后续集群正常运行一段时间,这时候磁盘占用会很高。对应的解决方法也比较简单,导入数据时调小 wal ttl 时间,比如只存半小时或者一个小时,这样磁盘占用率就会减少。当然磁盘空间够大你不做任何处理使用默认 4 小时也 ok。因为过了若干个小时后,有一个背景线程会不断去检查哪些 wal 可以删掉了,比如说默认值 4 个小时之后,一旦发现时过期的 wal 系统便会删掉。 除了初次导入会有个峰值之外,线上业务实时写入数据量并不会很大,wal 文件也相对小。这里不建议手动删 wal 文件,因为可能会出问题正常按照 ttl 来自动删除就行。
磁盘容量本身不均怎么处理
目前是不太好做,主要原因是存储 partition 分布查找是按照轮循形式进行的,另外一个原因是 Nebula 进行 Hash 分片,各个数据盘数据存储大小趋近。这会导致如果两个数据盘大小不一致,一个盘先满了后面的数据就写入不进去。解决方法可以从系统层进行处理,直接把两块盘绑成同一块盘,以同样一个路径挂载。 Nebula 的 RocksDB “魔改”
目前来说,其实我们完全没有用 column family,只用了default column family。后续可能会用,但是不会用来区分 vertex 属性,而是把不同 partition 数据分到不同 column family,这样的好处是直接物理隔离。
Nebula 社区首届征文活动进行中!🔗 奖品丰厚,全场景覆盖:撸码机械键盘??、手机无线充🔋、健康小助手智能手环??,更有数据库设计、知识图谱实践书籍📚 等你来领,还有 Nebula 精致周边送不停~🎁 欢迎对 Nebula 有兴趣、喜钻研的小伙伴来书写自己和 Nebula 有趣的故事呀~ 交流图数据库技术?加入 Nebula 交流群请先填写下你的 Nebula 名片,Nebula 小助手会拉你进群~~ |
|
|
上一篇文章 下一篇文章 查看所有文章 |
|
开发:
C++知识库
Java知识库
JavaScript
Python
PHP知识库
人工智能
区块链
大数据
移动开发
嵌入式
开发工具
数据结构与算法
开发测试
游戏开发
网络协议
系统运维
教程: HTML教程 CSS教程 JavaScript教程 Go语言教程 JQuery教程 VUE教程 VUE3教程 Bootstrap教程 SQL数据库教程 C语言教程 C++教程 Java教程 Python教程 Python3教程 C#教程 数码: 电脑 笔记本 显卡 显示器 固态硬盘 硬盘 耳机 手机 iphone vivo oppo 小米 华为 单反 装机 图拉丁 |
360图书馆 购物 三丰科技 阅读网 日历 万年历 2025年1日历 | -2025/1/16 8:39:52- |
|
网站联系: qq:121756557 email:121756557@qq.com IT数码 |