feat: enhance backups page with filtering and pagination features #10
@@ -3,10 +3,14 @@
|
||||
"build": {
|
||||
"dockerfile": "Dockerfile"
|
||||
},
|
||||
"settings": {},
|
||||
"customizations": {
|
||||
"vscode": {
|
||||
"extensions": ["ms-python.python"]
|
||||
"settings": {},
|
||||
"extensions": [
|
||||
"ms-python.python",
|
||||
"vivaxy.vscode-conventional-commits",
|
||||
"esbenp.prettier-vscode"
|
||||
]
|
||||
}
|
||||
},
|
||||
"postCreateCommand": "pip install -r requirements.txt; curl -sLO https://github.com/tailwindlabs/tailwindcss/releases/latest/download/tailwindcss-linux-x64; chmod +x tailwindcss-linux-x64; mv tailwindcss-linux-x64 tailwindcss",
|
||||
|
||||
@@ -232,17 +232,74 @@ def instance_detail(instance_id):
|
||||
@app.route('/backups')
|
||||
@login_required
|
||||
def backups():
|
||||
"""List all backups."""
|
||||
"""List all backups with pagination and filtering."""
|
||||
all_backups = db.get_all_backups()
|
||||
|
||||
# Add instance info to each backup
|
||||
instances_map = {}
|
||||
for backup in all_backups:
|
||||
instance = db.get_instance_by_id(backup['instance_id'])
|
||||
if backup['instance_id'] not in instances_map:
|
||||
instance = db.get_instance_by_id(backup['instance_id'])
|
||||
instances_map[backup['instance_id']] = instance
|
||||
|
||||
instance = instances_map.get(backup['instance_id'])
|
||||
if instance:
|
||||
backup['instance_name'] = instance['name']
|
||||
backup['instance_identifier'] = instance['identifier']
|
||||
|
||||
return render_template('backups.html', backups=all_backups)
|
||||
# Get unique instances for filter dropdown
|
||||
all_instances = []
|
||||
seen_ids = set()
|
||||
for backup in all_backups:
|
||||
instance_id = backup['instance_id']
|
||||
if instance_id not in seen_ids and backup.get('instance_name'):
|
||||
all_instances.append({
|
||||
'id': instance_id,
|
||||
'name': backup['instance_name'],
|
||||
'identifier': backup['instance_identifier']
|
||||
})
|
||||
seen_ids.add(instance_id)
|
||||
|
||||
# Sort instances by name
|
||||
all_instances.sort(key=lambda x: x['name'])
|
||||
|
||||
# Get filter parameter from query string
|
||||
filter_instance_id = request.args.get('instance_id', type=int)
|
||||
|
||||
# Filter backups if instance filter is applied
|
||||
if filter_instance_id:
|
||||
filtered_backups = [b for b in all_backups if b['instance_id'] == filter_instance_id]
|
||||
else:
|
||||
filtered_backups = all_backups
|
||||
|
||||
# Sort by upload date descending (newest first)
|
||||
filtered_backups.sort(key=lambda x: x['uploaded_at'] or datetime.min, reverse=True)
|
||||
|
||||
# Pagination
|
||||
page = request.args.get('page', 1, type=int)
|
||||
per_page = 10
|
||||
total_backups = len(filtered_backups)
|
||||
total_pages = (total_backups + per_page - 1) // per_page
|
||||
|
||||
# Ensure page is valid
|
||||
if page < 1:
|
||||
page = 1
|
||||
elif page > total_pages and total_pages > 0:
|
||||
page = total_pages
|
||||
|
||||
# Get backups for current page
|
||||
start_idx = (page - 1) * per_page
|
||||
end_idx = start_idx + per_page
|
||||
paginated_backups = filtered_backups[start_idx:end_idx]
|
||||
|
||||
return render_template('backups.html',
|
||||
backups=paginated_backups,
|
||||
all_instances=all_instances,
|
||||
current_page=page,
|
||||
total_pages=total_pages,
|
||||
total_backups=total_backups,
|
||||
filter_instance_id=filter_instance_id,
|
||||
per_page=per_page)
|
||||
|
||||
|
||||
@app.route('/backups/<int:backup_id>/download')
|
||||
|
||||
+92
-2
@@ -9,9 +9,40 @@
|
||||
<p class="text-neutral-400">View and download your backup files</p>
|
||||
</div>
|
||||
|
||||
<!-- Filter Section -->
|
||||
<div class="bg-neutral-800 rounded-lg border border-neutral-700 p-6">
|
||||
<h2 class="text-lg font-bold text-neutral-100 mb-4">Filters</h2>
|
||||
<form method="get" class="flex flex-col sm:flex-row gap-4">
|
||||
<div class="flex-1">
|
||||
<label for="instance_id" class="block text-sm font-medium text-neutral-300 mb-2">Instance</label>
|
||||
<select id="instance_id" name="instance_id" class="w-full px-4 py-2 bg-neutral-700 border border-neutral-600 rounded-md text-neutral-100 focus:outline-none focus:border-orange-500 transition-colors">
|
||||
<option value="">All Instances</option>
|
||||
{% for instance in all_instances %}
|
||||
<option value="{{ instance.id }}" {% if filter_instance_id == instance.id %}selected{% endif %}>
|
||||
{{ instance.name }}
|
||||
</option>
|
||||
{% endfor %}
|
||||
</select>
|
||||
</div>
|
||||
<div class="flex gap-2 sm:self-end">
|
||||
<button type="submit" class="px-4 py-2 bg-orange-600 hover:bg-orange-500 text-white rounded-md text-sm font-medium transition-colors hover:cursor-pointer">
|
||||
Apply Filter
|
||||
</button>
|
||||
{% if filter_instance_id %}
|
||||
<a href="{{ url_for('backups') }}" class="px-4 py-2 bg-neutral-700 hover:bg-neutral-600 text-neutral-300 rounded-md text-sm font-medium transition-colors">
|
||||
Clear Filter
|
||||
</a>
|
||||
{% endif %}
|
||||
</div>
|
||||
</form>
|
||||
</div>
|
||||
|
||||
<div class="bg-neutral-800 rounded-lg border border-neutral-700 overflow-hidden">
|
||||
<div class="px-6 py-4 border-b border-neutral-700">
|
||||
<div class="px-6 py-4 border-b border-neutral-700 flex justify-between items-center">
|
||||
<h2 class="text-xl font-bold text-neutral-100">Backup Files</h2>
|
||||
{% if total_backups > 0 %}
|
||||
<span class="text-sm text-neutral-400">{{ total_backups }} backup{% if total_backups != 1 %}s{% endif %}</span>
|
||||
{% endif %}
|
||||
</div>
|
||||
<div class="overflow-x-auto">
|
||||
<table class="w-full">
|
||||
@@ -51,7 +82,11 @@
|
||||
{% else %}
|
||||
<tr>
|
||||
<td colspan="5" class="px-6 py-8 text-center text-neutral-400">
|
||||
No backups found. Configure your OPNsense instances to start receiving backups.
|
||||
{% if filter_instance_id %}
|
||||
No backups found for the selected instance.
|
||||
{% else %}
|
||||
No backups found. Configure your OPNsense instances to start receiving backups.
|
||||
{% endif %}
|
||||
</td>
|
||||
</tr>
|
||||
{% endif %}
|
||||
@@ -59,6 +94,61 @@
|
||||
</table>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
<!-- Pagination -->
|
||||
{% if total_pages > 1 %}
|
||||
<div class="bg-neutral-800 rounded-lg border border-neutral-700 p-6">
|
||||
<div class="flex flex-col sm:flex-row gap-4 items-center justify-between">
|
||||
<div class="text-sm text-neutral-400">
|
||||
Page <span class="font-bold text-neutral-100">{{ current_page }}</span> of <span class="font-bold text-neutral-100">{{ total_pages }}</span>
|
||||
({{ (current_page - 1) * per_page + 1 }}-{{ [current_page * per_page, total_backups]|min }} of {{ total_backups }} backups)
|
||||
</div>
|
||||
<div class="flex gap-2 flex-wrap justify-center">
|
||||
{% if current_page > 1 %}
|
||||
<a href="{{ url_for('backups', page=1, instance_id=filter_instance_id or '') }}" class="px-3 py-2 bg-neutral-700 hover:bg-neutral-600 text-neutral-300 rounded-md text-sm transition-colors">
|
||||
First
|
||||
</a>
|
||||
<a href="{{ url_for('backups', page=current_page - 1, instance_id=filter_instance_id or '') }}" class="px-3 py-2 bg-neutral-700 hover:bg-neutral-600 text-neutral-300 rounded-md text-sm transition-colors">
|
||||
Previous
|
||||
</a>
|
||||
{% endif %}
|
||||
|
||||
<!-- Page Numbers -->
|
||||
{% set start_page = [current_page - 2, 1]|max %}
|
||||
{% set end_page = [current_page + 2, total_pages]|min %}
|
||||
|
||||
{% if start_page > 1 %}
|
||||
<span class="px-3 py-2 text-neutral-400">...</span>
|
||||
{% endif %}
|
||||
|
||||
{% for page_num in range(start_page, end_page + 1) %}
|
||||
{% if page_num == current_page %}
|
||||
<button class="px-3 py-2 bg-orange-600 text-white rounded-md text-sm font-medium">
|
||||
{{ page_num }}
|
||||
</button>
|
||||
{% else %}
|
||||
<a href="{{ url_for('backups', page=page_num, instance_id=filter_instance_id or '') }}" class="px-3 py-2 bg-neutral-700 hover:bg-neutral-600 text-neutral-300 rounded-md text-sm transition-colors">
|
||||
{{ page_num }}
|
||||
</a>
|
||||
{% endif %}
|
||||
{% endfor %}
|
||||
|
||||
{% if end_page < total_pages %}
|
||||
<span class="px-3 py-2 text-neutral-400">...</span>
|
||||
{% endif %}
|
||||
|
||||
{% if current_page < total_pages %}
|
||||
<a href="{{ url_for('backups', page=current_page + 1, instance_id=filter_instance_id or '') }}" class="px-3 py-2 bg-neutral-700 hover:bg-neutral-600 text-neutral-300 rounded-md text-sm transition-colors">
|
||||
Next
|
||||
</a>
|
||||
<a href="{{ url_for('backups', page=total_pages, instance_id=filter_instance_id or '') }}" class="px-3 py-2 bg-neutral-700 hover:bg-neutral-600 text-neutral-300 rounded-md text-sm transition-colors">
|
||||
Last
|
||||
</a>
|
||||
{% endif %}
|
||||
</div>
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
{% endblock %}
|
||||
|
||||
|
||||
+47
-4
@@ -7,18 +7,19 @@
|
||||
<link rel="icon" type="image/png" href="{{ url_for('static', filename='opnsense.png') }}">
|
||||
<link rel="stylesheet" href="{{ url_for('static', filename='css/output.css') }}">
|
||||
</head>
|
||||
<body class="bg-neutral-900 text-neutral-100 min-h-screen">
|
||||
<body class="bg-neutral-900 text-neutral-100 min-h-screen flex flex-col">
|
||||
<nav class="bg-neutral-800 border-b border-neutral-700">
|
||||
<div class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8">
|
||||
<div class="flex justify-between h-16">
|
||||
<div class="flex items-center">
|
||||
<a href="{{ url_for('dashboard') }}" class="flex items-center space-x-4">
|
||||
<img src="{{ url_for('static', filename='opnsense.png') }}" alt="OPNsense" class="h-8 w-8">
|
||||
<span class="text-neutral-400">Backup Manager</span>
|
||||
<span class="text-neutral-400 hidden sm:inline">Backup Manager</span>
|
||||
</a>
|
||||
</div>
|
||||
{% if session.user_id %}
|
||||
<div class="flex items-center space-x-4">
|
||||
<!-- Desktop Navigation -->
|
||||
<div class="hidden md:flex items-center space-x-4">
|
||||
<a href="{{ url_for('dashboard') }}" class="text-neutral-300 hover:text-orange-500 px-3 py-2 rounded-md text-sm font-medium transition-colors">
|
||||
Dashboard
|
||||
</a>
|
||||
@@ -34,12 +35,42 @@
|
||||
Logout
|
||||
</a>
|
||||
</div>
|
||||
|
||||
<!-- Mobile Menu Button -->
|
||||
<div class="md:hidden flex items-center">
|
||||
<button id="mobile-menu-btn" class="inline-flex items-center justify-center p-2 rounded-md text-neutral-300 hover:text-orange-500 hover:bg-neutral-700 focus:outline-none transition-colors">
|
||||
<svg class="h-6 w-6" fill="none" viewBox="0 0 24 24" stroke="currentColor">
|
||||
<path stroke-linecap="round" stroke-linejoin="round" stroke-width="2" d="M4 6h16M4 12h16M4 18h16" />
|
||||
</svg>
|
||||
</button>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
|
||||
<!-- Mobile Navigation Menu -->
|
||||
{% if session.user_id %}
|
||||
<div id="mobile-menu" class="hidden md:hidden pb-4 space-y-1">
|
||||
<a href="{{ url_for('dashboard') }}" class="block text-neutral-300 hover:text-orange-500 px-3 py-2 rounded-md text-sm font-medium transition-colors hover:bg-neutral-700">
|
||||
Dashboard
|
||||
</a>
|
||||
<a href="{{ url_for('instances') }}" class="block text-neutral-300 hover:text-orange-500 px-3 py-2 rounded-md text-sm font-medium transition-colors hover:bg-neutral-700">
|
||||
Instances
|
||||
</a>
|
||||
<a href="{{ url_for('backups') }}" class="block text-neutral-300 hover:text-orange-500 px-3 py-2 rounded-md text-sm font-medium transition-colors hover:bg-neutral-700">
|
||||
Backups
|
||||
</a>
|
||||
<div class="border-t border-neutral-700 pt-2 mt-2">
|
||||
<div class="px-3 py-2 text-sm text-neutral-400">{{ session.username }}</div>
|
||||
<a href="{{ url_for('logout') }}" class="block text-neutral-300 hover:text-orange-500 px-3 py-2 rounded-md text-sm font-medium transition-colors hover:bg-neutral-700">
|
||||
Logout
|
||||
</a>
|
||||
</div>
|
||||
</div>
|
||||
{% endif %}
|
||||
</div>
|
||||
</nav>
|
||||
|
||||
<main class="max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8">
|
||||
<main class="flex-grow max-w-7xl mx-auto px-4 sm:px-6 lg:px-8 py-8 w-full">
|
||||
{% with messages = get_flashed_messages(with_categories=true) %}
|
||||
{% if messages %}
|
||||
<div class="mb-6 space-y-2">
|
||||
@@ -65,6 +96,18 @@
|
||||
</p>
|
||||
</div>
|
||||
</footer>
|
||||
|
||||
<script>
|
||||
// Mobile menu toggle
|
||||
const mobileMenuBtn = document.getElementById('mobile-menu-btn');
|
||||
const mobileMenu = document.getElementById('mobile-menu');
|
||||
|
||||
if (mobileMenuBtn && mobileMenu) {
|
||||
mobileMenuBtn.addEventListener('click', function() {
|
||||
mobileMenu.classList.toggle('hidden');
|
||||
});
|
||||
}
|
||||
</script>
|
||||
</body>
|
||||
</html>
|
||||
|
||||
|
||||
Reference in New Issue
Block a user