mirror of
https://github.com/Django-LiveView/docs.git
synced 2025-12-31 05:32:23 +01:00
Add comprehensive documentation for advanced features
- Add Script Execution section with security warnings and examples - Add File Upload Handling with Base64 encoding guide - Add Message Queue System documentation - Add Network Connectivity Handling with reconnection behavior - Add Registry Management methods (get_handler, unregister, clear, get_all_handlers) - Add WebSocket Configuration options - Add Automatic Error Handling section - Update API Reference with additional registry methods - All examples include Python code, HTML templates, and best practices
This commit is contained in:
734
one.org
734
one.org
@@ -1086,6 +1086,529 @@ liveview_registry.add_middleware(auth_middleware)
|
|||||||
liveview_registry.add_middleware(logging_middleware)
|
liveview_registry.add_middleware(logging_middleware)
|
||||||
#+END_SRC
|
#+END_SRC
|
||||||
|
|
||||||
|
** Script Execution
|
||||||
|
|
||||||
|
Execute JavaScript code directly from your Python handlers.
|
||||||
|
|
||||||
|
⚠️ *Security Warning:* Only execute scripts from trusted sources. Never pass user input directly to the ~script~ parameter without sanitization, as this can lead to XSS (Cross-Site Scripting) vulnerabilities.
|
||||||
|
|
||||||
|
*** Basic Script Execution
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_handler, send
|
||||||
|
|
||||||
|
@liveview_handler("show_notification")
|
||||||
|
def show_notification(consumer, content):
|
||||||
|
message = content["form"]["message"]
|
||||||
|
|
||||||
|
# Execute JavaScript to show a browser notification
|
||||||
|
send(consumer, {
|
||||||
|
"script": f"""
|
||||||
|
if (Notification.permission === 'granted') {{
|
||||||
|
new Notification('New Message', {{
|
||||||
|
body: '{message}',
|
||||||
|
icon: '/static/icon.png'
|
||||||
|
}});
|
||||||
|
}}
|
||||||
|
"""
|
||||||
|
})
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Combining HTML and Script
|
||||||
|
|
||||||
|
You can combine HTML updates with script execution:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
@liveview_handler("load_chart")
|
||||||
|
def load_chart(consumer, content):
|
||||||
|
import json
|
||||||
|
chart_data = json.dumps(get_chart_data())
|
||||||
|
|
||||||
|
# Update HTML
|
||||||
|
html = render_to_string("chart_container.html", {
|
||||||
|
"chart_id": "sales-chart"
|
||||||
|
})
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#chart-container",
|
||||||
|
"html": html
|
||||||
|
})
|
||||||
|
|
||||||
|
# Initialize chart with JavaScript
|
||||||
|
send(consumer, {
|
||||||
|
"script": f"""
|
||||||
|
const ctx = document.getElementById('sales-chart');
|
||||||
|
new Chart(ctx, {{
|
||||||
|
type: 'bar',
|
||||||
|
data: {chart_data}
|
||||||
|
}});
|
||||||
|
"""
|
||||||
|
})
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Inline Scripts in HTML
|
||||||
|
|
||||||
|
Django LiveView automatically extracts and executes ~<script>~ tags from HTML responses:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
@liveview_handler("load_interactive_component")
|
||||||
|
def load_interactive_component(consumer, content):
|
||||||
|
html = '''
|
||||||
|
<div id="counter">
|
||||||
|
<button id="increment">Count: <span>0</span></button>
|
||||||
|
</div>
|
||||||
|
<script>
|
||||||
|
let count = 0;
|
||||||
|
document.getElementById('increment').addEventListener('click', () => {
|
||||||
|
count++;
|
||||||
|
document.querySelector('#increment span').textContent = count;
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
'''
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#component-container",
|
||||||
|
"html": html
|
||||||
|
})
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
The script will be automatically extracted and executed after the HTML is rendered.
|
||||||
|
|
||||||
|
*** Use Cases
|
||||||
|
|
||||||
|
- Integrating third-party JavaScript libraries (charts, maps, etc.)
|
||||||
|
- Triggering browser APIs (notifications, geolocation, etc.)
|
||||||
|
- Initializing complex UI components
|
||||||
|
- Playing sounds or animations
|
||||||
|
- Focusing specific elements with custom logic
|
||||||
|
|
||||||
|
*** Best Practices
|
||||||
|
|
||||||
|
1. ✓ Sanitize any user input before including in scripts
|
||||||
|
2. ✓ Use JSON serialization for data: ~import json; json.dumps(data)~
|
||||||
|
3. ✓ Prefer ~<script>~ tags in templates over the ~script~ parameter
|
||||||
|
4. ✓ Keep scripts focused and minimal
|
||||||
|
5. ✗ Don't use ~eval()~ or similar dangerous functions
|
||||||
|
6. ✗ Don't pass unsanitized user input to scripts
|
||||||
|
|
||||||
|
** File Upload Handling
|
||||||
|
|
||||||
|
Handle file uploads with Base64 encoding for WebSocket transmission.
|
||||||
|
|
||||||
|
*** Server-Side File Processing
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
import base64
|
||||||
|
from io import BytesIO
|
||||||
|
from PIL import Image
|
||||||
|
from django.core.files.base import ContentFile
|
||||||
|
|
||||||
|
@liveview_handler("upload_avatar")
|
||||||
|
def upload_avatar(consumer, content):
|
||||||
|
user = consumer.scope.get("user")
|
||||||
|
|
||||||
|
if not user or not user.is_authenticated:
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#upload-status",
|
||||||
|
"html": "<p class='error'>Please log in to upload</p>"
|
||||||
|
})
|
||||||
|
return
|
||||||
|
|
||||||
|
# Get Base64 data from form
|
||||||
|
base64_data = content["form"].get("avatar", "")
|
||||||
|
|
||||||
|
if not base64_data:
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#upload-status",
|
||||||
|
"html": "<p class='error'>No file selected</p>"
|
||||||
|
})
|
||||||
|
return
|
||||||
|
|
||||||
|
try:
|
||||||
|
# Extract format and data
|
||||||
|
# Format: "..."
|
||||||
|
format_str, img_str = base64_data.split(';base64,')
|
||||||
|
ext = format_str.split('/')[-1]
|
||||||
|
|
||||||
|
# Decode Base64
|
||||||
|
img_data = base64.b64decode(img_str)
|
||||||
|
|
||||||
|
# Validate it's an image
|
||||||
|
image = Image.open(BytesIO(img_data))
|
||||||
|
|
||||||
|
# Resize if needed
|
||||||
|
if image.width > 500 or image.height > 500:
|
||||||
|
image.thumbnail((500, 500), Image.Resampling.LANCZOS)
|
||||||
|
|
||||||
|
# Save resized image to bytes
|
||||||
|
buffer = BytesIO()
|
||||||
|
image.save(buffer, format=ext.upper())
|
||||||
|
img_data = buffer.getvalue()
|
||||||
|
|
||||||
|
# Save to user profile
|
||||||
|
filename = f"avatar_{user.id}.{ext}"
|
||||||
|
user.profile.avatar.save(
|
||||||
|
filename,
|
||||||
|
ContentFile(img_data),
|
||||||
|
save=True
|
||||||
|
)
|
||||||
|
|
||||||
|
# Show success
|
||||||
|
html = f'''
|
||||||
|
<p class='success'>Avatar uploaded successfully!</p>
|
||||||
|
<img src="{user.profile.avatar.url}" alt="Avatar" width="100">
|
||||||
|
'''
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#upload-status",
|
||||||
|
"html": html
|
||||||
|
})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
import logging
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
logger.error(f"Error uploading avatar: {e}")
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#upload-status",
|
||||||
|
"html": "<p class='error'>Upload failed. Please try again.</p>"
|
||||||
|
})
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** HTML Template
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<div>
|
||||||
|
<input type="file" id="avatar-upload" accept="image/*">
|
||||||
|
<button
|
||||||
|
data-liveview-function="upload_avatar"
|
||||||
|
data-action="click->page#run">
|
||||||
|
Upload Avatar
|
||||||
|
</button>
|
||||||
|
<div id="upload-status"></div>
|
||||||
|
</div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Encode file as Base64 before sending
|
||||||
|
document.querySelector('[data-liveview-function="upload_avatar"]')
|
||||||
|
.addEventListener('click', async (e) => {
|
||||||
|
const fileInput = document.getElementById('avatar-upload');
|
||||||
|
const file = fileInput.files[0];
|
||||||
|
|
||||||
|
if (file) {
|
||||||
|
const reader = new FileReader();
|
||||||
|
reader.onload = () => {
|
||||||
|
// Store Base64 in hidden input
|
||||||
|
let hiddenInput = document.getElementById('avatar-data');
|
||||||
|
if (!hiddenInput) {
|
||||||
|
hiddenInput = document.createElement('input');
|
||||||
|
hiddenInput.type = 'hidden';
|
||||||
|
hiddenInput.name = 'avatar';
|
||||||
|
hiddenInput.id = 'avatar-data';
|
||||||
|
document.querySelector('div').appendChild(hiddenInput);
|
||||||
|
}
|
||||||
|
hiddenInput.value = reader.result;
|
||||||
|
};
|
||||||
|
reader.readAsDataURL(file);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** File Size Limitations
|
||||||
|
|
||||||
|
WebSocket has practical limits for Base64-encoded files:
|
||||||
|
|
||||||
|
- ✓ *Small files* (< 1MB): Images, documents, avatars
|
||||||
|
- ⚠️ *Medium files* (1-5MB): May work but can be slow
|
||||||
|
- ✗ *Large files* (> 5MB): Not recommended, use traditional HTTP upload
|
||||||
|
|
||||||
|
For large files, use a traditional HTTP POST to upload, then notify via WebSocket.
|
||||||
|
|
||||||
|
*** Security Considerations
|
||||||
|
|
||||||
|
1. ✓ Validate file types (check magic bytes, not just extensions)
|
||||||
|
2. ✓ Limit file sizes on the server
|
||||||
|
3. ✓ Scan files for malware if accepting from untrusted users
|
||||||
|
4. ✓ Store files outside the web root
|
||||||
|
5. ✓ Use unique filenames to prevent overwrites
|
||||||
|
6. ✓ Validate image dimensions and format with Pillow/PIL
|
||||||
|
|
||||||
|
** Message Queue System
|
||||||
|
|
||||||
|
Django LiveView automatically queues messages when the WebSocket connection is not ready.
|
||||||
|
|
||||||
|
*** How It Works
|
||||||
|
|
||||||
|
When you call a LiveView handler but the WebSocket is:
|
||||||
|
- Still connecting
|
||||||
|
- Temporarily disconnected
|
||||||
|
- Reconnecting after a network failure
|
||||||
|
|
||||||
|
The message is automatically queued and sent once the connection is restored.
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<button
|
||||||
|
data-liveview-function="save_draft"
|
||||||
|
data-action="click->page#run">
|
||||||
|
Save Draft
|
||||||
|
</button>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
If the user clicks "Save Draft" while offline, the message is queued. When the connection is restored, all queued messages are sent automatically in order.
|
||||||
|
|
||||||
|
*** User Feedback During Queueing
|
||||||
|
|
||||||
|
Show users when their actions are being queued:
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<div id="connection-status"></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Monitor connection and queue status
|
||||||
|
setInterval(() => {
|
||||||
|
const statusEl = document.getElementById('connection-status');
|
||||||
|
const ws = window.myWebSocket;
|
||||||
|
|
||||||
|
if (ws && ws.readyState === WebSocket.OPEN) {
|
||||||
|
statusEl.innerHTML = '<span class="online">🟢 Connected</span>';
|
||||||
|
} else if (ws && ws.readyState === WebSocket.CONNECTING) {
|
||||||
|
statusEl.innerHTML = '<span class="connecting">🟡 Connecting...</span>';
|
||||||
|
} else {
|
||||||
|
statusEl.innerHTML = '<span class="offline">🔴 Disconnected</span>';
|
||||||
|
}
|
||||||
|
}, 1000);
|
||||||
|
</script>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
** Network Connectivity Handling
|
||||||
|
|
||||||
|
Django LiveView automatically handles network connectivity changes.
|
||||||
|
|
||||||
|
*** Automatic Detection
|
||||||
|
|
||||||
|
The framework detects when:
|
||||||
|
- Network goes offline (airplane mode, WiFi disconnect, etc.)
|
||||||
|
- Network comes back online
|
||||||
|
- Connection to the server is lost
|
||||||
|
- Connection to the server is restored
|
||||||
|
|
||||||
|
*** Visual Feedback
|
||||||
|
|
||||||
|
Create a connection status modal that appears when connectivity is lost:
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<!-- templates/base.html -->
|
||||||
|
{% load static %}
|
||||||
|
{% load liveview %}
|
||||||
|
<!DOCTYPE html>
|
||||||
|
<html lang="en" data-room="{% liveview_room_uuid %}">
|
||||||
|
<head>
|
||||||
|
<meta charset="UTF-8">
|
||||||
|
<title>{% block title %}My Site{% endblock %}</title>
|
||||||
|
<style>
|
||||||
|
.no-connection {
|
||||||
|
display: none;
|
||||||
|
position: fixed;
|
||||||
|
top: 0;
|
||||||
|
left: 0;
|
||||||
|
right: 0;
|
||||||
|
background: #ff6b6b;
|
||||||
|
color: white;
|
||||||
|
padding: 1rem;
|
||||||
|
text-align: center;
|
||||||
|
z-index: 9999;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-connection--show {
|
||||||
|
display: block;
|
||||||
|
}
|
||||||
|
|
||||||
|
.no-connection--hide {
|
||||||
|
display: none;
|
||||||
|
}
|
||||||
|
</style>
|
||||||
|
</head>
|
||||||
|
<body data-controller="page">
|
||||||
|
<!-- Connection status notification -->
|
||||||
|
<div id="no-connection" class="no-connection no-connection--hide">
|
||||||
|
⚠️ Connection lost. Reconnecting...
|
||||||
|
</div>
|
||||||
|
|
||||||
|
{% block content %}{% endblock %}
|
||||||
|
|
||||||
|
<script src="{% static 'liveview/liveview.min.js' %}" defer></script>
|
||||||
|
</body>
|
||||||
|
</html>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
The framework automatically shows/hides this modal when connectivity changes.
|
||||||
|
|
||||||
|
*** Reconnection Behavior
|
||||||
|
|
||||||
|
When the connection is lost, Django LiveView:
|
||||||
|
|
||||||
|
1. Shows the ~#no-connection~ modal (if it exists)
|
||||||
|
2. Queues any new messages
|
||||||
|
3. Attempts to reconnect automatically
|
||||||
|
4. Uses exponential backoff between attempts
|
||||||
|
5. Tries up to 5 times before giving up
|
||||||
|
|
||||||
|
Default reconnection settings:
|
||||||
|
|
||||||
|
- *Initial delay:* 3 seconds
|
||||||
|
- *Maximum attempts:* 5
|
||||||
|
- *Backoff multiplier:* 1.5x
|
||||||
|
- *Maximum delay:* 30 seconds
|
||||||
|
|
||||||
|
Reconnection delays: 3s → 4.5s → 6.75s → 10.12s → 15.18s
|
||||||
|
|
||||||
|
** Registry Management
|
||||||
|
|
||||||
|
Advanced control over LiveView handler registration.
|
||||||
|
|
||||||
|
*** Listing All Handlers
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_registry
|
||||||
|
|
||||||
|
# Get all registered handler names
|
||||||
|
handlers = liveview_registry.list_functions()
|
||||||
|
print(handlers) # ['say_hello', 'load_articles', 'submit_form', ...]
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Getting a Specific Handler
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
# Get handler by name
|
||||||
|
handler = liveview_registry.get_handler("say_hello")
|
||||||
|
|
||||||
|
if handler:
|
||||||
|
print(f"Handler found: {handler.__name__}")
|
||||||
|
else:
|
||||||
|
print("Handler not found")
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Unregistering Handlers
|
||||||
|
|
||||||
|
Remove a handler from the registry:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
# Unregister a specific handler
|
||||||
|
liveview_registry.unregister("old_handler")
|
||||||
|
|
||||||
|
# Verify it's gone
|
||||||
|
if liveview_registry.get_handler("old_handler") is None:
|
||||||
|
print("Handler successfully unregistered")
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
Use cases:
|
||||||
|
- Removing deprecated handlers
|
||||||
|
- Disabling features at runtime
|
||||||
|
- Testing and cleanup
|
||||||
|
|
||||||
|
*** Clearing All Handlers
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
# Remove all registered handlers
|
||||||
|
liveview_registry.clear()
|
||||||
|
|
||||||
|
# Verify
|
||||||
|
print(liveview_registry.list_functions()) # []
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
⚠️ *Warning:* This clears ALL handlers. Only use in testing or when reinitializing the application.
|
||||||
|
|
||||||
|
*** Dynamic Handler Registration
|
||||||
|
|
||||||
|
Register handlers programmatically without the decorator:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_registry, send
|
||||||
|
|
||||||
|
def dynamic_handler(consumer, content):
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#result",
|
||||||
|
"html": "<p>Dynamic handler executed!</p>"
|
||||||
|
})
|
||||||
|
|
||||||
|
# Register manually
|
||||||
|
handler_name = "dynamic_action"
|
||||||
|
decorated_func = liveview_registry.register(handler_name)(dynamic_handler)
|
||||||
|
|
||||||
|
# Now callable from frontend
|
||||||
|
# <button data-liveview-function="dynamic_action" data-action="click->page#run">
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
** WebSocket Configuration
|
||||||
|
|
||||||
|
Customize WebSocket connection settings.
|
||||||
|
|
||||||
|
*** Custom WebSocket Path
|
||||||
|
|
||||||
|
Change the default WebSocket URL path:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
# routing.py
|
||||||
|
from liveview.routing import get_liveview_path
|
||||||
|
|
||||||
|
websocket_urlpatterns = [
|
||||||
|
get_liveview_path("custom/path/<str:room_name>/"),
|
||||||
|
]
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
Update frontend configuration:
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<!-- templates/base.html -->
|
||||||
|
<script>
|
||||||
|
window.webSocketConfig = {
|
||||||
|
host: '{{ request.get_host }}',
|
||||||
|
protocol: '{% if request.is_secure %}wss{% else %}ws{% endif %}',
|
||||||
|
path: '/custom/path/' // Custom path
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
<script src="{% static 'liveview/liveview.min.js' %}" defer></script>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Custom Host and Protocol
|
||||||
|
|
||||||
|
For development or special deployments:
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<script>
|
||||||
|
window.webSocketConfig = {
|
||||||
|
host: 'api.example.com', // Different host
|
||||||
|
protocol: 'wss' // Force secure WebSocket
|
||||||
|
};
|
||||||
|
</script>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Reconnection Configuration
|
||||||
|
|
||||||
|
Modify reconnection behavior by editing ~frontend/webSocketsCli.js~ before building:
|
||||||
|
|
||||||
|
#+BEGIN_SRC javascript
|
||||||
|
// frontend/webSocketsCli.js
|
||||||
|
|
||||||
|
// Default values:
|
||||||
|
const RECONNECT_INTERVAL = 3000; // Initial delay: 3 seconds
|
||||||
|
const MAX_RECONNECT_ATTEMPTS = 5; // Maximum attempts: 5
|
||||||
|
const RECONNECT_BACKOFF_MULTIPLIER = 1.5; // Exponential multiplier
|
||||||
|
|
||||||
|
// Custom values (example):
|
||||||
|
const RECONNECT_INTERVAL = 5000; // Initial delay: 5 seconds
|
||||||
|
const MAX_RECONNECT_ATTEMPTS = 10; // Maximum attempts: 10
|
||||||
|
const RECONNECT_BACKOFF_MULTIPLIER = 2.0; // Double delay each time
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
Then rebuild the JavaScript:
|
||||||
|
|
||||||
|
#+BEGIN_SRC sh
|
||||||
|
cd frontend
|
||||||
|
npm run build:min
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
* Error Handling
|
* Error Handling
|
||||||
:PROPERTIES:
|
:PROPERTIES:
|
||||||
:ONE: one-custom-default-doc
|
:ONE: one-custom-default-doc
|
||||||
@@ -1412,6 +1935,136 @@ def important_operation(consumer, content):
|
|||||||
})
|
})
|
||||||
#+END_SRC
|
#+END_SRC
|
||||||
|
|
||||||
|
** Automatic Error Handling
|
||||||
|
|
||||||
|
Django LiveView automatically catches and reports handler exceptions.
|
||||||
|
|
||||||
|
*** Built-in Error Handling
|
||||||
|
|
||||||
|
When a handler raises an unhandled exception, the framework automatically:
|
||||||
|
|
||||||
|
1. Logs the error with ~logging.error()~
|
||||||
|
2. Sends an error message to the client
|
||||||
|
3. Re-raises the exception for debugging
|
||||||
|
|
||||||
|
Example of an error that will be automatically handled:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
@liveview_handler("risky_operation")
|
||||||
|
def risky_operation(consumer, content):
|
||||||
|
# This will raise a ZeroDivisionError
|
||||||
|
result = 10 / 0
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#result",
|
||||||
|
"html": f"<p>Result: {result}</p>"
|
||||||
|
})
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
The client will automatically receive:
|
||||||
|
|
||||||
|
#+BEGIN_SRC json
|
||||||
|
{
|
||||||
|
"error": "Handler error: division by zero",
|
||||||
|
"function": "risky_operation"
|
||||||
|
}
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Handling Errors on the Client
|
||||||
|
|
||||||
|
Detect and display errors sent from the server:
|
||||||
|
|
||||||
|
#+BEGIN_SRC html
|
||||||
|
<div id="result"></div>
|
||||||
|
<div id="error-display" class="error" style="display: none;"></div>
|
||||||
|
|
||||||
|
<script>
|
||||||
|
// Listen for error messages
|
||||||
|
window.myWebSocket.addEventListener('message', (event) => {
|
||||||
|
const data = JSON.parse(event.data);
|
||||||
|
|
||||||
|
if (data.error) {
|
||||||
|
// Show error to user
|
||||||
|
const errorDiv = document.getElementById('error-display');
|
||||||
|
errorDiv.textContent = `Error: ${data.error}`;
|
||||||
|
errorDiv.style.display = 'block';
|
||||||
|
|
||||||
|
// Hide after 5 seconds
|
||||||
|
setTimeout(() => {
|
||||||
|
errorDiv.style.display = 'none';
|
||||||
|
}, 5000);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
</script>
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Custom Error Responses
|
||||||
|
|
||||||
|
For better control, handle errors explicitly within your handler:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
@liveview_handler("safe_operation")
|
||||||
|
def safe_operation(consumer, content):
|
||||||
|
try:
|
||||||
|
result = perform_calculation()
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#result",
|
||||||
|
"html": f"<p class='success'>Result: {result}</p>"
|
||||||
|
})
|
||||||
|
|
||||||
|
except ValueError as e:
|
||||||
|
# Handle expected errors gracefully
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#result",
|
||||||
|
"html": f"<p class='error'>Invalid input: {e}</p>"
|
||||||
|
})
|
||||||
|
|
||||||
|
except Exception as e:
|
||||||
|
# Log and show generic error
|
||||||
|
import logging
|
||||||
|
logger = logging.getLogger(__name__)
|
||||||
|
logger.error(f"Unexpected error: {e}", exc_info=True)
|
||||||
|
|
||||||
|
send(consumer, {
|
||||||
|
"target": "#result",
|
||||||
|
"html": "<p class='error'>An unexpected error occurred</p>"
|
||||||
|
})
|
||||||
|
# Don't re-raise - this prevents automatic error handling
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
*** Configuring Error Logging
|
||||||
|
|
||||||
|
Configure logging for LiveView errors in ~settings.py~:
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
# settings.py
|
||||||
|
LOGGING = {
|
||||||
|
'version': 1,
|
||||||
|
'disable_existing_loggers': False,
|
||||||
|
'handlers': {
|
||||||
|
'file': {
|
||||||
|
'level': 'ERROR',
|
||||||
|
'class': 'logging.FileHandler',
|
||||||
|
'filename': '/var/log/django/liveview_errors.log',
|
||||||
|
},
|
||||||
|
'mail_admins': {
|
||||||
|
'level': 'ERROR',
|
||||||
|
'class': 'django.utils.log.AdminEmailHandler',
|
||||||
|
}
|
||||||
|
},
|
||||||
|
'loggers': {
|
||||||
|
'liveview': {
|
||||||
|
'handlers': ['file', 'mail_admins'],
|
||||||
|
'level': 'ERROR',
|
||||||
|
'propagate': False,
|
||||||
|
},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
Now all handler errors are logged to a file and emailed to admins.
|
||||||
|
|
||||||
* Testing
|
* Testing
|
||||||
:PROPERTIES:
|
:PROPERTIES:
|
||||||
:ONE: one-custom-default-doc
|
:ONE: one-custom-default-doc
|
||||||
@@ -2285,6 +2938,87 @@ handlers = liveview_registry.list_functions()
|
|||||||
print(handlers) # ['say_hello', 'load_articles', ...]
|
print(handlers) # ['say_hello', 'load_articles', ...]
|
||||||
#+END_SRC
|
#+END_SRC
|
||||||
|
|
||||||
|
**** ~get_handler(function_name)~
|
||||||
|
|
||||||
|
Get a specific handler by name.
|
||||||
|
|
||||||
|
**Parameters:**
|
||||||
|
- ~function_name~ (str): Name of the handler to retrieve
|
||||||
|
|
||||||
|
**Returns:** Callable or None
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_registry
|
||||||
|
|
||||||
|
handler = liveview_registry.get_handler("say_hello")
|
||||||
|
|
||||||
|
if handler:
|
||||||
|
print(f"Handler found: {handler.__name__}")
|
||||||
|
else:
|
||||||
|
print("Handler not found")
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
**** ~get_all_handlers()~
|
||||||
|
|
||||||
|
Get a dictionary of all registered handlers with their functions.
|
||||||
|
|
||||||
|
**Returns:** Dict[str, Callable]
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_registry
|
||||||
|
|
||||||
|
all_handlers = liveview_registry.get_all_handlers()
|
||||||
|
|
||||||
|
for name, func in all_handlers.items():
|
||||||
|
print(f"Handler: {name}, Function: {func.__name__}")
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
**** ~unregister(function_name)~
|
||||||
|
|
||||||
|
Remove a handler from the registry.
|
||||||
|
|
||||||
|
**Parameters:**
|
||||||
|
- ~function_name~ (str): Name of the handler to remove
|
||||||
|
|
||||||
|
**Returns:** Callable or None (the removed handler, or None if not found)
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_registry
|
||||||
|
|
||||||
|
# Unregister a handler
|
||||||
|
liveview_registry.unregister("old_handler")
|
||||||
|
|
||||||
|
# Verify it's gone
|
||||||
|
if liveview_registry.get_handler("old_handler") is None:
|
||||||
|
print("Handler successfully unregistered")
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
**** ~clear()~
|
||||||
|
|
||||||
|
Remove all registered handlers.
|
||||||
|
|
||||||
|
**Returns:** None
|
||||||
|
|
||||||
|
**Example:**
|
||||||
|
|
||||||
|
#+BEGIN_SRC python
|
||||||
|
from liveview import liveview_registry
|
||||||
|
|
||||||
|
# Remove all handlers
|
||||||
|
liveview_registry.clear()
|
||||||
|
|
||||||
|
# Verify
|
||||||
|
print(liveview_registry.list_functions()) # []
|
||||||
|
#+END_SRC
|
||||||
|
|
||||||
|
⚠️ *Warning:* This clears ALL handlers. Only use in testing or when reinitializing the application.
|
||||||
|
|
||||||
** Routing
|
** Routing
|
||||||
|
|
||||||
*** ~get_liveview_urlpatterns()~
|
*** ~get_liveview_urlpatterns()~
|
||||||
|
|||||||
Reference in New Issue
Block a user