Files
mqtt-explorer/.github/copilot-instructions.md

6.8 KiB

GitHub Copilot Agent Instructions for MQTT Explorer

Overview

MQTT Explorer is an Electron-based desktop application for exploring MQTT brokers. It provides a comprehensive UI for connecting to MQTT brokers, browsing topics, and analyzing message flows.

Technology Stack

  • Frontend: React 16.x with Material-UI
  • Backend: Node.js with TypeScript
  • Desktop Framework: Electron 29.x
  • MQTT Client: mqttjs v4.x
  • State Management: Redux with redux-thunk
  • Build Tools: webpack, TypeScript compiler
  • Testing: Mocha + Chai for unit tests, Playwright for MCP introspection tests

Project Setup

Building and Running

# Install dependencies
yarn install

# Build the project
yarn build

# Start the application
yarn start

# Start in development mode
yarn dev

Running with MCP Introspection (for testing)

# Build first
yarn build

# Start with MCP introspection enabled
electron . --enable-mcp-introspection

# Or with custom port
electron . --enable-mcp-introspection --remote-debugging-port=9223

Writing Tests

Requirements for All Tests

  1. Tests MUST be deterministic - They should produce the same results every time they run
  2. Tests MUST be independent - Each test should be able to run in isolation without depending on other tests
  3. Include screenshots - Visual verification is required for UI changes
  4. Handle asynchronous operations properly - This is an MQTT message queue tool

Handling MQTT Asynchronous Operations

MQTT is inherently asynchronous. When writing tests:

  • Wait for message propagation: Use proper wait strategies (e.g., await page.waitForSelector(), await sleep())
  • Don't assume immediate updates: Messages take time to send, receive, and update the UI
  • Use event-based waiting: Wait for specific UI elements or state changes rather than fixed timeouts when possible
  • Account for network latency: MQTT broker communication involves network round trips

Example Test Pattern

// 1. Perform action (e.g., publish message)
await publishMessage(topic, payload)

// 2. Wait for UI to update (not just arbitrary sleep)
await page.waitForSelector(`text="${expectedValue}"`, { timeout: 5000 })

// 3. Verify state
const value = await page.textContent('.message-value')
expect(value).toBe(expectedValue)

// 4. Take screenshot for verification
await page.screenshot({ path: 'test-result.png' })

Running Tests

# Run all tests
yarn test

# Run specific test suites
yarn test:app
yarn test:backend
yarn test:mcp

# Run linters
yarn lint
yarn lint:fix

MCP Introspection Testing

The project supports MCP (Model Context Protocol) for automated testing with Playwright:

  • Use yarn test:mcp to run automated UI tests
  • Tests launch the app with remote debugging enabled on port 9222
  • Connect to http://localhost:9222 via Chrome DevTools Protocol

Project Structure

  • app/ - Frontend React application
  • backend/ - Backend models, tests, and connection management
  • src/ - Electron main process and bindings
  • src/spec/ - Test specifications including MCP introspection tests

Code Style and Formatting

Linting

The project uses TSLint with Airbnb config and Prettier for code formatting:

# Run all linters
yarn lint

# Run linters individually
yarn lint:prettier    # Check Prettier formatting
yarn lint:tslint      # Check TSLint rules
yarn lint:spellcheck  # Check spelling in code

# Auto-fix issues
yarn lint:fix         # Fix TSLint and Prettier issues
yarn lint:tslint:fix  # Fix TSLint issues only
yarn lint:prettier:fix # Fix Prettier issues only

Code Style Rules

  • Semicolons: Never use semicolons (enforced by TSLint and Prettier)
  • Quotes: Single quotes for strings
  • Indentation: 2 spaces
  • Line length: Maximum 120 characters (Prettier) / 200 characters (TSLint)
  • Arrow functions: No parentheses for single parameters (x => x + 1)
  • Trailing commas: Required for multiline objects and arrays (ES5 compatible)

TypeScript Guidelines

  • Enable strict null checks and no implicit any
  • Use TypeScript interfaces for data structures
  • Prefer const over let, avoid var
  • Use type inference when possible, explicit types when clarity is needed

Dependency Management

Adding Dependencies

# Add to root project
yarn add <package-name>

# Add to app (frontend)
cd app && yarn add <package-name>

# Add to backend
cd backend && yarn add <package-name>

# Add dev dependencies
yarn add -D <package-name>

Important Dependency Notes

  • Main dependencies are in the root package.json
  • Frontend React app has its own dependencies in app/package.json
  • Backend models and logic have dependencies in backend/package.json
  • Always use --frozen-lockfile in CI to ensure reproducible builds
  • Run yarn install after pulling changes that modify yarn.lock

Debugging

Development Mode

# Start with hot reload for frontend
yarn dev

# This runs two processes in parallel:
# 1. webpack-dev-server for the React app (port varies)
# 2. Electron in development mode with the --development flag

Debugging TypeScript

  • Source maps are enabled in tsconfig.json
  • Use ts-node for running TypeScript files directly
  • Backend tests can be debugged with: cd backend && yarn test-inspect

Common Issues

  • Build fails: Clear dist/ and app/build/ directories, then rebuild
  • Electron won't start: Ensure yarn build completed successfully
  • Tests fail: Check if MQTT broker (mosquitto) is running for integration tests
  • UI not updating: In dev mode, ensure webpack-dev-server is running

Deployment and Packaging

Creating Releases

# Prepare release (updates version, changelog)
yarn prepare-release

# Package the application for distribution
yarn package

# Package with Docker (for consistent builds)
yarn package-with-docker

Release Workflow

  • Beta releases: Create PR to beta branch with "feat:" or "fix:" commits
  • Production releases: Create PR to release branch with "feat:" or "fix:" commits
  • Semantic release automatically handles versioning and changelog
  • Builds are created for Windows, macOS, and Linux

Build Artifacts

  • Output directory: build/
  • Supported formats: DMG (macOS), EXE/NSIS (Windows), AppImage/Snap (Linux), AppX (Windows Store)
  • Code signing is configured via res/ directory certificates and provisioning profiles

Important Notes

  • Always run yarn build before starting the application
  • The app uses Electron (see package.json for version)
  • MQTT communication is handled via mqttjs
  • All code changes should pass linting (yarn lint)
  • Node.js version requirement: >= 18
  • The project uses workspace-like structure with separate package.json files for app and backend