secubox-openwrt/package/secubox/luci-app-cdn-cache/htdocs/luci-static/resources/view/cdn-cache/policies.js
CyberMind-FR 31a87c5d7a feat(structure): reorganize luci-app packages into package/secubox/ + appstore migration
Major structural reorganization and feature additions:

## Folder Reorganization
- Move 17 luci-app-* packages to package/secubox/ (except luci-app-secubox core hub)
- Update all tooling to support new structure:
  - secubox-tools/quick-deploy.sh: search both locations
  - secubox-tools/validate-modules.sh: validate both directories
  - secubox-tools/fix-permissions.sh: fix permissions in both locations
  - .github/workflows/test-validate.yml: build from both paths
- Update README.md links to new package/secubox/ paths

## AppStore Migration (Complete)
- Add catalog entries for all remaining luci-app packages:
  - network-tweaks.json: Network optimization tools
  - secubox-bonus.json: Documentation & demos hub
- Total: 24 apps in AppStore catalog (22 existing + 2 new)
- New category: 'documentation' for docs/demos/tutorials

## VHost Manager v2.0 Enhancements
- Add profile activation system for Internal Services and Redirects
- Implement createVHost() API wrapper for template-based deployment
- Fix Virtual Hosts view rendering with proper LuCI patterns
- Fix RPCD backend shell script errors (remove invalid local declarations)
- Extend backend validation for nginx return directives (redirect support)
- Add section_id parameter for named VHost profiles
- Add Remove button to Redirects page for feature parity
- Update README to v2.0 with comprehensive feature documentation

## Network Tweaks Dashboard
- Close button added to component details modal

Files changed: 340+ (336 renames with preserved git history)
Packages affected: 19 luci-app, 2 secubox-app, 1 theme, 4 tools

🤖 Generated with [Claude Code](https://claude.com/claude-code)

Co-Authored-By: Claude Sonnet 4.5 <noreply@anthropic.com>
2026-01-01 14:59:38 +01:00

174 lines
7.1 KiB
JavaScript

'use strict';
'require view';
'require rpc';
'require ui';
'require form';
'require cdn-cache/nav as CdnNav';
var callPolicies = rpc.declare({
object: 'luci.cdn-cache',
method: 'policies',
expect: { policies: [] }
});
var callExclusions = rpc.declare({
object: 'luci.cdn-cache',
method: 'exclusions',
expect: { exclusions: [] }
});
var callRemovePolicy = rpc.declare({
object: 'luci.cdn-cache',
method: 'remove_policy',
params: ['id']
});
var callRemoveExclusion = rpc.declare({
object: 'luci.cdn-cache',
method: 'remove_exclusion',
params: ['id']
});
return view.extend({
load: function() {
return Promise.all([
callPolicies(),
callExclusions()
]);
},
render: function(data) {
var policies = data[0].policies || [];
var exclusions = data[1].exclusions || [];
return E('div', { 'class': 'cbi-map cdn-policies' }, [
E('link', { 'rel': 'stylesheet', 'href': L.resource('secubox-theme/secubox-theme.css') }),
E('link', { 'rel': 'stylesheet', 'href': L.resource('cdn-cache/common.css') }),
E('style', {}, `
.cdn-policies { font-family: -apple-system, BlinkMacSystemFont, 'Segoe UI', Roboto, sans-serif; }
.cdn-page-header { background: linear-gradient(135deg, #0891b2, #06b6d4); color: white; padding: 24px; border-radius: 12px; margin-bottom: 24px; }
.cdn-page-title { font-size: 24px; font-weight: 700; margin: 0 0 8px 0; }
.cdn-section { background: #1e293b; border-radius: 12px; padding: 20px; border: 1px solid #334155; margin-bottom: 20px; }
.cdn-section-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 16px; }
.cdn-section-title { font-size: 16px; font-weight: 600; color: #f1f5f9; display: flex; align-items: center; gap: 8px; }
.cdn-policy-card { background: #0f172a; border: 1px solid #334155; border-radius: 10px; padding: 16px; margin-bottom: 12px; }
.cdn-policy-card.enabled { border-left: 3px solid #22c55e; }
.cdn-policy-card.disabled { border-left: 3px solid #64748b; opacity: 0.7; }
.cdn-policy-header { display: flex; justify-content: space-between; align-items: center; margin-bottom: 12px; }
.cdn-policy-name { font-weight: 600; color: #f1f5f9; font-size: 15px; }
.cdn-policy-status { padding: 4px 10px; border-radius: 6px; font-size: 11px; font-weight: 600; }
.cdn-policy-status.active { background: rgba(34,197,94,0.15); color: #22c55e; }
.cdn-policy-status.inactive { background: rgba(100,116,139,0.15); color: #64748b; }
.cdn-policy-details { display: grid; grid-template-columns: repeat(auto-fit, minmax(200px, 1fr)); gap: 12px; }
.cdn-policy-field { }
.cdn-policy-label { font-size: 11px; color: #64748b; text-transform: uppercase; margin-bottom: 4px; }
.cdn-policy-value { font-size: 13px; color: #94a3b8; }
.cdn-tag { display: inline-block; background: rgba(6,182,212,0.15); color: #06b6d4; padding: 2px 8px; border-radius: 4px; font-size: 12px; margin: 2px; }
.cdn-btn { padding: 8px 16px; border-radius: 8px; font-size: 13px; font-weight: 600; cursor: pointer; border: none; transition: all 0.2s; }
.cdn-btn-primary { background: #06b6d4; color: white; }
.cdn-btn-sm { padding: 4px 10px; font-size: 12px; }
.cdn-btn-danger { background: transparent; color: #ef4444; border: 1px solid #ef4444; }
.cdn-empty { text-align: center; padding: 40px; color: #64748b; }
`),
CdnNav.renderTabs('policies'),
E('div', { 'class': 'cdn-page-header' }, [
E('h2', { 'class': 'cdn-page-title' }, '📋 Policies de Cache'),
E('p', { 'style': 'margin: 0; opacity: 0.9;' }, 'Règles de mise en cache par domaine et type de fichier')
]),
E('div', { 'class': 'cdn-section' }, [
E('div', { 'class': 'cdn-section-header' }, [
E('div', { 'class': 'cdn-section-title' }, ['✅ ', 'Policies Actives']),
E('a', { 'href': '/cgi-bin/luci/admin/services/cdn-cache/settings', 'class': 'cdn-btn cdn-btn-primary cdn-btn-sm' }, '+ Ajouter')
]),
policies.length > 0 ? policies.map(function(p) {
return E('div', { 'class': 'cdn-policy-card ' + (p.enabled ? 'enabled' : 'disabled') }, [
E('div', { 'class': 'cdn-policy-header' }, [
E('span', { 'class': 'cdn-policy-name' }, p.name),
E('div', { 'style': 'display: flex; gap: 8px; align-items: center;' }, [
E('span', { 'class': 'cdn-policy-status ' + (p.enabled ? 'active' : 'inactive') },
p.enabled ? 'Actif' : 'Inactif'),
E('button', {
'class': 'cdn-btn cdn-btn-sm cdn-btn-danger',
'click': function() {
if (confirm('Supprimer cette policy ?')) {
callRemovePolicy(p.id).then(function() {
window.location.reload();
});
}
}
}, '🗑️')
])
]),
E('div', { 'class': 'cdn-policy-details' }, [
E('div', { 'class': 'cdn-policy-field' }, [
E('div', { 'class': 'cdn-policy-label' }, 'Domaines'),
E('div', { 'class': 'cdn-policy-value' },
p.domains.split(' ').map(function(d) {
return E('span', { 'class': 'cdn-tag' }, d);
})
)
]),
E('div', { 'class': 'cdn-policy-field' }, [
E('div', { 'class': 'cdn-policy-label' }, 'Extensions'),
E('div', { 'class': 'cdn-policy-value' },
p.extensions.split(' ').map(function(e) {
return E('span', { 'class': 'cdn-tag' }, '.' + e);
})
)
]),
E('div', { 'class': 'cdn-policy-field' }, [
E('div', { 'class': 'cdn-policy-label' }, 'Durée cache'),
E('div', { 'class': 'cdn-policy-value' }, p.cache_time + ' minutes')
]),
E('div', { 'class': 'cdn-policy-field' }, [
E('div', { 'class': 'cdn-policy-label' }, 'Taille max'),
E('div', { 'class': 'cdn-policy-value' }, p.max_size + ' MB')
])
])
]);
}) : E('div', { 'class': 'cdn-empty' }, 'Aucune policy configurée')
]),
E('div', { 'class': 'cdn-section' }, [
E('div', { 'class': 'cdn-section-header' }, [
E('div', { 'class': 'cdn-section-title' }, ['🚫 ', 'Exclusions']),
]),
exclusions.length > 0 ? exclusions.map(function(e) {
return E('div', { 'class': 'cdn-policy-card ' + (e.enabled ? 'enabled' : 'disabled') }, [
E('div', { 'class': 'cdn-policy-header' }, [
E('span', { 'class': 'cdn-policy-name' }, e.name),
E('button', {
'class': 'cdn-btn cdn-btn-sm cdn-btn-danger',
'click': function() {
if (confirm('Supprimer cette exclusion ?')) {
callRemoveExclusion(e.id).then(function() {
window.location.reload();
});
}
}
}, '🗑️')
]),
E('div', { 'class': 'cdn-policy-details' }, [
E('div', { 'class': 'cdn-policy-field' }, [
E('div', { 'class': 'cdn-policy-label' }, 'Domaines exclus'),
E('div', { 'class': 'cdn-policy-value' }, e.domains)
]),
E('div', { 'class': 'cdn-policy-field' }, [
E('div', { 'class': 'cdn-policy-label' }, 'Raison'),
E('div', { 'class': 'cdn-policy-value' }, e.reason)
])
])
]);
}) : E('div', { 'class': 'cdn-empty' }, 'Aucune exclusion configurée')
])
]);
},
handleSaveApply: null,
handleSave: null,
handleReset: null
});