PostgreSQL Protocol
This document provides examples for using the ChronDB PostgreSQL protocol interface with psql and JavaScript clients.
PostgreSQL Protocol Overview
ChronDB implements a subset of the PostgreSQL protocol, allowing you to connect using standard PostgreSQL clients and drivers. This makes it easy to integrate with existing applications that already use PostgreSQL or to leverage the SQL query capability.
The PostgreSQL protocol server can be configured in the config.edn
file:
:servers {
:postgresql {
:enabled true
:host "0.0.0.0"
:port 5432
:username "chrondb"
:password "chrondb"
}
}
Data Model Mapping
ChronDB maps its document structure to SQL tables based on document keys:
Document keys with the format
collection:id
are mapped to tablesThe part before the colon becomes the table name
The part after the colon becomes the row's ID
Document fields become columns in the table
For example:
A document with key
user:1
becomes a row in theuser
table withid='1'
A document with key
product:xyz
becomes a row in theproduct
table withid='xyz'
Supported SQL Features
ChronDB supports the following SQL operations:
SELECT
- Query documentsINSERT
- Create documentsUPDATE
- Update documentsDELETE
- Delete documentsCREATE TABLE
- Create collection (optional, schemas are inferred)DROP TABLE
- Delete collection
Special Functions
ChronDB provides special SQL functions to access version control features:
chrondb_history(table_name, id)
- Get document historychrondb_at(table_name, id, timestamp)
- Get document at a point in timechrondb_diff(table_name, id, t1, t2)
- Compare document versionschrondb_branch_list()
- List brancheschrondb_branch_create(name)
- Create a new branchchrondb_branch_checkout(name)
- Switch to a branchchrondb_branch_merge(source, target)
- Merge branches
Examples with psql
Connecting to ChronDB
# Connect to ChronDB using psql
psql -h localhost -p 5432 -U chrondb -d chrondb
Document Operations
Creating Documents (INSERT)
-- Create a user
INSERT INTO user (id, name, email, age, roles)
VALUES ('1', 'John Doe', 'john@example.com', 30, '["admin", "user"]');
-- Create another user
INSERT INTO user (id, name, email, age, roles)
VALUES ('2', 'Jane Smith', 'jane@example.com', 28, '["editor"]');
-- Create a product
INSERT INTO product (id, name, price, category, in_stock)
VALUES ('laptop1', 'Premium Laptop', 1299.99, 'Electronics', true);
Querying Documents (SELECT)
-- Get all users
SELECT * FROM user;
-- Get a specific user
SELECT * FROM user WHERE id = '1';
-- Get users with specific attributes
SELECT * FROM user WHERE age > 25;
-- Get specific fields
SELECT name, email FROM user;
-- Count documents
SELECT COUNT(*) FROM user;
Updating Documents (UPDATE)
-- Update a user
UPDATE user
SET email = 'john.doe@example.com', roles = '["admin", "user", "reviewer"]'
WHERE id = '1';
-- Update with JSON operations
UPDATE product
SET price = 1199.99, tags = '["sale", "featured"]'
WHERE id = 'laptop1';
Deleting Documents (DELETE)
-- Delete a specific user
DELETE FROM user WHERE id = '2';
-- Delete all products in a category
DELETE FROM product WHERE category = 'Discontinued';
Version Control Operations
-- Get document history
SELECT * FROM chrondb_history('user', '1');
-- Get document at a specific point in time
SELECT * FROM chrondb_at('user', '1', '2023-10-10T09:15:00Z');
-- Compare document versions
SELECT * FROM chrondb_diff('user', '1',
'2023-10-10T09:15:00Z',
'2023-10-15T14:30:00Z');
Branch Operations
-- List all branches
SELECT * FROM chrondb_branch_list();
-- Create a new branch
SELECT chrondb_branch_create('feature-reporting');
-- Switch to a branch
SELECT chrondb_branch_checkout('feature-reporting');
-- Merge branches
SELECT chrondb_branch_merge('feature-reporting', 'main');
Advanced Queries
-- Join documents from different collections
SELECT u.name, u.email, o.total, o.status
FROM user u
JOIN order o ON o.user_id = u.id
WHERE o.status = 'pending';
-- Aggregation
SELECT category, COUNT(*) as product_count, AVG(price) as avg_price
FROM product
GROUP BY category
ORDER BY product_count DESC;
-- Filtering with JSON
SELECT * FROM user
WHERE roles::jsonb ? 'admin';
Examples with JavaScript (node-postgres)
The following examples use the node-postgres package, a popular PostgreSQL client for Node.js.
Setting Up
// Install the pg package:
// npm install pg
const { Pool } = require('pg');
// Create a connection pool
const pool = new Pool({
host: 'localhost',
port: 5432,
user: 'chrondb',
password: 'chrondb',
database: 'chrondb'
});
// Helper function for queries
async function query(text, params) {
const client = await pool.connect();
try {
const result = await client.query(text, params);
return result.rows;
} finally {
client.release();
}
}
Document Operations
// Create a document (INSERT)
async function createUser(id, userData) {
const { name, email, age, roles } = userData;
const text = `
INSERT INTO user (id, name, email, age, roles)
VALUES ($1, $2, $3, $4, $5)
RETURNING *
`;
// Convert JavaScript array to JSON string
const rolesJson = JSON.stringify(roles);
const values = [id, name, email, age, rolesJson];
const rows = await query(text, values);
return rows[0];
}
// Get a document (SELECT)
async function getUser(id) {
const text = 'SELECT * FROM user WHERE id = $1';
const values = [id];
const rows = await query(text, values);
if (rows.length === 0) {
return null;
}
// Parse JSON string fields back to JavaScript objects
const user = rows[0];
if (user.roles) {
user.roles = JSON.parse(user.roles);
}
return user;
}
// Update a document (UPDATE)
async function updateUser(id, updates) {
// First get the current document
const user = await getUser(id);
if (!user) {
throw new Error(`User ${id} not found`);
}
// Prepare the update fields
const fields = [];
const values = [];
let index = 1;
for (const [key, value] of Object.entries(updates)) {
let formattedValue = value;
// Convert arrays or objects to JSON strings
if (typeof value === 'object' && value !== null) {
formattedValue = JSON.stringify(value);
}
fields.push(`${key} = $${index}`);
values.push(formattedValue);
index++;
}
// Add the timestamp and id
fields.push(`updated_at = $${index}`);
values.push(new Date().toISOString());
index++;
values.push(id);
const text = `
UPDATE user
SET ${fields.join(', ')}
WHERE id = $${index}
RETURNING *
`;
const rows = await query(text, values);
// Parse JSON fields
const updatedUser = rows[0];
if (updatedUser.roles) {
updatedUser.roles = JSON.parse(updatedUser.roles);
}
return updatedUser;
}
// Delete a document (DELETE)
async function deleteUser(id) {
const text = 'DELETE FROM user WHERE id = $1 RETURNING id';
const values = [id];
const rows = await query(text, values);
return rows.length > 0;
}
// Search documents
async function searchUsers(conditions = {}, options = {}) {
const { limit = 10, offset = 0, orderBy = 'id', order = 'ASC' } = options;
const clauses = [];
const values = [];
let index = 1;
// Build WHERE clauses
for (const [key, value] of Object.entries(conditions)) {
clauses.push(`${key} = $${index}`);
values.push(value);
index++;
}
const whereClause = clauses.length > 0 ? `WHERE ${clauses.join(' AND ')}` : '';
const text = `
SELECT * FROM user
${whereClause}
ORDER BY ${orderBy} ${order}
LIMIT $${index} OFFSET $${index + 1}
`;
values.push(limit, offset);
const rows = await query(text, values);
// Parse JSON fields
return rows.map(user => {
if (user.roles) {
user.roles = JSON.parse(user.roles);
}
return user;
});
}
Version Control Operations
// Get document history
async function getUserHistory(id) {
const text = 'SELECT * FROM chrondb_history($1, $2)';
const values = ['user', id];
const rows = await query(text, values);
// Parse JSON data in each version
return rows.map(version => {
return {
timestamp: version.timestamp,
data: JSON.parse(version.data)
};
});
}
// Get document at a point in time
async function getUserAtTime(id, timestamp) {
const text = 'SELECT * FROM chrondb_at($1, $2, $3)';
const values = ['user', id, timestamp];
const rows = await query(text, values);
if (rows.length === 0) {
return null;
}
// Parse JSON fields
const user = rows[0];
if (user.roles) {
user.roles = JSON.parse(user.roles);
}
return user;
}
// Compare document versions
async function compareUserVersions(id, timestamp1, timestamp2) {
const text = 'SELECT * FROM chrondb_diff($1, $2, $3, $4)';
const values = ['user', id, timestamp1, timestamp2];
const rows = await query(text, values);
if (rows.length === 0) {
return null;
}
const diff = rows[0];
// Parse JSON diff data
return {
added: JSON.parse(diff.added || '{}'),
removed: JSON.parse(diff.removed || '{}'),
changed: JSON.parse(diff.changed || '{}')
};
}
Branch Operations
// List branches
async function listBranches() {
const text = 'SELECT * FROM chrondb_branch_list()';
const rows = await query(text);
return rows.map(row => row.name);
}
// Create a branch
async function createBranch(name) {
const text = 'SELECT chrondb_branch_create($1)';
const values = [name];
await query(text, values);
return true;
}
// Switch to a branch
async function switchBranch(name) {
const text = 'SELECT chrondb_branch_checkout($1)';
const values = [name];
await query(text, values);
return true;
}
// Merge branches
async function mergeBranches(source, target) {
const text = 'SELECT chrondb_branch_merge($1, $2)';
const values = [source, target];
await query(text, values);
return true;
}
Complete Example: Customer Order System
const { Pool } = require('pg');
const pool = new Pool({
host: 'localhost',
port: 5432,
user: 'chrondb',
password: 'chrondb',
database: 'chrondb'
});
// OrderSystem class
class OrderSystem {
constructor() {
this.pool = pool;
}
async query(text, params) {
const client = await this.pool.connect();
try {
const result = await client.query(text, params);
return result.rows;
} finally {
client.release();
}
}
// Initialize the database with tables
async initialize() {
const tables = [
`CREATE TABLE IF NOT EXISTS customer (
id TEXT PRIMARY KEY,
name TEXT,
email TEXT,
address TEXT,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)`,
`CREATE TABLE IF NOT EXISTS product (
id TEXT PRIMARY KEY,
name TEXT,
price NUMERIC,
description TEXT,
stock INTEGER,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)`,
`CREATE TABLE IF NOT EXISTS order_item (
id TEXT PRIMARY KEY,
order_id TEXT,
product_id TEXT,
quantity INTEGER,
price NUMERIC,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)`,
`CREATE TABLE IF NOT EXISTS customer_order (
id TEXT PRIMARY KEY,
customer_id TEXT,
total NUMERIC,
status TEXT,
created_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP,
updated_at TIMESTAMP DEFAULT CURRENT_TIMESTAMP
)`
];
for (const table of tables) {
await this.query(table);
}
console.log('Database initialized');
}
// Customer methods
async createCustomer(id, data) {
const { name, email, address } = data;
const text = `
INSERT INTO customer (id, name, email, address)
VALUES ($1, $2, $3, $4)
RETURNING *
`;
const values = [id, name, email, address];
const rows = await this.query(text, values);
return rows[0];
}
async getCustomer(id) {
const text = 'SELECT * FROM customer WHERE id = $1';
const rows = await this.query(text, [id]);
return rows[0] || null;
}
async updateCustomer(id, updates) {
const fields = [];
const values = [];
let index = 1;
for (const [key, value] of Object.entries(updates)) {
fields.push(`${key} = $${index}`);
values.push(value);
index++;
}
values.push(id);
const text = `
UPDATE customer
SET ${fields.join(', ')}
WHERE id = $${index}
RETURNING *
`;
const rows = await this.query(text, values);
return rows[0] || null;
}
// Product methods
async createProduct(id, data) {
const { name, price, description, stock } = data;
const text = `
INSERT INTO product (id, name, price, description, stock)
VALUES ($1, $2, $3, $4, $5)
RETURNING *
`;
const values = [id, name, price, description, stock];
const rows = await this.query(text, values);
return rows[0];
}
async getProduct(id) {
const text = 'SELECT * FROM product WHERE id = $1';
const rows = await this.query(text, [id]);
return rows[0] || null;
}
async updateProductStock(id, adjustment) {
const text = `
UPDATE product
SET stock = stock + $1
WHERE id = $2
RETURNING *
`;
const values = [adjustment, id];
const rows = await this.query(text, values);
return rows[0] || null;
}
// Order methods
async createOrder(data) {
const { id, customer_id, items } = data;
// Start transaction
const client = await this.pool.connect();
try {
await client.query('BEGIN');
// Calculate total
let total = 0;
for (const item of items) {
const product = await this.getProduct(item.product_id);
if (!product) {
throw new Error(`Product ${item.product_id} not found`);
}
if (product.stock < item.quantity) {
throw new Error(`Insufficient stock for product ${item.product_id}`);
}
total += product.price * item.quantity;
// Update stock
await client.query(
'UPDATE product SET stock = stock - $1 WHERE id = $2',
[item.quantity, item.product_id]
);
// Create order item
const orderItemId = `item:${id}-${item.product_id}`;
await client.query(
`INSERT INTO order_item (id, order_id, product_id, quantity, price)
VALUES ($1, $2, $3, $4, $5)`,
[orderItemId, id, item.product_id, item.quantity, product.price]
);
}
// Create the order
const orderResult = await client.query(
`INSERT INTO customer_order (id, customer_id, total, status)
VALUES ($1, $2, $3, $4)
RETURNING *`,
[id, customer_id, total, 'pending']
);
await client.query('COMMIT');
return orderResult.rows[0];
} catch (e) {
await client.query('ROLLBACK');
throw e;
} finally {
client.release();
}
}
async getOrder(id) {
const orderText = 'SELECT * FROM customer_order WHERE id = $1';
const orderRows = await this.query(orderText, [id]);
if (orderRows.length === 0) {
return null;
}
const order = orderRows[0];
// Get order items
const itemsText = 'SELECT * FROM order_item WHERE order_id = $1';
const items = await this.query(itemsText, [id]);
return {
...order,
items
};
}
async updateOrderStatus(id, status) {
const text = `
UPDATE customer_order
SET status = $1, updated_at = CURRENT_TIMESTAMP
WHERE id = $2
RETURNING *
`;
const values = [status, id];
const rows = await this.query(text, values);
return rows[0] || null;
}
// Reporting methods
async getCustomerOrders(customerId) {
const text = `
SELECT co.*, c.name as customer_name, c.email
FROM customer_order co
JOIN customer c ON co.customer_id = c.id
WHERE co.customer_id = $1
ORDER BY co.created_at DESC
`;
return await this.query(text, [customerId]);
}
async getOrderHistory(orderId) {
const text = 'SELECT * FROM chrondb_history($1, $2)';
const rows = await this.query(text, ['customer_order', orderId]);
return rows.map(version => ({
timestamp: version.timestamp,
data: JSON.parse(version.data)
}));
}
async getOrderStatistics() {
const text = `
SELECT
status,
COUNT(*) as count,
SUM(total) as total_value,
MIN(total) as min_value,
MAX(total) as max_value,
AVG(total) as avg_value
FROM customer_order
GROUP BY status
`;
return await this.query(text);
}
}
// Usage example
async function runOrderSystemExample() {
const orderSystem = new OrderSystem();
try {
// Initialize database
await orderSystem.initialize();
// Create customers
await orderSystem.createCustomer('cust1', {
name: 'Alice Johnson',
email: 'alice@example.com',
address: '123 Main St, Anytown, CA'
});
await orderSystem.createCustomer('cust2', {
name: 'Bob Smith',
email: 'bob@example.com',
address: '456 Oak St, Somewhere, NY'
});
// Create products
await orderSystem.createProduct('prod1', {
name: 'Mechanical Keyboard',
price: 129.99,
description: 'Mechanical keyboard with RGB lighting',
stock: 50
});
await orderSystem.createProduct('prod2', {
name: 'Wireless Mouse',
price: 49.99,
description: 'Ergonomic wireless mouse',
stock: 100
});
await orderSystem.createProduct('prod3', {
name: 'Monitor Stand',
price: 79.99,
description: 'Adjustable monitor stand',
stock: 30
});
// Create an order
const order = await orderSystem.createOrder({
id: 'order1',
customer_id: 'cust1',
items: [
{ product_id: 'prod1', quantity: 1 },
{ product_id: 'prod2', quantity: 2 }
]
});
console.log('Created order:', order);
// Get order details
const orderDetails = await orderSystem.getOrder('order1');
console.log('Order details:', orderDetails);
// Update order status
await orderSystem.updateOrderStatus('order1', 'shipped');
console.log('Updated order status');
// Create another order
await orderSystem.createOrder({
id: 'order2',
customer_id: 'cust2',
items: [
{ product_id: 'prod3', quantity: 1 },
{ product_id: 'prod1', quantity: 1 }
]
});
// Get order history
const history = await orderSystem.getOrderHistory('order1');
console.log('Order history:', history);
// Get customer orders
const customerOrders = await orderSystem.getCustomerOrders('cust1');
console.log('Customer orders:', customerOrders);
// Get order statistics
const statistics = await orderSystem.getOrderStatistics();
console.log('Order statistics:', statistics);
} catch (err) {
console.error('Error:', err);
} finally {
await pool.end();
}
}
// Run the example
runOrderSystemExample();
Last updated
Was this helpful?