Skip to content

Local Testing Guide

This guide covers comprehensive testing of RapidTriageME in your local development environment, from basic functionality verification to advanced testing scenarios.

Testing Overview

graph TB
    subgraph "Testing Layers"
        UNIT[Unit Tests<br/>Individual Components]
        INTEGRATION[Integration Tests<br/>Component Interaction]
        E2E[End-to-End Tests<br/>Full Workflow]
        MANUAL[Manual Testing<br/>User Scenarios]
    end

    subgraph "Test Environment"
        LOCAL[Local Development]
        CI[CI/CD Pipeline]
        STAGING[Staging Environment]
    end

    UNIT --> INTEGRATION
    INTEGRATION --> E2E
    E2E --> MANUAL

    LOCAL --> CI
    CI --> STAGING

    style UNIT fill:#e8f5e8
    style E2E fill:#e1f5fe
    style MANUAL fill:#fff3e0

Prerequisites

Before starting local testing, ensure you have:

# Required software
node --version    # v18.0.0 or higher
npm --version     # v9.0.0 or higher
google-chrome --version  # Latest stable

# Optional but recommended
docker --version  # For containerized testing
pytest --version  # For Python test utilities

Quick Test Setup

1. Clone and Install

# Clone the repository
git clone .git
cd rapidtriage

# Install all dependencies
npm run install:all

# Build all components
npm run build:all

2. Start Test Environment

# Start all services for testing
npm run test:setup

# This starts:
# - Browser connector on port 3025
# - Test web server on port 3000
# - Chrome with test extension loaded

3. Run Basic Tests

# Run all test suites
npm test

# Run specific test categories
npm run test:unit          # Unit tests only
npm run test:integration   # Integration tests
npm run test:e2e           # End-to-end tests

Unit Testing

Browser Connector Tests

// test/browser-connector.test.ts
import request from 'supertest';
import { BrowserConnectorServer } from '../src/browser-connector';

describe('Browser Connector Server', () => {
  let server: BrowserConnectorServer;
  let app: any;

  beforeAll(async () => {
    server = new BrowserConnectorServer({
      port: 0, // Use random port for testing
      host: 'localhost',
      environment: 'test'
    });
    app = server.getApp();
    await server.start();
  });

  afterAll(async () => {
    await server.stop();
  });

  describe('GET /.identity', () => {
    it('should return server identity', async () => {
      const response = await request(app)
        .get('/.identity')
        .expect(200);

      expect(response.body).toMatchObject({
        name: 'RapidTriageME Browser Connector',
        status: 'healthy',
        version: expect.any(String)
      });
    });
  });

  describe('GET /console-logs', () => {
    beforeEach(() => {
      // Clear cache before each test
      server.getDataCache().clearAll();
    });

    it('should return empty logs initially', async () => {
      const response = await request(app)
        .get('/console-logs')
        .expect(200);

      expect(response.body.logs).toEqual([]);
      expect(response.body.total).toBe(0);
    });

    it('should filter logs by level', async () => {
      // Add test logs
      server.getDataCache().addConsoleLog({
        id: 'log1',
        level: 'error',
        message: 'Test error',
        timestamp: Date.now()
      });

      server.getDataCache().addConsoleLog({
        id: 'log2',
        level: 'info',
        message: 'Test info',
        timestamp: Date.now()
      });

      const response = await request(app)
        .get('/console-logs?level=error')
        .expect(200);

      expect(response.body.logs).toHaveLength(1);
      expect(response.body.logs[0].level).toBe('error');
    });
  });

  describe('POST /capture-screenshot', () => {
    it('should handle screenshot request', async () => {
      const response = await request(app)
        .post('/capture-screenshot')
        .send({ options: { fullPage: true } })
        .expect(200);

      expect(response.body).toHaveProperty('screenshot');
      expect(response.body.screenshot).toMatch(/^data:image/);
    });

    it('should validate screenshot options', async () => {
      const response = await request(app)
        .post('/capture-screenshot')
        .send({ options: { quality: 150 } }) // Invalid quality
        .expect(400);

      expect(response.body.error).toContain('Invalid quality');
    });
  });
});

MCP Server Tests

// test/mcp-server.test.ts
import { MCPServer } from '../src/mcp-server';
import { BrowserConnectorMock } from './mocks/browser-connector-mock';

describe('MCP Server', () => {
  let mcpServer: MCPServer;
  let browserMock: BrowserConnectorMock;

  beforeEach(() => {
    browserMock = new BrowserConnectorMock();
    mcpServer = new MCPServer({
      browserConnectorUrl: browserMock.getUrl()
    });
  });

  describe('Tool Execution', () => {
    it('should execute screenshot_capture tool', async () => {
      browserMock.mockScreenshot('...');

      const result = await mcpServer.executeTool('screenshot_capture', {
        options: { fullPage: true }
      });

      expect(result.content).toHaveLength(2);
      expect(result.content[0].type).toBe('image');
      expect(result.content[1].type).toBe('text');
      expect(result.isError).toBe(false);
    });

    it('should handle tool execution errors gracefully', async () => {
      browserMock.mockError('Browser not connected');

      const result = await mcpServer.executeTool('screenshot_capture', {});

      expect(result.isError).toBe(true);
      expect(result.content[0].text).toContain('Screenshot capture failed');
    });
  });

  describe('Tool Registry', () => {
    it('should list all available tools', async () => {
      const tools = await mcpServer.listTools();

      const toolNames = tools.tools.map(tool => tool.name);
      expect(toolNames).toContain('screenshot_capture');
      expect(toolNames).toContain('get_console_logs');
      expect(toolNames).toContain('run_lighthouse_audit');
    });
  });
});

Chrome Extension Tests

// test/extension.test.js
describe('Chrome Extension', () => {
  let mockChrome;

  beforeEach(() => {
    mockChrome = {
      runtime: {
        onMessage: {
          addListener: jest.fn()
        },
        sendMessage: jest.fn()
      },
      devtools: {
        panels: {
          create: jest.fn()
        }
      },
      tabs: {
        captureVisibleTab: jest.fn()
      }
    };

    global.chrome = mockChrome;
  });

  it('should initialize DevTools panel', () => {
    require('../rapidtriage-extension/devtools.js');

    expect(mockChrome.devtools.panels.create).toHaveBeenCalledWith(
      'RapidTriage',
      expect.any(String),
      'panel.html',
      expect.any(Function)
    );
  });

  it('should capture screenshot when requested', async () => {
    mockChrome.tabs.captureVisibleTab.mockResolvedValue('');

    const { captureScreenshot } = require('../rapidtriage-extension/background.js');
    const result = await captureScreenshot({ fullPage: false });

    expect(result).toMatch(/^data:image/);
    expect(mockChrome.tabs.captureVisibleTab).toHaveBeenCalled();
  });
});

Integration Testing

Full Stack Integration

// test/integration/full-stack.test.ts
import { spawn, ChildProcess } from 'child_process';
import WebSocket from 'ws';
import puppeteer, { Browser, Page } from 'puppeteer';

describe('Full Stack Integration', () => {
  let browserConnector: ChildProcess;
  let mcpServer: ChildProcess;
  let browser: Browser;
  let page: Page;

  beforeAll(async () => {
    // Start browser connector server
    browserConnector = spawn('npm', ['run', 'start:browser-connector'], {
      env: { ...process.env, PORT: '3025' }
    });

    // Wait for server to be ready
    await waitForServer('http://localhost:3025');

    // Start MCP server
    mcpServer = spawn('npm', ['run', 'start:mcp-server']);

    // Launch browser with extension
    browser = await puppeteer.launch({
      headless: false,
      args: [
        '--load-extension=./rapidtriage-extension',
        '--disable-extensions-except=./rapidtriage-extension'
      ]
    });

    page = await browser.newPage();
  }, 30000);

  afterAll(async () => {
    if (browser) await browser.close();
    if (browserConnector) browserConnector.kill();
    if (mcpServer) mcpServer.kill();
  });

  it('should establish WebSocket connection between extension and server', async () => {
    await page.goto('http://localhost:3000/test-page.html');

    // Open DevTools to trigger extension
    const client = await page.target().createCDPSession();
    await client.send('Runtime.enable');

    // Verify WebSocket connection
    const ws = new WebSocket('ws://localhost:3025/ws');
    await new Promise(resolve => ws.on('open', resolve));

    expect(ws.readyState).toBe(WebSocket.OPEN);
    ws.close();
  });

  it('should capture and retrieve console logs end-to-end', async () => {
    await page.goto('http://localhost:3000/test-page.html');

    // Generate console log
    await page.evaluate(() => {
      console.error('Test integration error');
    });

    // Wait for log to be processed
    await new Promise(resolve => setTimeout(resolve, 1000));

    // Retrieve logs via API
    const response = await fetch('http://localhost:3025/console-logs');
    const data = await response.json();

    expect(data.logs).toHaveLength(1);
    expect(data.logs[0].message).toBe('Test integration error');
    expect(data.logs[0].level).toBe('error');
  });

  it('should complete screenshot workflow', async () => {
    await page.goto('http://localhost:3000/test-page.html');

    // Request screenshot via API
    const response = await fetch('http://localhost:3025/capture-screenshot', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ options: { fullPage: true } })
    });

    const data = await response.json();

    expect(response.status).toBe(200);
    expect(data.screenshot).toMatch(/^data:image/);
    expect(data.dimensions).toBeDefined();
  });
});

End-to-End Testing

Playwright E2E Tests

// test/e2e/user-workflows.spec.ts
import { test, expect, chromium } from '@playwright/test';

test.describe('RapidTriageME User Workflows', () => {
  test.beforeAll(async () => {
    // Ensure servers are running
    await fetch('http://localhost:3025/.identity');
  });

  test('complete debugging workflow', async () => {
    const browser = await chromium.launch({
      args: ['--load-extension=./rapidtriage-extension']
    });

    const context = await browser.newContext();
    const page = await context.newPage();

    // Navigate to test page
    await page.goto('http://localhost:3000/problematic-page.html');

    // Open DevTools (simulating user action)
    await page.keyboard.press('F12');

    // Wait for extension panel to load
    await page.waitForTimeout(2000);

    // Simulate AI assistant requesting screenshot
    const screenshotResponse = await fetch('http://localhost:3025/capture-screenshot', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({ options: { fullPage: true } })
    });

    expect(screenshotResponse.status).toBe(200);

    // Simulate AI assistant requesting console logs
    const logsResponse = await fetch('http://localhost:3025/console-logs');
    const logsData = await logsResponse.json();

    expect(logsResponse.status).toBe(200);
    expect(logsData.logs).toBeDefined();

    await browser.close();
  });

  test('Lighthouse audit workflow', async () => {
    const auditResponse = await fetch('http://localhost:3025/lighthouse-audit', {
      method: 'POST',
      headers: { 'Content-Type': 'application/json' },
      body: JSON.stringify({
        url: 'http://localhost:3000/test-page.html',
        categories: ['performance', 'accessibility']
      })
    });

    expect(auditResponse.status).toBe(200);

    const auditData = await auditResponse.json();
    expect(auditData.audit).toBeDefined();
    expect(auditData.audit.scores).toBeDefined();
    expect(auditData.audit.scores.performance).toBeGreaterThanOrEqual(0);
  });
});

Manual Testing Scenarios

Test Scenario 1: Basic Functionality

# 1. Start RapidTriageME
npm run dev

# 2. Open Chrome and load extension
# chrome://extensions/ -> Load unpacked -> select rapidtriage-extension

# 3. Navigate to test page
# http://localhost:3000/test-page.html

# 4. Open DevTools and check RapidTriage panel
# F12 -> RapidTriage tab -> Should show "Connected"

# 5. Test screenshot capture
curl -X POST http://localhost:3025/capture-screenshot \
  -H "Content-Type: application/json" \
  -d '{"options": {"fullPage": true}}'

# Expected: JSON response with base64 screenshot

Test Scenario 2: Error Handling

// Navigate to test page and run in console:
console.error('Test error message');
console.warn('Test warning message');
throw new Error('Test exception');

// Then check logs:
// curl http://localhost:3025/console-logs

Test Scenario 3: Network Monitoring

// On test page, trigger network requests:
fetch('/api/success').then(r => console.log('Success:', r.status));
fetch('/api/404').catch(e => console.log('Error:', e));
fetch('/api/slow').then(r => console.log('Slow request completed'));

// Check network logs:
// curl http://localhost:3025/network-requests

Performance Testing

Load Testing Script

// test/performance/load-test.ts
import autocannon from 'autocannon';

async function runLoadTest() {
  const result = await autocannon({
    url: 'http://localhost:3025',
    connections: 50,
    duration: 30, // 30 seconds
    requests: [
      {
        method: 'GET',
        path: '/.identity'
      },
      {
        method: 'GET', 
        path: '/console-logs?limit=100'
      },
      {
        method: 'POST',
        path: '/capture-screenshot',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify({ options: { fullPage: false } })
      }
    ]
  });

  console.log('Load test results:');
  console.log(`Requests/sec: ${result.requests.average}`);
  console.log(`Latency: ${result.latency.average}ms`);
  console.log(`Errors: ${result.errors}`);

  // Performance assertions
  expect(result.requests.average).toBeGreaterThan(100);
  expect(result.latency.average).toBeLessThan(100);
  expect(result.errors).toBe(0);
}

test('should handle load gracefully', runLoadTest, 60000);

Memory Testing

// test/performance/memory-test.ts
test('should not leak memory during extended operation', async () => {
  const initialMemory = process.memoryUsage().heapUsed;

  // Simulate extended operation
  for (let i = 0; i < 1000; i++) {
    await fetch('http://localhost:3025/console-logs');
    await new Promise(resolve => setTimeout(resolve, 10));
  }

  // Force garbage collection
  if (global.gc) global.gc();

  const finalMemory = process.memoryUsage().heapUsed;
  const memoryGrowth = finalMemory - initialMemory;

  // Memory growth should be reasonable (less than 50MB)
  expect(memoryGrowth).toBeLessThan(50 * 1024 * 1024);
}, 30000);

Test Utilities

Server Test Helper

// test/utils/server-helper.ts
export class ServerTestHelper {
  static async waitForServer(url: string, timeout = 30000): Promise<void> {
    const start = Date.now();

    while (Date.now() - start < timeout) {
      try {
        const response = await fetch(url);
        if (response.ok) return;
      } catch (error) {
        // Server not ready yet
      }

      await new Promise(resolve => setTimeout(resolve, 500));
    }

    throw new Error(`Server at ${url} not ready within ${timeout}ms`);
  }

  static async createTestData(count = 10): Promise<void> {
    for (let i = 0; i < count; i++) {
      await fetch('http://localhost:3025/test/add-log', {
        method: 'POST',
        headers: { 'Content-Type': 'application/json' },
        body: JSON.stringify({
          level: i % 2 === 0 ? 'error' : 'info',
          message: `Test log ${i}`,
          timestamp: Date.now() - (i * 1000)
        })
      });
    }
  }
}

Extension Test Helper

// test/utils/extension-helper.js
class ExtensionTestHelper {
  static mockChromeAPI() {
    global.chrome = {
      runtime: {
        onMessage: { addListener: jest.fn() },
        sendMessage: jest.fn(),
        getURL: jest.fn(path => `chrome-extension://test-id/${path}`)
      },
      devtools: {
        panels: {
          create: jest.fn((name, icon, page, callback) => {
            callback({ onShown: { addListener: jest.fn() } });
          })
        },
        inspectedWindow: {
          eval: jest.fn((code, callback) => callback('result', false))
        }
      },
      tabs: {
        captureVisibleTab: jest.fn(() => Promise.resolve(''))
      },
      storage: {
        local: {
          get: jest.fn(() => Promise.resolve({})),
          set: jest.fn(() => Promise.resolve())
        }
      }
    };
  }

  static async simulateExtensionInstall() {
    // Simulate extension installation process
    const installEvent = { reason: 'install' };
    chrome.runtime.onInstalled.addListener.mock.calls[0][0](installEvent);
  }
}

module.exports = ExtensionTestHelper;

Continuous Integration

GitHub Actions Workflow

# .github/workflows/test.yml
name: Test Suite

on:
  push:
    branches: [main, develop]
  pull_request:
    branches: [main]

jobs:
  test:
    runs-on: ubuntu-latest

    strategy:
      matrix:
        node-version: [18, 20]

    steps:
    - uses: actions/checkout@v4

    - name: Setup Node.js ${{ matrix.node-version }}
      uses: actions/setup-node@v4
      with:
        node-version: ${{ matrix.node-version }}
        cache: 'npm'

    - name: Install dependencies
      run: npm ci

    - name: Build all packages
      run: npm run build:all

    - name: Run unit tests
      run: npm run test:unit

    - name: Run integration tests
      run: npm run test:integration

    - name: Install Chrome
      run: |
        wget -q -O - https://dl.google.com/linux/linux_signing_key.pub | sudo apt-key add -
        sudo sh -c 'echo "deb [arch=amd64] http://dl.google.com/linux/chrome/deb/ stable main" >> /etc/apt/sources.list.d/google.list'
        sudo apt-get update
        sudo apt-get install google-chrome-stable

    - name: Run E2E tests
      run: npm run test:e2e

    - name: Upload coverage reports
      uses: codecov/codecov-action@v3
      with:
        files: ./coverage/lcov.info

Test Data Management

Test Fixtures

// test/fixtures/console-logs.ts
export const mockConsoleLogs = [
  {
    id: 'log_001',
    level: 'error',
    message: 'TypeError: Cannot read property \'foo\' of undefined',
    timestamp: 1704067200000,
    url: 'https://example.com/app.js',
    source: 'app.js:42:15',
    stack: [
      "TypeError: Cannot read property 'foo' of undefined",
      '    at app.js:42:15',
      '    at HTMLButtonElement.onclick (app.js:38:9)'
    ]
  },
  {
    id: 'log_002',
    level: 'warn',
    message: 'Deprecated API usage detected',
    timestamp: 1704067260000,
    url: 'https://example.com/legacy.js',
    source: 'legacy.js:123:20'
  }
];

export const mockNetworkRequests = [
  {
    id: 'req_001',
    url: 'https://api.example.com/users',
    method: 'GET',
    status: 200,
    statusText: 'OK',
    timestamp: 1704067200000,
    duration: 245,
    size: { request: 0, response: 1024 }
  },
  {
    id: 'req_002',
    url: 'https://api.example.com/posts',
    method: 'POST',
    status: 500,
    statusText: 'Internal Server Error',
    timestamp: 1704067300000,
    duration: 1500,
    size: { request: 512, response: 256 }
  }
];

Troubleshooting Tests

Common Test Issues

# Test server port conflicts
echo "Checking for port conflicts..."
lsof -i :3025 | grep LISTEN

# Clear test data
echo "Cleaning test environment..."
rm -rf ./test-data/*
npm run test:clean

# Reset test database
echo "Resetting test database..."
npm run test:db:reset

# Check Chrome processes
echo "Checking Chrome processes..."
ps aux | grep chrome | grep -v grep

Debug Test Failures

# Run tests with debug output
DEBUG=rapidtriage:* npm test

# Run specific test file
npm test -- --testPathPattern=browser-connector

# Run tests in watch mode
npm run test:watch

# Generate detailed coverage report
npm run test:coverage -- --coverage-reporters=html
open coverage/lcov-report/index.html

This comprehensive local testing guide ensures RapidTriageME works correctly across all scenarios before deployment to production.

Next Steps