webtap-tool 0.1.2__tar.gz → 0.1.3__tar.gz

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.

Potentially problematic release.


This version of webtap-tool might be problematic. Click here for more details.

Files changed (50) hide show
  1. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/CHANGELOG.md +24 -0
  2. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/PKG-INFO +1 -1
  3. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/extension/popup.html +9 -0
  4. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/extension/popup.js +28 -0
  5. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/pyproject.toml +1 -1
  6. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/__init__.py +6 -2
  7. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/api.py +71 -3
  8. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/app.py +3 -0
  9. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/query.py +2 -1
  10. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/launch.py +3 -17
  11. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/body.py +1 -1
  12. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/console.py +1 -1
  13. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/fetch.py +1 -1
  14. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/network.py +2 -2
  15. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/setup.py +20 -10
  16. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/.gitignore +0 -0
  17. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/ARCHITECTURE.md +0 -0
  18. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/README.md +0 -0
  19. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/data/filters.json +0 -0
  20. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/extension/manifest.json +0 -0
  21. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/llms.txt +0 -0
  22. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/VISION.md +0 -0
  23. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/README.md +0 -0
  24. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/__init__.py +0 -0
  25. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/schema/README.md +0 -0
  26. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/schema/cdp_protocol.json +0 -0
  27. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/schema/cdp_version.json +0 -0
  28. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/cdp/session.py +0 -0
  29. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/DEVELOPER_GUIDE.md +0 -0
  30. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/TIPS.md +0 -0
  31. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/__init__.py +0 -0
  32. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/_builders.py +0 -0
  33. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/_errors.py +0 -0
  34. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/_tips.py +0 -0
  35. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/_utils.py +0 -0
  36. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/body.py +0 -0
  37. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/connection.py +0 -0
  38. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/console.py +0 -0
  39. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/events.py +0 -0
  40. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/fetch.py +0 -0
  41. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/filters.py +0 -0
  42. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/inspect.py +0 -0
  43. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/javascript.py +0 -0
  44. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/navigation.py +0 -0
  45. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/network.py +0 -0
  46. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/commands/setup.py +0 -0
  47. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/filters.py +0 -0
  48. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/README.md +0 -0
  49. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/__init__.py +0 -0
  50. {webtap_tool-0.1.2 → webtap_tool-0.1.3}/src/webtap/services/main.py +0 -0
@@ -15,6 +15,30 @@ and this project adheres to [Semantic Versioning](https://semver.org/spec/v2.0.0
15
15
 
16
16
  ### Removed
17
17
 
18
+ ## [0.1.3] - 2025-09-05
19
+
20
+ ### Added
21
+ - Multi-instance WebTap support with first-come-first-serve port management
22
+ - `/instance` endpoint to show WebTap instance information (PID, connected page, events)
23
+ - `/release` endpoint for graceful API port handoff between instances
24
+ - Chrome extension "Switch WebTap Instance" button for managing multiple instances
25
+ - Instance status display in extension popup showing PID and event count
26
+ - Automatic reconnection in extension after instance switching
27
+
28
+ ### Changed
29
+ - API server now checks port availability before starting (port 8765)
30
+ - Chrome profile launch strategy uses bindfs mounting instead of symlinks
31
+ - Service docstrings simplified (removed redundant "Internal service for" prefixes)
32
+ - Added `api_thread` tracking to WebTapState for proper thread lifecycle management
33
+
34
+ ### Fixed
35
+ - SQL numeric comparisons in query builder now use string comparison instead of CAST
36
+ - Type annotations improved with proper union types (`threading.Thread | None`)
37
+ - Network service HTTP status filtering uses string comparison to prevent SQL errors
38
+ - Extension connection handling during instance transitions
39
+
40
+ ### Removed
41
+
18
42
  ## [0.1.2] - 2025-09-05
19
43
 
20
44
  ### Added
@@ -1,6 +1,6 @@
1
1
  Metadata-Version: 2.4
2
2
  Name: webtap-tool
3
- Version: 0.1.2
3
+ Version: 0.1.3
4
4
  Summary: Terminal-based web page inspector for AI debugging sessions
5
5
  Author-email: Fredrik Angelsen <fredrikangelsen@gmail.com>
6
6
  Classifier: Development Status :: 3 - Alpha
@@ -150,6 +150,15 @@
150
150
 
151
151
  <div class="divider"></div>
152
152
 
153
+ <button
154
+ id="switchInstance"
155
+ style="width: 100%; font-size: 11px; margin-bottom: 8px"
156
+ >
157
+ Switch WebTap Instance
158
+ </button>
159
+
160
+ <div class="divider"></div>
161
+
153
162
  <div class="filter-section">
154
163
  <div
155
164
  style="
@@ -25,6 +25,13 @@ async function loadPages() {
25
25
  return;
26
26
  }
27
27
 
28
+ // Also fetch instance info if available
29
+ const instanceInfo = await api("/instance");
30
+ if (instanceInfo && !instanceInfo.error) {
31
+ document.getElementById("switchInstance").title =
32
+ `PID: ${instanceInfo.pid} | Events: ${instanceInfo.events}`;
33
+ }
34
+
28
35
  const pages = result.pages || [];
29
36
  const select = document.getElementById("pageList");
30
37
 
@@ -247,6 +254,27 @@ document.getElementById("disableAllFilters").onclick = async () => {
247
254
  setTimeout(updateFilters, 100);
248
255
  };
249
256
 
257
+ // Switch to a different WebTap instance
258
+ document.getElementById("switchInstance").onclick = async () => {
259
+ const result = await api("/release", "POST");
260
+ if (!result.error) {
261
+ document.getElementById("status").innerHTML =
262
+ '<span style="color: #666">Port released. Start new WebTap.</span>';
263
+ // Disable controls until reconnected
264
+ document.getElementById("connect").disabled = true;
265
+ document.getElementById("disconnect").disabled = true;
266
+ document.getElementById("fetchToggle").disabled = true;
267
+ // Try to reconnect after delay
268
+ setTimeout(() => {
269
+ loadPages();
270
+ updateStatus();
271
+ }, 2000);
272
+ } else {
273
+ document.getElementById("status").innerHTML =
274
+ `<span class="error">Error: ${result.error}</span>`;
275
+ }
276
+ };
277
+
250
278
  // Update all status from server - single source of truth
251
279
  async function updateStatus() {
252
280
  const status = await api("/status");
@@ -1,6 +1,6 @@
1
1
  [project]
2
2
  name = "webtap-tool"
3
- version = "0.1.2"
3
+ version = "0.1.3"
4
4
  description = "Terminal-based web page inspector for AI debugging sessions"
5
5
  readme = "README.md"
6
6
  authors = [
@@ -47,8 +47,12 @@ def main():
47
47
  def _start_api_server_safely():
48
48
  """Start API server with error handling."""
49
49
  try:
50
- start_api_server(app.state)
51
- logger.info("API server started on http://localhost:8765")
50
+ thread = start_api_server(app.state)
51
+ if thread and app.state:
52
+ app.state.api_thread = thread
53
+ logger.info("API server started on port 8765")
54
+ else:
55
+ logger.info("Port 8765 in use by another instance")
52
56
  except Exception as e:
53
57
  logger.warning(f"Failed to start API server: {e}")
54
58
 
@@ -5,6 +5,8 @@ PUBLIC API:
5
5
  """
6
6
 
7
7
  import logging
8
+ import os
9
+ import socket
8
10
  import threading
9
11
  from typing import Any, Dict
10
12
 
@@ -184,7 +186,39 @@ async def disable_all_filters() -> Dict[str, Any]:
184
186
  return {"enabled": [], "total": 0}
185
187
 
186
188
 
187
- def start_api_server(state, host: str = "127.0.0.1", port: int = 8765):
189
+ @api.get("/instance")
190
+ async def get_instance_info() -> Dict[str, Any]:
191
+ """Get info about this WebTap instance."""
192
+ if not app_state:
193
+ return {"error": "WebTap not initialized"}
194
+
195
+ return {
196
+ "pid": os.getpid(),
197
+ "connected_to": app_state.cdp.current_page_title if app_state.cdp.is_connected else None,
198
+ "events": app_state.cdp.event_count,
199
+ }
200
+
201
+
202
+ @api.post("/release")
203
+ async def release_port() -> Dict[str, Any]:
204
+ """Release API port for another WebTap instance."""
205
+ logger.info("Releasing API port for another instance")
206
+
207
+ # Schedule graceful shutdown after response
208
+ def shutdown():
209
+ # Just set the flag to stop uvicorn, don't kill the whole process
210
+ global _shutdown_requested
211
+ _shutdown_requested = True
212
+
213
+ threading.Timer(0.5, shutdown).start()
214
+ return {"message": "Releasing port 8765"}
215
+
216
+
217
+ # Flag to signal shutdown
218
+ _shutdown_requested = False
219
+
220
+
221
+ def start_api_server(state, host: str = "127.0.0.1", port: int = 8765) -> threading.Thread | None:
188
222
  """Start the API server in a background thread.
189
223
 
190
224
  Args:
@@ -193,8 +227,16 @@ def start_api_server(state, host: str = "127.0.0.1", port: int = 8765):
193
227
  port: Port to bind to. Defaults to 8765.
194
228
 
195
229
  Returns:
196
- Thread instance running the server.
230
+ Thread instance running the server, or None if port is in use.
197
231
  """
232
+ # Check port availability first
233
+ try:
234
+ with socket.socket() as s:
235
+ s.bind((host, port))
236
+ except OSError:
237
+ logger.info(f"Port {port} already in use")
238
+ return None
239
+
198
240
  global app_state
199
241
  app_state = state
200
242
 
@@ -208,13 +250,39 @@ def start_api_server(state, host: str = "127.0.0.1", port: int = 8765):
208
250
  def run_server(host: str, port: int):
209
251
  """Run the FastAPI server in a thread."""
210
252
  try:
211
- uvicorn.run(
253
+ config = uvicorn.Config(
212
254
  api,
213
255
  host=host,
214
256
  port=port,
215
257
  log_level="error",
216
258
  access_log=False,
217
259
  )
260
+ server = uvicorn.Server(config)
261
+
262
+ # Run with checking for shutdown flag
263
+ import asyncio
264
+
265
+ loop = asyncio.new_event_loop()
266
+ asyncio.set_event_loop(loop)
267
+
268
+ async def serve():
269
+ await server.serve()
270
+
271
+ # Start serving
272
+ task = loop.create_task(serve())
273
+
274
+ # Check for shutdown flag
275
+ while not _shutdown_requested:
276
+ loop.run_until_complete(asyncio.sleep(0.1))
277
+ if task.done():
278
+ break
279
+
280
+ # Shutdown if requested
281
+ if _shutdown_requested:
282
+ logger.info("API server shutting down")
283
+ server.should_exit = True
284
+ loop.run_until_complete(server.shutdown())
285
+
218
286
  except Exception as e:
219
287
  logger.error(f"API server failed: {e}")
220
288
 
@@ -6,6 +6,7 @@ PUBLIC API:
6
6
  """
7
7
 
8
8
  import sys
9
+ import threading
9
10
  from dataclasses import dataclass, field
10
11
 
11
12
  from replkit2 import App
@@ -24,10 +25,12 @@ class WebTapState:
24
25
  Attributes:
25
26
  cdp: Chrome DevTools Protocol session instance.
26
27
  service: WebTapService orchestrating all domain services.
28
+ api_thread: Thread running the FastAPI server (if this instance owns the port).
27
29
  """
28
30
 
29
31
  cdp: CDPSession = field(default_factory=CDPSession)
30
32
  service: WebTapService = field(init=False)
33
+ api_thread: threading.Thread | None = None
31
34
 
32
35
  def __post_init__(self):
33
36
  """Initialize service with self reference after dataclass init."""
@@ -74,7 +74,8 @@ def build_query(
74
74
  pattern = value
75
75
  path_conditions.append(f"json_extract_string(event, '{json_path}') LIKE '{pattern}'")
76
76
  elif isinstance(value, (int, float)):
77
- path_conditions.append(f"CAST(json_extract_string(event, '{json_path}') AS NUMERIC) = {value}")
77
+ # Use string comparison for numeric values to avoid type conversion errors
78
+ path_conditions.append(f"json_extract_string(event, '{json_path}') = '{value}'")
78
79
  elif isinstance(value, bool):
79
80
  path_conditions.append(f"json_extract_string(event, '{json_path}') = '{str(value).lower()}'")
80
81
  elif value is None:
@@ -46,23 +46,9 @@ def run_chrome(state, detach: bool = True, port: int = 9222) -> dict:
46
46
  ],
47
47
  )
48
48
 
49
- # Setup temp profile with symlinks to real profile
49
+ # Simple: use clean temp profile for debugging
50
50
  temp_config = Path("/tmp/webtap-chrome-debug")
51
- real_config = Path.home() / ".config" / "google-chrome"
52
-
53
- if not temp_config.exists():
54
- temp_config.mkdir(parents=True)
55
-
56
- # Symlink Default profile
57
- default_profile = real_config / "Default"
58
- if default_profile.exists():
59
- (temp_config / "Default").symlink_to(default_profile)
60
-
61
- # Copy essential files
62
- for file in ["Local State", "First Run"]:
63
- src = real_config / file
64
- if src.exists():
65
- (temp_config / file).write_text(src.read_text())
51
+ temp_config.mkdir(parents=True, exist_ok=True)
66
52
 
67
53
  # Launch Chrome
68
54
  cmd = [chrome_exe, f"--remote-debugging-port={port}", "--remote-allow-origins=*", f"--user-data-dir={temp_config}"]
@@ -74,7 +60,7 @@ def run_chrome(state, detach: bool = True, port: int = 9222) -> dict:
74
60
  details={
75
61
  "Port": str(port),
76
62
  "Mode": "Background (detached)",
77
- "Profile": str(temp_config),
63
+ "Profile": "Temporary (clean)",
78
64
  "Next step": "Run connect() to attach WebTap",
79
65
  },
80
66
  )
@@ -12,7 +12,7 @@ logger = logging.getLogger(__name__)
12
12
 
13
13
 
14
14
  class BodyService:
15
- """Internal service for response body fetching and caching."""
15
+ """Response body fetching and caching."""
16
16
 
17
17
  def __init__(self):
18
18
  """Initialize body service."""
@@ -10,7 +10,7 @@ logger = logging.getLogger(__name__)
10
10
 
11
11
 
12
12
  class ConsoleService:
13
- """Internal service for console event queries and monitoring."""
13
+ """Console event queries and monitoring."""
14
14
 
15
15
  def __init__(self):
16
16
  """Initialize console service."""
@@ -13,7 +13,7 @@ logger = logging.getLogger(__name__)
13
13
 
14
14
 
15
15
  class FetchService:
16
- """Internal service for fetch interception with explicit actions."""
16
+ """Fetch interception with explicit actions."""
17
17
 
18
18
  def __init__(self):
19
19
  """Initialize fetch service."""
@@ -10,7 +10,7 @@ logger = logging.getLogger(__name__)
10
10
 
11
11
 
12
12
  class NetworkService:
13
- """Internal service for network event queries and monitoring."""
13
+ """Network event queries and monitoring."""
14
14
 
15
15
  def __init__(self):
16
16
  """Initialize network service."""
@@ -81,7 +81,7 @@ class NetworkService:
81
81
  json_extract_string(event, '$.params.response.statusText') as StatusText
82
82
  FROM events
83
83
  WHERE json_extract_string(event, '$.method') = 'Network.responseReceived'
84
- AND CAST(json_extract_string(event, '$.params.response.status') AS INTEGER) >= 400
84
+ AND json_extract_string(event, '$.params.response.status') >= '400'
85
85
  ORDER BY rowid DESC LIMIT {limit}
86
86
  """
87
87
 
@@ -177,22 +177,32 @@ class SetupService:
177
177
  }
178
178
 
179
179
  wrapper_script = """#!/bin/bash
180
- # Chrome wrapper to always enable debugging
180
+ # Chrome wrapper using bindfs for perfect state sync with debug port
181
181
 
182
- REAL_CONFIG="$HOME/.config/google-chrome"
183
- DEBUG_CONFIG="/tmp/chrome-debug-profile"
182
+ DEBUG_DIR="$HOME/.config/google-chrome-debug"
183
+ REAL_DIR="$HOME/.config/google-chrome"
184
184
 
185
- if [ ! -d "$DEBUG_CONFIG" ]; then
186
- mkdir -p "$DEBUG_CONFIG"
187
- ln -sf "$REAL_CONFIG/Default" "$DEBUG_CONFIG/Default"
188
- cp "$REAL_CONFIG/Local State" "$DEBUG_CONFIG/" 2>/dev/null || true
189
- cp "$REAL_CONFIG/First Run" "$DEBUG_CONFIG/" 2>/dev/null || true
185
+ # Check if bindfs is installed
186
+ if ! command -v bindfs &>/dev/null; then
187
+ echo "Error: bindfs not installed. Install with: yay -S bindfs" >&2
188
+ exit 1
190
189
  fi
191
190
 
191
+ # Mount real profile via bindfs if not already mounted
192
+ if ! mountpoint -q "$DEBUG_DIR" 2>/dev/null; then
193
+ mkdir -p "$DEBUG_DIR"
194
+ if ! bindfs --no-allow-other "$REAL_DIR" "$DEBUG_DIR"; then
195
+ echo "Error: Failed to mount Chrome profile via bindfs" >&2
196
+ exit 1
197
+ fi
198
+ echo "Chrome debug profile mounted. To unmount: fusermount -u $DEBUG_DIR" >&2
199
+ fi
200
+
201
+ # Launch Chrome with debugging on bindfs mount
192
202
  exec /usr/bin/google-chrome-stable \\
193
203
  --remote-debugging-port=9222 \\
194
- --remote-allow-origins=* \\
195
- --user-data-dir="$DEBUG_CONFIG" \\
204
+ --remote-allow-origins='*' \\
205
+ --user-data-dir="$DEBUG_DIR" \\
196
206
  "$@"
197
207
  """
198
208
 
File without changes
File without changes
File without changes
File without changes