基于PostgreSQL的图数据库AgensGraph简介

一 AgensGraph简介

AgensGraph 是一个基于 PostgreSQL 的新一代多模型图数据库。它提供图形分析环境,用户可以同时编写、编辑和执行 SQL 和 Cypher 查询。AgensGraph 带有 PostgreSQL 兼容性和 PostgreSQL扩展,能够帮助PostgreSQL用户摆脱数据迁移的痛苦,轻松开发提供高级数据分析的服务。

1.1 AgensGraph特点

  • 1 万物互联,无论是蛋白网络还是电网,图数据库是必然需求,标准Cypher图查询。
  • 2 运维配置同PostgreSQL几乎一样,熟悉PG的DBA和开发人员上手容易,开发效率高。
  • 3 支持sql和Cypher混合查询。两种查询可以实现join操作,从而实现图查询结果匹配对应业务,整体结果输出,减少后端开发工作。
  • 4 支持csv导入导出,方便rdb数据迁移。
  • 5 可以使用PG库丰富的其他特性,比如 图+PotGIS,或者数据库自己的事务隔离等等,分布式的话,可以参考pg的分布式方案(个人认为是不改源码的那种,比如fdw或者citus方案,其他pgxc/xl和gp不适合,因为这个agens不是插件,是单独的改的pg的数据库,插件可以用)。

1.2 AgensGraph vs NoSQL

AgensGraph即支持图模型,还支持关系模型,文档模型,kv模型。


AgensGraph支持多种模型

二 AgnesGraph安装

安装和PostgreSQL几乎一模一样,详情如下:

2.1 安装依赖

 yum install gcc glibc glib-common readline readline-devel zlib zlib-devel flex bison

2.2 创建用户

[root@zz ~]# useradd agens
[root@zz ~]# chown -R agens.agens /home/agens

2.3 源码安装

#下载release
[root@zz ~]#wget https://github.com/bitnine-oss/agensgraph/archive/v2.1.1.tar.gz
#解压
[root@zz ~]# tar -zxvf v2.1.1.tar.gz
#进入目录安装
[root@zz ~]# cd agensgraph-2.1.1
[root@zz agensgraph-2.1.1]# ./configure --prefix=/home/agens
[root@zz agensgraph-2.1.1]# make
[root@zz agensgraph-2.1.1]# make install
#安装contrib的pg扩展工具
[root@zz agensgraph-2.1.1]# cd contrib
[root@zz contrib]# make
[root@zz contrib]# make install

2.4 数据库初始化

2.4.1 设置环境变量

[root@zz contrib]# su - agens
[agens@zz ~]$ vi .bashrc
#编辑内如如下:
AGHOME=/home/agens
export AGHOME
AGDATA=$AGHOME/data
export AGDATA
PATH=$PATH:$HOME/.local/bin:$HOME/bin:$AGHOME/bin
export PATH

#保存启用环境变量
[agens@zz ~]$ source .bashrc
#初始化数据库
[agens@zz ~]$ initdb -D $AGDATA
#启动数据库
[agens@zz ~]$ ag_ctl -D $AGDATA
#psql登录数据库,进入我们熟悉的psql环境
[agens@zz ~]$ psql -d postgres
psql (10.4)
Type "help" for help.

postgres=# 

结论:和postgresql安装和操作一模一样,其他的如登录授权的pg_hba.conf和服务设置文件postgresql.conf设置和pg也是一样的。

三 AgensGraph概念

3.1 AgensGraph数据库架构

AgensGraph是一个多模型数据库,既支持带属性的图模型,也支持关系模型。


Agens数据库

Agens是基于PostgreSQL数据库的,通过schema,支持原生的关系模型,也就是普通的关系表等;通过graph,支持图模型的vertice(顶点)和edge(边)。一句话,在Agens你可以操作关系表,也可以操作图。

3.2 图模型概念

带属性的图模型

顶点(Vertices):一个实体一个顶点。一个实体可以有多个属性。
边(Edges):两个实体之间的连接线。当试图删除顶点时,要先删除连接该顶点的边才行(AgensGraph中不允许存在断边,也就是边一定会有连接的两个顶点)。
属性:实体和边都可以有多个属性。形象举个例子,一个实体对应关系表中一行记录,一个实体的属性代表关系表中这行记录的所有字段和值构成的键值对。
标签:AgensGraph中有VLABEL ,ELABEL 两个对象,分别定义顶点,边的分组。形象举个例子,标签类似与关系表的表名称,那么一个标签分组下的实体和边,其属性的键名称和数量一致。(ps:理论上是可以不一致,属性是nosql的json形式,但是,实际中的图大部分是关系表导出,所以实际中几乎一致)。

四 AgensGraph练习

方便后面描述,新建一个测试数据库作图数据库的测试库。

[agens@zz ~]$ psql -d postgres
psql (10.4)
Type "help" for help.
#新建测试库
postgres=# create database agens_test;
#切换测试库
postgres=# \c agens_test;

4.1 图管理

4.1.1 创建

agens_test=# create graph dlxx;

4.1.2 查询当前应用图

agens_test=# SHOW graph_path; 
 graph_path 
------------
 dlxx
(1 row)

4.1.3 设置当前应用图

假设当前应用图为空,可以先设置一个应用图,如下:

agens_test=# SHOW graph_path; 
 graph_path 
------------
 
(1 row)

agens_test=# set graph_path='dlxx';
SET
agens_test=# SHOW graph_path; 
 graph_path 
------------
 dlxx
(1 row)

4.1.4 删除图

cacsade级联删除,会将该图下的对象都删除:

agens_test=# drop graph dlxx cascade;

4.2 标签(Lables)管理

4.2.1 标签创建

#创建顶点标签
agens_test=# CREATE VLABEL person;
CREATE VLABEL
#创建顶点标签,标签friend继承标签person。
agens_test=# CREATE VLABEL friend inherits (person);
CREATE VLABEL

#创建边标签
agens_test=# CREATE ELABEL knows;
CREATE ELABEL
agens_test=# CREATE ELABEL live_together;
CREATE ELABEL
#创建边标签,标签room_mate继承标签knows和live_together
agens_test=# CREATE ELABEL room_mate inherits (knows, live_together);
CREATE ELABEL

4.2.2 标签删除

agens_test=# drop ELABEL knows;
ERROR:  cannot drop elabel knows because other objects depend on it
DETAIL:  elabel room_mate depends on elabel knows
HINT:  Use DROP ... CASCADE to drop the dependent objects too.
# 因为ELABEL room_mate继承knows,所以删除不了。可以通过两个方式删除
# 方式一
agens_test=# drop ELABEL room_mate;
agens_test=# drop ELABEL knows;
#方式二
agens_test=# drop ELABEL knows cascade;

这些操作其实都是pg的,一模一样。

4.3 索引和约束

待补充

4.4 图查询

4.4.1 创建图的连接性

顶点数据格式:(variable:label {property: value, ...})
边数据格式:-[variable:label {property: value, ...}]-
最左边的附加<或最右边的>用于表示边的方向

CREATE VLABEL person;
CREATE ELABEL knows;
#name=tom的顶点,通过边(knows),连接最右边的name=summer的顶点
CREATE (:person {name: 'Tom'})-[:knows {fromdate:'2011-11-24'}]->(:person {name: 'Summer'});
CREATE (:person {name: 'Pat'})-[:knows {fromdate:'2013-12-25'}]->(:person {name: 'Nikki'});
CREATE (:person {name: 'Olive'})-[:knows {fromdate:'2015-01-26'}]->(:person {name: 'Todd'});
#使用MATCH命令,从Person的顶点标签 选择name属性为tom和pat的两个顶点
#这两个顶点分别设置代指p和k,创建p-edge-k的图连接关系
MATCH (p:Person {name: 'Tom'}),(k:Person{name: 'Pat'}) 
CREATE (p)-[:KNOWS {fromdate:'2017-02-27'} ]->(k);

图形如下:


示例图

示例1:查询与属性name=‘Tom’的顶点有直接连接关系的其他标签为person的顶点:

MATCH (n:person {name: 'Tom'})-[:knows]->(m:person) RETURN n.name AS n, m.name AS m;
   n   |    m
-------+----------
 "Tom" | "Summer"
 "Tom" | "Pat"
(2 rows)

示例2:查询与属性name=‘Tom’的顶点有两层连接关系的其他标签为person的顶点:

MATCH (p:person {name: 'Tom'})-[:knows]->()-[:knows]->(f:person) RETURN f.name;
  name   
---------
 "Nikki"
(1 row)

示例3:使用union将图查询结果合并输出:

MATCH (p:person {name: 'Tom'})-[:knows]->(f:person) RETURN f.name
 UNION ALL
MATCH (p:person {name: 'Tom'})-[:knows]->()-[:knows]->(f:person) RETURN f.name;
   name   
----------
 "Summer"
 "Pat"
 "Nikki"
(3 rows)

以上语句可以使用以下等效,注意[r:knows*1..2]写法,1..2代表1级到2级:

MATCH (p:person {name: 'Tom'})-[r:knows*1..2]->(f:person)
RETURN f.name, r[1].fromdate;
   name   |   fromdate   
----------+--------------
 "Summer" | 
 "Pat"    | 
 "Nikki"  | "2013-12-25"
(3 rows)

在图数据库中,典型的查询是查找位于可变连接长度的边-顶点路径之后的顶点。 边中使用的* 1..2表示这样的变长边。 其中1是边的最小长度,2是最大长度。
不指定值,默认值应该是1,就是就一层连接(直接连接关系,有一个边即可达):

MATCH (p:person {name: 'Tom'})-[r:knows]->(f:person) RETURN f.name;
   name   
----------
 "Summer"
 "Pat"
(2 rows)

指定*,就是有多个边可达,都属于查询要求:

MATCH (p:person {name: 'Tom'})-[r:knows*]->(f:person) RETURN f.name;
   name   
----------
 "Summer"
 "Pat"
 "Nikki"
(3 rows)

4.5 图处理(增删改)

4.5.1 create

CREATE (:person {name: 'Tom'})-[:knows {fromdate:'2011-11-24'}]->(:person {name: 'Summer'});

4.5.2 set

MATCH (:person {name: 'Tom'})-[r:knows]->(:person {name: 'Summer'})
SET r.since = '2009-01-08';

4.5.3 delete

MATCH (n:person {name: 'Pat'}) DETACH DELETE (n);

DETACH DELETE将相关的顶点和边一次性全部删除。类似sql的 cascade,级联删除,因为agens不允许断边存在,删除了顶点,这个顶点相关的边都要删了,使用DETACH DELETE更常用。

4.5.4 merge

这个和sql中的insert into on confict do很像,没有就插入,有就更新,是个命令合并语法。
merge的意思是,如果图中有查询条件的数据,等同于match,就是查找。如果图中没有查询条件的数据,等同于create,就是创建。
merge就是有就match,没有就create的复合语法。

CREATE VLABEL customer;
CREATE VLABEL

CREATE VLABEL city;
CREATE VLABEL

CREATE (:customer {name:'Tom', city:'santa clara'}),
(:customer {name:'Summer ', city:'san jose'}),
(:customer {name:'Pat', city:'santa clara'}),
(:customer {name:'Nikki', city:'san jose'}),
(:customer {name:'Olive', city:'san francisco'});
GRAPH WRITE (INSERT VERTEX 5, INSERT EDGE 0)

#检索所有的custome,将city属性通过merge语法到city顶点标签。
MATCH (a:customer) MERGE (c:city {name:a.city});
GRAPH WRITE (INSERT VERTEX 3, INSERT EDGE 0)

#原来city标签是未赋值的,merge之后,属性值发生变化。
MATCH (c:city) RETURN properties(c);
        properties         
---------------------------
 {"name": "santa clara"}
 {"name": "san jose"}
 {"name": "san francisco"}
(3 rows)

MERGE可以通过 ON MATCH SET和 ON Create SET修改对应图中顶点或者边的属性。

CREATE (:customer {name:'Todd', city:'palo alto'});

MATCH (a:customer)
MERGE (c:city {name:a.city})
    ON MATCH SET c.matched = 'true'
    ON CREATE SET c.created = 'true';
    
#匹配到的是matched=true,未匹配的会create,设置created=true
MATCH (c:city) RETURN properties(c);
                  properties
----------------------------------------------
 {"name": "santa clara", "matched": "true"}
 {"name": "san jose", "matched": "true"}
 {"name": "san francisco", "matched": "true"}
 {"name": "palo alto", "created": "true"}
(4 rows)

4.5.5 事务隔离

BEGIN;
SET TRANSACTION ISOLATION LEVEL SERIALIZABLE;
MATCH (a:customer)
MERGE (c:city {name:a.city});
COMMIT;

4.5.6 最短路径

新建一个连接,tom指向olive:

MATCH (p:person {name:'Tom'}), (f:person {name:'Olive'}) CREATE (p)-[:knows]->(f);
GRAPH WRITE (INSERT VERTEX 0, INSERT EDGE 1)

新建后的图更新如下:

示例图更新

shortestpath方法用于查询最短路径,需要制定起点,终点的顶点,可以通过-[:knows*1..5]-这只edge的查找深度,比如1-5层连接关系:

MATCH (p1:person {name: 'Tom'}), (p2:person {name: 'Todd'}),
path=shortestpath((p1)-[:knows*1..5]->(p2)) RETURN path;
                                                                           path                                                                            
-----------------------------------------------------------------------------------------------------------------------------------------------------------
 [person[3.1]{"name": "Tom"},knows[5.5][3.1,3.5]{},person[3.5]{"name": "Olive"},knows[5.3][3.5,3.6]{"fromdate": "2015-01-26"},person[3.6]{"name": "Todd"}]
(1 row)

查询结果是:tom到olive到todd,上图示意符合一致。

4.5.7 sql和图查询的混合查询

构建下测试图数据与关系表数据:

CREATE GRAPH bitnine;
CREATE VLABEL dev;
CREATE (:dev {name: 'someone', year: 2015});
CREATE (:dev {name: 'somebody', year: 2016});

CREATE TABLE history (year, event)
AS VALUES (1996, 'PostgreSQL'), (2016, 'AgensGraph'),(2019, 'sssss');

执行查询,查询图中年份小于history的年份。图在agens其实都是jsonb的nosql格式,可以通过->>等pg中json,jsonb格式根据键名称获取值等函数,详细查看pg的jsonb function文档:
cypher in sql:

SELECT history.*,n->>'name' as name 
FROM history, (MATCH (n:dev) RETURN n) as dev 
WHERE history.year > (n->>'year')::int;
 year |   event    |   name   
------+------------+----------
 2016 | AgensGraph | someone
 2019 | sssss      | someone
 2019 | sssss      | somebody
(3 rows)

sql in cypher:

MATCH (n:dev) WHERE n.year < (SELECT year FROM history WHERE 
event = 'AgensGraph') RETURN properties(n) AS n;
                 n                 
-----------------------------------
 {"name": "someone", "year": 2015}
(1 row)

最后编辑于
©著作权归作者所有,转载或内容合作请联系作者
  • 序言:七十年代末,一起剥皮案震惊了整个滨河市,随后出现的几起案子,更是在滨河造成了极大的恐慌,老刑警刘岩,带你破解...
    沈念sama阅读 214,922评论 6 497
  • 序言:滨河连续发生了三起死亡事件,死亡现场离奇诡异,居然都是意外死亡,警方通过查阅死者的电脑和手机,发现死者居然都...
    沈念sama阅读 91,591评论 3 389
  • 文/潘晓璐 我一进店门,熙熙楼的掌柜王于贵愁眉苦脸地迎上来,“玉大人,你说我怎么就摊上这事。” “怎么了?”我有些...
    开封第一讲书人阅读 160,546评论 0 350
  • 文/不坏的土叔 我叫张陵,是天一观的道长。 经常有香客问我,道长,这世上最难降的妖魔是什么? 我笑而不...
    开封第一讲书人阅读 57,467评论 1 288
  • 正文 为了忘掉前任,我火速办了婚礼,结果婚礼上,老公的妹妹穿的比我还像新娘。我一直安慰自己,他们只是感情好,可当我...
    茶点故事阅读 66,553评论 6 386
  • 文/花漫 我一把揭开白布。 她就那样静静地躺着,像睡着了一般。 火红的嫁衣衬着肌肤如雪。 梳的纹丝不乱的头发上,一...
    开封第一讲书人阅读 50,580评论 1 293
  • 那天,我揣着相机与录音,去河边找鬼。 笑死,一个胖子当着我的面吹牛,可吹牛的内容都是我干的。 我是一名探鬼主播,决...
    沈念sama阅读 39,588评论 3 414
  • 文/苍兰香墨 我猛地睁开眼,长吁一口气:“原来是场噩梦啊……” “哼!你这毒妇竟也来了?” 一声冷哼从身侧响起,我...
    开封第一讲书人阅读 38,334评论 0 270
  • 序言:老挝万荣一对情侣失踪,失踪者是张志新(化名)和其女友刘颖,没想到半个月后,有当地人在树林里发现了一具尸体,经...
    沈念sama阅读 44,780评论 1 307
  • 正文 独居荒郊野岭守林人离奇死亡,尸身上长有42处带血的脓包…… 初始之章·张勋 以下内容为张勋视角 年9月15日...
    茶点故事阅读 37,092评论 2 330
  • 正文 我和宋清朗相恋三年,在试婚纱的时候发现自己被绿了。 大学时的朋友给我发了我未婚夫和他白月光在一起吃饭的照片。...
    茶点故事阅读 39,270评论 1 344
  • 序言:一个原本活蹦乱跳的男人离奇死亡,死状恐怖,灵堂内的尸体忽然破棺而出,到底是诈尸还是另有隐情,我是刑警宁泽,带...
    沈念sama阅读 34,925评论 5 338
  • 正文 年R本政府宣布,位于F岛的核电站,受9级特大地震影响,放射性物质发生泄漏。R本人自食恶果不足惜,却给世界环境...
    茶点故事阅读 40,573评论 3 322
  • 文/蒙蒙 一、第九天 我趴在偏房一处隐蔽的房顶上张望。 院中可真热闹,春花似锦、人声如沸。这庄子的主人今日做“春日...
    开封第一讲书人阅读 31,194评论 0 21
  • 文/苍兰香墨 我抬头看了看天上的太阳。三九已至,却和暖如春,着一层夹袄步出监牢的瞬间,已是汗流浃背。 一阵脚步声响...
    开封第一讲书人阅读 32,437评论 1 268
  • 我被黑心中介骗来泰国打工, 没想到刚下飞机就差点儿被人妖公主榨干…… 1. 我叫王不留,地道东北人。 一个月前我还...
    沈念sama阅读 47,154评论 2 366
  • 正文 我出身青楼,却偏偏与公主长得像,于是被迫代替她去往敌国和亲。 传闻我的和亲对象是个残疾皇子,可洞房花烛夜当晚...
    茶点故事阅读 44,127评论 2 352

推荐阅读更多精彩内容