REST API
This document provides examples for using the ChronDB REST API with curl and JavaScript.
REST API Overview
ChronDB provides a RESTful API that allows you to interact with the database over HTTP. This makes it easy to integrate with any programming language or environment that can make HTTP requests.
The REST API server can be configured in the config.edn
file:
:servers {
:rest {
:enabled true
:host "0.0.0.0"
:port 3000
}
}
Examples with curl
Document Operations
Create or Update a Document
# Create a new document
curl -X POST http://localhost:3000/api/v1/documents/user:1 \
-H "Content-Type: application/json" \
-d '{
"name": "John Doe",
"email": "john@example.com",
"age": 30,
"roles": ["admin", "user"]
}'
# Response
# {"status":"success","key":"user:1"}
Get a Document
# Retrieve a document by key
curl http://localhost:3000/api/v1/documents/user:1
# Response
# {"name":"John Doe","email":"john@example.com","age":30,"roles":["admin","user"]}
Delete a Document
# Delete a document by key
curl -X DELETE http://localhost:3000/api/v1/documents/user:1
# Response
# {"status":"success","key":"user:1"}
Search Operations
# Search for documents by field
curl http://localhost:3000/api/v1/search?q=name:John
# Search with multiple criteria
curl http://localhost:3000/api/v1/search?q=name:John%20AND%20age:%5B25%20TO%2035%5D
# Search with pagination
curl http://localhost:3000/api/v1/search?q=roles:admin&limit=5&offset=0
Version Control Operations
Get Document History
# Get the history of a document
curl http://localhost:3000/api/v1/documents/user:1/history
# Response
# [
# {
# "timestamp": "2023-10-15T14:30:45Z",
# "data": {"name":"John Doe","email":"john@example.com","age":30,"roles":["admin","user"]}
# },
# {
# "timestamp": "2023-10-10T09:15:22Z",
# "data": {"name":"John Doe","email":"john@example.com","roles":["user"]}
# }
# ]
Get Document at a Point in Time
# Get a document as it was at a specific time
curl http://localhost:3000/api/v1/documents/user:1/at/2023-10-10T09:15:22Z
# Response
# {"name":"John Doe","email":"john@example.com","roles":["user"]}
Compare Document Versions
# Compare two versions of a document
curl http://localhost:3000/api/v1/documents/user:1/diff?t1=2023-10-10T09:15:22Z&t2=2023-10-15T14:30:45Z
# Response
# {
# "added": {"age":30,"roles":["admin"]},
# "removed": {"roles":["user"]},
# "changed": {}
# }
Branch Operations
List Branches
# List all branches
curl http://localhost:3000/api/v1/branches
# Response
# ["main","dev","test"]
Create a Branch
# Create a new branch
curl -X POST http://localhost:3000/api/v1/branches/feature-login
# Response
# {"status":"success","branch":"feature-login"}
Switch Branch
# Switch to a different branch
curl -X PUT http://localhost:3000/api/v1/branches/feature-login/checkout
# Response
# {"status":"success","branch":"feature-login"}
Merge Branches
# Merge one branch into another
curl -X POST http://localhost:3000/api/v1/branches/feature-login/merge/main
# Response
# {"status":"success","source":"feature-login","target":"main"}
Examples with JavaScript
Setting Up
// Utility function for API requests
async function chronDBRequest(endpoint, method = 'GET', body = null) {
const options = {
method,
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
};
if (body) {
options.body = JSON.stringify(body);
}
const response = await fetch(`http://localhost:3000/api/v1/${endpoint}`, options);
return response.json();
}
Document Operations
// Create or update a document
async function saveDocument(key, data) {
return chronDBRequest(`documents/${key}`, 'POST', data);
}
// Get a document
async function getDocument(key) {
return chronDBRequest(`documents/${key}`);
}
// Delete a document
async function deleteDocument(key) {
return chronDBRequest(`documents/${key}`, 'DELETE');
}
// Usage examples
async function documentExamples() {
// Create a user
const user = {
name: "Jane Smith",
email: "jane@example.com",
age: 28,
roles: ["editor"]
};
await saveDocument('user:2', user);
// Retrieve the user
const savedUser = await getDocument('user:2');
console.log('Retrieved user:', savedUser);
// Update the user
savedUser.roles.push('reviewer');
await saveDocument('user:2', savedUser);
// Delete the user
await deleteDocument('user:2');
}
Search Operations
// Search for documents
async function searchDocuments(query, limit = 10, offset = 0) {
const params = new URLSearchParams({
q: query,
limit,
offset
});
return chronDBRequest(`search?${params.toString()}`);
}
// Usage examples
async function searchExamples() {
// Simple search
const admins = await searchDocuments('roles:admin');
console.log('Admins:', admins);
// Complex search
const activeUsers = await searchDocuments('lastLogin:[NOW-7DAYS TO NOW]');
console.log('Recently active users:', activeUsers);
// Pagination
const page1 = await searchDocuments('type:article', 5, 0);
const page2 = await searchDocuments('type:article', 5, 5);
console.log('Articles page 1:', page1);
console.log('Articles page 2:', page2);
}
Version Control Operations
// Get document history
async function getDocumentHistory(key) {
return chronDBRequest(`documents/${key}/history`);
}
// Get document at a point in time
async function getDocumentAt(key, timestamp) {
return chronDBRequest(`documents/${key}/at/${encodeURIComponent(timestamp)}`);
}
// Compare document versions
async function compareDocuments(key, timestamp1, timestamp2) {
const params = new URLSearchParams({
t1: timestamp1,
t2: timestamp2
});
return chronDBRequest(`documents/${key}/diff?${params.toString()}`);
}
// Usage examples
async function versionControlExamples() {
// Get document history
const history = await getDocumentHistory('user:1');
console.log('User history:', history);
// Get version from a week ago
const oneWeekAgo = new Date();
oneWeekAgo.setDate(oneWeekAgo.getDate() - 7);
const oldVersion = await getDocumentAt('user:1', oneWeekAgo.toISOString());
console.log('User from one week ago:', oldVersion);
// Compare current version with week-old version
const diff = await compareDocuments('user:1', oneWeekAgo.toISOString(), new Date().toISOString());
console.log('Changes in the last week:', diff);
}
Branch Operations
// List all branches
async function listBranches() {
return chronDBRequest('branches');
}
// Create a new branch
async function createBranch(name) {
return chronDBRequest(`branches/${name}`, 'POST');
}
// Switch to a branch
async function switchBranch(name) {
return chronDBRequest(`branches/${name}/checkout`, 'PUT');
}
// Merge branches
async function mergeBranches(source, target) {
return chronDBRequest(`branches/${source}/merge/${target}`, 'POST');
}
// Usage examples
async function branchExamples() {
// List all branches
const branches = await listBranches();
console.log('Available branches:', branches);
// Create a feature branch
await createBranch('feature-newui');
// Switch to the feature branch
await switchBranch('feature-newui');
// Make changes in the feature branch
await saveDocument('ui:config', { theme: 'dark', layout: 'responsive' });
// Merge back to main
await mergeBranches('feature-newui', 'main');
}
Complete Example: User Management System
// User management module
const UserAPI = {
baseUrl: 'http://localhost:3000/api/v1',
async request(endpoint, method = 'GET', body = null) {
const options = {
method,
headers: {
'Content-Type': 'application/json',
'Accept': 'application/json'
}
};
if (body) {
options.body = JSON.stringify(body);
}
const response = await fetch(`${this.baseUrl}/${endpoint}`, options);
if (!response.ok) {
throw new Error(`API request failed: ${response.statusText}`);
}
return response.json();
},
// User operations
async createUser(userData) {
const userId = `user:${Date.now()}`;
await this.request(`documents/${userId}`, 'POST', {
...userData,
createdAt: new Date().toISOString(),
updatedAt: new Date().toISOString()
});
return userId;
},
async updateUser(userId, updates) {
const user = await this.request(`documents/${userId}`);
const updatedUser = {
...user,
...updates,
updatedAt: new Date().toISOString()
};
return this.request(`documents/${userId}`, 'POST', updatedUser);
},
async getUser(userId) {
return this.request(`documents/${userId}`);
},
async deleteUser(userId) {
return this.request(`documents/${userId}`, 'DELETE');
},
async searchUsers(query) {
return this.request(`search?q=${encodeURIComponent(query)}`);
},
async getUserHistory(userId) {
return this.request(`documents/${userId}/history`);
}
};
// Usage
async function userManagementExample() {
try {
// Create a user
const userId = await UserAPI.createUser({
name: "Alice Johnson",
email: "alice@example.com",
role: "developer",
skills: ["JavaScript", "React", "Node.js"]
});
console.log(`Created user with ID: ${userId}`);
// Get the user
const user = await UserAPI.getUser(userId);
console.log("Retrieved user:", user);
// Update the user
await UserAPI.updateUser(userId, {
role: "senior developer",
skills: [...user.skills, "TypeScript"]
});
console.log("Updated user skills and role");
// Get user history
const history = await UserAPI.getUserHistory(userId);
console.log("User modification history:", history);
// Search for developers
const developers = await UserAPI.searchUsers("role:*developer*");
console.log("Found developers:", developers);
} catch (error) {
console.error("Error in user management:", error);
}
}
// Run the example
userManagementExample();
Last updated
Was this helpful?