Troubleshooting
This guide provides solutions for common issues you may encounter when using QueryLeaf.
Common Error Messages
SQL Parsing Errors
Error: SQL parsing error: syntax error at position X
Possible causes: - Syntax error in SQL query - Using unsupported SQL syntax - Missing commas, parentheses, or quotes
Solutions: 1. Check your SQL syntax for typos or errors 2. Make sure you're using SQL syntax compatible with PostgreSQL 3. Try simplifying your query to isolate the issue 4. Use the DummyQueryLeaf to debug the query parsing
Example:
// Import from @queryleaf/lib instead of queryleaf
import { DummyQueryLeaf } from '@queryleaf/lib';
// Use DummyQueryLeaf to debug
const dummy = new DummyQueryLeaf('debug_db');
try {
await dummy.execute('SELECT * FORM users'); // Error: typo in FROM
} catch (error) {
console.error('Parsing error:', error.message);
}
MongoDB Connection Issues
Error: MongoServerSelectionError: connection timed out
Possible causes: - MongoDB server is not running - Network connectivity issues - Incorrect connection string - Firewall blocking connections
Solutions: 1. Verify MongoDB server is running 2. Check network connectivity 3. Confirm your connection string is correct 4. Ensure firewall allows MongoDB connections 5. Add timeouts and retry logic
Example:
// Add retry logic for connection issues
async function executeWithRetry(sqlQuery, maxRetries = 3) {
let retries = 0;
while (retries < maxRetries) {
try {
const client = new MongoClient('mongodb://localhost:27017');
await client.connect();
const queryLeaf = new QueryLeaf(client, 'mydb');
return await queryLeaf.execute(sqlQuery);
} catch (error) {
retries++;
console.log(`Connection attempt ${retries} failed:`, error.message);
if (retries >= maxRetries) {
throw new Error(`Failed to connect after ${maxRetries} attempts`);
}
// Wait before retrying
await new Promise(r => setTimeout(r, 1000 * retries));
}
}
}
Nested Field Access Errors
Error: Cannot read property of undefined
Possible causes: - Accessing a nested field that doesn't exist - Incorrect dot notation in SQL query - Missing intermediate objects in the document structure
Solutions: 1. Check document structure to ensure nested fields exist 2. Use conditional operators in MongoDB to handle missing fields 3. Add validation for nested fields before accessing them
Example:
// In MongoDB, you can handle this with $ifNull or similar operators
// In your application code, add validation:
function validateNestedPath(obj, path) {
const parts = path.split('.');
let current = obj;
for (const part of parts) {
if (current === undefined || current === null) {
return false;
}
current = current[part];
}
return true;
}
Array Access Issues
Error: Error in array index syntax
or Cannot convert undefined or null to object
Possible causes: - Incorrect array access syntax - Accessing an array index that doesn't exist - Array field is empty or not an array
Solutions: 1. Verify array access syntax (use items[0].name
format) 2. Check array bounds before accessing elements 3. Use conditional logic to handle missing array elements
Example:
// Check array existence and bounds before accessing
const query = `
SELECT _id, customer,
CASE
WHEN items[0] IS NOT NULL THEN items[0].name
ELSE NULL
END as first_item
FROM orders
`;
// In MongoDB, you can use $arrayElemAt with conditional logic
// But QueryLeaf may need a simpler approach for now
JOIN Operation Issues
Error: Error executing $lookup stage
or Join condition not properly specified
Possible causes: - Incorrect JOIN syntax - JOIN fields have different types - Missing or invalid JOIN conditions - Referenced collection doesn't exist
Solutions: 1. Verify JOIN syntax and field names 2. Make sure JOIN fields have compatible types 3. Check that referenced collections exist 4. Simplify JOIN conditions for troubleshooting
Example:
// Simplified JOIN for troubleshooting
await queryLeaf.execute(`
SELECT u.name, o._id
FROM users u
JOIN orders o ON u._id = o.userId
`);
// If issues persist, examine the MongoDB schema and indexes
Debugging Techniques
Using DummyQueryLeaf for Debugging
The DummyQueryLeaf class is invaluable for debugging SQL translation issues:
import { DummyQueryLeaf } from 'queryleaf';
async function debugQuery(sqlQuery) {
const dummy = new DummyQueryLeaf('debug_db');
console.log('\n=== SQL Query ===');
console.log(sqlQuery);
try {
await dummy.execute(sqlQuery);
console.log('✅ Query parsed and compiled successfully');
} catch (error) {
console.error('❌ Error:', error.message);
}
}
// Debug a specific query
debugQuery('SELECT * FROM users WHERE address.city = "New York"');
Enabling Verbose Logging
You can enable more verbose logging in your application:
// Set environment variable for debugging
process.env.DEBUG_QUERYLEAF = 'true';
// In your QueryLeaf implementation, you could add:
function log(message) {
if (process.env.DEBUG_QUERYLEAF) {
console.log(`[QueryLeaf Debug] ${message}`);
}
}
Checking MongoDB Commands
Examine the MongoDB commands being generated:
// Import from @queryleaf/lib
import { DummyQueryLeaf } from '@queryleaf/lib';
// Using DummyQueryLeaf
const dummy = new DummyQueryLeaf('debug_db');
console.log('Translating SQL to MongoDB commands...');
await dummy.execute('SELECT name, email FROM users WHERE age > 21');
// This will log the MongoDB commands that would be executed
Using MongoDB Explain
For performance issues, you can use MongoDB's explain feature:
// After identifying the MongoDB query using DummyQueryLeaf,
// you can run MongoDB's explain directly:
const client = new MongoClient('mongodb://localhost:27017');
await client.connect();
const db = client.db('mydb');
const explanation = await db.collection('users').find({ age: { $gt: 21 } })
.explain('executionStats');
console.log(JSON.stringify(explanation, null, 2));
Performance Issues
Slow Queries
Symptoms: - Queries take longer than expected - CPU usage spikes during queries - Memory usage increases significantly
Possible causes: - Missing indexes on queried fields - Complex aggregation pipelines - Large result sets - Complex JOIN operations
Solutions: 1. Add appropriate indexes to MongoDB collections 2. Limit result sets with LIMIT clause 3. Optimize WHERE conditions to use indexed fields 4. Reduce complexity of aggregation queries 5. Consider denormalizing data to avoid JOINs
Example - Creating indexes:
// Create indexes on frequently queried fields
const client = new MongoClient('mongodb://localhost:27017');
await client.connect();
const db = client.db('mydb');
// Create index on frequently queried field
await db.collection('users').createIndex({ age: 1 });
// Create compound index for queries that filter on multiple fields
await db.collection('orders').createIndex({ status: 1, date: -1 });
// Create index on join fields
await db.collection('orders').createIndex({ userId: 1 });
// Create index on nested fields
await db.collection('users').createIndex({ 'address.city': 1 });
Memory Issues
Symptoms: - Out of memory errors - Application crashes with large datasets - Gradually increasing memory usage
Possible causes: - Large result sets being loaded into memory - Missing LIMIT clauses in queries - Memory leaks in connection handling - Inadequate server resources
Solutions: 1. Add LIMIT clauses to all queries 2. Implement pagination for large result sets 3. Close MongoDB connections properly 4. Use streaming for large result processing 5. Increase server memory or scale horizontally
Example - Implementing pagination:
async function paginateResults(query, pageSize = 100, page = 1) {
const client = new MongoClient('mongodb://localhost:27017');
await client.connect();
try {
const queryLeaf = new QueryLeaf(client, 'mydb');
// Add LIMIT and calculate offset
const offset = (page - 1) * pageSize;
const paginatedQuery = `${query} LIMIT ${pageSize} OFFSET ${offset}`;
return await queryLeaf.execute(paginatedQuery);
} finally {
await client.close();
}
}
// Use pagination
const page1 = await paginateResults('SELECT * FROM users ORDER BY name', 50, 1);
const page2 = await paginateResults('SELECT * FROM users ORDER BY name', 50, 2);
Common Patterns and Solutions
Handling MongoDB ObjectIDs
QueryLeaf automatically converts string representations of MongoDB ObjectIDs to actual ObjectID objects when:
- The field name is
_id
- The field name ends with
Id
(e.g.,userId
) - The field name ends with
Ids
(for arrays of IDs) - The string follows the MongoDB ObjectID format (24 hex characters)
-- These will automatically have the string converted to ObjectID
SELECT * FROM users WHERE _id = '507f1f77bcf86cd799439011'
SELECT * FROM orders WHERE userId = '507f1f77bcf86cd799439011'
If you're working directly with ObjectIDs in your application code:
import { ObjectId } from 'mongodb';
// Converting string IDs to ObjectID in your application
function stringToObjectId(id) {
try {
return new ObjectId(id);
} catch (error) {
throw new Error(`Invalid ObjectID format: ${id}`);
}
}
// Using with QueryLeaf
const id = new ObjectId('5f8d94c3e6b5c3a99ceafb53');
await queryLeaf.execute(`SELECT * FROM users WHERE _id = '${id.toString()}'`);
Common ObjectID Issues
Invalid ObjectID Format:
Solution: Ensure you're using a valid 24-character hexadecimal string.
ObjectID Comparison with String:
If your query isn't returning expected results when filtering by ID, make sure:
- The ID string has the correct format
- The field name follows the convention (_id, userId, etc.)
- You're using single quotes around the ObjectID string
-- Correct:
SELECT * FROM users WHERE _id = '507f1f77bcf86cd799439011'
-- Incorrect (missing quotes):
SELECT * FROM users WHERE _id = 507f1f77bcf86cd799439011
Handling NULL Values
Work with NULL values in SQL queries:
-- Find documents where a field is NULL
SELECT * FROM users WHERE profile IS NULL
-- Find documents where a field is NOT NULL
SELECT * FROM users WHERE email IS NOT NULL
-- Handle NULLs in results with COALESCE
SELECT name, COALESCE(email, 'No Email') as contact FROM users
Debugging Complex Aggregations
For complex GROUP BY operations:
// Import from @queryleaf/lib
import { DummyQueryLeaf } from '@queryleaf/lib';
// Break down the aggregation into stages
const dummy = new DummyQueryLeaf('debug_db');
console.log('=== Stage 1: Simple GROUP BY ===');
await dummy.execute(`
SELECT category, COUNT(*) as count
FROM products
GROUP BY category
`);
console.log('\n=== Stage 2: Multiple Aggregations ===');
await dummy.execute(`
SELECT category, COUNT(*) as count, AVG(price) as avg_price
FROM products
GROUP BY category
`);
console.log('\n=== Stage 3: With WHERE Clause ===');
await dummy.execute(`
SELECT category, COUNT(*) as count, AVG(price) as avg_price
FROM products
WHERE stock > 0
GROUP BY category
`);
Getting Additional Help
If you're still experiencing issues after trying these troubleshooting steps:
- Check Documentation: Review the detailed documentation for specific features
- Example Projects: Look at example projects for reference implementations
- GitHub Issues: Search the GitHub repository for similar issues
- Community Support: Ask questions in the GitHub Discussions section
- Professional Support: For commercial license holders, contact [email protected]
For bug reports or feature requests, please open an issue on the GitHub repository with:
- A minimal, reproducible example
- Your SQL query and expected results
- Error messages and stack traces
- QueryLeaf version and MongoDB version