- Add Remote Monitoring Station with PIAP device profile system - Add SSH/SSHD manager with fail2ban integration - Add privileged daemon architecture for safe root operations - Add encrypted vault, HAL memory, HAL auto-analyst - Add network security suite, module creator, codex training - Add start.sh launcher script and GTK3 desktop launcher - Remove Output/ build artifacts, installer files, loose docs - Update .gitignore for runtime data and build artifacts - Update README for v1.9 with new launch method, screenshots, and features Co-Authored-By: Claude Opus 4.6 (1M context) <noreply@anthropic.com>
399 lines
14 KiB
HTML
399 lines
14 KiB
HTML
{% extends "base.html" %}
|
|
{% block title %}Module Creator - AUTARCH{% endblock %}
|
|
|
|
{% block content %}
|
|
<div class="page-header">
|
|
<h1>Module Creator</h1>
|
|
<p style="margin:0;font-size:0.85rem;color:var(--text-secondary)">
|
|
Create, edit, validate, and manage AUTARCH modules
|
|
</p>
|
|
</div>
|
|
|
|
<style>
|
|
#module-code {
|
|
font-family: 'Courier New', monospace;
|
|
font-size: 0.82rem;
|
|
background: var(--bg-primary);
|
|
color: var(--text-primary);
|
|
border: 1px solid var(--border);
|
|
border-radius: var(--radius);
|
|
padding: 0.75rem;
|
|
width: 100%;
|
|
min-height: 500px;
|
|
resize: vertical;
|
|
tab-size: 4;
|
|
line-height: 1.5;
|
|
}
|
|
#module-code:focus {
|
|
outline: none;
|
|
border-color: var(--accent);
|
|
}
|
|
.mc-layout {
|
|
display: grid;
|
|
grid-template-columns: 1fr 340px;
|
|
gap: 1.5rem;
|
|
margin-top: 1rem;
|
|
}
|
|
@media (max-width: 960px) {
|
|
.mc-layout { grid-template-columns: 1fr; }
|
|
}
|
|
.mc-editor, .mc-list {
|
|
background: var(--bg-card);
|
|
border: 1px solid var(--border);
|
|
border-radius: var(--radius);
|
|
padding: 1rem;
|
|
}
|
|
.mc-form-grid {
|
|
display: grid;
|
|
grid-template-columns: 1fr 1fr;
|
|
gap: 0.6rem 1rem;
|
|
margin-bottom: 0.75rem;
|
|
}
|
|
.mc-form-grid .form-group { margin-bottom: 0; }
|
|
.mc-form-grid label {
|
|
display: block;
|
|
font-size: 0.78rem;
|
|
color: var(--text-secondary);
|
|
margin-bottom: 2px;
|
|
}
|
|
.mc-form-grid input, .mc-form-grid select {
|
|
width: 100%;
|
|
padding: 0.4rem 0.6rem;
|
|
background: var(--bg-input);
|
|
border: 1px solid var(--border);
|
|
border-radius: var(--radius);
|
|
color: var(--text-primary);
|
|
font-size: 0.82rem;
|
|
}
|
|
.mc-btn-row {
|
|
display: flex;
|
|
gap: 0.5rem;
|
|
margin-top: 0.75rem;
|
|
flex-wrap: wrap;
|
|
}
|
|
.mc-btn-row .btn { font-size: 0.82rem; padding: 0.4rem 0.9rem; }
|
|
.mc-status {
|
|
margin-top: 0.75rem;
|
|
padding: 0.6rem 0.8rem;
|
|
border-radius: var(--radius);
|
|
font-size: 0.8rem;
|
|
display: none;
|
|
white-space: pre-wrap;
|
|
}
|
|
.mc-status.success { display: block; background: rgba(34,197,94,0.12); color: #4ade80; border: 1px solid rgba(34,197,94,0.25); }
|
|
.mc-status.error { display: block; background: rgba(239,68,68,0.12); color: #f87171; border: 1px solid rgba(239,68,68,0.25); }
|
|
.mc-status.info { display: block; background: rgba(99,102,241,0.12); color: var(--accent-hover); border: 1px solid rgba(99,102,241,0.25); }
|
|
.mc-list h2 {
|
|
font-size: 1rem;
|
|
margin-bottom: 0.5rem;
|
|
display: flex;
|
|
align-items: center;
|
|
justify-content: space-between;
|
|
}
|
|
.mc-modules-scroll {
|
|
max-height: 620px;
|
|
overflow-y: auto;
|
|
padding-right: 0.25rem;
|
|
}
|
|
.mc-cat-group { margin-bottom: 0.75rem; }
|
|
.mc-cat-group h3 {
|
|
font-size: 0.78rem;
|
|
text-transform: uppercase;
|
|
color: var(--text-muted);
|
|
letter-spacing: 0.05em;
|
|
margin-bottom: 0.3rem;
|
|
padding-bottom: 0.2rem;
|
|
border-bottom: 1px solid var(--border);
|
|
}
|
|
.mc-mod-item {
|
|
padding: 0.4rem 0.5rem;
|
|
border-radius: calc(var(--radius) - 2px);
|
|
cursor: pointer;
|
|
font-size: 0.8rem;
|
|
transition: background 0.15s;
|
|
}
|
|
.mc-mod-item:hover { background: var(--bg-input); }
|
|
.mc-mod-item .mod-name { font-weight: 600; color: var(--text-primary); }
|
|
.mc-mod-item .mod-desc {
|
|
font-size: 0.72rem;
|
|
color: var(--text-secondary);
|
|
white-space: nowrap;
|
|
overflow: hidden;
|
|
text-overflow: ellipsis;
|
|
}
|
|
.mc-mod-item .mod-meta {
|
|
font-size: 0.68rem;
|
|
color: var(--text-muted);
|
|
}
|
|
</style>
|
|
|
|
<div class="mc-layout">
|
|
<!-- Left column: editor -->
|
|
<div class="mc-editor">
|
|
<div class="mc-form-grid">
|
|
<div class="form-group">
|
|
<label for="mod-name">Module Name</label>
|
|
<input type="text" id="mod-name" placeholder="my_module" autocomplete="off">
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="mod-category">Category</label>
|
|
<select id="mod-category">
|
|
<option value="defense">defense</option>
|
|
<option value="offense">offense</option>
|
|
<option value="counter">counter</option>
|
|
<option value="analyze">analyze</option>
|
|
<option value="osint">osint</option>
|
|
<option value="simulate">simulate</option>
|
|
<option value="core">core</option>
|
|
<option value="hardware">hardware</option>
|
|
</select>
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="mod-desc">Description</label>
|
|
<input type="text" id="mod-desc" placeholder="Short description of the module">
|
|
</div>
|
|
<div class="form-group">
|
|
<label for="mod-author">Author</label>
|
|
<input type="text" id="mod-author" value="darkHal">
|
|
</div>
|
|
</div>
|
|
|
|
<div style="margin-bottom:0.5rem">
|
|
<button class="btn btn-sm" onclick="loadTemplate()" style="font-size:0.78rem">Load Template</button>
|
|
</div>
|
|
|
|
<textarea id="module-code" rows="25" placeholder="# Module code will appear here... # Click 'Load Template' or select a module from the list." spellcheck="false"></textarea>
|
|
|
|
<div class="mc-btn-row">
|
|
<button class="btn" onclick="validateCode()">Validate</button>
|
|
<button class="btn" onclick="createModule()" style="background:var(--accent);color:#fff">Create Module</button>
|
|
<button class="btn" onclick="saveModule()">Save Changes</button>
|
|
</div>
|
|
|
|
<div id="mc-status" class="mc-status"></div>
|
|
</div>
|
|
|
|
<!-- Right column: module list -->
|
|
<div class="mc-list">
|
|
<h2>
|
|
Modules
|
|
<button class="btn btn-sm" onclick="refreshModuleList()" style="font-size:0.72rem;padding:0.25rem 0.6rem">Refresh</button>
|
|
</h2>
|
|
<div id="module-list" class="mc-modules-scroll">
|
|
<p style="color:var(--text-muted);font-size:0.8rem">Loading...</p>
|
|
</div>
|
|
</div>
|
|
</div>
|
|
|
|
<script>
|
|
const API = '/module-creator';
|
|
const statusEl = document.getElementById('mc-status');
|
|
const codeEl = document.getElementById('module-code');
|
|
let currentEditName = null;
|
|
|
|
function showStatus(msg, type) {
|
|
statusEl.className = 'mc-status ' + type;
|
|
statusEl.textContent = msg;
|
|
}
|
|
|
|
function clearStatus() {
|
|
statusEl.className = 'mc-status';
|
|
statusEl.textContent = '';
|
|
}
|
|
|
|
/* Tab key support in textarea */
|
|
codeEl.addEventListener('keydown', function(e) {
|
|
if (e.key === 'Tab') {
|
|
e.preventDefault();
|
|
const start = this.selectionStart;
|
|
const end = this.selectionEnd;
|
|
this.value = this.value.substring(0, start) + ' ' + this.value.substring(end);
|
|
this.selectionStart = this.selectionEnd = start + 4;
|
|
}
|
|
});
|
|
|
|
/* Load template skeleton for selected category */
|
|
function loadTemplate() {
|
|
const cat = document.getElementById('mod-category').value;
|
|
fetch(API + '/templates')
|
|
.then(r => r.json())
|
|
.then(templates => {
|
|
const t = templates.find(x => x.category === cat);
|
|
if (t) {
|
|
codeEl.value = t.code;
|
|
if (!document.getElementById('mod-desc').value) {
|
|
document.getElementById('mod-desc').value = t.description;
|
|
}
|
|
currentEditName = null;
|
|
showStatus('Template loaded for category: ' + cat, 'info');
|
|
}
|
|
})
|
|
.catch(err => showStatus('Failed to load templates: ' + err, 'error'));
|
|
}
|
|
|
|
/* Validate code syntax */
|
|
function validateCode() {
|
|
const code = codeEl.value.trim();
|
|
if (!code) { showStatus('No code to validate', 'error'); return; }
|
|
fetch(API + '/validate', {
|
|
method: 'POST',
|
|
headers: {'Content-Type': 'application/json'},
|
|
body: JSON.stringify({code: code})
|
|
})
|
|
.then(r => r.json())
|
|
.then(data => {
|
|
if (data.valid) {
|
|
showStatus('Validation passed. ' + (data.warnings || []).join('; '), 'success');
|
|
} else {
|
|
showStatus('Validation failed:\n' + (data.errors || []).join('\n'), 'error');
|
|
}
|
|
})
|
|
.catch(err => showStatus('Validation request failed: ' + err, 'error'));
|
|
}
|
|
|
|
/* Create new module */
|
|
function createModule() {
|
|
const name = document.getElementById('mod-name').value.trim();
|
|
const category = document.getElementById('mod-category').value;
|
|
const description = document.getElementById('mod-desc').value.trim();
|
|
const author = document.getElementById('mod-author').value.trim() || 'darkHal';
|
|
const code = codeEl.value;
|
|
|
|
if (!name) { showStatus('Module name is required', 'error'); return; }
|
|
if (!code.trim()) { showStatus('Module code is empty', 'error'); return; }
|
|
|
|
fetch(API + '/create', {
|
|
method: 'POST',
|
|
headers: {'Content-Type': 'application/json'},
|
|
body: JSON.stringify({name, category, description, author, code})
|
|
})
|
|
.then(r => r.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showStatus(data.message, 'success');
|
|
currentEditName = name;
|
|
refreshModuleList();
|
|
} else {
|
|
showStatus(data.error || 'Creation failed', 'error');
|
|
}
|
|
})
|
|
.catch(err => showStatus('Request failed: ' + err, 'error'));
|
|
}
|
|
|
|
/* Save changes to existing module */
|
|
function saveModule() {
|
|
const name = currentEditName || document.getElementById('mod-name').value.trim();
|
|
const code = codeEl.value;
|
|
|
|
if (!name) { showStatus('No module selected to save. Enter a name or click a module from the list.', 'error'); return; }
|
|
if (!code.trim()) { showStatus('Module code is empty', 'error'); return; }
|
|
|
|
fetch(API + '/save', {
|
|
method: 'POST',
|
|
headers: {'Content-Type': 'application/json'},
|
|
body: JSON.stringify({name, code})
|
|
})
|
|
.then(r => r.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
showStatus(data.message, 'success');
|
|
refreshModuleList();
|
|
} else {
|
|
showStatus(data.error || 'Save failed', 'error');
|
|
}
|
|
})
|
|
.catch(err => showStatus('Request failed: ' + err, 'error'));
|
|
}
|
|
|
|
/* Load a module into the editor */
|
|
function loadModule(name) {
|
|
fetch(API + '/preview', {
|
|
method: 'POST',
|
|
headers: {'Content-Type': 'application/json'},
|
|
body: JSON.stringify({name: name})
|
|
})
|
|
.then(r => r.json())
|
|
.then(data => {
|
|
if (data.success) {
|
|
codeEl.value = data.code;
|
|
currentEditName = name;
|
|
const m = data.metadata || {};
|
|
document.getElementById('mod-name').value = name;
|
|
if (m.category) document.getElementById('mod-category').value = m.category;
|
|
if (m.description) document.getElementById('mod-desc').value = m.description;
|
|
if (m.author) document.getElementById('mod-author').value = m.author;
|
|
showStatus('Loaded module: ' + name, 'info');
|
|
} else {
|
|
showStatus(data.error || 'Failed to load module', 'error');
|
|
}
|
|
})
|
|
.catch(err => showStatus('Failed to load module: ' + err, 'error'));
|
|
}
|
|
|
|
/* Refresh the module list */
|
|
function refreshModuleList() {
|
|
const container = document.getElementById('module-list');
|
|
container.innerHTML = '<p style="color:var(--text-muted);font-size:0.8rem">Loading...</p>';
|
|
|
|
fetch(API + '/list')
|
|
.then(r => r.json())
|
|
.then(modules => {
|
|
if (!modules.length) {
|
|
container.innerHTML = '<p style="color:var(--text-muted);font-size:0.8rem">No modules found.</p>';
|
|
return;
|
|
}
|
|
/* Group by category */
|
|
const groups = {};
|
|
modules.forEach(m => {
|
|
const cat = m.category || 'unknown';
|
|
if (!groups[cat]) groups[cat] = [];
|
|
groups[cat].push(m);
|
|
});
|
|
|
|
let html = '';
|
|
const catOrder = ['defense','offense','counter','analyze','osint','simulate','core','hardware','unknown'];
|
|
catOrder.forEach(cat => {
|
|
if (!groups[cat]) return;
|
|
html += '<div class="mc-cat-group">';
|
|
html += '<h3>' + cat + ' (' + groups[cat].length + ')</h3>';
|
|
groups[cat].forEach(m => {
|
|
const desc = m.description ? m.description.substring(0, 60) : '';
|
|
const ver = m.version ? 'v' + m.version : '';
|
|
html += '<div class="mc-mod-item" onclick="loadModule(\'' + m.name.replace(/'/g, "\\'") + '\')">';
|
|
html += '<div class="mod-name">' + m.name + '</div>';
|
|
if (desc) html += '<div class="mod-desc">' + desc + '</div>';
|
|
html += '<div class="mod-meta">' + [ver, m.last_modified].filter(Boolean).join(' · ') + '</div>';
|
|
html += '</div>';
|
|
});
|
|
html += '</div>';
|
|
});
|
|
|
|
/* Any categories not in catOrder */
|
|
Object.keys(groups).forEach(cat => {
|
|
if (catOrder.includes(cat)) return;
|
|
html += '<div class="mc-cat-group">';
|
|
html += '<h3>' + cat + ' (' + groups[cat].length + ')</h3>';
|
|
groups[cat].forEach(m => {
|
|
const desc = m.description ? m.description.substring(0, 60) : '';
|
|
const ver = m.version ? 'v' + m.version : '';
|
|
html += '<div class="mc-mod-item" onclick="loadModule(\'' + m.name.replace(/'/g, "\\'") + '\')">';
|
|
html += '<div class="mod-name">' + m.name + '</div>';
|
|
if (desc) html += '<div class="mod-desc">' + desc + '</div>';
|
|
html += '<div class="mod-meta">' + [ver, m.last_modified].filter(Boolean).join(' · ') + '</div>';
|
|
html += '</div>';
|
|
});
|
|
html += '</div>';
|
|
});
|
|
|
|
container.innerHTML = html;
|
|
})
|
|
.catch(err => {
|
|
container.innerHTML = '<p style="color:var(--danger);font-size:0.8rem">Failed to load modules</p>';
|
|
});
|
|
}
|
|
|
|
/* Initial load */
|
|
refreshModuleList();
|
|
</script>
|
|
{% endblock %}
|