Skip to content

Dummy Client for Testing

QueryLeaf provides a DummyQueryLeaf class that helps you test and debug your SQL queries without connecting to a real MongoDB database. This is useful for:

  • Development without a MongoDB instance
  • Unit testing without database dependencies
  • Debugging SQL to MongoDB translations
  • Generating MongoDB commands for documentation

This guide explains how to use the dummy client effectively.

Basic Usage

The DummyQueryLeaf class mimics the real QueryLeaf but logs operations instead of executing them:

import { DummyQueryLeaf } from '@queryleaf/lib';

// Create a dummy client (no MongoDB client required)
const dummyLeaf = new DummyQueryLeaf('mydb');

// Execute a query (will be logged, not executed)
await dummyLeaf.execute('SELECT name, email FROM users WHERE age > 21');
// [DUMMY MongoDB] FIND in mydb.users with filter: {"age":{"$gt":21}}
// [DUMMY MongoDB] Executing find on users with projection: {"name":1,"email":1}

Debugging SQL Translation

The dummy client is particularly useful for understanding how SQL queries are translated to MongoDB commands:

import { DummyQueryLeaf } from '@queryleaf/lib';

async function debugSqlTranslation() {
  const dummyLeaf = new DummyQueryLeaf('testdb');

  console.log('===== SELECT with WHERE =====');
  await dummyLeaf.execute(
    'SELECT name, email FROM users WHERE age > 21 AND (status = "active" OR role = "admin")'
  );

  console.log('\n===== JOIN =====');
  await dummyLeaf.execute(`
    SELECT u.name, o.total 
    FROM users u 
    JOIN orders o ON u._id = o.userId 
    WHERE o.status = "completed"
  `);

  console.log('\n===== GROUP BY with Aggregation =====');
  await dummyLeaf.execute(`
    SELECT category, COUNT(*) as count, AVG(price) as avg_price 
    FROM products 
    GROUP BY category
  `);
}

debugSqlTranslation().catch(console.error);

This will output detailed logs showing how each SQL query is translated to MongoDB operations.

Using in Unit Tests

The dummy client is especially useful in unit tests where you want to verify that your application generates the correct SQL queries without actually executing them:

import { DummyQueryLeaf } from '@queryleaf/lib';

// Mock console.log to capture output
let consoleOutput: string[] = [];
const originalLog = console.log;
console.log = (message: string) => {
  consoleOutput.push(message);
  originalLog(message);
};

describe('User Service', () => {
  let dummyLeaf: DummyQueryLeaf;
  let userService: UserService;

  beforeEach(() => {
    consoleOutput = [];
    dummyLeaf = new DummyQueryLeaf('testdb');
    userService = new UserService(dummyLeaf);
  });

  afterAll(() => {
    console.log = originalLog;
  });

  test('getActiveUsers should generate correct SQL', async () => {
    await userService.getActiveUsers();

    // Check that the correct query was generated
    expect(consoleOutput.some(log => 
      log.includes('FIND in testdb.users with filter: {"status":"active"}')
    )).toBe(true);
  });

  test('createUser should generate correct INSERT', async () => {
    const user = { name: 'Test User', email: '[email protected]', age: 30 };
    await userService.createUser(user);

    // Check that the correct insert was generated
    expect(consoleOutput.some(log => 
      log.includes('INSERT in testdb.users')
    )).toBe(true);
  });
});

How the Dummy Client Works

The DummyQueryLeaf class follows the same flow as the real QueryLeaf:

  1. Parses the SQL query to an AST
  2. Compiles the AST to MongoDB commands
  3. Instead of executing the commands, it logs them

This means it validates that your SQL syntax is correct and shows how it would be executed, without actually running the operations.

Customizing the Dummy Client

You can extend the DummyQueryLeaf class for custom testing scenarios:

import { DummyQueryLeaf, Command } from '@queryleaf/lib';

class TestingQueryLeaf extends DummyQueryLeaf {
  public commands: Command[] = [];

  // Override to capture commands for assertions
  protected async executeDummyCommand(command: Command): Promise<any> {
    this.commands.push(command);
    return await super.executeDummyCommand(command);
  }

  // Override to provide mock data
  protected executeDummyFind(command: Command): any[] {
    // Return fake data based on the collection
    if (command.collection === 'users') {
      return [
        { _id: '1', name: 'Alice', email: '[email protected]', age: 28 },
        { _id: '2', name: 'Bob', email: '[email protected]', age: 35 }
      ];
    }
    return [];
  }

  // Clear captured commands between tests
  public reset(): void {
    this.commands = [];
  }
}

// Usage in tests
const testLeaf = new TestingQueryLeaf('testdb');
await testLeaf.execute('SELECT * FROM users WHERE age > 30');

// Now you can assert on the captured commands
expect(testLeaf.commands.length).toBe(1);
expect(testLeaf.commands[0].type).toBe('FIND');
expect(testLeaf.commands[0].filter).toEqual({ age: { $gt: 30 } });

Common Testing Patterns

Testing Query Generation

test('should generate correct MongoDB filter from SQL WHERE clause', async () => {
  const dummyLeaf = new DummyQueryLeaf('testdb');

  // Capture console output
  const spy = jest.spyOn(console, 'log');

  await dummyLeaf.execute(
    'SELECT * FROM users WHERE status = "active" AND age BETWEEN 18 AND 65'
  );

  // Verify the correct filter was generated
  expect(spy).toHaveBeenCalledWith(
    expect.stringContaining('"status":"active"')
  );
  expect(spy).toHaveBeenCalledWith(
    expect.stringContaining('"age":{"$gte":18,"$lte":65}')
  );

  spy.mockRestore();
});

Testing Error Handling

test('should throw error on invalid SQL syntax', async () => {
  const dummyLeaf = new DummyQueryLeaf('testdb');

  // Test invalid SQL
  await expect(
    dummyLeaf.execute('SELECT * FORM users') // Typo in FROM
  ).rejects.toThrow();
});

Next Steps

Now that you understand how to use the dummy client for testing, you can: