MongoDB Client Integration
QueryLeaf is designed to work with your existing MongoDB client instead of managing connections itself. This approach provides several benefits:
- You have full control over connection management
- You can use your existing connection pools and configurations
- You can customize authentication and encryption
- QueryLeaf remains lightweight and focused on SQL translation
This guide explains how to integrate QueryLeaf with the MongoDB Node.js driver.
Creating a MongoDB Client
Before using QueryLeaf, you need to create and configure a MongoDB client:
import { MongoClient } from 'mongodb';
// Create a MongoDB client with connection options
const mongoClient = new MongoClient('mongodb://localhost:27017', {
  // Optional: Connection pool settings
  maxPoolSize: 50,
  minPoolSize: 5,
  // Optional: Authentication
  auth: {
    username: 'username',
    password: 'password'
  },
  // Optional: TLS/SSL settings
  tls: true,
  tlsCAFile: '/path/to/ca.pem',
  // Optional: Other settings
  connectTimeoutMS: 5000,
  socketTimeoutMS: 30000
});
// Connect to MongoDB
await mongoClient.connect();
Integrating with QueryLeaf
Once you have a MongoDB client, you can create a QueryLeaf instance:
import { QueryLeaf } from '@queryleaf/lib';
// Create a QueryLeaf instance with your MongoDB client
const queryLeaf = new QueryLeaf(mongoClient, 'mydatabase');
// Execute SQL queries and get all results at once
const results = await queryLeaf.execute('SELECT * FROM users');
// For large result sets, use cursor execution for better memory efficiency
const cursor = await queryLeaf.executeCursor('SELECT * FROM users');
try {
  // Process results one at a time
  await cursor.forEach(user => {
    console.log(`User: ${user.name}`);
  });
} finally {
  // Always close the cursor when done
  await cursor.close();
}
Connection Management
Since QueryLeaf doesn't manage connections, you're responsible for:
- Creating the MongoDB client
- Connecting before using QueryLeaf
- Closing the connection when done
Here's a complete example with proper connection management:
import { MongoClient } from 'mongodb';
import { QueryLeaf } from '@queryleaf/lib';
async function main() {
  const client = new MongoClient('mongodb://localhost:27017');
  try {
    // Connect to MongoDB
    await client.connect();
    console.log('Connected to MongoDB');
    // Create QueryLeaf instance
    const queryLeaf = new QueryLeaf(client, 'mydatabase');
    // Execute queries - get all results at once
    const users = await queryLeaf.execute('SELECT * FROM users LIMIT 10');
    console.log(`Found ${users.length} users`);
    // You can execute multiple queries with the same instance
    const products = await queryLeaf.execute(
      'SELECT name, price FROM products WHERE category = "Electronics"'
    );
    console.log(`Found ${products.length} electronic products`);
    // For large result sets, use cursor execution
    const ordersCursor = await queryLeaf.executeCursor(
      'SELECT * FROM orders WHERE total > 1000'
    );
    try {
      // Process results in a memory-efficient way
      let count = 0;
      await ordersCursor.forEach(order => {
        console.log(`Processing order #${order.orderId}`);
        count++;
      });
      console.log(`Processed ${count} high-value orders`);
    } finally {
      // Always close the cursor when done
      await ordersCursor.close();
    }
  } catch (error) {
    console.error('Error:', error);
  } finally {
    // Always close the MongoDB client when done
    await client.close();
    console.log('MongoDB connection closed');
  }
}
main().catch(console.error);
Advanced Connection Options
Connection Pooling
For production applications, you should configure connection pooling:
const client = new MongoClient('mongodb://localhost:27017', {
  maxPoolSize: 100,   // Maximum connections in the pool
  minPoolSize: 10,    // Minimum connections to maintain
  maxIdleTimeMS: 30000  // Close connections after 30 seconds of inactivity
});
Replica Sets
For high availability, connect to a MongoDB replica set:
const client = new MongoClient(
  'mongodb://server1:27017,server2:27017,server3:27017/?replicaSet=myReplicaSet',
  {
    readPreference: 'secondaryPreferred'  // Read from secondary nodes when possible
  }
);
MongoDB Atlas
To connect to MongoDB Atlas:
const client = new MongoClient(
  'mongodb+srv://username:[email protected]/mydb?retryWrites=true&w=majority'
);
Connection Timeouts
Configure timeouts for better error handling:
const client = new MongoClient('mongodb://localhost:27017', {
  connectTimeoutMS: 5000,    // Give up connecting after 5 seconds
  socketTimeoutMS: 30000,    // Socket timeout for operations
  serverSelectionTimeoutMS: 5000  // Timeout for server selection
});
Using Multiple Databases
You can create multiple QueryLeaf instances to work with different databases:
// Create separate instances for different databases
const usersDb = new QueryLeaf(mongoClient, 'users_db');
const productsDb = new QueryLeaf(mongoClient, 'products_db');
// Query the users database
const activeUsers = await usersDb.execute(
  'SELECT * FROM users WHERE status = "active"'
);
// Query the products database
const featuredProducts = await productsDb.execute(
  'SELECT * FROM products WHERE featured = true'
);
Connection Error Handling
Implement proper error handling for MongoDB connection issues:
import { MongoClient, MongoServerError } from 'mongodb';
import { QueryLeaf } from '@queryleaf/lib';
async function executeWithRetry(sqlQuery: string, maxRetries = 3) {
  const client = new MongoClient('mongodb://localhost:27017');
  let retries = 0;
  while (retries < maxRetries) {
    try {
      await client.connect();
      const queryLeaf = new QueryLeaf(client, 'mydatabase');
      // Execute the query
      const results = await queryLeaf.execute(sqlQuery);
      // Success - return results
      return results;
    } catch (error) {
      retries++;
      if (error instanceof MongoServerError) {
        // Handle specific MongoDB errors
        if (error.code === 13) {
          console.error('Authentication failed');
          throw error; // Don't retry auth failures
        }
      }
      if (retries >= maxRetries) {
        console.error(`Failed after ${maxRetries} retries`, error);
        throw error;
      }
      // Wait before retrying (exponential backoff)
      const waitTime = Math.pow(2, retries) * 100;
      console.log(`Retrying in ${waitTime}ms...`);
      await new Promise(resolve => setTimeout(resolve, waitTime));
    } finally {
      // Always close the connection
      await client.close().catch(console.error);
    }
  }
}
Next Steps
Now that you understand how to integrate QueryLeaf with MongoDB clients, you can:
- Learn about Using the Dummy Client for testing
- Explore SQL Syntax Support for details on supported features
- See practical Examples of using QueryLeaf in different scenarios