async function loadResources(){ const resources = await api(API + "/resources/active"); const mappings = await api(API + "/domainmap"); const table = document.getElementById("resources"); table.innerHTML = ""; resources.forEach(r => { let ips = ""; if(Array.isArray(r.ips)){ ips = r.ips.map(ip => "" + (ip.type || "") + " " + ip.ip + "" ).join(""); } let domains = mappings .filter(m => m.resource_id == r.id) .map(m => "🌐 " + m.domain_name + "") .join(""); const tr = document.createElement("tr"); tr.innerHTML = ` ${r.name} ${r.produkt || ""} ${r.provider || ""} ...
${r.cpu ? r.cpu + " CPU" : ""} ${r.ram ? " • " + r.ram + " RAM" : ""} ${r.disk ? " • " + r.disk : ""} ${r.os ? " • " + r.os : ""}
${domains} ${ips}
`; table.appendChild(tr); }); } async function loadDomains(){ const domains = await api(API + "/domains") const table = document.getElementById("domains") table.innerHTML = "" for(const d of domains){ const tr = document.createElement("tr") tr.innerHTML = ` ${d.domain_name} ${d.provider || ""} ${d.ip_address || ""} ${d.resource_name ? d.resource_name : "⚠ no server"} ... ${d.yearly_cost || ""} ` table.appendChild(tr) try{ const dnsRes = await fetch(API + "/dns/" + d.domain_name) const result = await dnsRes.json() let status = "❌" if(result.ips && result.ips.length){ const ipList = result.ips.join("
") if(d.ip_address && result.ips.includes(d.ip_address)){ status = "✅ " + ipList }else{ status = "⚠ " + ipList } } document.getElementById("dns-"+d.id).innerHTML = status }catch(e){ document.getElementById("dns-"+d.id).innerHTML = "❌" } } } async function deleteDomain(id){ if(!confirm("Domain löschen?")) return; await api(API + "/domains/" + id, { method: "DELETE" }); loadDomains(); } async function loadMapping(){ const data = await api(API + "/domainmap"); const table = document.getElementById("mapping"); table.innerHTML = ""; data.forEach(m => { const tr = document.createElement("tr"); tr.innerHTML = ` ${m.domain_name} ${m.ip_address} ${m.server_name || ""} `; table.appendChild(tr); }); } function openIPManager(resourceId){ document.getElementById("ip_resource_id").value = resourceId document.getElementById("ipModal").style.display = "block" loadIPs(resourceId) } function closeIPModal(){ document.getElementById("ipModal").style.display = "none" } async function loadIPs(resourceId){ const ips = await api(API + "/resources/" + resourceId + "/ips") const container = document.getElementById("ipList") container.innerHTML = "" ips.forEach(ip=>{ const div = document.createElement("div") div.innerHTML = ` ${ip.ip} (${ip.type||""}) ${ip.comment||""} ` container.appendChild(div) }) } async function saveIP(){ const resourceId = document.getElementById("ip_resource_id").value const ip = document.getElementById("new_ip").value const type = document.getElementById("new_type").value const comment = document.getElementById("new_comment").value await api("/resman/api/resources/"+resourceId+"/ips",{ method:"POST", headers:{'Content-Type':'application/json'}, body:JSON.stringify({ip,type,comment}) }) loadIPs(resourceId) loadResources() } function openCreate(){ document.getElementById("modalTitle").innerText = "Create Resource" document.getElementById("resource_id").value = "" document.querySelectorAll("#resourceModal input, #resourceModal textarea") .forEach(e=>e.value="") document.getElementById("status").value = "aktiv" document.getElementById("resourceModal").style.display = "block" } function openEdit(resource){ document.getElementById("modalTitle").innerText = "Edit Resource" document.getElementById("resource_id").value = resource.id Object.keys(resource).forEach(k=>{ const el = document.getElementById(k) if(el) el.value = resource[k] || "" }) document.getElementById("resourceModal").style.display = "block" } function closeModal(){ document.getElementById("resourceModal").style.display="none" } async function saveResource(){ const id = document.getElementById("resource_id").value let kostenMonat=document.getElementById("kosten_monat").value let kostenJahr=document.getElementById("kosten_jahr").value if(kostenMonat) kostenMonat=kostenMonat.replace(",",".") if(kostenJahr) kostenJahr=kostenJahr.replace(",",".") const data={ name:document.getElementById("name").value, produkt:document.getElementById("produkt").value, provider:document.getElementById("provider").value, art:document.getElementById("art").value, cpu:document.getElementById("cpu").value, ram:document.getElementById("ram").value, disk:document.getElementById("disk").value, os:document.getElementById("os").value, kosten_monat:kostenMonat || null, kosten_jahr:kostenJahr || null, providername:document.getElementById("providername").value, ipv6_net:document.getElementById("ipv6_net").value, bestelldatum:document.getElementById("bestelldatum").value || null, kuendbar_ab:document.getElementById("kuendbar_ab").value || null, kuendigungsdatum:document.getElementById("kuendigungsdatum").value || null, status:document.getElementById("status").value, bemerkung:document.getElementById("bemerkung").value } if(id){ await api(API+"/resources/"+id,{ method:"PUT", headers:{'Content-Type':'application/json'}, body:JSON.stringify(data) }) }else{ await api(API+"/resources",{ method:"POST", headers:{'Content-Type':'application/json'}, body:JSON.stringify(data) }) } closeModal() loadResources() loadCancelled() loadCosts() loadInfrastructure() } function openDomainCreate(){ document.getElementById("domainModalTitle").innerText="Create Domain" document.getElementById("domain_id").value="" document.getElementById("domain_name").value="" document.getElementById("domain_provider").value="" document.getElementById("domain_ip").value="" document.getElementById("domain_cost").value="" document.getElementById("domain_notes").value="" document.getElementById("domainModal").style.display="block" } function openDomainEdit(d){ document.getElementById("domainModalTitle").innerText="Edit Domain" document.getElementById("domain_id").value=d.id document.getElementById("domain_name").value=d.domain_name||"" document.getElementById("domain_provider").value=d.provider||"" document.getElementById("domain_ip").value=d.ip_address||"" document.getElementById("domain_cost").value=d.yearly_cost||"" document.getElementById("domain_notes").value=d.notes||"" document.getElementById("domainModal").style.display="block" } function closeDomainModal(){ document.getElementById("domainModal").style.display="none" } async function saveDomain(){ const id=document.getElementById("domain_id").value let cost=document.getElementById("domain_cost").value if(cost){ cost=cost.replace(",",".") } const data={ domain_name:domain_name.value, provider:domain_provider.value, ip_address:domain_ip.value, yearly_cost:cost || null, notes:domain_notes.value } if(id){ await api(API+"/domains/"+id,{ method:"PUT", headers:{'Content-Type':'application/json'}, body:JSON.stringify(data) }) }else{ await api(API+"/domains",{ method:"POST", headers:{'Content-Type':'application/json'}, body:JSON.stringify(data) }) } closeDomainModal() loadDomains() loadMapping() loadCosts() } async function loadCancelled(){ const res = await api(API+"/resources/cancelled") const table = document.getElementById("cancelled") table.innerHTML="" res.forEach(r=>{ const tr=document.createElement("tr") tr.innerHTML=` ${r.name} ${r.produkt||""} ${r.provider || ""} ${r.cpu||""} ${r.ram||""} ${r.disk||""} gekündigt ` table.appendChild(tr) }) } async function loadCosts(){ const resources = await api(API+"/resources/active") const domains = await api(API+"/domains") let month = 0 let year = 0 let domainYear = 0 resources.forEach(r=>{ let m = Number(r.kosten_monat || 0) let y = Number(r.kosten_jahr || 0) if(m){ month += m }else if(y){ month += y / 12 } if(y){ year += y }else if(m){ year += m * 12 } }) domains.forEach(d=>{ domainYear += Number(d.yearly_cost || 0) }) document.getElementById("costMonth").innerText = month.toFixed(2) document.getElementById("costYear").innerText = year.toFixed(2) document.getElementById("costDomain").innerText = domainYear.toFixed(2) document.getElementById("costTotal").innerText = (year + domainYear).toFixed(2) } async function loadInfrastructure(){ const resources = await api(API + "/resources/active") const mappings = await api(API + "/domainmap") let html = "" resources.forEach(r => { html += `
${r.name}
Produkt: ${r.produkt || ""}
CPU: ${r.cpu || ""} | RAM: ${r.ram || ""} | Disk: ${r.disk || ""}
OS: ${r.os || ""}

` if(Array.isArray(r.ips)){ r.ips.forEach(ip => { html += `
IP: ${ip.ip} (${ip.type || ""}) ` const domains = mappings.filter(m => m.ip_address == ip.ip) domains.forEach(d => { html += `
🌐 ${d.domain_name}
` }) html += "
" }) } html += "
" }) document.getElementById("infraView").innerHTML = html } loadResources(); loadDomains(); loadMapping(); loadCosts() loadInfrastructure()