Skip to content

Commit 5a5fd6d

Browse files
authoredJun 25, 2024··
Merge pull request #24 from share/setup-ci
💥 Comply with `sharedb` spec
2 parents 36afc17 + aabcda4 commit 5a5fd6d

File tree

7 files changed

+205
-108
lines changed

7 files changed

+205
-108
lines changed
 

‎.github/workflows/test.yml

+56
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,56 @@
1+
name: Test
2+
3+
on:
4+
push:
5+
branches:
6+
- main
7+
- setup-ci # TODO: Remove
8+
pull_request:
9+
branches:
10+
- main
11+
12+
jobs:
13+
test:
14+
name: Node.js ${{ matrix.node }} + PostgreSQL ${{ matrix.postgres }}
15+
runs-on: ubuntu-latest
16+
strategy:
17+
fail-fast: false
18+
matrix:
19+
node:
20+
- 16
21+
- 18
22+
- 20
23+
postgres:
24+
- 13
25+
- 14
26+
- 15
27+
- 16
28+
services:
29+
postgres:
30+
image: postgres:${{ matrix.postgres }}
31+
env:
32+
POSTGRES_HOST_AUTH_METHOD: trust
33+
POSTGRES_DB: postgres
34+
POSTGRES_USER: postgres
35+
POSTGRES_PASSWORD: postgres
36+
options: >-
37+
--health-cmd pg_isready
38+
--health-interval 10s
39+
--health-timeout 5s
40+
--health-retries 5
41+
ports:
42+
- 5432:5432
43+
timeout-minutes: 10
44+
steps:
45+
- uses: actions/checkout@v4
46+
- uses: actions/setup-node@v4
47+
with:
48+
node-version: ${{ matrix.node }}
49+
- name: Install
50+
run: npm install
51+
- name: Test
52+
run: npm test
53+
env:
54+
PGUSER: postgres
55+
PGPASSWORD: postgres
56+
PGDATABASE: postgres

‎.mocharc.js

+5
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,5 @@
1+
module.exports = {
2+
timeout: 5_000,
3+
file: './test/setup.js',
4+
spec: '**/*.spec.js',
5+
};

‎index.js

+74-105
Original file line numberDiff line numberDiff line change
@@ -1,6 +1,8 @@
11
var DB = require('sharedb').DB;
22
var pg = require('pg');
33

4+
const PG_UNIQUE_VIOLATION = '23505';
5+
46
// Postgres-backed ShareDB database
57

68
function PostgresDB(options) {
@@ -9,24 +11,32 @@ function PostgresDB(options) {
911

1012
this.closed = false;
1113

12-
this.pg_config = options;
13-
this.pool = new pg.Pool(options);
14+
this._pool = new pg.Pool(options);
1415
};
1516
module.exports = PostgresDB;
1617

1718
PostgresDB.prototype = Object.create(DB.prototype);
1819

19-
PostgresDB.prototype.close = function(callback) {
20-
this.closed = true;
21-
this.pool.end();
22-
23-
if (callback) callback();
20+
PostgresDB.prototype.close = async function(callback) {
21+
let error;
22+
try {
23+
if (!this.closed) {
24+
this.closed = true;
25+
await this._pool.end();
26+
}
27+
} catch (err) {
28+
error = err;
29+
}
30+
31+
// FIXME: Don't swallow errors. Emit 'error' event?
32+
if (callback) callback(error);
2433
};
2534

2635

2736
// Persists an op and snapshot if it is for the next version. Calls back with
2837
// callback(err, succeeded)
29-
PostgresDB.prototype.commit = function(collection, id, op, snapshot, options, callback) {
38+
PostgresDB.prototype.commit = async function(collection, id, op, snapshot, options, callback) {
39+
try {
3040
/*
3141
* op: CreateOp {
3242
* src: '24545654654646',
@@ -37,34 +47,28 @@ PostgresDB.prototype.commit = function(collection, id, op, snapshot, options, ca
3747
* }
3848
* snapshot: PostgresSnapshot
3949
*/
40-
this.pool.connect((err, client, done) => {
41-
if (err) {
42-
done(client);
43-
callback(err);
44-
return;
45-
}
4650
/*
47-
* This query uses common table expression to upsert the snapshot table
51+
* This query uses common table expression to upsert the snapshot table
4852
* (iff the new version is exactly 1 more than the latest table or if
4953
* the document id does not exists)
5054
*
51-
* It will then insert into the ops table if it is exactly 1 more than the
55+
* It will then insert into the ops table if it is exactly 1 more than the
5256
* latest table or it the first operation and iff the previous insert into
5357
* the snapshot table is successful.
5458
*
5559
* This result of this query the version of the newly inserted operation
5660
* If either the ops or the snapshot insert fails then 0 rows are returned
5761
*
58-
* If 0 zeros are return then the callback must return false
62+
* If 0 zeros are return then the callback must return false
5963
*
6064
* Casting is required as postgres thinks that collection and doc_id are
61-
* not varchar
62-
*/
65+
* not varchar
66+
*/
6367
const query = {
6468
name: 'sdb-commit-op-and-snap',
6569
text: `WITH snapshot_id AS (
66-
INSERT INTO snapshots (collection, doc_id, doc_type, version, data)
67-
SELECT $1::varchar collection, $2::varchar doc_id, $4 doc_type, $3 v, $5 d
70+
INSERT INTO snapshots (collection, doc_id, version, doc_type, data, metadata)
71+
SELECT $1::varchar collection, $2::varchar doc_id, $3 v, $4 doc_type, $5 d, $6 m
6872
WHERE $3 = (
6973
SELECT version+1 v
7074
FROM snapshots
@@ -76,11 +80,11 @@ PostgresDB.prototype.commit = function(collection, id, op, snapshot, options, ca
7680
WHERE collection = $1 AND doc_id = $2
7781
FOR UPDATE
7882
)
79-
ON CONFLICT (collection, doc_id) DO UPDATE SET version = $3, data = $5, doc_type = $4
83+
ON CONFLICT (collection, doc_id) DO UPDATE SET version = $3, data = $5, doc_type = $4, metadata = $5
8084
RETURNING version
8185
)
8286
INSERT INTO ops (collection, doc_id, version, operation)
83-
SELECT $1::varchar collection, $2::varchar doc_id, $3 v, $6 operation
87+
SELECT $1::varchar collection, $2::varchar doc_id, $3 v, $7 operation
8488
WHERE (
8589
$3 = (
8690
SELECT max(version)+1
@@ -93,66 +97,47 @@ WHERE (
9397
)
9498
) AND EXISTS (SELECT 1 FROM snapshot_id)
9599
RETURNING version`,
96-
values: [collection,id,snapshot.v, snapshot.type, snapshot.data,op]
100+
values: [collection, id, snapshot.v, snapshot.type, JSON.stringify(snapshot.data), JSON.stringify(snapshot.m), JSON.stringify(op)]
97101
}
98-
client.query(query, (err, res) => {
99-
if (err) {
100-
callback(err)
101-
} else if(res.rows.length === 0) {
102-
done(client);
103-
callback(null,false)
104-
}
105-
else {
106-
done(client);
107-
callback(null,true)
108-
}
109-
})
110-
111-
})
102+
const result = await this._pool.query(query);
103+
const success = result.rowCount > 0;
104+
callback(null, success);
105+
} catch (error) {
106+
// Return non-success instead of duplicate key error, since this is
107+
// expected to occur during simultaneous creates on the same id
108+
if (error.code === PG_UNIQUE_VIOLATION) callback(null, false);
109+
else callback(error);
110+
}
112111
};
113112

114113
// Get the named document from the database. The callback is called with (err,
115114
// snapshot). A snapshot with a version of zero is returned if the docuemnt
116115
// has never been created in the database.
117-
PostgresDB.prototype.getSnapshot = function(collection, id, fields, options, callback) {
118-
this.pool.connect(function(err, client, done) {
119-
if (err) {
120-
done(client);
121-
callback(err);
122-
return;
123-
}
124-
client.query(
125-
'SELECT version, data, doc_type FROM snapshots WHERE collection = $1 AND doc_id = $2 LIMIT 1',
116+
PostgresDB.prototype.getSnapshot = async function(collection, id, fields, options, callback) {
117+
fields ||= {};
118+
options ||= {};
119+
const wantsMetadata = fields.$submit || options.metadata;
120+
try {
121+
const result = await this._pool.query(
122+
'SELECT version, data, doc_type, metadata FROM snapshots WHERE collection = $1 AND doc_id = $2 LIMIT 1',
126123
[collection, id],
127-
function(err, res) {
128-
done();
129-
if (err) {
130-
callback(err);
131-
return;
132-
}
133-
if (res.rows.length) {
134-
var row = res.rows[0]
135-
var snapshot = new PostgresSnapshot(
136-
id,
137-
row.version,
138-
row.doc_type,
139-
row.data,
140-
undefined // TODO: metadata
141-
)
142-
callback(null, snapshot);
143-
} else {
144-
var snapshot = new PostgresSnapshot(
145-
id,
146-
0,
147-
null,
148-
undefined,
149-
undefined
150-
)
151-
callback(null, snapshot);
152-
}
153-
}
154-
)
155-
})
124+
);
125+
126+
var row = result.rows[0]
127+
const snapshot = {
128+
id,
129+
v: row?.version || 0,
130+
type: row?.doc_type || null,
131+
data: row?.data || undefined,
132+
m: wantsMetadata ?
133+
// Postgres returns null but ShareDB expects undefined
134+
(row?.metadata || undefined) :
135+
null,
136+
};
137+
callback(null, snapshot);
138+
} catch (error) {
139+
callback(error);
140+
}
156141
};
157142

158143
// Get operations between [from, to) noninclusively. (Ie, the range should
@@ -164,37 +149,21 @@ PostgresDB.prototype.getSnapshot = function(collection, id, fields, options, cal
164149
// The version will be inferred from the parameters if it is missing.
165150
//
166151
// Callback should be called as callback(error, [list of ops]);
167-
PostgresDB.prototype.getOps = function(collection, id, from, to, options, callback) {
168-
this.pool.connect(function(err, client, done) {
169-
if (err) {
170-
done(client);
171-
callback(err);
172-
return;
173-
}
174-
152+
PostgresDB.prototype.getOps = async function(collection, id, from, to, options, callback) {
153+
from ||= 0;
154+
options ||= {};
155+
const wantsMetadata = options.metadata;
156+
try {
175157
var cmd = 'SELECT version, operation FROM ops WHERE collection = $1 AND doc_id = $2 AND version > $3 ';
176158
var params = [collection, id, from];
177159
if(to || to == 0) { cmd += ' AND version <= $4'; params.push(to)}
178160
cmd += ' order by version';
179-
client.query( cmd, params,
180-
function(err, res) {
181-
done();
182-
if (err) {
183-
callback(err);
184-
return;
185-
}
186-
callback(null, res.rows.map(function(row) {
187-
return row.operation;
188-
}));
189-
}
190-
)
191-
})
161+
const result = await this._pool.query(cmd, params);
162+
callback(null, result.rows.map(({operation}) => {
163+
if (!wantsMetadata) delete operation.m;
164+
return operation;
165+
}));
166+
} catch (error) {
167+
callback(error);
168+
}
192169
};
193-
194-
function PostgresSnapshot(id, version, type, data, meta) {
195-
this.id = id;
196-
this.v = version;
197-
this.type = type;
198-
this.data = data;
199-
this.m = meta;
200-
}

‎index.spec.js

+37
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,37 @@
1+
const PostgresDB = require('.');
2+
const {Pool} = require('pg');
3+
const fs = require('node:fs');
4+
5+
const DB_NAME = 'sharedbtest';
6+
7+
function create(callback) {
8+
var db = new PostgresDB({database: DB_NAME});
9+
callback(null, db);
10+
};
11+
12+
describe('PostgresDB', function() {
13+
let pool;
14+
let client;
15+
16+
beforeEach(async () => {
17+
pool = new Pool({database: 'postgres'});
18+
client = await pool.connect();
19+
await client.query(`DROP DATABASE IF EXISTS ${DB_NAME}`);
20+
await client.query(`CREATE DATABASE ${DB_NAME}`);
21+
22+
const testPool = new Pool({database: DB_NAME});
23+
const testClient = await testPool.connect();
24+
const structure = fs.readFileSync('./structure.sql', 'utf8');
25+
await testClient.query(structure);
26+
await testClient.release(true);
27+
await testPool.end();
28+
});
29+
30+
afterEach(async function() {
31+
await client.query(`DROP DATABASE IF EXISTS ${DB_NAME}`);
32+
await client.release(true);
33+
await pool.end();
34+
});
35+
36+
require('sharedb/test/db')({create: create});
37+
});

‎package.json

+9-3
Original file line numberDiff line numberDiff line change
@@ -4,7 +4,7 @@
44
"description": "PostgreSQL adapter for ShareDB. forked from share/sharedb-postgres",
55
"main": "index.js",
66
"scripts": {
7-
"test": "echo \"Error: no test specified\" && exit 1"
7+
"test": "mocha"
88
},
99
"author": "Jeremy Apthorp <nornagon@nornagon.net>",
1010
"license": "MIT",
@@ -19,8 +19,14 @@
1919
"url": "https://github.com/share/sharedb-postgres"
2020
},
2121
"dependencies": {
22-
"pg": "^8.5.1",
23-
"pg-pool": "^3.2.1",
22+
"pg": "^8.12.0",
2423
"sharedb": "^1.6.0 || ^2.0.0 || ^3.0.0 || ^4.0.0 || ^5.0.0"
24+
},
25+
"devDependencies": {
26+
"chai": "^4.4.1",
27+
"mocha": "^10.4.0",
28+
"ot-json1": "^1.0.2",
29+
"rich-text": "^4.1.0",
30+
"sinon": "^18.0.0"
2531
}
2632
}

‎structure.sql

+14
Original file line numberDiff line numberDiff line change
@@ -26,3 +26,17 @@ ALTER TABLE snapshots
2626
ALTER COLUMN data
2727
SET DATA TYPE jsonb
2828
USING data::jsonb;
29+
30+
31+
-- v5.0.0 --
32+
33+
ALTER TABLE snapshots
34+
ALTER column doc_type
35+
DROP NOT NULL;
36+
37+
ALTER TABLE snapshots
38+
ALTER column data
39+
DROP NOT NULL;
40+
41+
ALTER TABLE snapshots
42+
ADD metadata jsonb;

‎test/setup.js

+10
Original file line numberDiff line numberDiff line change
@@ -0,0 +1,10 @@
1+
var logger = require('sharedb/lib/logger');
2+
3+
if (process.env.LOGGING !== 'true') {
4+
// Silence the logger for tests by setting all its methods to no-ops
5+
logger.setMethods({
6+
info: function() {},
7+
warn: function() {},
8+
error: function() {}
9+
});
10+
}

0 commit comments

Comments
 (0)
Please sign in to comment.