vue-network-dashboard
v0.2.8
Published
Universal network logger for Vue 3 applications - intercepts HTTP and WebSocket traffic
Maintainers
Readme
Universal network monitoring plugin for Vue 3. Intercepts all HTTP (Fetch / XHR), WebSocket, and Server-Sent Events (SSE) traffic, logs them in a unified format, automatically sanitizes sensitive data, and exposes reactive storage with rich statistics.
Table of Contents
- Features
- How It Works
- Installation
- Quick Start
- Vue Plugin Registration
- Nuxt 3 Module
- Usage
- Built-in UI Component
- Component Props
- Hotkey Configuration
- Pending Requests
- Mock Mode
- Mock Groups
- Mock Config Import / Export
- Session Compare
- Diff View
- Waterfall Timeline
- Grouping
- Virtual Scroll
- URL Highlight
- Copy as cURL
- Replay with Editing
- Mock from Log
- Mock Conditions
- Response Transform
- Network Throttling
- GraphQL Detection
- N+1 Detection
- Breakpoints
- OpenAPI Import
- HAR Import
- Filter Persistence
- WebSocket Message Filter
- Traffic Sparkline
- Configuration Reference
- Log Entry Structure
- Instance API
- Vue Router Integration
- Advanced Usage
- Security & Sanitization
- Statistics
- Architecture
- License
Features
| Feature | Description |
|---|---|
| Full Coverage | Intercepts Fetch API, XMLHttpRequest (XHR), WebSocket, and Server-Sent Events |
| Unified Format | All network events share one consistent log structure regardless of transport type |
| Security First | Auto-redacts sensitive headers, removes sensitive body fields, masks PII |
| Rich Metrics | Tracks request/response sizes, duration, data transfer volumes, per-method and per-status breakdowns |
| Vue 3 Native | Uses Vue ref for reactive storage — no Pinia or Vuex required |
| Zero Configuration | Works immediately after app.use() with sensible defaults |
| Highly Configurable | Extensive options for interceptor selection, URL filtering, sanitization, callbacks, and UI behaviour |
| Pending Requests | In-flight requests appear as live entries and update in place on completion — just like browser DevTools |
| Mock Groups | Organise mock rules into named groups — enable/disable a whole group at once, collapse/expand, rename inline, import/export as JSON |
| Session Compare | Load two HAR files side by side and see an instant diff — added, removed, and changed requests with per-field deltas |
| Mock Mode | Define URL/method rules to intercept requests and return custom responses without touching the backend |
| HAR Export | Export session as an HTTP Archive file — open directly in Chrome DevTools, Postman, or Charles Proxy |
| Waterfall Timeline | Visual bar chart of all requests on a shared time axis |
| Diff View | Select any two log entries to see a side-by-side diff of headers and body |
| Grouping | Collapse repeated requests to the same endpoint into a single row with a count badge |
| Body Search | Filter log entries by content of request or response body |
| Regex Search | Prefix any URL or body filter with regex: to match by regular expression |
| URL Highlight | Matched URL fragment is highlighted inline in the log list when an active URL filter is set |
| Debounced Filters | Text filters (URL, Body, Route) update with a 180 ms debounce; method/status/toggles apply instantly |
| Filter Persistence | Active filters are saved to sessionStorage and restored automatically on next page load |
| Route Context | Attach the current Vue Router route to every log entry; filter by route in the UI |
| Filtered Export | Export modal with format selector (JSON / CSV / HAR), protocol checkboxes, status checkboxes, and dual time-range slider |
| HAR Import | Load any .har file into the debugger to inspect a recorded session; live traffic is paused while a file is active |
| Copy as cURL | One-click copy of any HTTP request as a ready-to-run curl command (with headers and body) |
| Replay with Editing | Edit URL, method, headers, and body before re-sending any request — JSON body is validated inline |
| Mock from Log | One-click mock creation from any captured response — rule is pre-filled with URL, method, status, and body |
| Mock Conditions | Add fine-grained match conditions to a mock rule: query params, request headers, and body fields (all AND-combined) |
| Response Transform | Modify real responses on the fly without a full mock — override status, merge/delete JSON body fields, add headers |
| Network Throttling | Simulate slow connections with built-in presets (Fast 3G / Slow 3G / Offline-ish); delay applied per request via a getter, no interceptor restart needed |
| GraphQL Detection | Automatically detects GraphQL operations in POST requests; shows operation type, name, and variables in a dedicated section with a colour badge |
| N+1 Detection | Highlights duplicate requests to the same URL+method within a 5-second window with an orange ×N badge |
| Breakpoints | Pause any outgoing request before it is sent — inspect and edit URL, method, headers, and body — then release or cancel |
| OpenAPI Import | Load an OpenAPI 3.x or Swagger 2.x JSON spec to auto-generate a full set of mock rules from all paths |
| Virtual Scroll | Log list renders at most 100 rows; additional rows load incrementally on scroll via IntersectionObserver |
| WebSocket Message Filter | When the WS type filter is active, a toggle hides connection/open/close events and shows only message entries |
| Traffic Sparkline | Statistics tab shows a live SVG sparkline of request volume over time (5-second buckets, last ~3 min) |
| Fullscreen Mode | Expand the debugger panel to fill the entire viewport with a single click |
| Built-in Debugger UI | Draggable, resizable panel with filters, 3-tab detail view, stats, timeline, mock editor, and export |
| Nuxt 3 Module | First-class Nuxt integration with auto-registration and useNetworkDashboard() auto-import |
| Vue DevTools | Optional inspector tab and timeline layer in Vue DevTools (browser extension + vite-plugin-vue-devtools) |
| Sentry / OpenTelemetry | Ready-made adapters for error tracking and distributed tracing |
| TypeScript | Full type definitions included |
How It Works
The plugin replaces global browser APIs at the JavaScript layer before any application code runs. When a network call is made anywhere in your app (including third-party libraries), the plugin captures the event, formats it into a UnifiedLogEntry, and stores it reactively.
Your App Code
│
▼
Patched Globals
┌────────────────────────────────────┐
│ window.fetch (FetchInterceptor)│
│ XMLHttpRequest (XHRInterceptor) │
│ window.WebSocket (WSInterceptor) │
│ window.EventSource (SSEInterceptor)│
└────────────────────────────────────┘
│
▼
Formatter (HTTP / WebSocket / SSE)
│
▼
LogStore ──► Vue Reactive Refs ──► Your ComponentsInterception strategies:
| Transport | Strategy |
|---|---|
| Fetch | Wraps window.fetch globally; captures request/response including body, headers, timing |
| XHR | Overrides XMLHttpRequest.prototype.open, send, setRequestHeader; uses a WeakMap to associate per-request data without polluting instances |
| WebSocket | Replaces window.WebSocket constructor; wraps all event listeners and the send method to capture both directions |
| SSE | Replaces window.EventSource constructor; wraps event listeners to capture open, message, error, and named custom events |
The original native implementations are preserved internally and called through normally, so interception is completely transparent to the rest of the application.
Installation
npm install vue-network-dashboardQuick Start
// main.ts
import { createApp } from 'vue'
import App from './App.vue'
import NetworkDashboard from 'vue-network-dashboard'
const app = createApp(App)
app.use(NetworkDashboard, {
devOnly: true, // only active in development builds
maxLogs: 500,
})
app.mount('#app')That's it. All Fetch, XHR, WebSocket, and SSE calls are now captured.
Vue Plugin Registration
Pass an options object as the second argument to app.use():
app.use(NetworkDashboard, {
enabled: true,
maxLogs: 1000,
devOnly: false,
persistToStorage: false,
interceptors: {
fetch: true,
xhr: true,
websocket: true,
sse: true,
},
filters: {
excludeUrlPattern: /\/(health|metrics|favicon)/,
methods: ['GET', 'POST', 'PUT', 'DELETE'],
},
sanitization: {
sensitiveHeaders: ['authorization', 'x-api-key'],
sensitiveFields: ['password', 'token', 'secret'],
maskFields: ['email', 'phone'],
},
callbacks: {
onLog: (entry) => console.log('[net]', entry.url),
onError: (err) => console.error('[net error]', err),
},
// Configure the hotkey for showing/hiding the debugger panel
ui: {
hotkey: 'd',
hotkeyModifiers: { ctrl: true, shift: true }, // Ctrl+Shift+D
},
})Nuxt 3 Module
Install and add to modules:
// nuxt.config.ts
export default defineNuxtConfig({
modules: ['vue-network-dashboard/nuxt'],
networkDashboard: {
devOnly: true,
maxLogs: 500,
ui: { hotkey: 'd', hotkeyModifiers: { ctrl: true, shift: true } }
}
})<NetworkDebugger> and useNetworkDashboard() are auto-imported — no manual import required. The plugin runs client-side only, so SSR is unaffected.
Usage
Composition API
<script setup lang="ts">
import { useNetworkDashboard } from 'vue-network-dashboard'
const {
logs, // Ref<UnifiedLogEntry[]> — reactive log array
totalRequests, // Ref<number>
totalErrors, // Ref<number>
averageDuration, // Ref<number> — milliseconds
totalDataSent, // Ref<number> — bytes
totalDataReceived, // Ref<number> — bytes
clear, // () => void
enable, // () => void
disable, // () => void
isEnabled, // () => boolean
getStats, // () => NetworkStats
getStatsSummary, // () => string (human-readable)
getLogsByType, // (type: 'http'|'websocket'|'sse') => UnifiedLogEntry[]
getLogsByUrl, // (pattern: string | RegExp) => UnifiedLogEntry[]
getLogsByStatus, // ([min, max]: [number, number]) => UnifiedLogEntry[]
getLogsByMethod, // (method: string) => UnifiedLogEntry[]
getErrorLogs, // () => UnifiedLogEntry[]
queryLogs, // (filters: QueryFilters) => UnifiedLogEntry[]
subscribe, // (cb: (entry) => void) => () => void (returns unsubscribe)
export: exportLogs, // (format?: 'json' | 'csv') => string
} = useNetworkDashboard()
</script>
<template>
<div>
<p>Requests: {{ totalRequests }} | Errors: {{ totalErrors }} | Avg: {{ averageDuration }}ms</p>
<button @click="clear()">Clear</button>
<ul>
<li v-for="log in logs" :key="log.id">
{{ log.method }} {{ log.url }} — {{ log.http?.status }} ({{ log.duration }}ms)
</li>
</ul>
</div>
</template>Options API
<template>
<div>
<p>Total Requests: {{ $networkDashboard.totalRequests }}</p>
<p>Errors: {{ $networkDashboard.totalErrors }}</p>
<button @click="$networkDashboard.clear()">Clear Logs</button>
</div>
</template>
<script>
export default {
mounted() {
console.log(this.$networkDashboard.getStatsSummary())
this._unsubscribe = this.$networkDashboard.subscribe((entry) => {
if (entry.error.occurred) {
console.warn('Request failed:', entry.url)
}
})
},
beforeUnmount() {
this._unsubscribe?.()
}
}
</script>Without Vue (standalone)
You can use the core logger independently of Vue, for example in a testing environment or a non-Vue web app:
import { createNetworkDashboard } from 'vue-network-dashboard'
const logger = createNetworkDashboard({
maxLogs: 200,
interceptors: { fetch: true, xhr: false, websocket: false, sse: false },
})
logger.enable()
const stats = logger.getStats()
console.log(stats.totalRequests)
logger.disable()Built-in UI Component
The plugin ships a production-grade debugger panel. Add it once to your root layout and it appears as a floating overlay:
<!-- App.vue -->
<script setup>
import { NetworkDebugger } from 'vue-network-dashboard'
</script>
<template>
<RouterView />
<NetworkDebugger />
</template>The panel includes:
- Filter bar — filter by type (HTTP / WS / SSE), URL, body (both support
regex:pattern), HTTP method, status code, route, minimum duration, errors-only toggle, and (when WS type is active) a messages-only toggle - URL highlight — active URL filter highlights the matching fragment directly in the log list
- Filter persistence — filters are saved to
sessionStorageand restored on next load - Log list — virtual scroll (100 rows at a time); colour-coded entries with expandable detail view (Request / Response / Meta tabs); Meta tab shows the Vue Router route when available
- Copy as cURL — one-click copy of any HTTP request as a
curlcommand from the detail panel - Replay — re-send any HTTP request from the detail panel; the response appears as a new log entry
- Statistics panel — live traffic sparkline, request counts, error rate, average duration, data transfer, method/status distribution, slowest and largest requests
- HAR import — load a
.harfile to browse a recorded session; a banner indicates import mode with a dismiss button to return to live traffic - Export modal — format selector (JSON / CSV / HAR), filter by protocol and status codes, dual time-range slider showing
N / M logscount before confirming - Fullscreen mode — expand to full viewport; page scroll is blocked while active; exit with the button or
Escape - Pin — pin the panel so it stays open when clicking outside
- Drag & resize — drag the panel by its header; resize from any of the 8 edges and corners
- Toggle FAB — compact floating button with request count and error indicator when the panel is hidden
- Keyboard shortcut —
Ctrl+Shift+Dby default (configurable)
Pending Requests
In-flight requests appear immediately as a live row with an animated spinner. When the response arrives the entry updates in place — no duplicate rows, exactly like the browser Network tab.
Pending entries are excluded from HAR export and the waterfall timeline.
Mock Mode
Create rules in the UI (Mocks tab) or programmatically. The legacy flat API still works and proxies into a built-in default group:
const { addMock, removeMock } = useNetworkDashboard()
const rule = addMock({
name: 'Mock /api/users',
urlPattern: '/api/users',
method: 'GET',
enabled: true,
response: {
status: 200,
body: [{ id: 1, name: 'Alice' }],
delay: 200 // optional artificial latency in ms
}
})
removeMock(rule.id)Mocked responses are logged normally with a mock badge and metadata.mocked = true, so you can tell at a glance which entries were intercepted. Both Fetch and XHR are supported.
Mock Groups
Rules can be organised into named groups. Each group has its own enable/disable toggle — turning a group off suspends all its rules without losing their individual enabled states. Groups are collapsible; double-clicking a group name renames it inline.
const {
addMockGroup,
renameMockGroup,
toggleMockGroup,
removeMockGroup,
addMockToGroup,
updateMockInGroup,
removeMockFromGroup,
mockGroups, // Ref<readonly MockRulesGroup[]>
} = useNetworkDashboard()
// Create a group
const groupId = addMockGroup('Auth mocks')
// Add a rule to it
addMockToGroup(groupId, {
urlPattern: '/api/refresh',
method: 'POST',
response: {
status: 401,
body: { error: 'token_expired' },
delay: 300,
}
})
// Disable the whole group (individual rule states are preserved)
toggleMockGroup(groupId, false)
// Rename
renameMockGroup(groupId, 'Auth & Session')
// Remove
removeMockGroup(groupId)In the UI, clicking + in a group header opens an inline add/edit form directly within that group. Groups are persisted to localStorage under vue-network-dashboard:mockGroups when persistToStorage is enabled.
Mock Config Import / Export
The Mocks toolbar has Import and Export buttons.
Export serialises all groups and rules to a JSON file (mock-config.json):
{
"version": 1,
"groups": [
{
"id": "grp_a1b2c3",
"name": "Auth mocks",
"enabled": true,
"rules": [
{
"id": "rule_x7y8z9",
"urlPattern": "/api/refresh",
"method": "POST",
"enabled": true,
"response": { "status": 401, "body": { "error": "token_expired" }, "delay": 300 }
}
]
}
]
}Import reads the file and calls replaceMockGroups(), which replaces the entire configuration atomically — no ID conflicts, predictable behaviour.
// Programmatic replacement
const { replaceMockGroups } = useNetworkDashboard()
replaceMockGroups(parsedGroups)Typical use-cases: switching between pre-built scenario files (happy path, auth errors, backend down), sharing a mock set with teammates, or loading a prepared configuration for a demo without touching the server.
Session Compare
The Compare tab lets you load two HAR files and see an instant diff of the recorded sessions.
Session A (before.har) Session B (after.har)
──────────────────────────────────────────────────────
= GET /api/users 200 42ms = GET /api/users 200 87ms +45ms
~ POST /api/orders 200 130ms ~ POST /api/orders 500 ← changed
+ GET /api/products + GET /api/products 200 55ms added
- DELETE /api/cart 204 18ms removedMatching is done on the composite key METHOD URL. For entries present in both sessions the tool compares HTTP status, duration, and response size — a difference greater than 20 % is flagged as changed.
Filter chips above the list let you show only Added, Removed, Changed, or Unchanged entries. Each side shows the session filename and entry count in the column header.
Both drag-and-drop and click-to-select file loading are supported. HAR files exported by the plugin itself are accepted directly.
Diff View
Click the Group button in the header to enter diff mode. Every log row gains a diff-select button. Select exactly two entries — the diff panel opens automatically above the log list and shows:
- Changed request and response headers
- Unified line diff of request body
- Unified line diff of response body
Waterfall Timeline
Switch to the Timeline tab to see all completed requests as horizontal bars on a shared time axis. Bars are colour-coded by result (green = 2xx, yellow = 4xx, red = 5xx, purple = WebSocket, teal = SSE). Hover a bar to see the exact duration.
Grouping
Toggle the Group button in the header to collapse all requests to the same URL+method into a single row showing the total count. Expand a group to see individual entries inside it.
Virtual Scroll
The log list renders at most 100 rows at a time. As you scroll toward the bottom, the next 50 rows load automatically via an IntersectionObserver sentinel element. Switching filters or clearing logs resets the counter back to 100. This keeps DOM size bounded even with thousands of requests captured.
URL Highlight
While a URL filter is active, the matched substring (or regex capture) is highlighted directly in the log list row using an amber accent. Regex mode (regex:pattern) is also supported.
Copy as cURL
Open any HTTP request's detail view and click cURL to copy the full request as a curl command to the clipboard. The command includes the method, all non-redundant headers, and the request body. Common internal headers (host, content-length, transfer-encoding, connection, keep-alive) are omitted automatically.
Replay with Editing
Click Replay in an HTTP request's detail view to open the Edit & Replay modal. Before sending you can change the URL, HTTP method, add or remove individual headers, and edit the JSON body. Invalid JSON is highlighted with a warning. The response is captured by the interceptor and appears as a new log entry, so you can compare it with the original using Diff view.
Mock from Log
In the detail view of any captured HTTP request, click Mock to create a mock rule pre-filled with the URL path, method, status code, and response body from that log entry. The Mocks tab opens automatically with the new rule ready to edit or enable.
Mock Conditions
Mock rules can be narrowed with additional match conditions beyond URL and method. Add conditions in the rule editor under the Conditions section:
interface MockRule {
// ...
conditions?: {
queryParams?: Record<string, string> // URL ?key=value pairs (all must match)
headers?: Record<string, string> // Request header values (all must match)
bodyFields?: Record<string, unknown> // JSON body field values (all must match)
}
}All conditions within a rule are AND-combined. For example, route the same POST /api/search to different mock responses depending on a field in the request body:
// Returns search results for user queries
dashboard.addMock({
urlPattern: '/api/search', method: 'POST',
conditions: { bodyFields: { type: 'user' } },
response: { status: 200, body: { items: [{ id: 1, name: 'Alice' }] } },
})
// Returns empty list for product queries
dashboard.addMock({
urlPattern: '/api/search', method: 'POST',
conditions: { bodyFields: { type: 'product' } },
response: { status: 200, body: { items: [] } },
})Response Transform
Transform mode lets you modify a real server response without replacing it entirely. The request goes to the actual backend; after the response arrives the interceptor applies the declared transforms and returns the modified Response to your app. The log entry is marked mock to indicate the response was altered.
interface MockRule {
mode?: 'mock' | 'transform' // default: 'mock'
transform?: {
status?: number // Override HTTP status code
headers?: Record<string, string> // Add or overwrite response headers
bodyMerge?: Record<string, unknown> // Deep-merge fields into JSON body
bodyDelete?: string[] // Remove fields from JSON body
}
}Example — inject an isAdmin flag and remove an internal field:
dashboard.addMock({
urlPattern: '/api/me',
method: 'GET',
mode: 'transform',
enabled: true,
response: { status: 200 }, // required field; ignored in transform mode
transform: {
bodyMerge: { isAdmin: true, beta: true },
bodyDelete: ['internalId'],
},
})In the UI, switch between Mock and Transform using the mode toggle in the rule editor — the visible form fields update accordingly.
Network Throttling
Simulate slow or unreliable connections using the throttle selector in the Logs toolbar:
| Preset | Added latency | |-------------|---------------| | No throttle | 0 ms | | Fast 3G | 400 ms | | Slow 3G | 2 000 ms | | Offline-ish | 5 000 ms |
The delay is applied per request via a getter that the interceptor reads immediately before each call — switching presets takes effect on the next request without restarting or re-registering interceptors.
GraphQL Detection
When a POST request contains a JSON body with a query string field, the dashboard automatically treats it as a GraphQL operation. A purple badge with the operation name appears in the log list row:
[POST] /graphql [query GetUser] 200 142msThe expanded detail view gains a GraphQL section showing:
- Operation type (
query,mutation, orsubscription) - Operation name (from the query string or
operationNamefield) - Variables (parsed from
body.variables)
No external GraphQL library is required.
N+1 Detection
If more than one request with the same HTTP method and URL appears within a 5-second window, an orange ×N badge is shown on each matching row in the log list. The count updates reactively as requests arrive or leave the window — no timers or polling are used.
This makes it easy to spot N+1 patterns such as a list component that fires one detail request per item.
Breakpoints
Breakpoints pause outgoing requests before they are sent, letting you inspect and modify every field — then decide whether to release or cancel.
┌─────────────────────────────────────────────────┐
│ ● POST /api/checkout 14:32:07 │
│ │
│ URL [https://api.example/checkout] │
│ Method [POST ] │
│ Headers Authorization: Bearer eyJ... │
│ Content-Type: application/json │
│ Body { "items": [{ "id": 1 }] } │
│ │
│ [Cancel request] [▶ Release] │
└─────────────────────────────────────────────────┘Adding rules programmatically:
const dashboard = useNetworkDashboard()
dashboard.addBreakpointRule({
urlPattern: '/api/checkout', // substring or /regex/
method: 'POST', // omit to match any method
name: 'Pause checkout',
enabled: true,
})Managing paused requests:
// Release with optional edits (url, method, headers, body)
dashboard.releaseBreakpoint(id, { url, method, headers, body })
// Cancel — throws AbortError in the calling code
dashboard.cancelBreakpoint(id)Reactive state:
const { breakpointRules, activeBreakpoints } = useNetworkDashboard()
// breakpointRules — Ref<BreakpointRule[]>
// activeBreakpoints — Ref<ActiveBreakpoint[]> (currently paused)The number of paused requests is shown as a badge on the Breakpoints tab. Rules can be added, edited (inline, replacing the row), toggled, and deleted from the UI.
OpenAPI Import
Click OpenAPI in the Mocks toolbar to load an OpenAPI 3.x or Swagger 2.x JSON spec file. The parser generates one mock rule per path + method combination, builds an example response body from the schema, and adds all rules to a new group named after info.title.
// Example: load the Petstore spec → creates ~18 mock rules in one click
// All rules are added to a group: "Swagger Petstore"The generated rules use mode: 'mock' with status 200 and a body derived from the first successful response schema. $ref references are resolved automatically. No external dependencies — the parser is ~120 lines of TypeScript.
After import, rules can be enabled individually, edited, or exported as a JSON config file like any other mock group.
HAR Import
Click Import in the panel header to load a .har file. The imported entries are displayed using the same filters and detail view as live traffic. A banner at the top of the panel shows the file name and entry count; click × to dismiss and return to live mode.
┌─────────────────────────────────────────┐
│ ↑ Imported session my-session.har │
│ 247 entries × │
└─────────────────────────────────────────┘Filter Persistence
The active filter state (URL, body, method, status, type, duration threshold, and toggles) is automatically saved to sessionStorage under the key vue-network-dashboard:filters. On the next page load the filters are restored without any configuration required. Clicking Reset clears both the UI state and the stored value.
WebSocket Message Filter
When the WS type tab is active, a Messages only toggle appears in the filter bar. Enabling it hides connection, open, error, and close events and shows only message entries — useful for high-frequency connections where lifecycle noise would otherwise bury the actual data frames.
Traffic Sparkline
The Statistics tab displays a live SVG sparkline at the top of the panel showing request volume over time. Requests are bucketed into 5-second intervals across the last ~3 minutes (40 buckets). The chart redraws automatically when the panel is resized or switched to fullscreen mode. The peak value (requests per bucket) is shown in the section header.
Component Props
| Prop | Type | Default | Description |
|---|---|---|---|
| defaultVisible | boolean | false | Whether the panel is open on mount |
| defaultPinned | boolean | false | Whether the panel is pinned on mount |
| hotkey | string | 'd' | Key character for the toggle shortcut |
| hotkeyModifiers | object | { ctrl: true, shift: true } | Modifier keys: ctrl, alt, shift, meta |
Props take effect as component-level defaults. If you configure hotkeys via the plugin's ui option (see below), you do not need to set these props.
Hotkey Configuration
The toggle hotkey can be configured at two levels. Plugin level (recommended) — set once at registration, applies automatically to all <NetworkDebugger> instances:
app.use(NetworkDashboard, {
ui: {
hotkey: 'd',
hotkeyModifiers: { ctrl: true, shift: true }, // Ctrl+Shift+D
},
})Component level — pass props directly to the component (overrides plugin-level config):
<NetworkDebugger
hotkey="d"
:hotkeyModifiers="{ ctrl: true, shift: true }"
/>The panel ignores hotkeys when focus is inside an <input>, <textarea>, or a contenteditable element. When pinned, the hotkey is also disabled to prevent accidental closes.
Programmatic Control
The component exposes methods via ref:
<script setup>
import { ref } from 'vue'
import { NetworkDebugger } from 'vue-network-dashboard'
const debugger = ref()
</script>
<template>
<NetworkDebugger ref="debugger" />
<button @click="debugger.show()">Open</button>
<button @click="debugger.hide()">Close</button>
<button @click="debugger.toggle()">Toggle</button>
<button @click="debugger.clear()">Clear logs</button>
<button @click="debugger.export('json')">Export JSON</button>
</template>Configuration Reference
Top-level options
| Option | Type | Default | Description |
|---|---|---|---|
| enabled | boolean | true | Master on/off switch |
| maxLogs | number | 1000 | Maximum entries kept in memory. Oldest are evicted when the limit is reached |
| devOnly | boolean | false | When true, the plugin is a no-op in production (import.meta.env.PROD) |
| persistToStorage | boolean | false | Persist logs to localStorage and restore them on page reload |
| router | Router | — | Vue Router instance. Required for route context enrichment (see Vue Router Integration) |
| enrichWithRoute | boolean | false | Attach router.currentRoute.value.fullPath to every log entry as entry.route |
interceptors
| Option | Type | Default | Description |
|---|---|---|---|
| interceptors.fetch | boolean | true | Intercept window.fetch |
| interceptors.xhr | boolean | true | Intercept XMLHttpRequest |
| interceptors.websocket | boolean | true | Intercept WebSocket |
| interceptors.sse | boolean | true | Intercept EventSource (SSE) |
filters
Pre-filter events before they are stored. Entries that do not match are discarded.
| Option | Type | Description |
|---|---|---|
| filters.urlPattern | RegExp | Only store entries whose URL matches |
| filters.excludeUrlPattern | RegExp | Discard entries whose URL matches |
| filters.methods | string[] | Only store entries with these HTTP methods (e.g. ['GET','POST']) |
| filters.statusCodes | number[] | Only store entries with these exact status codes |
| filters.bodySizeThreshold | number | Only store entries whose body is at least this many bytes |
sanitization
| Option | Type | Description |
|---|---|---|
| sanitization.sensitiveHeaders | string[] | Header names to redact entirely ([REDACTED]). Merged with built-in defaults |
| sanitization.sensitiveFields | string[] | Body field names to remove entirely. Merged with built-in defaults |
| sanitization.maskFields | string[] | Body field names to partially mask (e.g. us***@example.com). Merged with built-in defaults |
metrics
| Option | Type | Default | Description |
|---|---|---|---|
| metrics.calculateTTFB | boolean | true | Track Time To First Byte for HTTP requests |
| metrics.trackRetries | boolean | true | Increment metadata.retryCount on repeated requests to the same URL |
callbacks
| Option | Type | Description |
|---|---|---|
| callbacks.onLog | (entry: UnifiedLogEntry) => void | Called after every log entry is stored |
| callbacks.onError | (error: Error) => void | Called when an internal plugin error occurs |
| callbacks.onFlush | (logs: UnifiedLogEntry[]) => void | Called after clear() with the entries that were removed |
ui
Controls the built-in <NetworkDebugger> panel. These values act as defaults and can be overridden by component props.
| Option | Type | Default | Description |
|---|---|---|---|
| ui.hotkey | string | 'd' | Key character for the toggle shortcut |
| ui.hotkeyModifiers.ctrl | boolean | true | Require Ctrl key |
| ui.hotkeyModifiers.alt | boolean | — | Require Alt key |
| ui.hotkeyModifiers.shift | boolean | true | Require Shift key |
| ui.hotkeyModifiers.meta | boolean | — | Require Meta (⌘ / Win) key |
Log Entry Structure
Every captured event, regardless of transport type, is stored as a UnifiedLogEntry:
interface UnifiedLogEntry {
id: string // Unique UUID-like identifier
type: 'http' | 'websocket' | 'sse'
// Timing
startTime: number // Unix timestamp at request start (ms)
endTime: number | null // Unix timestamp at completion
duration: number | null // endTime - startTime, milliseconds
// Request identity
url: string
method: string // HTTP verb, or WebSocket/SSE event type
// HTTP-specific (null for WebSocket / SSE)
http: {
status: number | null // e.g. 200, 404, 500
statusText: string | null // e.g. "OK", "Not Found"
protocol: string | null // e.g. "HTTP/1.1", "HTTP/2"
} | null
// WebSocket-specific (null for HTTP / SSE)
websocket: {
readyState: number // 0 CONNECTING | 1 OPEN | 2 CLOSING | 3 CLOSED
eventType: 'connection' | 'open' | 'message' | 'error' | 'close'
direction: 'incoming' | 'outgoing' | null
code: number | null // Close code (1000 = normal, 1001 = going away, …)
reason: string | null
wasClean: boolean | null
} | null
// SSE-specific (null for HTTP / WebSocket)
sse: {
readyState: number // 0 CONNECTING | 1 OPEN | 2 CLOSED
eventType: string | null // Named event type, or null for default "message"
lastEventId: string | null
} | null
// Headers
requestHeaders: Record<string, string> // Sanitized request headers
responseHeaders: Record<string, string> // Sanitized response headers
// Bodies
request: {
body: any | null // Parsed body (object, string, FormData, etc.)
bodyRaw: string | null // Raw serialized body string
bodySize: number | null // Size in bytes
bodyType: string | null // MIME type derived from Content-Type
}
response: {
body: any | null
bodyRaw: string | null
bodySize: number | null
bodyType: string | null
}
// Errors
error: {
occurred: boolean
message: string | null
name: string | null // e.g. "TypeError", "NetworkError"
stack: string | null
}
// Metadata
metadata: {
clientType: 'fetch' | 'xhr' | 'websocket' | 'eventsource'
redirected: boolean
retryCount: number
timestamp: string // ISO 8601 timestamp at log creation
}
// Vue Router route path at the moment the request was initiated.
// Populated only when `enrichWithRoute: true` is set in plugin options.
route?: string
}Instance API
All methods below are available on the object returned by useNetworkDashboard(), on this.$networkDashboard in the Options API, or on the instance returned by createNetworkDashboard().
Reactive state
| Property | Type | Description |
|---|---|---|
| logs | Ref<UnifiedLogEntry[]> | All stored log entries |
| totalRequests | Ref<number> | Total number of entries |
| totalErrors | Ref<number> | Entries where error.occurred === true or http.status >= 400 |
| averageDuration | Ref<number> | Mean duration in milliseconds |
| totalDataSent | Ref<number> | Sum of request.bodySize across all entries |
| totalDataReceived | Ref<number> | Sum of response.bodySize across all entries |
| mockGroups | Ref<readonly MockRulesGroup[]> | All mock groups with their rules |
| breakpointRules | Ref<BreakpointRule[]> | All configured breakpoint rules |
| activeBreakpoints | Ref<ActiveBreakpoint[]> | Requests currently paused at a breakpoint |
Control methods
enable() // Start capturing
disable() // Stop capturing (existing logs are preserved)
isEnabled() // Returns boolean
clear() // Remove all stored logs (triggers onFlush callback)Query methods
// Filter by transport type
getLogsByType('http' | 'websocket' | 'sse'): UnifiedLogEntry[]
// Filter by URL string or regex
getLogsByUrl('/api/users'): UnifiedLogEntry[]
getLogsByUrl(/^https:\/\/api\./): UnifiedLogEntry[]
// Filter by HTTP status range
getLogsByStatus([400, 599]): UnifiedLogEntry[] // all 4xx and 5xx
// Filter by HTTP method
getLogsByMethod('POST'): UnifiedLogEntry[]
// Only failed entries
getErrorLogs(): UnifiedLogEntry[]
// Composable query with multiple filters at once
queryLogs({
type: 'http',
method: 'GET',
url: /\/api\//,
minDuration: 500,
hasError: false,
}): UnifiedLogEntry[]Export
export() // JSON string (default)
export('json') // JSON string
export('csv') // CSV string with header rowStatistics
getStats(): NetworkStats // Full statistics object (see Statistics section)
getStatsSummary(): string // Multi-line human-readable textSubscription
const unsubscribe = subscribe((entry: UnifiedLogEntry) => {
// called synchronously after each log entry is stored
})
unsubscribe() // stop receiving eventsMock methods
// Legacy flat API (proxied into the default group)
addMock(rule): MockRule
removeMock(id: string): void
getMocks(): MockRule[]
// Group management
addMockGroup(name: string): string // returns groupId
renameMockGroup(groupId: string, name: string): void
toggleMockGroup(groupId: string, enabled: boolean): void
removeMockGroup(groupId: string): void
// Rules within a group
addMockToGroup(groupId: string, rule: Partial<MockRule>): MockRule
updateMockInGroup(groupId: string, ruleId: string, patch: Partial<MockRule>): void
removeMockFromGroup(groupId: string, ruleId: string): void
// Bulk replace (used by import)
replaceMockGroups(groups: MockRulesGroup[]): voidThrottle methods
setThrottle(preset: 'none' | 'fast3g' | 'slow3g' | 'offline'): void
getThrottleDelay(): number // current delay in millisecondsBreakpoint methods
// Rule management
addBreakpointRule(rule: Omit<BreakpointRule, 'id'>): BreakpointRule
removeBreakpointRule(id: string): void
updateBreakpointRule(id: string, patch: Partial<BreakpointRule>): void
getBreakpointRules(): BreakpointRule[]
// Controlling paused requests
releaseBreakpoint(id: string, edits: BreakpointEdits): void
cancelBreakpoint(id: string): void // throws AbortError in the caller
getActiveBreakpoints(): ActiveBreakpoint[]
// Subscribe to changes in the paused set
onActiveBreakpointsChange(cb: (list: ActiveBreakpoint[]) => void): () => voidTypes:
interface BreakpointRule {
id: string
urlPattern: string | RegExp
method?: string // e.g. 'POST'; omit to match any method
name?: string
enabled: boolean
}
interface ActiveBreakpoint {
id: string
url: string
method: string
requestHeaders: Record<string, string>
requestBody: unknown
timestamp: number
}
interface BreakpointEdits {
url: string
method: string
headers: Record<string, string>
body: string | null
}Vue Router Integration
Enable route context to automatically tag every log entry with the Vue Router route path that was active when the request was made. This makes it easy to trace which page triggered a slow request or an error without manually correlating by timestamp.
Vue 3 SPA
// main.ts
import { createApp } from 'vue'
import { createRouter, createWebHistory } from 'vue-router'
import NetworkDashboard from 'vue-network-dashboard'
import App from './App.vue'
const router = createRouter({ history: createWebHistory(), routes })
const app = createApp(App)
app.use(router)
app.use(NetworkDashboard, {
router,
enrichWithRoute: true,
})
app.mount('#app')Nuxt 3
// plugins/network-dashboard.client.ts
export default defineNuxtPlugin((nuxtApp) => {
nuxtApp.vueApp.use(NetworkDashboardPlugin, {
router: nuxtApp.$router,
enrichWithRoute: true,
})
})What it adds
When enrichWithRoute: true is configured, each UnifiedLogEntry gains a route field:
entry.route // e.g. "/dashboard/users/42"In the debugger UI:
- A Route filter input appears in the filter bar (only when at least one log carries a route). Supports
regex:prefix. - The Meta tab of each expanded log entry shows a highlighted Route chip.
- The Export modal counts are route-aware — filtering by route reduces the exported set.
The feature has no runtime dependency on vue-router — the plugin accepts any object that satisfies the minimal RouterInstance interface (currentRoute + afterEach).
Advanced Usage
Filtering Logs
Use queryLogs for complex, multi-criteria filtering:
const { queryLogs } = useNetworkDashboard()
const slowPosts = queryLogs({
type: 'http',
method: 'POST',
url: /\/api\//,
minDuration: 1000,
})For reactive in-component filtering, use the bundled useLogFilter composable:
import { useLogFilter } from 'vue-network-dashboard'
const { filters, filteredLogs, resetFilters } = useLogFilter()
filters.value.url = 'api/users'
filters.value.type = 'http'
filters.value.hasError = true
// filteredLogs is a computed ref that reacts to both filters and the live log storeSubscribing to New Logs
const { subscribe } = useNetworkDashboard()
const stop = subscribe((entry) => {
if (entry.error.occurred || (entry.http?.status ?? 0) >= 500) {
myMonitoring.capture(entry)
}
})
onUnmounted(stop)Exporting Logs
const dashboard = useNetworkDashboard()
function downloadJson() {
const json = dashboard.export('json')
const blob = new Blob([json], { type: 'application/json' })
const url = URL.createObjectURL(blob)
Object.assign(document.createElement('a'), { href: url, download: 'network-logs.json' }).click()
}
function downloadCsv() {
const csv = dashboard.export('csv')
const blob = new Blob([csv], { type: 'text/csv' })
// ...
}Logs can also be exported directly from the panel UI via the Export button in the header.
Sentry Integration
import * as Sentry from '@sentry/vue'
import { createSentryAdapter } from 'vue-network-dashboard'
app.use(NetworkDashboardPlugin, {
callbacks: createSentryAdapter(Sentry, {
errorStatusThreshold: 500, // send Sentry event for 5xx (default)
includeBodies: false // omit bodies from breadcrumbs (default)
})
})Every request becomes a Sentry breadcrumb. HTTP 5xx and network errors are also sent as Sentry events via captureMessage.
OpenTelemetry Integration
import { trace } from '@opentelemetry/api'
import { createOpenTelemetryAdapter } from 'vue-network-dashboard'
const tracer = trace.getTracer('my-app')
app.use(NetworkDashboardPlugin, {
callbacks: createOpenTelemetryAdapter(tracer, {
httpOnly: true, // skip WebSocket/SSE (default)
includeBodySize: true // add body size attributes (default)
})
})Creates one OTel span per HTTP request with semantic HTTP attributes (http.request.method, http.response.status_code, url.full, etc.) per semconv 1.23.
Vue DevTools Integration
Adds a Network inspector tab and timeline layer to Vue DevTools (browser extension ≥ 6.5 and vite-plugin-vue-devtools ≥ 7.x). Requires @vue/devtools-api as a peer dependency.
import { setupDevtools } from 'vue-network-dashboard'
import { useNetworkDashboard } from 'vue-network-dashboard'
if (import.meta.env.DEV) {
app.use(NetworkDashboardPlugin)
const dashboard = useNetworkDashboard()
setupDevtools(app, dashboard)
}The inspector shows every log entry as a tree node with full request/response state. The timeline layer records a marker for each completed HTTP request.
Callbacks
React to logs outside of Vue components — useful for integrating with error tracking or analytics:
app.use(NetworkDashboard, {
callbacks: {
onLog(entry) {
analytics.trackEvent('network_request', {
url: entry.url,
method: entry.method,
status: entry.http?.status,
duration: entry.duration,
})
},
onError(err) {
Sentry.captureException(err)
},
onFlush(cleared) {
console.log(`Flushed ${cleared.length} log entries`)
},
},
})Security & Sanitization
Sanitization runs automatically before each entry is written to the store. Custom lists are merged with the built-in defaults, so you never lose the baseline protection.
Header redaction
The following request and response headers are replaced with [REDACTED] by default:
authorization, cookie, x-api-key, api-key, x-auth-token, bearer, token, password, secret, x-secret, x-token
Body field removal
The following fields are deleted from JSON request/response bodies:
password, token, secret, apiKey, api_key, accessToken, access_token, refreshToken, refresh_token, authorization, creditCard, credit_card, cvv, ssn, socialSecurity
Field masking (PII)
The following fields are partially masked rather than removed:
email, phone, phoneNumber, phone_number, address, firstName, first_name, lastName, last_name
Examples:
[email protected]→us***@example.com1234567890→12***890John→Jo***n
Customising sanitization
app.use(NetworkDashboard, {
sanitization: {
sensitiveHeaders: ['x-custom-token'], // added on top of defaults
sensitiveFields: ['pin', 'securityAnswer'], // added on top of defaults
maskFields: ['nationalId'], // added on top of defaults
},
})Statistics
getStats() returns a NetworkStats object:
interface NetworkStats {
totalRequests: number
totalErrors: number
averageDuration: number // Mean response time in ms
totalDataSent: number // Total request body bytes
totalDataReceived: number // Total response body bytes
requestsByMethod: Record<string, number> // { GET: 42, POST: 17, … }
requestsByStatus: Record<string, number> // { '200': 55, '404': 3, '500': 1, … }
slowestRequests: UnifiedLogEntry[] // Top 10 by duration (descending)
largestRequests: UnifiedLogEntry[] // Top 10 by total body size (descending)
sseEventCount: number // Total SSE events captured
}getStatsSummary() returns the same data as a formatted multi-line string, useful for logging to the console.
Architecture
vue-network-dashboard/
├── src/
│ ├── core/
│ │ ├── NetworkDashboard.ts # Orchestrator — interceptors, mock registry, breakpoints, lifecycle
│ │ ├── formatters.ts # HTTPFormatter, WebSocketFormatter, SSEFormatter
│ │ ├── openApiParser.ts # Parses OpenAPI 3.x / Swagger 2.x JSON → MockRule[]
│ │ └── types.ts # UnifiedLogEntry, MockRule, BreakpointRule, NetworkStats, …
│ ├── interceptors/
│ │ ├── fetchInterceptor.ts # Patches window.fetch (pending state + mock support)
│ │ ├── xhrInterceptor.ts # Patches XMLHttpRequest prototype (WeakMap + mock support)
│ │ ├── websocketInterceptor.ts # Replaces window.WebSocket
│ │ └── sseInterceptor.ts # Replaces window.EventSource
│ ├── store/
│ │ └── logStore.ts # Vue reactive storage — addLog, updateLog, HAR/JSON/CSV export
│ ├── plugins/
│ │ └── vuePlugin.ts # Vue 3 plugin + useNetworkDashboard composable + mock API
│ ├── adapters/
│ │ ├── sentry.ts # createSentryAdapter — breadcrumbs + Sentry events
│ │ └── opentelemetry.ts # createOpenTelemetryAdapter — OTel spans per request
│ ├── utils/
│ │ ├── sanitizer.ts # Header redaction, field removal, PII masking
│ │ ├── helpers.ts # generateId, formatBytes, getContentType
│ │ └── sizeCalculator.ts # calculateSize, getDataType, safeStringify
│ ├── view/
│ │ ├── components/
│ │ │ ├── NetworkDebugger.vue # Main draggable panel (logs / stats / timeline / mocks tabs)
│ │ │ ├── LogEntry.vue # Single row — pending state, mocked badge, diff select
│ │ │ ├── FilterBar.vue # Type tabs, URL, body, method, status, duration filters
│ │ │ ├── StatsPanel.vue # Live statistics with distribution bars
│ │ │ ├── MockPanel.vue # Mock rule editor — groups, conditions, transform, OpenAPI import
│ │ │ ├── BreakpointPanel.vue # Breakpoint rules + paused-request cards with editable fields
│ │ │ ├── ReplayModal.vue # Edit & Replay modal — URL, method, headers, body
│ │ │ ├── SessionComparePanel.vue # HAR diff view — two sessions side by side
│ │ │ ├── NetworkTimeline.vue # Waterfall bar chart
│ │ │ └── DiffPanel.vue # LCS-based header and body diff between two log entries
│ │ ├── composables/
│ │ │ ├── useLogFilter.ts # Reactive filter state (url, body, method, status, …)
│ │ │ └── useHotkey.ts # Keyboard shortcut binding helper
│ │ └── styles/
│ │ ├── variables.scss # Design tokens (colours, spacing, typography)
│ │ └── debugger.scss # All component styles
│ ├── devtools.ts # setupDevtools() — Vue DevTools inspector + timeline layer
│ ├── nuxt.ts # Nuxt 3 module (defineNuxtModule)
│ └── runtime/
│ └── nuxt-plugin.ts # Nuxt client plugin — registered automatically by nuxt.ts
└── demo/ # Demo app (Vite + Vue 3)The plugin has no runtime dependencies besides Vue 3. It relies only on standard browser APIs (window.fetch, XMLHttpRequest, WebSocket, EventSource).
License
MIT
Author
Danil Lisin Vladimirovich aka Macrulez
GitHub: macrulezru · Website: macrulez.ru/en
Bugs and questions — issues
💖 Support the project
Open source takes time and effort. If my work saves you time or brings value, consider supporting further development.
Thank you for being part of this journey. ❤️
