在 Postgres 里克隆一个 MongoDB

开发 前端 MongoDB
Postgres 社区在 NoSQL采取一系列动作后并没有坐以待毙. Postgres一直在进步: 集成了 JSON和 PLV8. PLV8 引入了 V8 Javascript引擎 . 操作 JSON也更简单了 (需要验证).

世上本没有路,走的人多了也就成了路。为什么不能在 Postgres上建一个MongoDB 呢? 

Postgres 社区在 NoSQL采取一系列动作后并没有坐以待毙.  Postgres一直在进步: 集成了 JSON和 PLV8.  PLV8 引入了 V8 Javascript引擎 . 操作 JSON也更简单了 (需要验证).

开始前需要做的准备:

 MongoDB的最低级别是集合.  集合可以用表来表示:

  1. CREATE TABLE some_collection ( 
  2.   some_collection_id SERIAL NOT NULL PRIMARY KEY
  3.   data JSON 
  4. ); 

字符型的JSON 被保存在 Postgres 表里,简单易行 (现在看是这样).

下面实现自动创建集合.  保存在集合表里:

  1. CREATE TABLE collection ( 
  2.   collection_id SERIAL NOT NULL PRIMARY KEY
  3.   name VARCHAR 
  4. ); 
  5.  
  6. -- make sure the name is unique 
  7. CREATE UNIQUE INDEX idx_collection_constraint ON collection (name); 

一旦表建好了,就可以通过存储过程自动创建集合.  方法就是先建表,然后插入建表序列.

  1. CREATE OR REPLACE FUNCTION create_collection(collection varcharRETURNS 
  2. boolean AS $$ 
  3.   var plan1 = plv8.prepare('INSERT INTO collection (name) VALUES ($1)', [ 'varchar' ]); 
  4.   var plan2 = plv8.prepare('CREATE TABLE col_' + collection + 
  5.     ' (col_' + collection + '_id INT NOT NULL PRIMARY KEY, data JSON)'); 
  6.   var plan3 = plv8.prepare('CREATE SEQUENCE seq_col_' + collection); 
  7.  
  8.   var ret; 
  9.  
  10.   try { 
  11.     plv8.subtransaction(function () { 
  12.       plan1.execute([ collection ]); 
  13.       plan2.execute([ ]); 
  14.       plan3.execute([ ]); 
  15.    
  16.       ret = true
  17.     }); 
  18.   } catch (err) { 
  19.     ret = false
  20.   } 
  21.  
  22.   plan1.free(); 
  23.   plan2.free(); 
  24.   plan3.free(); 
  25.  
  26.   return ret; 
  27. $$ LANGUAGE plv8 IMMUTABLE STRICT; 

有了存储过程,就方便多了:

  1. SELECT create_collection('my_collection'); 

解决了集合存储的问题,下面看看MongoDB数据解析.  MongoDB 通过点式注解方法操作完成这一动作:

  1. CREATE OR REPLACE FUNCTION find_in_obj(data json, key varcharRETURNS 
  2. VARCHAR AS $$ 
  3.   var obj = JSON.parse(data); 
  4.   var parts = key.split('.'); 
  5.  
  6.   var part = parts.shift(); 
  7.   while (part && (obj = obj[part]) !== undefined) { 
  8.     part = parts.shift(); 
  9.   } 
  10.  
  11.   // this will either be the value, or undefined 
  12.   return obj; 
  13. $$ LANGUAGE plv8 STRICT; 

上述功能返回VARCHAR,并不适用所有情形,但对于字符串的比较很有用:

  1. SELECT data 
  2.   FROM col_my_collection 
  3.  WHERE find_in_obj(data, 'some.element') = 'something cool' 

除了字符串的比较, MongoDB还提供了数字类型的比较并提供关键字exists .  下面是find_in_obj() 方法的不同实现:

  1. CREATE OR REPLACE FUNCTION find_in_obj_int(data json, key varcharRETURNS 
  2. INT AS $$ 
  3.   var obj = JSON.parse(data); 
  4.   var parts = key.split('.'); 
  5.  
  6.   var part = parts.shift(); 
  7.   while (part && (obj = obj[part]) !== undefined) { 
  8.     part = parts.shift(); 
  9.   } 
  10.  
  11.   return Number(obj); 
  12. $$ LANGUAGE plv8 STRICT; 
  13.  
  14. CREATE OR REPLACE FUNCTION find_in_obj_exists(data json, key varcharRETURNS 
  15. BOOLEAN AS $$ 
  16.   var obj = JSON.parse(data); 
  17.   var parts = key.split('.'); 
  18.  
  19.   var part = parts.shift(); 
  20.   while (part && (obj = obj[part]) !== undefined) { 
  21.     part = parts.shift(); 
  22.   } 
  23.  
  24.   return (obj === undefined ? 'f' : 't'); 
  25. $$ LANGUAGE plv8 STRICT; 

接下来是数据查询.  通过现有的材料来实现 find() 方法.

#p#

在本部分中将覆盖保存数据以及从MongDB查询中构建WHERE从句,以便检索我们已经写入的数据。

保存数据到集合中很简单。首先,我们需要检查JSON对象并寻找一个_id值。这部分代码是原生的假设,如果_id已存在这意味着一个更新,否则就意味着一个插入。请注意,我们目前还没有创建objectID,只使用了一个序列待其发生:

  1. CREATE OR REPLACE FUNCTION save(collection varchar, data json) RETURNS 
  2. BOOLEAN AS $$ 
  3.   var obj = JSON.parse(data); 
  4.  
  5.   var id = obj._id; 
  6.  
  7.   // if there is no id, naively assume an insert 
  8.   if (id === undefined) { 
  9.     // get the next value from the sequence for the ID 
  10.     var seq = plv8.prepare("SELECT nextval('seq_col_" + 
  11.         collection + "') AS id"); 
  12.     var rows = seq.execute([ ]); 
  13.    
  14.     id = rows[0].id; 
  15.     obj._id = id; 
  16.  
  17.     seq.free(); 
  18.  
  19.     var insert = plv8.prepare("INSERT INTO col_" + collection + 
  20.         "  (col_" + collection + "_id, data) VALUES ($1, $2)"
  21.         [ 'int''json']); 
  22.  
  23.     insert.execute([ id, JSON.stringify(obj) ]); 
  24.     insert.free(); 
  25.   } else { 
  26.     var update = plv8.prepare("UPDATE col_" + collection + 
  27.       " SET data = $1 WHERE col_" + collection + "_id = $2"
  28.      [ 'json''int' ]); 
  29.  
  30.     update.execute([ data, id ]); 
  31.   } 
  32.  
  33.   return true
  34. $$ LANGUAGE plv8 IMMUTABLE STRICT; 

基于这个观点,我们可以构建一些插入的简单文档:

  1.   "name""Jane Doe"
  2.   "address": { 
  3.     "street""123 Fake Street"
  4.     "city""Portland"
  5.     "state""OR" 
  6.   }, 
  7.   "age": 33 
  8.  
  9.   "name""Sarah Smith"
  10.   "address": { 
  11.     "street""456 Real Ave"
  12.     "city""Seattle"
  13.     "state""WA" 
  14.   } 
  15.  
  16.   "name""James Jones"
  17.   "address": { 
  18.     "street""789 Infinity Way"
  19.     "city""Oakland"
  20.     "state""CA" 
  21.   }, 
  22.   "age": 23 

让我们创建一个集合并插入一些数据:

  1. work=# SELECT create_collection('data'); 
  2.  create_collection 
  3. ------------------- 
  4.  t 
  5. (1 row) 
  6.  
  7. work=# SELECT save('data''{ our object }'); 
  8.  save 
  9. ------ 
  10.  t 
  11. (1 row) 

英文原文:Building a MongoDB Clone in Postgres: Part 1

译文链接:http://www.oschina.net/translate/building_a_mongodb_clone_in_postgres_part_1

责任编辑:林师授 来源: 中国开源社区 编译
相关推荐

2011-07-20 10:02:01

Xcode cocoa 窗口

2024-12-10 13:11:36

2016-04-08 15:13:29

人工智能阿里小Ai

2013-04-22 10:15:27

GoogleGoogle管理

2019-07-05 09:20:30

1G5G4G

2021-07-26 17:18:03

Linux进程通信

2021-01-04 09:12:31

集合变量

2009-09-22 11:54:42

ibmdwPHP

2023-03-15 09:00:43

SwiftUISlider

2009-04-22 15:16:30

2023-05-30 11:30:54

云原生多云IT

2021-06-03 10:00:47

JavaScript 前端数克隆对象

2011-09-08 10:46:12

Widget

2013-05-13 10:24:44

谷歌开发团队开发管理

2015-08-06 13:44:21

swiftcocoapods

2022-02-10 22:34:51

对象JVM收集器

2021-12-27 10:11:22

加密后门密码学数据安全

2017-11-23 08:30:27

数据库Postgres树莓派

2021-06-26 16:24:21

Linux命令系统

2017-08-17 14:38:39

JavaAbstract抽象
点赞
收藏

51CTO技术栈公众号