aboutsummaryrefslogtreecommitdiff
diff options
context:
space:
mode:
authorShulhan <ms@kilabit.info>2021-01-14 00:24:24 +0700
committerShulhan <ms@kilabit.info>2021-01-20 02:14:39 +0700
commit4f51cd8d9dee4fe57bf060ae66b9dd180e2625d3 (patch)
treeaa5a131727ac33e3ebf1835d4045f2577fab98f6
parent312d77f0905c7c98d9c76c99946b2cf1e24353c4 (diff)
downloadrescached-4f51cd8d9dee4fe57bf060ae66b9dd180e2625d3.tar.xz
all: rewrite the web user interface and the APIs
-rw-r--r--Makefile2
-rw-r--r--README.adoc25
-rw-r--r--_doc/rescached.cfg.5.gzbin2636 -> 2643 bytes
-rw-r--r--_doc/rescached.cfg.adoc8
-rw-r--r--_www/environment/index.html327
-rw-r--r--_www/favicon.png (renamed from _www/public/favicon.png)bin12685 -> 12685 bytes
-rw-r--r--_www/hosts.d/index.html258
-rw-r--r--_www/hosts_blocks/index.html133
-rw-r--r--_www/index.css139
-rw-r--r--_www/index.html131
-rw-r--r--_www/index.js28
-rw-r--r--_www/package-lock.json627
-rw-r--r--_www/package.json22
-rw-r--r--_www/public/build/bundle.css13
-rw-r--r--_www/public/build/bundle.css.map32
-rw-r--r--_www/public/build/bundle.js8001
-rw-r--r--_www/public/build/bundle.js.map1
-rw-r--r--_www/public/global.css79
-rw-r--r--_www/public/index.html17
-rw-r--r--_www/rescached.js289
-rw-r--r--_www/rollup.config.js71
-rw-r--r--_www/src/App.svelte116
-rw-r--r--_www/src/Dashboard.svelte136
-rw-r--r--_www/src/Environment.svelte286
-rw-r--r--_www/src/HostsBlock.svelte112
-rw-r--r--_www/src/HostsDir.svelte250
-rw-r--r--_www/src/MasterDir.svelte532
-rw-r--r--_www/src/common.js29
-rw-r--r--_www/src/environment.js19
-rw-r--r--_www/src/main.js7
-rw-r--r--_www/zone.d/index.html513
-rw-r--r--environment.go4
-rw-r--r--go.mod4
-rw-r--r--go.sum4
-rw-r--r--httpd.go415
-rw-r--r--internal/generate_memfs.go2
-rw-r--r--rescached.1.gzbin5613 -> 5619 bytes
-rw-r--r--rescached.go17
38 files changed, 2091 insertions, 10558 deletions
diff --git a/Makefile b/Makefile
index e9c2824..6b80a87 100644
--- a/Makefile
+++ b/Makefile
@@ -87,7 +87,7 @@ clean:
install-common:
mkdir -p $(PREFIX)/etc/rescached
mkdir -p $(PREFIX)/etc/rescached/hosts.d
- mkdir -p $(PREFIX)/etc/rescached/master.d
+ mkdir -p $(PREFIX)/etc/rescached/zone.d
cp $(RESCACHED_CFG) $(PREFIX)/etc/rescached/
cp testdata/localhost.cert.pem $(PREFIX)/etc/rescached/
cp testdata/localhost.key.pem $(PREFIX)/etc/rescached/
diff --git a/README.adoc b/README.adoc
index ff01fb6..f9debc7 100644
--- a/README.adoc
+++ b/README.adoc
@@ -1,7 +1,7 @@
RESCACHED(1)
============
-M. Shulhan
-2018.09.29
+M. Shulhan <ms@kilabit.info>
+20 January 2021
:doctype: manpage
:mansource: rescached
:manmanual: rescached
@@ -41,14 +41,15 @@ List of current features,
- Enable to handle request from UDP and TCP connections
- Enable to forward request using UDP or TCP
-- Load and serve addresses and hostnames in +/etc/hosts+
+- Load and serve addresses and host names in +/etc/hosts+
- Load and serve hosts formatted files inside directory
+/etc/rescached/hosts.d/+
- Blocking ads and/or malicious websites through host list in
+/etc/rescached/hosts.d/+
-- Support loading and serving master (zone) file format from
- +/etc/rescached/master.d+
+- Support loading and serving zone file format from
+ +/etc/rescached/zone.d+
- Integration with openresolv
+- Support DNS over TLS (DoH) (RFC 7858)
- Support DNS over HTTPS (DoH) (RFC 8484)
@@ -56,7 +57,7 @@ List of current features,
When you open a website, let say 'kilabit.info', in a browser, the first thing
that browser do is to translate name address 'kilabit.info' into an internet
-address (for example to 35.240.172.103) so browser can make a connection to
+address (for example to 18.136.35.199) so browser can make a connection to
'kilabit.info' server.
How browser do that?
@@ -245,16 +246,16 @@ invoking,
All rescached configuration located in file +/etc/rescached/rescached.cfg+.
See manual page of *rescached.cfg*(5) for more information.
-=== MASTER FILE
+=== ZONE FILE
-Rescached support loading master file format.
+Rescached support loading zone file format.
Unlike hosts file format, where each domain name is only mapped to type A
-(IPv4 address), in master file, one can define other type that known to
+(IPv4 address), in zone file, one can define other type that known to
rescached.
-All files defined +dir.master+ configuration are considered as master file and
+All files defined +zone.d+ configuration are considered as zone file and
will be loaded by rescached only if the configuration is not empty.
-Example of master file,
+Example of zone file,
----
$ORIGIN my-site.vm.
@@ -294,7 +295,7 @@ The third RR define a mapping for another relative subdomain
The last RR define a mapping for absolute domain "my-site.com." to IP
address "10.8.0.1".
-For more information about format of master file see RFC 1035 section 5.
+For more information about format of zone file see RFC 1035 section 5.
=== INTEGRATION WITH OPENRESOLV
diff --git a/_doc/rescached.cfg.5.gz b/_doc/rescached.cfg.5.gz
index d40a07c..da08b6e 100644
--- a/_doc/rescached.cfg.5.gz
+++ b/_doc/rescached.cfg.5.gz
Binary files differ
diff --git a/_doc/rescached.cfg.adoc b/_doc/rescached.cfg.adoc
index 5f24a06..f8739d3 100644
--- a/_doc/rescached.cfg.adoc
+++ b/_doc/rescached.cfg.adoc
@@ -200,16 +200,16 @@ Its value must be negative and greater or equal than -1 hour (-3600 seconds).
== FILES
-[[dir.hosts]]
+[[hosts.d]]
=== /etc/rescached/hosts.d
Path to hosts directory where rescached will load all hosts formatted files.
-[[dir.master]]
-=== /etc/rescached/master.d
+[[zone.d]]
+=== /etc/rescached/zone.d
-Path to master directory where rescached will load all master files.
+Path to zone directory where rescached will load all zone files.
== EXAMPLE
diff --git a/_www/environment/index.html b/_www/environment/index.html
new file mode 100644
index 0000000..bca0535
--- /dev/null
+++ b/_www/environment/index.html
@@ -0,0 +1,327 @@
+<!DOCTYPE html>
+<html lang="en">
+ <head>
+ <meta charset="utf-8" />
+ <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
+ <link rel="icon" type="image/png" href="/favicon.png" />
+ <link rel="stylesheet" href="/index.css" />
+ <title>rescached | Environment</title>
+
+ <style>
+ .input > label {
+ width: 11em;
+ display: inline-block;
+ }
+ .input > input,
+ .input > select {
+ width: calc(100% - 13em);
+ display: inline-block;
+ }
+ .input {
+ margin-top: 1em;
+ }
+ .input-deletable {
+ width: 100%;
+ }
+ .input-deletable > input {
+ max-width: calc(100% - 6em);
+ }
+ .input-deletable > button {
+ width: 5em;
+ }
+ .input-checkbox {
+ width: calc(100% - 13em);
+ display: inline-block;
+ }
+ .input-checkbox input[type="checkbox"] {
+ width: auto;
+ }
+ .section-bottom {
+ margin: 2em 0px;
+ }
+ </style>
+ </head>
+ <body onload="onLoad()">
+ <nav class="menu">
+ <a href="/"> rescached </a>
+ /
+ <a href="/environment/" class="active"> Environment </a>
+ /
+ <a href="/hosts_blocks/"> Hosts blocks </a>
+ /
+ <a href="/hosts.d/"> hosts.d </a>
+ /
+ <a href="/zone.d/"> zone.d </a>
+ </nav>
+
+ <div id="notif"></div>
+
+ <div class="environment">
+ <p>
+ This page allow you to change the rescached environment. Upon save, the rescached service
+ will be restarted.
+ </p>
+
+ <h3>rescached</h3>
+
+ <div class="input">
+ <label for="FileResolvConf"> System resolv.conf </label>
+ <input id="FileResolvConf" oninput="onInput('FileResolvConf', this.value)" />
+ <span class="input-info-toggler" onclick="toggleInfo('FileResolvConf_info')">?</span>
+ <div id="FileResolvConf_info" class="input-info" style="display: none">
+ A path to dynamically generated resolv.conf(5) by resolvconf(8). If set, the
+ nameserver values in referenced file will replace 'parent' value and 'parent' will
+ become a fallback in case the referenced file being deleted or can not be parsed.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="Debug"> Debug level </label>
+ <input id="Debug" type="number" min="0" max="3" oninput="onInput('Debug', this.value)" />
+ <span class="input-info-toggler" onclick="toggleInfo('Debug_info')">?</span>
+ <div id="Debug_info" class="input-info" style="display: none">
+ This option only used for debugging program or if user want to monitor what kind of
+ traffic goes in and out of rescached.
+ </div>
+ </div>
+
+ <h3>DNS server</h3>
+
+ <div class="input">
+ <label for="nameservers"> Parent name servers </label>
+ <span class="input-info-toggler" onclick="toggleInfo('nameservers_info')">?</span>
+ <div id="nameservers_info" class="input-info" style="display: none">
+ List of parent DNS servers.
+ </div>
+ <div id="nameservers"></div>
+ <button onclick="handleAddNameserver()">Add</button>
+ </div>
+
+ <div class="input">
+ <label for="ListenAddress"> Listen address </label>
+ <input id="ListenAddress" oninput="onInput('ListenAddress', this.value)" />
+ <span class="input-info-toggler" onclick="toggleInfo('ListenAddress_info')">?</span>
+ <div id="ListenAddress_info" class="input-info" style="display: none">
+ Address in local network where rescached will listening for query from client
+ through UDP and TCP.
+ <br />
+ If you want rescached to serve a query from another host in your local network,
+ change this value to <tt>0.0.0.0:53</tt>.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="HTTPPort"> HTTP listen port </label>
+ <input
+ id="HTTPPort"
+ type="number"
+ min="0"
+ max="65535"
+ oninput="onInput('HTTPPort', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('HTTPPort_info')">?</span>
+ <div id="HTTPPort_info" class="input-info" style="display: none">
+ Port to serve DNS over HTTP
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="TLSPort"> TLS listen port </label>
+ <input
+ id="TLSPort"
+ type="number"
+ min="0"
+ max="65535"
+ oninput="onInput('TLSPort', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('TLSPort_info')">?</span>
+ <div id="TLSPort_info" class="input-info" style="display: none">
+ Port to serve DNS over TLS
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="TLSCertFile"> TLS certificate </label>
+ <input
+ id="TLSCertFile"
+ placeholder="/path/to/certificate"
+ oninput="onInput('TLSCertFile', this.value)"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('TLSCertFile_info')">?</span>
+ <div id="TLSCertFile_info" class="input-info" style="display: none">
+ Path to certificate file to serve DNS over TLS and HTTPS
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="TLSPrivateKey"> TLS private key </label>
+ <input
+ id="TLSPrivateKey"
+ placeholder="/path/to/certificate/private.key"
+ oninput="onInput('TLSPrivateKey', this.value)"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('TLSPrivateKey_info')">?</span>
+ <div id="TLSPrivateKey_info" class="input-info" style="display: none">
+ Path to certificate private key file to serve DNS over TLS and HTTPS.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="TLSAllowInsecure"> TLS allow insecure </label>
+ <div class="input-checkbox">
+ <input
+ id="TLSAllowInsecure"
+ type="checkbox"
+ oninput="onInput('TLSAllowInsecure', this.checked)"
+ />
+ <span class="suffix"> Yes </span>
+ </div>
+ <span class="input-info-toggler" onclick="toggleInfo('TLSAllowInsecure_info')">?</span>
+ <div id="TLSAllowInsecure_info" class="input-info" style="display: none">
+ If its true, allow serving DoH and DoT with self signed certificate.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="DoHBehindProxy"> DoH behind proxy </label>
+ <div class="input-checkbox">
+ <input
+ id="DoHBehindProxy"
+ type="checkbox"
+ oninput="onInput('DoHBehindProxy', this.checked)"
+ />
+ <span class="suffix"> Yes </span>
+ </div>
+ <span class="input-info-toggler" onclick="toggleInfo('DoHBehindProxy_info')">?</span>
+ <div id="DoHBehindProxy_info" class="input-info" style="display: none">
+ If its true, serve DNS over HTTP only, even if certificate files is defined. This
+ allow serving DNS request forwarded by another proxy server.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="PruneDelay"> Prune delay </label>
+ <input
+ id="PruneDelay"
+ type="number"
+ min="3600"
+ max="36000"
+ oninput="onInput('PruneDelay', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('PruneDelay_info')">?</span>
+ <div id="PruneDelay_info" class="input-info" style="display: none">
+ Delay for pruning caches. Every N seconds, rescached will traverse all caches and
+ remove response that has not been accessed less than cache.prune_threshold. Its
+ value must be equal or greater than 1 hour (3600 seconds).
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="PruneThreshold"> Prune threshold </label>
+ <input
+ id="PruneThreshold"
+ type="number"
+ min="-36000"
+ max="-3600"
+ oninput="onInput('PruneThreshold', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('PruneThreshold_info')">?</span>
+ <div id="PruneThreshold_info" class="input-info" style="display: none">
+ The duration when the cache will be considered expired. Its value must be negative
+ and greater or equal than -1 hour (-3600 seconds).
+ </div>
+ </div>
+
+ <div class="section-bottom">
+ <div>
+ <button onclick="handleSave()">Save</button>
+ </div>
+ </div>
+ </div>
+
+ <script src="/index.js"></script>
+ <script src="/rescached.js"></script>
+ <script>
+ let resc = null
+
+ async function getEnvironment() {
+ const res = await resc.getEnvironment()
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ const env = res.data
+
+ // Set all input values using env data.
+ document.getElementById("FileResolvConf").value = env.FileResolvConf
+ document.getElementById("Debug").value = env.Debug
+
+ renderNameservers()
+
+ document.getElementById("ListenAddress").value = env.ListenAddress
+ document.getElementById("HTTPPort").value = env.HTTPPort
+ document.getElementById("TLSPort").value = env.TLSPort
+ document.getElementById("TLSCertFile").value = env.TLSCertFile
+ document.getElementById("TLSPrivateKey").value = env.TLSPrivateKey
+
+ document.getElementById("TLSAllowInsecure").checked = env.TLSAllowInsecure
+ document.getElementById("DoHBehindProxy").checked = env.DoHBehindProxy
+
+ document.getElementById("PruneDelay").value = env.PruneDelay
+ document.getElementById("PruneThreshold").value = env.PruneThreshold
+ }
+
+ function handleAddNameserver() {
+ resc.env.NameServers.push("")
+ renderNameservers()
+ }
+
+ async function handleSave() {
+ console.log("handleSave: ", resc.env)
+ let res = await resc.updateEnvironment()
+ if (res.code !== 200) {
+ notifError(res.message)
+ return
+ }
+ notifInfo("Environment has been saved!")
+ }
+
+ function onDeleteNameserver(x) {
+ resc.env.NameServers.splice(x, 1)
+ renderNameservers()
+ }
+
+ function onInput(key, value) {
+ console.log("onInput ", key, ": ", value)
+ resc.env[key] = value
+ }
+
+ function onInputNameserver(x, newv) {
+ resc.env.NameServers[x] = newv
+ }
+
+ function onLoad() {
+ resc = new Rescached("")
+ getEnvironment()
+ }
+
+ function renderNameservers() {
+ let listNS = document.getElementById("nameservers")
+ listNS.innerHTML = ""
+
+ for (let x = 0; x < resc.env.NameServers.length; x++) {
+ let ns = resc.env.NameServers[x]
+
+ let el = document.createElement("div")
+ el.classList.add("input-deletable")
+ el.innerHTML = `
+ <input value="${ns}" oninput="onInputNameserver(${x}, this.value)">
+ <button onclick="onDeleteNameserver(${x})">
+ Delete
+ </button>`
+ listNS.appendChild(el)
+ }
+ }
+ </script>
+ </body>
+</html>
diff --git a/_www/public/favicon.png b/_www/favicon.png
index 0a3c077..0a3c077 100644
--- a/_www/public/favicon.png
+++ b/_www/favicon.png
Binary files differ
diff --git a/_www/hosts.d/index.html b/_www/hosts.d/index.html
new file mode 100644
index 0000000..40d63eb
--- /dev/null
+++ b/_www/hosts.d/index.html
@@ -0,0 +1,258 @@
+<!DOCTYPE html>
+<html lang="en">
+ <head>
+ <meta charset="utf-8" />
+ <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
+ <link rel="icon" type="image/png" href="/favicon.png" />
+ <link rel="stylesheet" href="/index.css" />
+ <title>rescached | hosts.d</title>
+
+ <style>
+ .nav-left {
+ padding: 0px;
+ width: 280px;
+ float: left;
+ }
+ .nav-left .item {
+ margin: 4px 0px;
+ }
+ #activeHostsFile {
+ float: left;
+ width: calc(100% - 300px);
+ }
+ .host {
+ font-family: monospace;
+ width: 100%;
+ }
+ .host.header {
+ margin: 1em 0px;
+ font-weight: bold;
+ border-bottom: 1px solid silver;
+ }
+ .host_name {
+ display: inline-block;
+ width: 240px;
+ word-wrap: break-word;
+ }
+ .host_value {
+ display: inline-block;
+ width: 140px;
+ }
+ </style>
+ </head>
+ <body onload="onLoad()">
+ <nav class="menu">
+ <a href="/"> rescached </a>
+ /
+ <a href="/environment/"> Environment </a>
+ /
+ <a href="/hosts_blocks/"> Hosts blocks </a>
+ /
+ <a href="/hosts.d/" class="active"> hosts.d </a>
+ /
+ <a href="/zone.d/"> zone.d </a>
+ </nav>
+
+ <div id="notif"></div>
+
+ <div class="hosts_d">
+ <div class="nav-left">
+ <h3>Hosts files</h3>
+ <div id="HostsFiles"></div>
+
+ <br />
+
+ <label>
+ <span>New hosts file:</span>
+ <br />
+ <input id="newHostsFile" oninput="onInputNewHostsFile(this.value)" />
+ </label>
+ <button onclick="onCreateHostsFile()">Create</button>
+ </div>
+
+ <div id="activeHostsFile">
+ <p>Select one of the hosts file to manage.</p>
+ </div>
+ </div>
+
+ <script src="/index.js"></script>
+ <script src="/rescached.js"></script>
+ <script>
+ let resc = null
+ let activeHostsFile = null
+ let newHostsFile = ""
+ let newRecord = {
+ Name: "",
+ Value: "",
+ }
+
+ async function getHostsFile(name) {
+ activeHostsFile = resc.env.HostsFiles[name]
+ if (typeof activeHostsFile.Records === "undefined") {
+ activeHostsFile.Records = []
+ }
+ if (activeHostsFile.Records === null) {
+ activeHostsFile.Records = []
+ }
+ if (activeHostsFile.Records.length === 0) {
+ const res = await resc.HostsFileGet(name)
+ activeHostsFile.Records = res.data
+ }
+ renderHostsFile(activeHostsFile)
+ newRecord.Name = ""
+ newRecord.Value = ""
+ }
+
+ async function onCreateHostsFile() {
+ if (newHostsFile === "") {
+ notifError("Please fill the hosts file name first")
+ return
+ }
+
+ let res = await resc.HostsFileCreate(newHostsFile)
+
+ if (res.code >= 400) {
+ notifError("ERROR: HostsFileCreate: " + res.message)
+ return
+ }
+ renderHostsFiles(resc.env.HostsFiles)
+ notifInfo(res.message)
+ resetInputs()
+ }
+
+ async function onDeleteActiveHostsFile() {
+ const res = await resc.HostsFileDelete(activeHostsFile.Name)
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ renderHostsFiles(resc.env.HostsFiles)
+ document.getElementById(
+ "activeHostsFile",
+ ).innerHTML = `<p>Select one of the hosts file to manage.</p>`
+ notifInfo(`Hosts file "${activeHostsFile.Name}" has been deleted`)
+ activeHostsFile = null
+ }
+
+ async function onCreateRecord(hostsFile) {
+ if (hostsFile === "") {
+ notifError("invalid or empty hosts file name: " + hostsFile)
+ return
+ }
+ if (newRecord.Name === "") {
+ notifError("invalid or empty domain name")
+ return
+ }
+ if (newRecord.Value === "") {
+ notifError("invalid or empty IP address")
+ return
+ }
+ let res = await resc.HostsFileRecordAdd(hostsFile, newRecord.Name, newRecord.Value)
+ if (res.code >= 400) {
+ notifError("failed to add record for " + hostsFile + ": " + res.message)
+ return
+ }
+ renderNewRecord(res.data)
+ }
+
+ async function onDeleteRecord(domain) {
+ let res = await resc.HostsFileRecordDelete(activeHostsFile.Name, domain)
+ if (res.code !== 200) {
+ notifError("Failed to delete record " + domain)
+ return
+ }
+ activeHostsFile = resc.env.HostsFiles[activeHostsFile.Name]
+ renderHostsFile(activeHostsFile)
+ }
+
+ function onInputNewHostsFile(v) {
+ newHostsFile = v
+ }
+
+ function onInputNewRecord(k, v) {
+ newRecord[k] = v
+ }
+
+ async function onLoad() {
+ resc = new Rescached("")
+
+ let res = await resc.getEnvironment()
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+
+ renderHostsFiles(res.data.HostsFiles)
+ resetInputs()
+ }
+
+ function renderHostsFile(hf) {
+ let content = document.getElementById("activeHostsFile")
+ let innerHTML = `
+ <p>
+ ${hf.Name} (${hf.Records.length} records)
+ <button onclick="onDeleteActiveHostsFile()">Delete</button>
+ </p>
+ <div class="host">
+ <input class="host_name" placeholder="Domain name" value=""
+ oninput="onInputNewRecord('Name', this.value)"
+ />
+ <input class="host_value" placeholder="IP address" value=""
+ oninput="onInputNewRecord('Value', this.value)"
+ />
+ <button onclick="onCreateRecord('${hf.Name}')">Create</button>
+ </div>
+ <div class="host header">
+ <span class="host_name"> Domain name </span>
+ <span class="host_value"> IP address </span>
+ </div>
+ <div id="records">
+ `
+ for (let x = 0; x < hf.Records.length; x++) {
+ let rr = hf.Records[x]
+ innerHTML += `
+ <div class="host">
+ <span class="host_name"> ${rr.Name} </span>
+ <span class="host_value"> ${rr.Value} </span>
+ <button onclick="onDeleteRecord('${rr.Name}')">X</button>
+ </div>`
+ }
+ innerHTML += "</div>"
+ content.innerHTML = innerHTML
+ }
+
+ function renderHostsFiles(hostsFiles) {
+ let parent = document.getElementById("HostsFiles")
+ parent.innerHTML = ""
+
+ for (let k in hostsFiles) {
+ if (!hostsFiles.hasOwnProperty(k)) {
+ continue
+ }
+ let hf = hostsFiles[k]
+ let item = document.createElement("div")
+ item.classList.add("item")
+ item.innerHTML = `<a href="#" onclick="getHostsFile('${k}')"> ${hf.Name} </a>`
+ parent.appendChild(item)
+ }
+ }
+
+ // renderNewRecord prepend the new record on top of the list.
+ function renderNewRecord(rr) {
+ let div = document.getElementById("records")
+ innerHTML = `
+ <div class="host">
+ <span class="host_name"> ${rr.Name} </span>
+ <span class="host_value"> ${rr.Value} </span>
+ <button onclick="onDeleteRecord('${rr.Name}')">X</button>
+ </div>`
+ div.innerHTML = innerHTML + div.innerHTML
+ }
+
+ function resetInputs() {
+ document.getElementById("newHostsFile").value = ""
+ newHostsFile = ""
+ }
+ </script>
+ </body>
+</html>
diff --git a/_www/hosts_blocks/index.html b/_www/hosts_blocks/index.html
new file mode 100644
index 0000000..10c3d99
--- /dev/null
+++ b/_www/hosts_blocks/index.html
@@ -0,0 +1,133 @@
+<!DOCTYPE html>
+<html lang="en">
+ <head>
+ <meta charset="utf-8" />
+ <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
+ <link rel="icon" type="image/png" href="/favicon.png" />
+ <link rel="stylesheet" href="/index.css" />
+ <title>rescached | hosts blocks</title>
+
+ <style>
+ .block_source {
+ width: calc(100% - 2em);
+ overflow: auto;
+ }
+ .block_source input:disabled {
+ color: black;
+ }
+ .item {
+ width: 100%;
+ margin-bottom: 1em;
+ }
+ .item.header {
+ font-weight: bold;
+ margin-bottom: 1em;
+ border-bottom: 1px solid silver;
+ }
+ .item .is-enabled {
+ display: inline-block;
+ width: 4em;
+ vertical-align: top;
+ }
+ .item .info {
+ display: inline-block;
+ width: calc(100% - 6em);
+ }
+ .item .info input {
+ width: calc(100% - 4em);
+ }
+ </style>
+ </head>
+ <body onload="onLoad()">
+ <nav class="menu">
+ <a href="/"> rescached </a>
+ /
+ <a href="/environment/"> Environment </a>
+ /
+ <a href="/hosts_blocks/" class="active"> Hosts blocks </a>
+ /
+ <a href="/hosts.d/"> hosts.d </a>
+ /
+ <a href="/zone.d/"> zone.d </a>
+ </nav>
+
+ <div id="notif"></div>
+
+ <p>Configure the source of blocked hosts file.</p>
+
+ <div class="block_source">
+ <div class="item header">
+ <span class="is-enabled"> Enabled </span>
+ <span class="info"> Name </span>
+ </div>
+ </div>
+ <div id="HostsBlocks" class="block_source"></div>
+
+ <div>
+ <button onclick="updateHostsBlocks()">Save</button>
+ </div>
+
+ <script src="/index.js"></script>
+ <script src="/rescached.js"></script>
+ <script>
+ let resc = null
+
+ function onLoad() {
+ resc = new Rescached("")
+ getEnvironment()
+ }
+
+ async function getEnvironment() {
+ const res = await resc.getEnvironment()
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+
+ let env = res.data
+ renderHostsBlocks(env.HostsBlocks)
+ }
+
+ function onCheckHostblock(idx, val) {
+ resc.env.HostsBlocks[idx].IsEnabled = val
+ }
+
+ async function updateHostsBlocks() {
+ const res = await resc.updateHostsBlocks(resc.env.HostsBlocks)
+ if (res.code != 200) {
+ notifError("updateHostsBlocks: ", res.message)
+ return
+ }
+
+ renderHostsBlocks(res.data)
+ notifInfo("The hosts blocks has been updated.")
+ }
+
+ function renderHostsBlocks(hostsBlocks) {
+ let parent = document.getElementById("HostsBlocks")
+ parent.innerHTML = ""
+
+ for (let x = 0; x < hostsBlocks.length; x++) {
+ let hostsBlock = hostsBlocks[x]
+ let item = document.createElement("div")
+ item.classList.add("item")
+ item.innerHTML = `
+ <span class="is-enabled">
+ <input
+ type="checkbox"
+ ${hostsBlock.IsEnabled ? "checked" : ""}
+ oninput="onCheckHostblock('${x}', this.checked)"
+ />
+ </span>
+ <span class="info">
+ <div>${hostsBlock.Name}</div>
+ <div> URL: <input value="${hostsBlock.URL}" disabled /> </div>
+ <div> Last updated at ${hostsBlock.LastUpdated} </div>
+ </span>`
+
+ parent.appendChild(item)
+ }
+ }
+ </script>
+ </body>
+</html>
diff --git a/_www/index.css b/_www/index.css
new file mode 100644
index 0000000..34dd919
--- /dev/null
+++ b/_www/index.css
@@ -0,0 +1,139 @@
+html,
+body {
+ position: relative;
+ width: 100%;
+ height: 100%;
+}
+
+body {
+ background-color: floralwhite;
+ color: #333;
+ margin: 0;
+ padding: 8px;
+ box-sizing: border-box;
+ font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto,
+ Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif;
+}
+
+a {
+ color: rgb(0, 100, 200);
+ text-decoration: none;
+}
+
+a:hover {
+ text-decoration: underline;
+}
+
+a:visited {
+ color: rgb(0, 80, 160);
+}
+
+input,
+button,
+select,
+textarea {
+ font-family: inherit;
+ font-size: inherit;
+ padding: 0.4em;
+ margin: 0.5em 0;
+ box-sizing: border-box;
+ border: 1px solid #ccc;
+ border-radius: 2px;
+}
+
+input:disabled {
+ color: #ccc;
+}
+
+input[type="range"] {
+ height: 0;
+}
+
+button {
+ color: #333;
+ background-color: lavender;
+ outline: none;
+}
+
+button:disabled {
+ color: #999;
+}
+
+button:not(:disabled):active {
+ background-color: #ddd;
+}
+
+button:focus {
+ border-color: #666;
+}
+
+h1,
+h2 {
+ color: #ff3e00;
+ text-transform: uppercase;
+ font-weight: 100;
+}
+
+body {
+ margin: 0 auto;
+ width: 800px;
+ padding: 1em;
+}
+
+#notif {
+ position: fixed;
+ top: 1em;
+ width: 70%;
+}
+#notif > .error {
+ background-color: salmon;
+ padding: 1em;
+}
+#notif > .info {
+ background-color: lightblue;
+ padding: 1em;
+}
+
+nav.menu {
+ color: #ff3e00;
+ text-transform: uppercase;
+ font-weight: 100;
+ margin-bottom: 2em;
+}
+.active {
+ padding-bottom: 4px;
+ border-bottom: 4px solid #ff3e00;
+}
+
+.input > label {
+ width: 8em;
+ display: inline-block;
+}
+.input > input,
+.input > select {
+ width: calc(100% - 11em);
+ display: inline-block;
+}
+.input > .input-info-toggler {
+ border-radius: 50%;
+ border: 1px solid grey;
+ cursor: pointer;
+ display: inline-block;
+ font-size: 12px;
+ height: 14px;
+ line-height: 14px;
+ padding: 2px;
+ text-align: center;
+ width: 14px;
+}
+.input > .input-info {
+ background-color: #eee;
+ margin: 8px 0px;
+ padding: 1em;
+}
+
+@media (max-width: 900px) {
+ body {
+ width: calc(100% - 2em);
+ }
+}
diff --git a/_www/index.html b/_www/index.html
new file mode 100644
index 0000000..3403751
--- /dev/null
+++ b/_www/index.html
@@ -0,0 +1,131 @@
+<!DOCTYPE html>
+<html lang="en">
+ <head>
+ <meta charset="utf-8" />
+ <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
+ <link rel="icon" type="image/png" href="/favicon.png" />
+ <link rel="stylesheet" href="/index.css" />
+ <title>rescached</title>
+
+ <style>
+ .message {
+ padding: 1em 0px;
+ border-bottom: 1px solid silver;
+ }
+ .rr {
+ margin-left: 1em;
+ width: 100%;
+ }
+ .rr.header {
+ font-weight: bold;
+ }
+ .rr span {
+ display: inline-block;
+ }
+ .kind {
+ width: 9em;
+ }
+ .type {
+ width: 5em;
+ }
+ .ttl {
+ width: 6em;
+ }
+ .value {
+ word-wrap: anywhere;
+ width: calc(100% - 24em);
+ }
+ </style>
+ </head>
+ <body>
+ <nav class="menu">
+ <a href="/" class="active"> rescached </a>
+ /
+ <a href="/environment/"> Environment </a>
+ /
+ <a href="/hosts_blocks/"> Hosts blocks </a>
+ /
+ <a href="/hosts.d/"> hosts.d </a>
+ /
+ <a href="/zone.d/"> zone.d </a>
+ </nav>
+ <div class="search">
+ Caches:
+ <input name="query" />
+ <button onclick="doSearch()">Search</button>
+ </div>
+ <div id="result"></div>
+ <div id="notif"></div>
+
+ <script src="/index.js"></script>
+ <script src="/rescached.js"></script>
+ <script>
+ const resc = new Rescached("")
+
+ async function doSearch() {
+ const query = document.getElementsByName("query")[0].value
+ console.log("doSearch: ", query)
+ const res = await resc.Search(query)
+ if (res.code != 200) {
+ notifError(`doSearch ${query}: ${res.message}`)
+ return
+ }
+ onSearchResult(res.data)
+ }
+
+ function onSearchResult(dnsRecords) {
+ const elResult = document.getElementById("result")
+ elResult.innerHTML = ""
+
+ for (let x = 0; x < dnsRecords.length; x++) {
+ const record = dnsRecords[x]
+ const divRecord = document.createElement("div")
+ divRecord.classList.add("message")
+ innerHTML = `
+ <div class="qname">
+ ${record.Question.Name}
+ <button class="b-remove"
+ onclick="doRemove('${record.Question.Name}')"
+ >Remove from cache</button>
+ </div>
+ <div class="rr header">
+ <span class="kind"></span>
+ <span class="type"> Type </span>
+ <span class="ttl"> TTL </span>
+ <span class="value"> Value </span>
+ </div>
+ `
+ if (record.Answer !== null && record.Answer.length > 0) {
+ innerHTML += renderRR(record.Answer, "Answer")
+ }
+ if (record.Authority !== null && record.Authority.length > 0) {
+ innerHTML += renderRR(record.Authority, "Authority")
+ }
+ if (record.Additional !== null && record.Additional.length > 0) {
+ innerHTML += renderRR(record.Additional, "Additional")
+ }
+
+ divRecord.innerHTML = innerHTML
+ elResult.appendChild(divRecord)
+ }
+ }
+
+ function renderRR(listRR, title) {
+ let innerHTML = ""
+
+ for (let x = 0; x < listRR.length; x++) {
+ const rr = listRR[x]
+ innerHTML += `
+ <div class="rr">
+ <span class="kind"> ${title} </span>
+ <span class="type"> ${getRRTypeName(rr.Type)} </span>
+ <span class="ttl"> ${rr.TTL} </span>
+ <span class="value"> ${JSON.stringify(rr.Value, null, 2)} </span>
+ </div>
+ `
+ }
+ return innerHTML
+ }
+ </script>
+ </body>
+</html>
diff --git a/_www/index.js b/_www/index.js
new file mode 100644
index 0000000..2f4ed6e
--- /dev/null
+++ b/_www/index.js
@@ -0,0 +1,28 @@
+function notifError(msg) {
+ displayNotif("error", msg)
+}
+
+function notifInfo(msg) {
+ displayNotif("info", msg)
+}
+
+function displayNotif(className, msg) {
+ let notif = document.getElementById("notif")
+ let el = document.createElement("div")
+ el.classList.add(className)
+ el.innerHTML = msg
+ notif.appendChild(el)
+
+ setTimeout(function () {
+ notif.removeChild(notif.children[0])
+ }, 5000)
+}
+
+function toggleInfo(id) {
+ let el = document.getElementById(id)
+ if (el.style.display === "none") {
+ el.style.display = "block"
+ } else {
+ el.style.display = "none"
+ }
+}
diff --git a/_www/package-lock.json b/_www/package-lock.json
deleted file mode 100644
index 7078c4e..0000000
--- a/_www/package-lock.json
+++ /dev/null
@@ -1,627 +0,0 @@
-{
- "name": "rescached-wui",
- "version": "1.0.0",
- "lockfileVersion": 1,
- "requires": true,
- "dependencies": {
- "@babel/code-frame": {
- "version": "7.10.4",
- "resolved": "https://registry.npmjs.org/@babel/code-frame/-/code-frame-7.10.4.tgz",
- "integrity": "sha512-vG6SvB6oYEhvgisZNFRmRCUkLz11c7rp+tbNTynGqc6mS1d5ATd/sGyV6W0KZZnXRKMTzZDRgQT3Ou9jhpAfUg==",
- "dev": true,
- "requires": {
- "@babel/highlight": "^7.10.4"
- }
- },
- "@babel/helper-validator-identifier": {
- "version": "7.10.4",
- "resolved": "https://registry.npmjs.org/@babel/helper-validator-identifier/-/helper-validator-identifier-7.10.4.tgz",
- "integrity": "sha512-3U9y+43hz7ZM+rzG24Qe2mufW5KhvFg/NhnNph+i9mgCtdTCtMJuI1TMkrIUiK7Ix4PYlRF9I5dhqaLYA/ADXw==",
- "dev": true
- },
- "@babel/highlight": {
- "version": "7.10.4",
- "resolved": "https://registry.npmjs.org/@babel/highlight/-/highlight-7.10.4.tgz",
- "integrity": "sha512-i6rgnR/YgPEQzZZnbTHHuZdlE8qyoBNalD6F+q4vAFlcMEcqmkoG+mPqJYJCo63qPf74+Y1UZsl3l6f7/RIkmA==",
- "dev": true,
- "requires": {
- "@babel/helper-validator-identifier": "^7.10.4",
- "chalk": "^2.0.0",
- "js-tokens": "^4.0.0"
- }
- },
- "@polka/url": {
- "version": "0.5.0",
- "resolved": "https://registry.npmjs.org/@polka/url/-/url-0.5.0.tgz",
- "integrity": "sha512-oZLYFEAzUKyi3SKnXvj32ZCEGH6RDnao7COuCVhDydMS9NrCSVXhM79VaKyP5+Zc33m0QXEd2DN3UkU7OsHcfw=="
- },
- "@rollup/plugin-commonjs": {
- "version": "11.0.2",
- "resolved": "https://registry.npmjs.org/@rollup/plugin-commonjs/-/plugin-commonjs-11.0.2.tgz",
- "integrity": "sha512-MPYGZr0qdbV5zZj8/2AuomVpnRVXRU5XKXb3HVniwRoRCreGlf5kOE081isNWeiLIi6IYkwTX9zE0/c7V8g81g==",
- "dev": true,
- "requires": {
- "@rollup/pluginutils": "^3.0.0",
- "estree-walker": "^1.0.1",
- "is-reference": "^1.1.2",
- "magic-string": "^0.25.2",
- "resolve": "^1.11.0"
- }
- },
- "@rollup/plugin-node-resolve": {
- "version": "7.1.3",
- "resolved": "https://registry.npmjs.org/@rollup/plugin-node-resolve/-/plugin-node-resolve-7.1.3.tgz",
- "integrity": "sha512-RxtSL3XmdTAE2byxekYLnx+98kEUOrPHF/KRVjLH+DEIHy6kjIw7YINQzn+NXiH/NTrQLAwYs0GWB+csWygA9Q==",
- "dev": true,
- "requires": {
- "@rollup/pluginutils": "^3.0.8",
- "@types/resolve": "0.0.8",
- "builtin-modules": "^3.1.0",
- "is-module": "^1.0.0",
- "resolve": "^1.14.2"
- }
- },
- "@rollup/pluginutils": {
- "version": "3.0.10",
- "resolved": "https://registry.npmjs.org/@rollup/pluginutils/-/pluginutils-3.0.10.tgz",
- "integrity": "sha512-d44M7t+PjmMrASHbhgpSbVgtL6EFyX7J4mYxwQ/c5eoaE6N2VgCgEcWVzNnwycIloti+/MpwFr8qfw+nRw00sw==",
- "dev": true,
- "requires": {
- "@types/estree": "0.0.39",
- "estree-walker": "^1.0.1",
- "picomatch": "^2.2.2"
- }
- },
- "@types/estree": {
- "version": "0.0.39",
- "resolved": "https://registry.npmjs.org/@types/estree/-/estree-0.0.39.tgz",
- "integrity": "sha512-EYNwp3bU+98cpU4lAWYYL7Zz+2gryWH1qbdDTidVd6hkiR6weksdbMadyXKXNPEkQFhXM+hVO9ZygomHXp+AIw==",
- "dev": true
- },
- "@types/node": {
- "version": "13.13.5",
- "resolved": "https://registry.npmjs.org/@types/node/-/node-13.13.5.tgz",
- "integrity": "sha512-3ySmiBYJPqgjiHA7oEaIo2Rzz0HrOZ7yrNO5HWyaE5q0lQ3BppDZ3N53Miz8bw2I7gh1/zir2MGVZBvpb1zq9g==",
- "dev": true
- },
- "@types/resolve": {
- "version": "0.0.8",
- "resolved": "https://registry.npmjs.org/@types/resolve/-/resolve-0.0.8.tgz",
- "integrity": "sha512-auApPaJf3NPfe18hSoJkp8EbZzer2ISk7o8mCC3M9he/a04+gbMF97NkpD2S8riMGvm4BMRI59/SZQSaLTKpsQ==",
- "dev": true,
- "requires": {
- "@types/node": "*"
- }
- },
- "acorn": {
- "version": "7.2.0",
- "resolved": "https://registry.npmjs.org/acorn/-/acorn-7.2.0.tgz",
- "integrity": "sha512-apwXVmYVpQ34m/i71vrApRrRKCWQnZZF1+npOD0WV5xZFfwWOmKGQ2RWlfdy9vWITsenisM8M0Qeq8agcFHNiQ==",
- "dev": true
- },
- "ansi-styles": {
- "version": "3.2.1",
- "resolved": "https://registry.npmjs.org/ansi-styles/-/ansi-styles-3.2.1.tgz",
- "integrity": "sha512-VT0ZI6kZRdTh8YyJw3SMbYm/u+NqfsAxEpWO0Pf9sq8/e94WxxOpPKx9FR1FlyCtOVDNOQ+8ntlqFxiRc+r5qA==",
- "dev": true,
- "requires": {
- "color-convert": "^1.9.0"
- }
- },
- "anymatch": {
- "version": "3.1.1",
- "resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.1.tgz",
- "integrity": "sha512-mM8522psRCqzV+6LhomX5wgp25YVibjh8Wj23I5RPkPppSVSjyKD2A2mBJmWGa+KN7f2D6LNh9jkBCeyLktzjg==",
- "dev": true,
- "requires": {
- "normalize-path": "^3.0.0",
- "picomatch": "^2.0.4"
- }
- },
- "async-limiter": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/async-limiter/-/async-limiter-1.0.1.tgz",
- "integrity": "sha512-csOlWGAcRFJaI6m+F2WKdnMKr4HhdhFVBk0H/QbJFMCr+uO2kwohwXQPxw/9OCxp05r5ghVBFSyioixx3gfkNQ==",
- "dev": true
- },
- "binary-extensions": {
- "version": "2.0.0",
- "resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.0.0.tgz",
- "integrity": "sha512-Phlt0plgpIIBOGTT/ehfFnbNlfsDEiqmzE2KRXoX1bLIlir4X/MR+zSyBEkL05ffWgnRSf/DXv+WrUAVr93/ow==",
- "dev": true
- },
- "braces": {
- "version": "3.0.2",
- "resolved": "https://registry.npmjs.org/braces/-/braces-3.0.2.tgz",
- "integrity": "sha512-b8um+L1RzM3WDSzvhm6gIz1yfTbBt6YTlcEKAvsmqCZZFw46z626lVj9j1yEPW33H5H+lBQpZMP1k8l+78Ha0A==",
- "dev": true,
- "requires": {
- "fill-range": "^7.0.1"
- }
- },
- "buffer-from": {
- "version": "1.1.1",
- "resolved": "https://registry.npmjs.org/buffer-from/-/buffer-from-1.1.1.tgz",
- "integrity": "sha512-MQcXEUbCKtEo7bhqEs6560Hyd4XaovZlO/k9V3hjVUF/zwW7KBVdSK4gIt/bzwS9MbR5qob+F5jusZsb0YQK2A==",
- "dev": true
- },
- "builtin-modules": {
- "version": "3.1.0",
- "resolved": "https://registry.npmjs.org/builtin-modules/-/builtin-modules-3.1.0.tgz",
- "integrity": "sha512-k0KL0aWZuBt2lrxrcASWDfwOLMnodeQjodT/1SxEQAXsHANgo6ZC/VEaSEHCXt7aSTZ4/4H5LKa+tBXmW7Vtvw==",
- "dev": true
- },
- "chalk": {
- "version": "2.4.2",
- "resolved": "https://registry.npmjs.org/chalk/-/chalk-2.4.2.tgz",
- "integrity": "sha512-Mti+f9lpJNcwF4tWV8/OrTTtF1gZi+f8FqlyAdouralcFWFQWF2+NgCHShjkCb+IFBLq9buZwE1xckQU4peSuQ==",
- "dev": true,
- "requires": {
- "ansi-styles": "^3.2.1",
- "escape-string-regexp": "^1.0.5",
- "supports-color": "^5.3.0"
- }
- },
- "chokidar": {
- "version": "3.4.0",
- "resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.4.0.tgz",
- "integrity": "sha512-aXAaho2VJtisB/1fg1+3nlLJqGOuewTzQpd/Tz0yTg2R0e4IGtshYvtjowyEumcBv2z+y4+kc75Mz7j5xJskcQ==",
- "dev": true,
- "requires": {
- "anymatch": "~3.1.1",
- "braces": "~3.0.2",
- "fsevents": "~2.1.2",
- "glob-parent": "~5.1.0",
- "is-binary-path": "~2.1.0",
- "is-glob": "~4.0.1",
- "normalize-path": "~3.0.0",
- "readdirp": "~3.4.0"
- }
- },
- "color-convert": {
- "version": "1.9.3",
- "resolved": "https://registry.npmjs.org/color-convert/-/color-convert-1.9.3.tgz",
- "integrity": "sha512-QfAUtd+vFdAtFQcC8CCyYt1fYWxSqAiK2cSD6zDB8N3cpsEBAvRxp9zOGg6G/SHHJYAT88/az/IuDGALsNVbGg==",
- "dev": true,
- "requires": {
- "color-name": "1.1.3"
- }
- },
- "color-name": {
- "version": "1.1.3",
- "resolved": "https://registry.npmjs.org/color-name/-/color-name-1.1.3.tgz",
- "integrity": "sha1-p9BVi9icQveV3UIyj3QIMcpTvCU=",
- "dev": true
- },
- "commander": {
- "version": "2.20.3",
- "resolved": "https://registry.npmjs.org/commander/-/commander-2.20.3.tgz",
- "integrity": "sha512-GpVkmM8vF2vQUkj2LvZmD35JxeJOLCwJ9cUkugyk2nuhbv3+mJvpLYYt+0+USMxE+oj+ey/lJEnhZw75x/OMcQ==",
- "dev": true
- },
- "console-clear": {
- "version": "1.1.1",
- "resolved": "https://registry.npmjs.org/console-clear/-/console-clear-1.1.1.tgz",
- "integrity": "sha512-pMD+MVR538ipqkG5JXeOEbKWS5um1H4LUUccUQG68qpeqBYbzYy79Gh55jkd2TtPdRfUaLWdv6LPP//5Zt0aPQ=="
- },
- "escape-string-regexp": {
- "version": "1.0.5",
- "resolved": "https://registry.npmjs.org/escape-string-regexp/-/escape-string-regexp-1.0.5.tgz",
- "integrity": "sha1-G2HAViGQqN/2rjuyzwIAyhMLhtQ=",
- "dev": true
- },
- "estree-walker": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-1.0.1.tgz",
- "integrity": "sha512-1fMXF3YP4pZZVozF8j/ZLfvnR8NSIljt56UhbZ5PeeDmmGHpgpdwQt7ITlGvYaQukCvuBRMLEiKiYC+oeIg4cg==",
- "dev": true
- },
- "fill-range": {
- "version": "7.0.1",
- "resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.0.1.tgz",
- "integrity": "sha512-qOo9F+dMUmC2Lcb4BbVvnKJxTPjCm+RRpe4gDuGrzkL7mEVl/djYSu2OdQ2Pa302N4oqkSg9ir6jaLWJ2USVpQ==",
- "dev": true,
- "requires": {
- "to-regex-range": "^5.0.1"
- }
- },
- "fsevents": {
- "version": "2.1.3",
- "resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.1.3.tgz",
- "integrity": "sha512-Auw9a4AxqWpa9GUfj370BMPzzyncfBABW8Mab7BGWBYDj4Isgq+cDKtx0i6u9jcX9pQDnswsaaOTgTmA5pEjuQ==",
- "dev": true,
- "optional": true
- },
- "get-port": {
- "version": "3.2.0",
- "resolved": "https://registry.npmjs.org/get-port/-/get-port-3.2.0.tgz",
- "integrity": "sha1-3Xzn3hh8Bsi/NTeWrHHgmfCYDrw="
- },
- "glob-parent": {
- "version": "5.1.1",
- "resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.1.tgz",
- "integrity": "sha512-FnI+VGOpnlGHWZxthPGR+QhR78fuiK0sNLkHQv+bL9fQi57lNNdquIbna/WrfROrolq8GK5Ek6BiMwqL/voRYQ==",
- "dev": true,
- "requires": {
- "is-glob": "^4.0.1"
- }
- },
- "has-flag": {
- "version": "3.0.0",
- "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-3.0.0.tgz",
- "integrity": "sha1-tdRU3CGZriJWmfNGfloH87lVuv0=",
- "dev": true
- },
- "is-binary-path": {
- "version": "2.1.0",
- "resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
- "integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
- "dev": true,
- "requires": {
- "binary-extensions": "^2.0.0"
- }
- },
- "is-extglob": {
- "version": "2.1.1",
- "resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
- "integrity": "sha1-qIwCU1eR8C7TfHahueqXc8gz+MI=",
- "dev": true
- },
- "is-glob": {
- "version": "4.0.1",
- "resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.1.tgz",
- "integrity": "sha512-5G0tKtBTFImOqDnLB2hG6Bp2qcKEFduo4tZu9MT/H6NQv/ghhy30o55ufafxJ/LdH79LLs2Kfrn85TLKyA7BUg==",
- "dev": true,
- "requires": {
- "is-extglob": "^2.1.1"
- }
- },
- "is-module": {
- "version": "1.0.0",
- "resolved": "https://registry.npmjs.org/is-module/-/is-module-1.0.0.tgz",
- "integrity": "sha1-Mlj7afeMFNW4FdZkM2tM/7ZEFZE=",
- "dev": true
- },
- "is-number": {
- "version": "7.0.0",
- "resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
- "integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
- "dev": true
- },
- "is-reference": {
- "version": "1.1.4",
- "resolved": "https://registry.npmjs.org/is-reference/-/is-reference-1.1.4.tgz",
- "integrity": "sha512-uJA/CDPO3Tao3GTrxYn6AwkM4nUPJiGGYu5+cB8qbC7WGFlrKZbiRo7SFKxUAEpFUfiHofWCXBUNhvYJMh+6zw==",
- "dev": true,
- "requires": {
- "@types/estree": "0.0.39"
- }
- },
- "jest-worker": {
- "version": "26.3.0",
- "resolved": "https://registry.npmjs.org/jest-worker/-/jest-worker-26.3.0.tgz",
- "integrity": "sha512-Vmpn2F6IASefL+DVBhPzI2J9/GJUsqzomdeN+P+dK8/jKxbh8R3BtFnx3FIta7wYlPU62cpJMJQo4kuOowcMnw==",
- "dev": true,
- "requires": {
- "@types/node": "*",
- "merge-stream": "^2.0.0",
- "supports-color": "^7.0.0"
- },
- "dependencies": {
- "has-flag": {
- "version": "4.0.0",
- "resolved": "https://registry.npmjs.org/has-flag/-/has-flag-4.0.0.tgz",
- "integrity": "sha512-EykJT/Q1KjTWctppgIAgfSO0tKVuZUjhgMr17kqTumMl6Afv3EISleU7qZUzoXDFTAHTDC4NOoG/ZxU3EvlMPQ==",
- "dev": true
- },
- "supports-color": {
- "version": "7.2.0",
- "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-7.2.0.tgz",
- "integrity": "sha512-qpCAvRl9stuOHveKsn7HncJRvv501qIacKzQlO/+Lwxc9+0q2wLyv4Dfvt80/DPn2pqOBsJdDiogXGR9+OvwRw==",
- "dev": true,
- "requires": {
- "has-flag": "^4.0.0"
- }
- }
- }
- },
- "js-tokens": {
- "version": "4.0.0",
- "resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
- "integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
- "dev": true
- },
- "kleur": {
- "version": "3.0.3",
- "resolved": "https://registry.npmjs.org/kleur/-/kleur-3.0.3.tgz",
- "integrity": "sha512-eTIzlVOSUR+JxdDFepEYcBMtZ9Qqdef+rnzWdRZuMbOywu5tO2w2N7rqjoANZ5k9vywhL6Br1VRjUIgTQx4E8w=="
- },
- "livereload": {
- "version": "0.9.1",
- "resolved": "https://registry.npmjs.org/livereload/-/livereload-0.9.1.tgz",
- "integrity": "sha512-9g7sua11kkyZNo2hLRCG3LuZZwqexoyEyecSlV8cAsfAVVCZqLzVir6XDqmH0r+Vzgnd5LrdHDMyjtFnJQLAYw==",
- "dev": true,
- "requires": {
- "chokidar": "^3.3.0",
- "livereload-js": "^3.1.0",
- "opts": ">= 1.2.0",
- "ws": "^6.2.1"
- }
- },
- "livereload-js": {
- "version": "3.2.2",
- "resolved": "https://registry.npmjs.org/livereload-js/-/livereload-js-3.2.2.tgz",
- "integrity": "sha512-xhScbNeC687ZINjEf/bD+BMiPx4s4q0mehcLb3zCc8+mykOtmaBR4vqzyIV9rIGdG9JjHaT0LiFdscvivCjX1Q==",
- "dev": true
- },
- "local-access": {
- "version": "1.0.1",
- "resolved": "https://registry.npmjs.org/local-access/-/local-access-1.0.1.tgz",
- "integrity": "sha512-ykt2pgN0aqIy6KQC1CqdWTWkmUwNgaOS6dcpHVjyBJONA+Xi7AtSB1vuxC/U/0tjIP3wcRudwQk1YYzUvzk2bA=="
- },
- "magic-string": {
- "version": "0.25.7",
- "resolved": "https://registry.npmjs.org/magic-string/-/magic-string-0.25.7.tgz",
- "integrity": "sha512-4CrMT5DOHTDk4HYDlzmwu4FVCcIYI8gauveasrdCu2IKIFOJ3f0v/8MDGJCDL9oD2ppz/Av1b0Nj345H9M+XIA==",
- "dev": true,
- "requires": {
- "sourcemap-codec": "^1.4.4"
- }
- },
- "merge-stream": {
- "version": "2.0.0",
- "resolved": "https://registry.npmjs.org/merge-stream/-/merge-stream-2.0.0.tgz",
- "integrity": "sha512-abv/qOcuPfk3URPfDzmZU1LKmuw8kT+0nIHvKrKgFrwifol/doWcdA4ZqsWQ8ENrFKkd67Mfpo/LovbIUsbt3w==",
- "dev": true
- },
- "mime": {
- "version": "2.4.6",
- "resolved": "https://registry.npmjs.org/mime/-/mime-2.4.6.tgz",
- "integrity": "sha512-RZKhC3EmpBchfTGBVb8fb+RL2cWyw/32lshnsETttkBAyAUXSGHxbEJWWRXc751DrIxG1q04b8QwMbAwkRPpUA=="
- },
- "mri": {
- "version": "1.1.5",
- "resolved": "https://registry.npmjs.org/mri/-/mri-1.1.5.tgz",
- "integrity": "sha512-d2RKzMD4JNyHMbnbWnznPaa8vbdlq/4pNZ3IgdaGrVbBhebBsGUUE/6qorTMYNS6TwuH3ilfOlD2bf4Igh8CKg=="
- },
- "normalize-path": {
- "version": "3.0.0",
- "resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
- "integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
- "dev": true
- },
- "opts": {
- "version": "1.2.7",
- "resolved": "https://registry.npmjs.org/opts/-/opts-1.2.7.tgz",
- "integrity": "sha512-hwZhzGGG/GQ7igxAVFOEun2N4fWul31qE9nfBdCnZGQCB5+L7tN9xZ+94B4aUpLOJx/of3zZs5XsuubayQYQjA==",
- "dev": true
- },
- "path-parse": {
- "version": "1.0.6",
- "resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.6.tgz",
- "integrity": "sha512-GSmOT2EbHrINBf9SR7CDELwlJ8AENk3Qn7OikK4nFYAu3Ote2+JYNVvkpAEQm3/TLNEJFD/xZJjzyxg3KBWOzw==",
- "dev": true
- },
- "picomatch": {
- "version": "2.2.2",
- "resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.2.2.tgz",
- "integrity": "sha512-q0M/9eZHzmr0AulXyPwNfZjtwZ/RBZlbN3K3CErVrk50T2ASYI7Bye0EvekFY3IP1Nt2DHu0re+V2ZHIpMkuWg==",
- "dev": true
- },
- "randombytes": {
- "version": "2.1.0",
- "resolved": "https://registry.npmjs.org/randombytes/-/randombytes-2.1.0.tgz",
- "integrity": "sha512-vYl3iOX+4CKUWuxGi9Ukhie6fsqXqS9FE2Zaic4tNFD2N2QQaXOMFbuKK4QmDHC0JO6B1Zp41J0LpT0oR68amQ==",
- "dev": true,
- "requires": {
- "safe-buffer": "^5.1.0"
- }
- },
- "readdirp": {
- "version": "3.4.0",
- "resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.4.0.tgz",
- "integrity": "sha512-0xe001vZBnJEK+uKcj8qOhyAKPzIT+gStxWr3LCB0DwcXR5NZJ3IaC+yGnHCYzB/S7ov3m3EEbZI2zeNvX+hGQ==",
- "dev": true,
- "requires": {
- "picomatch": "^2.2.1"
- }
- },
- "require-relative": {
- "version": "0.8.7",
- "resolved": "https://registry.npmjs.org/require-relative/-/require-relative-0.8.7.tgz",
- "integrity": "sha1-eZlTn8ngR6N5KPoZb44VY9q9Nt4=",
- "dev": true
- },
- "resolve": {
- "version": "1.17.0",
- "resolved": "https://registry.npmjs.org/resolve/-/resolve-1.17.0.tgz",
- "integrity": "sha512-ic+7JYiV8Vi2yzQGFWOkiZD5Z9z7O2Zhm9XMaTxdJExKasieFCr+yXZ/WmXsckHiKl12ar0y6XiXDx3m4RHn1w==",
- "dev": true,
- "requires": {
- "path-parse": "^1.0.6"
- }
- },
- "rollup": {
- "version": "1.32.1",
- "resolved": "https://registry.npmjs.org/rollup/-/rollup-1.32.1.tgz",
- "integrity": "sha512-/2HA0Ec70TvQnXdzynFffkjA6XN+1e2pEv/uKS5Ulca40g2L7KuOE3riasHoNVHOsFD5KKZgDsMk1CP3Tw9s+A==",
- "dev": true,
- "requires": {
- "@types/estree": "*",
- "@types/node": "*",
- "acorn": "^7.1.0"
- }
- },
- "rollup-plugin-livereload": {
- "version": "1.3.0",
- "resolved": "https://registry.npmjs.org/rollup-plugin-livereload/-/rollup-plugin-livereload-1.3.0.tgz",
- "integrity": "sha512-abyqXaB21+nFHo+vJULBqfzNx6zXABC19UyvqgDfdoxR/8pFAd041GO+GIUe8ZYC2DbuMUmioh1Lvbk14YLZgw==",
- "dev": true,
- "requires": {
- "livereload": "^0.9.1"
- }
- },
- "rollup-plugin-svelte": {
- "version": "5.2.3",
- "resolved": "https://registry.npmjs.org/rollup-plugin-svelte/-/rollup-plugin-svelte-5.2.3.tgz",
- "integrity": "sha512-513vOht9A93OV7fvmpIq8mD1JFgTZ5LidmpULKM2Od9P1l8oI5KwvO32fwCnASuVJS1EkRfvCnS7vKQ8DF4srg==",
- "dev": true,
- "requires": {
- "require-relative": "^0.8.7",
- "rollup-pluginutils": "^2.8.2",
- "sourcemap-codec": "^1.4.8"
- }
- },
- "rollup-plugin-terser": {
- "version": "7.0.1",
- "resolved": "https://registry.npmjs.org/rollup-plugin-terser/-/rollup-plugin-terser-7.0.1.tgz",
- "integrity": "sha512-HL0dgzSxBYG/Ly9i/E5Sc+PuKKZ0zBzk11VmLCfdUtpqH4yYqkLclPkTqRy85FU9246yetImOClaQ/ufnj08vg==",
- "dev": true,
- "requires": {
- "@babel/code-frame": "^7.10.4",
- "jest-worker": "^26.2.1",
- "serialize-javascript": "^4.0.0",
- "terser": "^5.0.0"
- }
- },
- "rollup-pluginutils": {
- "version": "2.8.2",
- "resolved": "https://registry.npmjs.org/rollup-pluginutils/-/rollup-pluginutils-2.8.2.tgz",
- "integrity": "sha512-EEp9NhnUkwY8aif6bxgovPHMoMoNr2FulJziTndpt5H9RdwC47GSGuII9XxpSdzVGM0GWrNPHV6ie1LTNJPaLQ==",
- "dev": true,
- "requires": {
- "estree-walker": "^0.6.1"
- },
- "dependencies": {
- "estree-walker": {
- "version": "0.6.1",
- "resolved": "https://registry.npmjs.org/estree-walker/-/estree-walker-0.6.1.tgz",
- "integrity": "sha512-SqmZANLWS0mnatqbSfRP5g8OXZC12Fgg1IwNtLsyHDzJizORW4khDfjPqJZsemPWBB2uqykUah5YpQ6epsqC/w==",
- "dev": true
- }
- }
- },
- "sade": {
- "version": "1.7.3",
- "resolved": "https://registry.npmjs.org/sade/-/sade-1.7.3.tgz",
- "integrity": "sha512-m4BctppMvJ60W1dXnHq7jMmFe3hPJZDAH85kQ3ACTo7XZNVUuTItCQ+2HfyaMeV5cKrbw7l4vD/6We3GBxvdJw==",
- "requires": {
- "mri": "^1.1.0"
- }
- },
- "safe-buffer": {
- "version": "5.2.1",
- "resolved": "https://registry.npmjs.org/safe-buffer/-/safe-buffer-5.2.1.tgz",
- "integrity": "sha512-rp3So07KcdmmKbGvgaNxQSJr7bGVSVk5S9Eq1F+ppbRo70+YeaDxkw5Dd8NPN+GD6bjnYm2VuPuCXmpuYvmCXQ==",
- "dev": true
- },
- "serialize-javascript": {
- "version": "4.0.0",
- "resolved": "https://registry.npmjs.org/serialize-javascript/-/serialize-javascript-4.0.0.tgz",
- "integrity": "sha512-GaNA54380uFefWghODBWEGisLZFj00nS5ACs6yHa9nLqlLpVLO8ChDGeKRjZnV4Nh4n0Qi7nhYZD/9fCPzEqkw==",
- "dev": true,
- "requires": {
- "randombytes": "^2.1.0"
- }
- },
- "sirv": {
- "version": "0.4.6",
- "resolved": "https://registry.npmjs.org/sirv/-/sirv-0.4.6.tgz",
- "integrity": "sha512-rYpOXlNbpHiY4nVXxuDf4mXPvKz1reZGap/LkWp9TvcZ84qD/nPBjjH/6GZsgIjVMbOslnY8YYULAyP8jMn1GQ==",
- "requires": {
- "@polka/url": "^0.5.0",
- "mime": "^2.3.1"
- }
- },
- "sirv-cli": {
- "version": "0.4.6",
- "resolved": "https://registry.npmjs.org/sirv-cli/-/sirv-cli-0.4.6.tgz",
- "integrity": "sha512-/Vj85/kBvPL+n9ibgX6FicLE8VjidC1BhlX67PYPBfbBAphzR6i0k0HtU5c2arejfU3uzq8l3SYPCwl1x7z6Ww==",
- "requires": {
- "console-clear": "^1.1.0",
- "get-port": "^3.2.0",
- "kleur": "^3.0.0",
- "local-access": "^1.0.1",
- "sade": "^1.4.0",
- "sirv": "^0.4.6",
- "tinydate": "^1.0.0"
- }
- },
- "source-map": {
- "version": "0.6.1",
- "resolved": "https://registry.npmjs.org/source-map/-/source-map-0.6.1.tgz",
- "integrity": "sha512-UjgapumWlbMhkBgzT7Ykc5YXUT46F0iKu8SGXq0bcwP5dz/h0Plj6enJqjz1Zbq2l5WaqYnrVbwWOWMyF3F47g==",
- "dev": true
- },
- "source-map-support": {
- "version": "0.5.19",
- "resolved": "https://registry.npmjs.org/source-map-support/-/source-map-support-0.5.19.tgz",
- "integrity": "sha512-Wonm7zOCIJzBGQdB+thsPar0kYuCIzYvxZwlBa87yi/Mdjv7Tip2cyVbLj5o0cFPN4EVkuTwb3GDDyUx2DGnGw==",
- "dev": true,
- "requires": {
- "buffer-from": "^1.0.0",
- "source-map": "^0.6.0"
- }
- },
- "sourcemap-codec": {
- "version": "1.4.8",
- "resolved": "https://registry.npmjs.org/sourcemap-codec/-/sourcemap-codec-1.4.8.tgz",
- "integrity": "sha512-9NykojV5Uih4lgo5So5dtw+f0JgJX30KCNI8gwhz2J9A15wD0Ml6tjHKwf6fTSa6fAdVBdZeNOs9eJ71qCk8vA==",
- "dev": true
- },
- "supports-color": {
- "version": "5.5.0",
- "resolved": "https://registry.npmjs.org/supports-color/-/supports-color-5.5.0.tgz",
- "integrity": "sha512-QjVjwdXIt408MIiAqCX4oUKsgU2EqAGzs2Ppkm4aQYbjm+ZEWEcW4SfFNTr4uMNZma0ey4f5lgLrkB0aX0QMow==",
- "dev": true,
- "requires": {
- "has-flag": "^3.0.0"
- }
- },
- "svelte": {
- "version": "3.24.1",
- "resolved": "https://registry.npmjs.org/svelte/-/svelte-3.24.1.tgz",
- "integrity": "sha512-OX/IBVUJSFo1rnznXdwf9rv6LReJ3qQ0PwRjj76vfUWyTfbHbR9OXqJBnUrpjyis2dwYcbT2Zm1DFjOOF1ZbbQ==",
- "dev": true
- },
- "terser": {
- "version": "5.3.0",
- "resolved": "https://registry.npmjs.org/terser/-/terser-5.3.0.tgz",
- "integrity": "sha512-XTT3D3AwxC54KywJijmY2mxZ8nJiEjBHVYzq8l9OaYuRFWeQNBwvipuzzYEP4e+/AVcd1hqG/CqgsdIRyT45Fg==",
- "dev": true,
- "requires": {
- "commander": "^2.20.0",
- "source-map": "~0.6.1",
- "source-map-support": "~0.5.12"
- }
- },
- "tinydate": {
- "version": "1.2.0",
- "resolved": "https://registry.npmjs.org/tinydate/-/tinydate-1.2.0.tgz",
- "integrity": "sha512-3GwPk8VhDFnUZ2TrgkhXJs6hcMAIIw4x/xkz+ayK6dGoQmp2nUwKzBXK0WnMsqkh6vfUhpqQicQF3rbshfyJkg=="
- },
- "to-regex-range": {
- "version": "5.0.1",
- "resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
- "integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
- "dev": true,
- "requires": {
- "is-number": "^7.0.0"
- }
- },
- "ws": {
- "version": "6.2.1",
- "resolved": "https://registry.npmjs.org/ws/-/ws-6.2.1.tgz",
- "integrity": "sha512-GIyAXC2cB7LjvpgMt9EKS2ldqr0MTrORaleiOno6TweZ6r3TKtoFQWay/2PceJ3RuBasOHzXNn5Lrw1X0bEjqA==",
- "dev": true,
- "requires": {
- "async-limiter": "~1.0.0"
- }
- },
- "wui.svelte": {
- "version": "0.2.2",
- "resolved": "https://registry.npmjs.org/wui.svelte/-/wui.svelte-0.2.2.tgz",
- "integrity": "sha512-lz6ZTUo/sxnjOTrn71pV9XlDdPrX1oIPFrL5iAfvKDjQVMUTsNUr9CeDcW0asT+uWS3SRPw34Zwb4p0YBJuYPQ=="
- }
- }
-}
diff --git a/_www/package.json b/_www/package.json
deleted file mode 100644
index 6d942c9..0000000
--- a/_www/package.json
+++ /dev/null
@@ -1,22 +0,0 @@
-{
- "name": "rescached-wui",
- "version": "1.0.0",
- "scripts": {
- "build": "rollup -c",
- "dev": "rollup -c -w",
- "start": "sirv public"
- },
- "devDependencies": {
- "@rollup/plugin-commonjs": "11.0.2",
- "@rollup/plugin-node-resolve": "^7.0.0",
- "rollup": "^1.20.0",
- "rollup-plugin-livereload": "^1.0.0",
- "rollup-plugin-svelte": "^5.2.3",
- "rollup-plugin-terser": "^7.0.1",
- "svelte": "^3.24.1"
- },
- "dependencies": {
- "sirv-cli": "^0.4.6",
- "wui.svelte": "^0.2.2"
- }
-}
diff --git a/_www/public/build/bundle.css b/_www/public/build/bundle.css
deleted file mode 100644
index 50afc4e..0000000
--- a/_www/public/build/bundle.css
+++ /dev/null
@@ -1,13 +0,0 @@
-.wui-input-ipport.svelte-sy39ke.svelte-sy39ke{display:inline-block;width:100%}.wui-input-ipport.svelte-sy39ke input.svelte-sy39ke{width:100%}.invalid.svelte-sy39ke.svelte-sy39ke{color:red}div.invalid.svelte-sy39ke.svelte-sy39ke{font-size:12px}
-.wui-input-number.svelte-1qrd8wr.svelte-1qrd8wr{width:100%}.wui-input-number.svelte-1qrd8wr input.svelte-1qrd8wr{display:inline-block;width:70%}.wui-input-number.svelte-1qrd8wr .suffix.svelte-1qrd8wr{width:30%}
-.label-hint.svelte-1weevo5{display:inline-flex;margin-top:1em;width:100%}.title.svelte-1weevo5{margin-bottom:4px}.title.svelte-1weevo5{display:inline-block}.toggle.svelte-1weevo5{border-radius:50%;border:1px solid grey;cursor:pointer;display:inline-block;font-size:12px;height:14px;line-height:14px;padding:2px;text-align:center;width:14px}.info.svelte-1weevo5{background-color:#eee;margin:8px 0px;padding:1em}@media(max-width: 720px){.label-hint.svelte-1weevo5{display:block}}
-.wui-notif.svelte-xdooa2{position:fixed;top:5px;left:calc((100% - 400px)/2);width:400px}@media(max-width: 500px){.wui-notif.svelte-xdooa2{left:1em;width:calc(100% - 2em)}}
-.wui-notif-item.svelte-1n99njq{background-color:white;border:1px solid black;box-shadow:3px 3px;padding:1em;margin-bottom:1em;z-index:1000}.wui-notif-item.error.svelte-1n99njq{border:1px solid red;box-shadow:3px 3px red}
-div.main.svelte-jyzzth{margin:0 auto;width:800px;padding:0px 1em}nav.menu.svelte-jyzzth{color:#ff3e00;text-transform:uppercase;font-weight:100;margin-bottom:2em}.active.svelte-jyzzth{padding-bottom:4px;border-bottom:4px solid #ff3e00}@media(max-width: 900px){div.main.svelte-jyzzth{width:calc(100% - 2em)}}
-.message.svelte-1d4jf0i.svelte-1d4jf0i{padding:1em 0px;border-bottom:1px solid silver}.rr.svelte-1d4jf0i.svelte-1d4jf0i{margin-left:1em;width:100%}.rr.header.svelte-1d4jf0i.svelte-1d4jf0i{font-weight:bold}.rr.svelte-1d4jf0i span.svelte-1d4jf0i{display:inline-block}.kind.svelte-1d4jf0i.svelte-1d4jf0i{width:9em}.type.svelte-1d4jf0i.svelte-1d4jf0i{width:5em}.ttl.svelte-1d4jf0i.svelte-1d4jf0i{width:6em}.value.svelte-1d4jf0i.svelte-1d4jf0i{word-wrap:anywhere;width:calc(100% - 24em)}
-input.svelte-ivqrh9.svelte-ivqrh9{width:100%}.input-deletable.svelte-ivqrh9.svelte-ivqrh9{width:100%}.input-deletable.svelte-ivqrh9>input.svelte-ivqrh9{max-width:calc(100% - 100px)}.input-deletable.svelte-ivqrh9>button.svelte-ivqrh9{width:80px}.input-checkbox.svelte-ivqrh9.svelte-ivqrh9{width:100%}.input-checkbox.svelte-ivqrh9 input[type="checkbox"].svelte-ivqrh9{width:auto}.section-bottom.svelte-ivqrh9.svelte-ivqrh9{margin:2em 0px 0px 0px;padding:1em;border-top:1px solid black}
-.block_source.svelte-ze2due.svelte-ze2due{width:calc(100% - 2em);overflow:auto;font-size:12px}.block_source.svelte-ze2due input.svelte-ze2due:disabled{color:black}.item.svelte-ze2due span.svelte-ze2due{display:inline-block;margin-right:1em}.item.header.svelte-ze2due.svelte-ze2due{font-weight:bold;margin-bottom:1em;border-bottom:1px solid silver}.item.svelte-ze2due>span.svelte-ze2due:nth-child(1){width:4em}.item.svelte-ze2due>span.svelte-ze2due:nth-child(2){width:15em}.item.svelte-ze2due>span.svelte-ze2due:nth-child(3){width:23em}.item.svelte-ze2due>span:nth-child(3) input.svelte-ze2due{width:100%}.item.svelte-ze2due>span.svelte-ze2due:nth-child(4){width:16em}
-.nav-left.svelte-1w15y7z.svelte-1w15y7z{padding:0px;width:280px;float:left}.nav-left.svelte-1w15y7z .item.svelte-1w15y7z{margin:4px 0px}.content.svelte-1w15y7z.svelte-1w15y7z{float:left;width:calc(100% - 300px)}.host.svelte-1w15y7z.svelte-1w15y7z{font-family:monospace;width:100%}.host.header.svelte-1w15y7z.svelte-1w15y7z{margin:1em 0px;font-weight:bold;border-bottom:1px solid silver}.host_name.svelte-1w15y7z.svelte-1w15y7z{display:inline-block;width:240px;word-wrap:break-word}.host_value.svelte-1w15y7z.svelte-1w15y7z{display:inline-block;width:140px}
-h4.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{border-bottom:1px solid silver}.nav-left.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{padding:0px;width:280px;float:left}.nav-left.svelte-yu4l71.svelte-yu4l71 .item.svelte-yu4l71.svelte-yu4l71{margin:4px 0px;cursor:pointer;color:rgb(0, 100, 200)}.content.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{float:left;width:calc(100% - 300px)}.action-delete.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{margin-left:1em}.actions.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{padding:1em}.actions.svelte-yu4l71.svelte-yu4l71 button.svelte-yu4l71.svelte-yu4l71{width:100%}form.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{margin:1em 0px;padding:10px 10px 0px 10px;border:1px solid silver}form.svelte-yu4l71>label.svelte-yu4l71>span.svelte-yu4l71{width:7em;display:inline-block}form.svelte-yu4l71>label.svelte-yu4l71>input.svelte-yu4l71{width:calc(100% - 8em)}form.svelte-yu4l71>label.svelte-yu4l71>input.name.svelte-yu4l71{width:12em}.rr.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{font-family:monospace;width:100%;padding:1em 0px}.rr.header.svelte-yu4l71.svelte-yu4l71.svelte-yu4l71{font-weight:bold}.rr.svelte-yu4l71.svelte-yu4l71>.name.svelte-yu4l71.svelte-yu4l71{width:20em;display:inline-block;word-wrap:break-word}.rr.svelte-yu4l71.svelte-yu4l71>.type.svelte-yu4l71.svelte-yu4l71{width:4em;display:inline-block}.rr.svelte-yu4l71.svelte-yu4l71>.value.svelte-yu4l71.svelte-yu4l71{display:inline-block;word-wrap:break-word}
-
-/*# sourceMappingURL=bundle.css.map */ \ No newline at end of file
diff --git a/_www/public/build/bundle.css.map b/_www/public/build/bundle.css.map
deleted file mode 100644
index 59cba63..0000000
--- a/_www/public/build/bundle.css.map
+++ /dev/null
@@ -1,32 +0,0 @@
-{
- "version": 3,
- "file": "bundle.css",
- "sources": [
- "../../InputIPPort.svelte",
- "../../InputNumber.svelte",
- "../../LabelHint.svelte",
- "../../Notif.svelte",
- "../../NotifItem.svelte",
- "../../App.svelte",
- "../../Dashboard.svelte",
- "../../Environment.svelte",
- "../../HostsBlock.svelte",
- "../../HostsDir.svelte",
- "../../MasterDir.svelte"
- ],
- "sourcesContent": [
- "<script>\n\texport let value = \"\";\n\tlet isInvalid = false;\n\tlet error = \"\";\n\n\tfunction onBlur() {\n\t\tif (value === \"\") {\n\t\t\tisInvalid = false;\n\t\t\terror = \"\";\n\t\t\treturn;\n\t\t}\n\t\tconst ipport = value.split(\":\");\n\t\tif (ipport.length !== 2) {\n\t\t\tisInvalid = true;\n\t\t\terror = \"missing port number\";\n\t\t\treturn;\n\t\t}\n\t\tconst ip = ipport[0];\n\t\tconst port = parseInt(ipport[1]);\n\t\tif (isNaN(port) || port <= 0 || port >= 65535) {\n\t\t\tisInvalid = true;\n\t\t\terror = \"invalid port number\";\n\t\t\treturn;\n\t\t}\n\t\tisInvalid = false;\n\t\tvalue = ip +\":\"+ port;\n\t}\n</script>\n\n<style>\n\t.wui-input-ipport {\n\t\tdisplay: inline-block;\n\t\twidth: 100%;\n\t}\n\t.wui-input-ipport input {\n\t\twidth: 100%;\n\t}\n\t.invalid {\n\t\tcolor: red;\n\t}\n\tdiv.invalid {\n\t\tfont-size: 12px;\n\t}\n</style>\n\n<div class=\"wui-input-ipport\">\n\t<input\n\t\ttype=\"text\"\n\t\tbind:value={value}\n\t\ton:blur={onBlur}\n\t\tclass:invalid={isInvalid}\n\t\tplaceholder=\"127.0.0.1:8080\"\n\t>\n\t{#if isInvalid}\n\t<div class=\"invalid\">{error}</div>\n\t{/if}\n</div>\n",
- "<script>\n\texport let min;\n\texport let max;\n\texport let value = 0;\n\texport let unit;\n\n\tfunction onBlur() {\n\t\tvalue = +value\n\t\tif (isNaN(value)) {\n\t\t\tvalue = max\n\t\t} else if (value < min) {\n\t\t\tvalue = min\n\t\t} else if (value > max) {\n\t\t\tvalue = max\n\t\t}\n\t}\n</script>\n\n<style>\n\t.wui-input-number {\n\t\twidth: 100%;\n\t}\n\t.wui-input-number input {\n\t\tdisplay: inline-block;\n\t\twidth: 70%;\n\t}\n\t.wui-input-number .suffix {\n\t\twidth: 30%;\n\t}\n</style>\n\n<div class=\"wui-input-number\">\n\t<input type=\"number\" on:blur={onBlur} bind:value={value}>\n\t{#if unit !== ''}\n\t\t<span class=\"suffix\">\n\t\t\t{unit}\n\t\t</span>\n\t{/if}\n</div>\n",
- "<script>\n\texport let title;\n\texport let info;\n\texport let title_width = \"300px\";\n\tlet showInfo = false;\n</script>\n\n<style>\n\t.label-hint {\n\t\tdisplay: inline-flex;\n\t\tmargin-top: 1em;\n\t\twidth: 100%;\n\t}\n\t.title {\n\t\tmargin-bottom: 4px;\n\t}\n\t.title {\n\t\tdisplay: inline-block;\n\t}\n\t.toggle {\n\t\tborder-radius: 50%;\n\t\tborder: 1px solid grey;\n\t\tcursor: pointer;\n\t\tdisplay: inline-block;\n\t\tfont-size: 12px;\n\t\theight: 14px;\n\t\tline-height: 14px;\n\t\tpadding: 2px;\n\t\ttext-align: center;\n\t\twidth: 14px;\n\t}\n\t.info {\n\t\tbackground-color: #eee;\n\t\tmargin: 8px 0px;\n\t\tpadding: 1em;\n\t}\n\t@media (max-width: 720px) {\n\t\t.label-hint {\n\t\t\tdisplay: block;\n\t\t}\n\t}\n</style>\n\n<label class=\"label-hint\">\n\t<span class=\"title\" style=\"width:{title_width};\">\n\t\t{title}\n\t\t<span class=\"toggle\" on:click={() => showInfo = !showInfo}>\n\t\t\t?\n\t\t</span>\n\t</span>\n\n\t<slot>\n\t</slot>\n\n</label>\n\n{#if showInfo}\n<div class=\"info\">\n\t{@html info}\n</div>\n{/if}\n",
- "<script context=\"module\">\n\timport { messages } from \"./Notif.store.js\"\n\timport NotifItem from \"./NotifItem.svelte\"\n\n\texport const WuiPushNotif = {\n\t\tInfo: function(text) {\n\t\t\tconst msg = {\n\t\t\t\ttext: text,\n\t\t\t}\n\t\t\tmessages.update(msgs => msgs = [...msgs, msg])\n\t\t},\n\t\tError: function(text) {\n\t\t\tconst msg = {\n\t\t\t\ttext: text,\n\t\t\t\tkind: \"error\",\n\t\t\t}\n\t\t\tmessages.update(msgs => msgs = [...msgs, msg])\n\t\t}\n\t}\n</script>\n<script>\n\texport let timeout = 5000;\n</script>\n\n<style>\n\t.wui-notif {\n\t\tposition: fixed;\n\t\ttop: 5px;\n\t\tleft: calc((100% - 400px)/2);\n\t\twidth: 400px;\n\t}\n\t@media (max-width: 500px) {\n\t\t.wui-notif {\n\t\t\tleft: 1em;\n\t\t\twidth: calc(100% - 2em);\n\t\t}\n\t}\n</style>\n\n<div class=\"wui-notif\">\n\t{#each $messages as msg (msg)}\n\t<NotifItem text={msg.text} kind=\"{msg.kind}\" {timeout}/>\n\t{/each}\n</div>\n",
- "<script>\n\timport { onMount } from 'svelte'\n\timport { fade } from 'svelte/transition'\n\timport { messages } from \"./Notif.store.js\"\n\n\texport let text = \"\"\n\texport let kind = \"\"\n\texport let timeout = 5000\n\n\tonMount(() => {\n\t\tlet timerID = setTimeout(()=> {\n\t\t\tmessages.update(msgs => {\n\t\t\t\tmsgs.splice(0, 1);\n\t\t\t\tmsgs = msgs\n\t\t\t\treturn msgs\n\t\t\t})\n\t\t}, timeout)\n\t})\n</script>\n\n<style>\n\t.wui-notif-item {\n\t\tbackground-color: white;\n\t\tborder: 1px solid black;\n\t\tbox-shadow: 3px 3px;\n\t\tpadding: 1em;\n\t\tmargin-bottom: 1em;\n\t\tz-index: 1000;\n\t}\n\t.wui-notif-item.error {\n\t\tborder: 1px solid red;\n\t\tbox-shadow: 3px 3px red;\n\t}\n</style>\n\n<div transition:fade class=\"wui-notif-item {kind}\">\n\t{text}\n</div>\n",
- "<script>\n\timport { onMount } from 'svelte';\n\timport { WuiNotif, WuiPushNotif } from 'wui.svelte';\n\n\timport { apiEnvironment, environment, nanoSeconds, setEnvironment } from './environment.js';\n\timport Dashboard from './Dashboard.svelte';\n\timport Environment from './Environment.svelte';\n\timport HostsBlock from './HostsBlock.svelte';\n\timport HostsDir from './HostsDir.svelte';\n\timport MasterDir from './MasterDir.svelte';\n\n\tconst stateEnvironment = \"environment\";\n\tconst stateHostsBlock = \"hosts_block\";\n\tconst stateHostsDir = \"hosts_d\";\n\tconst stateMasterDir = \"master_d\";\n\n\tlet state;\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: {},\n\t};\n\n\tonMount(async () => {\n\t\tconst res = await fetch(apiEnvironment);\n\t\tif (res.status >= 400) {\n\t\t\tWuiPushNotif.Error(\"ERROR: {apiEnvironment}: \",\n\t\t\t\tres.status, res.statusText);\n\t\t\treturn;\n\t\t}\n\n\t\tsetEnvironment(await res.json());\n \t\tstate = window.location.hash.slice(1);\n\t});\n</script>\n\n<style>\n\tdiv.main {\n\t\tmargin: 0 auto;\n\t\twidth: 800px;\n\t\tpadding: 0px 1em;\n\t}\n\tnav.menu {\n\t\tcolor: #ff3e00;\n\t\ttext-transform: uppercase;\n\t\tfont-weight: 100;\n\t\tmargin-bottom: 2em;\n\t}\n\t.active {\n\t\tpadding-bottom: 4px;\n\t\tborder-bottom: 4px solid #ff3e00;\n\t}\n\t@media (max-width: 900px) {\n\t\tdiv.main {\n\t\t\twidth: calc(100% - 2em);\n\t\t}\n\t}\n</style>\n\n<WuiNotif timeout=3000 />\n\n<div class=\"main\">\n\t<nav class=\"menu\">\n\t\t<a\n\t\t\thref=\"#home\"\n\t\t\ton:click={()=>state=\"\"}\n\t\t\tclass:active=\"{state===''||state==='home'}\"\n\t\t>\n\t\t\trescached\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#environment\"\n\t\t\ton:click={()=>state=stateEnvironment}\n\t\t\tclass:active=\"{state===stateEnvironment}\"\n\t\t>\n\t\t\tEnvironment\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#{stateHostsBlock}\"\n\t\t\ton:click={()=>state=stateHostsBlock}\n\t\t\tclass:active=\"{state===stateHostsBlock}\"\n\t\t>\n\t\t\tHosts blocks\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#{stateHostsDir}\"\n\t\t\ton:click={()=>state=stateHostsDir}\n\t\t\tclass:active=\"{state === stateHostsDir}\"\n\t\t>\n\t\t\thosts.d\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#{stateMasterDir}\"\n\t\t\ton:click={()=>state=stateMasterDir}\n\t\t\tclass:active=\"{state === stateMasterDir}\"\n\t\t>\n\t\t\tmaster.d\n\t\t</a>\n\t</nav>\n\n\t{#if state === stateEnvironment}\n\t\t<Environment/>\n\t{:else if state === stateHostsBlock}\n\t\t<HostsBlock/>\n\t{:else if state === stateHostsDir}\n\t\t<HostsDir/>\n\t{:else if state === stateMasterDir}\n\t\t<MasterDir/>\n\t{:else}\n\t\t<Dashboard/>\n\t{/if}\n</div>\n",
- "<script>\n\timport {WuiPushNotif} from 'wui.svelte';\n\timport {getRRTypeName} from './common.js';\n\n\tconst apiCaches = \"/api/caches\";\n\n\tlet query = \"\";\n\tlet listMsg = [];\n\n\tasync function handleSearch() {\n\t\tconst res = await fetch(apiCaches+\"?query=\"+ query)\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resbody = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resbody.message)\n\t\t\treturn;\n\t\t}\n\n\t\tlistMsg = await res.json()\n\t}\n\n\tasync function handleRemoveFromCaches(name) {\n\t\tconst res = await fetch(apiCaches+\"?name=\"+ name, {\n\t\t\tmethod: \"DELETE\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resbody = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resbody.message)\n\t\t\treturn;\n\t\t}\n\n\t\tfor (let x = 0; x < listMsg.length; x++) {\n\t\t\tif (listMsg[x].Question.Name === name) {\n\t\t\t\tlistMsg.splice(x, 1)\n\t\t\t\tlistMsg = listMsg\n\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\n\t\tconst msg = await res.json()\n\n\t\tWuiPushNotif.Info(msg.message)\n\t}\n</script>\n\n<style>\n\t.message {\n\t\tpadding: 1em 0px;\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.rr {\n\t\tmargin-left: 1em;\n\t\twidth: 100%;\n\t}\n\t.rr.header {\n\t\tfont-weight: bold;\n\t}\n\t.rr span {\n\t\tdisplay: inline-block;\n\t}\n\t.kind {\n\t\twidth: 9em;\n\t}\n\t.type {\n\t\twidth: 5em;\n\t}\n\t.ttl {\n\t\twidth: 6em;\n\t}\n\t.value {\n\t\tword-wrap: anywhere;\n\t\twidth: calc(100% - 24em);\n\t}\n</style>\n\n<div class=\"dashboard\">\n\t<div class=\"search\">\n\t\tCaches:\n\t\t<input bind:value={query}>\n\t\t<button on:click={handleSearch}>\n\t\t\tSearch\n\t\t</button>\n\t</div>\n\t{#each listMsg as msg (msg)}\n\t\t<div class=\"message\">\n\t\t\t<div class=\"qname\">\n\t\t\t\t{msg.Question.Name}\n\n\t\t\t\t<button\n\t\t\t\t\tclass=\"b-remove\"\n\t\t\t\t\ton:click={handleRemoveFromCaches(msg.Question.Name)}\n\t\t\t\t>\n\t\t\t\t\tRemove from caches\n\t\t\t\t</button>\n\t\t\t</div>\n\t\t\t<div class=\"rr header\">\n\t\t\t\t<span class=\"kind\"> </span>\n\t\t\t\t<span class=\"type\"> Type </span>\n\t\t\t\t<span class=\"ttl\"> TTL </span>\n\t\t\t\t<span class=\"value\"> Value </span>\n\t\t\t</div>\n\n\t\t\t{#if msg.Answer !== null && msg.Answer.length > 0}\n\t\t\t\t{#each msg.Answer as rr}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"kind\"> Answer </span>\n\t\t\t\t\t\t<span class=\"type\"> {getRRTypeName(rr.Type)} </span>\n\t\t\t\t\t\t<span class=\"ttl\"> {rr.TTL} </span>\n\t\t\t\t\t\t<span class=\"value\"> {JSON.stringify(rr.Value, null, 2)} </span>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/if}\n\t\t\t{#if msg.Authority !== null && msg.Authority.length > 0}\n\t\t\t\t{#each msg.Authority as rr}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"kind\"> Authority </span>\n\t\t\t\t\t\t<span class=\"type\"> {getRRTypeName(rr.Type)} </span>\n\t\t\t\t\t\t<span class=\"ttl\"> {rr.TTL} </span>\n\t\t\t\t\t\t<span class=\"value\"> {JSON.stringify(rr.Value, null, 2)} </span>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/if}\n\t\t\t{#if msg.Additional !== null && msg.Additional.length > 0}\n\t\t\t\t{#each msg.Additional as rr}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"kind\"> Additional </span>\n\t\t\t\t\t\t<span class=\"type\"> {getRRTypeName(rr.Type)} </span>\n\t\t\t\t\t\t<span class=\"ttl\"> {rr.TTL} </span>\n\t\t\t\t\t\t<span class=\"value\"> {JSON.stringify(rr.Value, null, 2)} </span>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/if}\n\t\t</div>\n\t{/each}\n</div>\n",
- "<script>\n\timport { onDestroy } from 'svelte';\n\n\timport { apiEnvironment, environment, nanoSeconds } from './environment.js';\n\timport { WuiPushNotif } from \"wui.svelte\";\n\timport { WuiLabelHint, WuiInputNumber, WuiInputIPPort } from \"wui.svelte\";\n\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: {},\n\t};\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\n\tonDestroy(envUnsubscribe);\n\n\tconst defTitleWidth = \"300px\";\n\n\tfunction addNameServer() {\n\t\tenv.NameServers = [...env.NameServers, '']\n\t}\n\n\tfunction deleteNameServer(ns) {\n\t\tfor (let x = 0; x < env.NameServers.length; x++) {\n\t\t\tif (env.NameServers[x] === ns) {\n\t\t\t\tenv.NameServers.splice(x, 1);\n\t\t\t\tenv.NameServers = env.NameServers;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n\n\tasync function updateEnvironment() {\n\t\tlet got = {};\n\n\t\tObject.assign(got, env)\n\t\tenvironment.set(env)\n\n\t\tgot.PruneDelay = got.PruneDelay * nanoSeconds;\n\t\tgot.PruneThreshold = got.PruneThreshold * nanoSeconds;\n\n\t\tconst res = await fetch(apiEnvironment, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tbody: JSON.stringify(got),\n\t\t});\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resbody = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \", resbody.message)\n\t\t\treturn;\n\t\t}\n\n\t\tWuiPushNotif.Info(\"The environment succesfully updated ...\")\n\t}\n</script>\n\n<style>\n\tinput {\n\t\twidth: 100%;\n\t}\n\t.input-deletable {\n\t\twidth: 100%;\n\t}\n\t.input-deletable > input {\n\t\tmax-width: calc(100% - 100px);\n\t}\n\t.input-deletable > button {\n\t\twidth: 80px;\n\t}\n\t.input-checkbox {\n\t\twidth: 100%;\n\t}\n\t.input-checkbox input[type=\"checkbox\"] {\n\t\twidth: auto;\n\t}\n\t.section-bottom {\n\t\tmargin: 2em 0px 0px 0px;\n\t\tpadding: 1em;\n\t\tborder-top: 1px solid black;\n\t}\n</style>\n\n<div class=\"environment\">\n<p>\nThis page allow you to change the rescached environment.\nUpon save, the rescached service will be restarted.\n</p>\n\n<h3>rescached</h3>\n<div>\n\t<WuiLabelHint\n\t\ttitle=\"System resolv.conf\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"A path to dynamically generated resolv.conf(5) by\nresolvconf(8). If set, the nameserver values in referenced file will\nreplace 'parent' value and 'parent' will become a fallback in\ncase the referenced file being deleted or can not be parsed.\"\n\t>\n\t\t<input\n\t\t\tbind:value={env.FileResolvConf}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"Debug level\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"This option only used for debugging program or if user\nwant to monitor what kind of traffic goes in and out of rescached.\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=0\n\t\t\tmax=3\n\t\t\tbind:value={env.Debug}\n\t\t\tunit=\"\"\n\t\t/>\n\t</WuiLabelHint>\n</div>\n\n<h3>DNS server</h3>\n<div>\n\t<WuiLabelHint\n\t\ttitle=\"Parent name servers\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"List of parent DNS servers.\"\n\t>\n\t</WuiLabelHint>\n\n\t{#each env.NameServers as ns}\n\t<div class=\"input-deletable\">\n\t\t<input bind:value={ns}>\n\t\t<button on:click={deleteNameServer(ns)}>\n\t\t\tDelete\n\t\t</button>\n\t</div>\n\t{/each}\n\n\t<button\n\t\ton:click={addNameServer}\n\t>\n\t\tAdd\n\t</button>\n\n\t<WuiLabelHint\n\t\ttitle=\"Listen address\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Address in local network where rescached will\nlistening for query from client through UDP and TCP.\n<br/>\nIf you want rescached to serve a query from another host in your local\nnetwork, change this value to <tt>0.0.0.0:53</tt>.\"\n\t>\n\t\t<WuiInputIPPort\n\t\t\tbind:value={env.ListenAddress}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"HTTP listen port\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Port to serve DNS over HTTP\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=0\n\t\t\tmax=65535\n\t\t\tbind:value={env.HTTPPort}\n\t\t\tunit=\"\"\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS listen port\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Port to listen for DNS over TLS\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=0\n\t\t\tmax=65535\n\t\t\tbind:value={env.TLSPort}\n\t\t\tunit=\"\"\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS certificate\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Path to certificate file to serve DNS over TLS and\nHTTPS\">\n\t\t<input\n\t\t\tplaceholder=\"/path/to/certificate\"\n\t\t\tbind:value={env.TLSCertFile}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS private key\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Path to certificate private key file to serve DNS over TLS and\nHTTPS.\"\n\t>\n\t\t<input\n\t\t\tplaceholder=\"/path/to/certificate/private.key\"\n\t\t\tbind:value={env.TLSPrivateKey}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS allow insecure\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"If its true, allow serving DoH and DoT with self signed\ncertificate.\"\n\t>\n\t\t<div class=\"input-checkbox\">\n\t\t\t<input\n\t\t\t\ttype=checkbox\n\t\t\t\tbind:checked={env.TLSAllowInsecure}\n\t\t\t>\n\t\t\t<span class=\"suffix\">\n\t\t\t\tYes\n\t\t\t</span>\n\t\t</div>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"DoH behind proxy\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"If its true, serve DNS over HTTP only, even if\ncertificate files is defined.\nThis allow serving DNS request forwarded by another proxy server.\"\n\t>\n\t\t<div class=\"input-checkbox\">\n\t\t\t<input\n\t\t\t\ttype=checkbox\n\t\t\t\tbind:checked={env.DoHBehindProxy}\n\t\t\t>\n\t\t\t<span class=\"suffix\">\n\t\t\t\tYes\n\t\t\t</span>\n\t\t</div>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"Prune delay\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Delay for pruning caches.\nEvery N seconds, rescached will traverse all caches and remove response that\nhas not been accessed less than cache.prune_threshold.\nIts value must be equal or greater than 1 hour (3600 seconds).\n\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=3600\n\t\t\tmax=36000\n\t\t\tbind:value={env.PruneDelay}\n\t\t\tunit=\"seconds\"\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"Prune threshold\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"The duration when the cache will be considered expired.\nIts value must be negative and greater or equal than -1 hour (-3600 seconds).\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=-36000\n\t\t\tmax=-3600\n\t\t\tbind:value={env.PruneThreshold}\n\t\t\tunit=\"seconds\"\n\t\t/>\n\t</WuiLabelHint>\n</div>\n\n\t<div class=\"section-bottom\">\n\t\t<div>\n\t\t\t<button on:click={updateEnvironment}>\n\t\t\t\tSave\n\t\t\t</button>\n\t\t</div>\n\t</div>\n</div>\n",
- "<script>\n\timport { onDestroy } from 'svelte';\n\timport { WuiPushNotif } from 'wui.svelte';\n\timport { environment, nanoSeconds, setEnvironment } from './environment.js';\n\n\tconst apiHostsBlock = \"/api/hosts_block\"\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: {},\n\t};\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\tonDestroy(envUnsubscribe);\n\n\tasync function updateHostsBlocks() {\n\t\tconst res = await fetch(apiHostsBlock, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tbody: JSON.stringify(env.HostsBlocks),\n\t\t});\n\n\t\tif (res.status >= 400) {\n\t\t\tWuiPushNotif.Error(\"ERROR: \", res.status, res.statusText)\n\t\t\treturn;\n\t\t}\n\n\t\tsetEnvironment(await res.json());\n\t}\n</script>\n\n<style>\n\t.block_source {\n\t\twidth: calc(100% - 2em);\n\t\toverflow: auto;\n\t\tfont-size: 12px;\n\t}\n\t.block_source input:disabled {\n\t\tcolor: black;\n\t}\n\t.item span {\n\t\tdisplay: inline-block;\n\t\tmargin-right: 1em;\n\t}\n\t.item.header {\n\t\tfont-weight: bold;\n\t\tmargin-bottom: 1em;\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.item > span:nth-child(1) {\n\t\twidth: 4em;\n\t}\n\t.item > span:nth-child(2) {\n\t\twidth: 15em;\n\t}\n\t.item > span:nth-child(3) {\n\t\twidth: 23em;\n\t}\n\t.item > span:nth-child(3) input {\n\t\twidth: 100%;\n\t}\n\t.item > span:nth-child(4) {\n\t\twidth: 16em;\n\t}\n</style>\n\n<div class=\"hosts-block\">\n\t<p>\n\tConfigure the source of blocked hosts file.\n\t</p>\n\n\t<div class=\"block_source\">\n\t\t<div class=\"item header\">\n\t\t\t<span> Enabled </span>\n\t\t\t<span> Name </span>\n\t\t\t<span> URL </span>\n\t\t\t<span> Last updated </span>\n\t\t</div>\n\t\t{#each env.HostsBlocks as hostsBlock}\n\t\t<div class=\"item\">\n\t\t\t<span>\n\t\t\t\t<input\n\t\t\t\t\ttype=checkbox\n\t\t\t\t\tbind:checked={hostsBlock.IsEnabled}\n\t\t\t\t>\n\t\t\t</span>\n\t\t\t<span>\n\t\t\t\t{hostsBlock.Name}\n\t\t\t</span>\n\t\t\t<span>\n\t\t\t\t<input\n\t\t\t\t\tbind:value={hostsBlock.URL}\n\t\t\t\t\tdisabled\n\t\t\t\t>\n\t\t\t</span>\n\t\t\t<span>\n\t\t\t\t{hostsBlock.LastUpdated}\n\t\t\t</span>\n\t\t</div>\n\t\t{/each}\n\t</div>\n\n\t<div>\n\t\t<button on:click={updateHostsBlocks}>\n\t\t\tSave\n\t\t</button>\n\t</div>\n</div>\n",
- "<script> import { onDestroy } from 'svelte';\n\timport { WuiPushNotif } from 'wui.svelte';\n\timport { apiEnvironment, environment, nanoSeconds } from './environment.js';\n\n\tconst apiHostsDir = \"/api/hosts.d\"\n\n\tlet env = {\n\t\tHostsFiles: {},\n\t};\n\tlet hostsFile = {\n\t\tName: \"\",\n\t\tRecords: [],\n\t};\n\tlet newRecord = null;\n\tlet newHostsFile = \"\";\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\tonDestroy(envUnsubscribe);\n\n\tasync function getHostsFile(hf) {\n\t\tif (hf.Records === null) {\n\t\t\thf.Records = []\n\t\t}\n\t\tif (hf.Records.length > 0) {\n\t\t\thostsFile = hf;\n\t\t\treturn;\n\t\t}\n\t\tconst res = await fetch(apiHostsDir +\"/\"+ hf.Name);\n\t\thf.Records = await res.json();\n\t\thostsFile = hf;\n\t}\n\n\tasync function createHostsFile() {\n\t\tif (newHostsFile === \"\") {\n\t\t\treturn;\n\t\t}\n\n\t\tconst res = await fetch(apiHostsDir+ \"/\"+ newHostsFile, {\n\t\t\tmethod: \"PUT\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: createHostsFile: \", resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tconst hf = {\n\t\t\tName: newHostsFile,\n\t\t\tRecords: [],\n\t\t}\n\t\tenv.HostsFiles[newHostsFile] = hf\n\t\tenv = env\n\n\t\tWuiPushNotif.Info(\"The new host file '\"+ newHostsFile +\"' has been created\")\n\t}\n\n\tasync function updateHostsFile() {\n\t\tconst res = await fetch(apiHostsDir+\"/\"+ hostsFile.Name, {\n\t\t\tmethod: \"POST\",\n\t\t\tbody: JSON.stringify(hostsFile.Records),\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: updateHostsFile: \", resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\thostsFile.Records = await res.json()\n\n\t\tWuiPushNotif.Info(\"The host file '\"+ hostsFile.Name +\"' has been updated\")\n\t}\n\n\tfunction addRecord() {\n\t\tif (newRecord !== null) {\n\t\t\treturn\n\t\t}\n\t\tnewRecord = {\n\t\t\tName: \"\",\n\t\t\tValue: \"\",\n\t\t}\n\t}\n\n\tasync function handleHostsRecordCreate() {\n\t\tconst api = apiHostsDir +\"/\"+ hostsFile.Name +\"/rr\"\n\t\t\t+\"?domain=\"+ newRecord.Name\n\t\t\t+\"&value=\"+ newRecord.Value\n\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"POST\"\n\t\t})\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\t\tconst rr = await res.json()\n\t\thostsFile.Records.push(rr)\n\t\thostsFile.Records = hostsFile.Records\n\t\tnewRecord = null\n\t\tWuiPushNotif.Info(\"Record '\"+ rr.Name +\"' has been created\")\n\t}\n\n\tasync function handleHostsRecordDelete(rr, idx) {\n\t\tconst api = apiHostsDir +\"/\"+ hostsFile.Name +\"/rr\"+\n\t\t\t\"?domain=\"+rr.Name\n\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"DELETE\"\n\t\t})\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\t\thostsFile.Records.splice(idx, 1);\n\t\thostsFile.Records = hostsFile.Records;\n\t\tWuiPushNotif.Info(\"Record '\"+ rr.Name +\"' has been deleted\")\n\t}\n\n\tasync function deleteHostsFile(hfile) {\n\t\tconst res = await fetch(apiHostsDir+\"/\"+hfile.Name, {\n\t\t\tmethod: \"DELETE\",\n\t\t});\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: deleteHostsFile: \", resError.message)\n\t\t\treturn;\n\t\t}\n\t\tdelete env.HostsFiles[hfile.Name]\n\t\tenv = env;\n\t\thostsFile = {\n\t\t\tName: \"\",\n\t\t\tRecords: [],\n\t\t}\n\t\tWuiPushNotif.Info(\"The host file '\"+ hfile.Name +\"' has been deleted\")\n\t}\n</script>\n\n<style>\n\t.nav-left {\n\t\tpadding: 0px;\n\t\twidth: 280px;\n\t\tfloat: left;\n\t}\n\t.nav-left .item {\n\t\tmargin: 4px 0px;\n\t}\n\t.content {\n\t\tfloat: left;\n\t\twidth: calc(100% - 300px);\n\t}\n\t.host {\n\t\tfont-family: monospace;\n\t\twidth: 100%;\n\t}\n\t.host.header {\n\t\tmargin: 1em 0px;\n\t\tfont-weight: bold;\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.host_name {\n\t\tdisplay: inline-block;\n\t\twidth: 240px;\n\t\tword-wrap: break-word;\n\t}\n\t.host_value {\n\t\tdisplay: inline-block;\n\t\twidth: 140px;\n\t}\n</style>\n\n<div class=\"hosts_d\">\n\t<div class=\"nav-left\">\n\t\t{#each Object.entries(env.HostsFiles) as [name,hf], name }\n\t\t<div class=\"item\">\n\t\t\t<a href=\"#\" on:click={getHostsFile(hf)}>\n\t\t\t\t{hf.Name}\n\t\t\t</a>\n\t\t</div>\n\t\t{/each}\n\n\t\t<br/>\n\n\t\t<label>\n\t\t\t<span>New hosts file:</span>\n\t\t\t<br/>\n\t\t\t<input bind:value={newHostsFile}>\n\t\t</label>\n\t\t<button on:click={createHostsFile}>\n\t\t\tCreate\n\t\t</button>\n\t</div>\n\n\t<div class=\"content\">\n\t\t{#if hostsFile.Name === \"\"}\n\t\t\t<div>\n\t\t\t\tSelect one of the hosts file to manage.\n\t\t\t</div>\n\t\t{:else}\n\t\t\t<p>\n\t\t\t\t{hostsFile.Name} ({hostsFile.Records.length} records)\n\t\t\t\t<button on:click={deleteHostsFile(hostsFile)}>\n\t\t\t\t\tDelete\n\t\t\t\t</button>\n\t\t\t</p>\n\t\t\t<div>\n\t\t\t\t<button on:click={addRecord}>\n\t\t\t\t\tAdd\n\t\t\t\t</button>\n\t\t\t</div>\n\n\t\t\t{#if newRecord !== null}\n\t\t\t\t<div class=\"host\">\n\t\t\t\t\t<input\n\t\t\t\t\t\tclass=\"host_name\"\n\t\t\t\t\t\tplaceholder=\"Domain name\"\n\t\t\t\t\t\tbind:value={newRecord.Name}\n\t\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tclass=\"host_value\"\n\t\t\t\t\t\tplaceholder=\"IP address\"\n\t\t\t\t\t\tbind:value={newRecord.Value}\n\t\t\t\t\t>\n\t\t\t\t\t<button on:click={handleHostsRecordCreate}>\n\t\t\t\t\t\tCreate\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t{/if}\n\n\t\t\t<div class=\"host header\">\n\t\t\t\t<span class=\"host_name\"> Domain name </span>\n\t\t\t\t<span class=\"host_value\"> IP address </span>\n\t\t\t</div>\n\n\t\t\t{#each hostsFile.Records as rr, idx (idx)}\n\t\t\t\t<div class=\"host\">\n\t\t\t\t\t<span class=\"host_name\"> {rr.Name} </span>\n\t\t\t\t\t<span class=\"host_value\"> {rr.Value} </span>\n\t\t\t\t\t<button on:click={handleHostsRecordDelete(rr, idx)}>\n\t\t\t\t\t\tX\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t{/each}\n\t\t{/if}\n\t</div>\n</div>\n",
- "<script>\n\timport { onDestroy } from 'svelte';\n\timport { WuiPushNotif, WuiLabelHint } from 'wui.svelte';\n\timport { environment, nanoSeconds, setEnvironment } from './environment.js';\n\n\tconst apiMasterd = \"/api/master.d/\"\n\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: [],\n\t\tZoneFiles: {},\n\t};\n\tlet newZoneFile = \"\";\n\tlet activeZone = {\n\t\tName: \"\",\n\t};\n\n\tlet RRTypes = {\n\t\t1: 'A',\n\t\t2: 'NS',\n\t\t5: 'CNAME',\n\t\t12: 'PTR',\n\t\t15: 'MX',\n\t\t16: 'TXT',\n\t\t28: 'AAAA',\n\t};\n\n\tlet _rr = newRR()\n\tlet _rrMX = newMX();\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\tonDestroy(envUnsubscribe);\n\n\tfunction setActiveZone(zone) {\n\t\tif (zone.SOA === null) {\n\t\t\tzone.SOA = newSOA()\n\t\t}\n\t\tactiveZone = zone\n\t}\n\n\tasync function handleZoneFileCreate() {\n\t\tconst res = await fetch(apiMasterd + newZoneFile, {\n\t\t\tmethod: \"PUT\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleZoneFileCreate: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tactiveZone = await res.json()\n\t\tactiveZone.SOA = newSOA()\n\t\tenv.ZoneFiles[activeZone.Name] = activeZone\n\n\t\tWuiPushNotif.Info(\"The new zone file '\"+ newZoneFile +\"' has been created\")\n\t}\n\n\tasync function handleZoneFileDelete() {\n\t\tlet api = apiMasterd + activeZone.Name\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"DELETE\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleZoneFileDelete: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tWuiPushNotif.Info(\"The zone file '\"+ activeZone.Name + \"' has beed deleted\")\n\n\t\tdelete env.ZoneFiles[activeZone.Name]\n\t\tactiveZone = {\n\t\t\tName: \"\",\n\t\t}\n\t\tenv.ZoneFiles = env.ZoneFiles\n\n\t}\n\n\tfunction onSelectRRType() {\n\t\tswitch (_rr.Type) {\n\t\tcase 15:\n\t\t\t_rrMX = newMX()\n\t\t\tbreak\n\t\t}\n\t}\n\n\tasync function handleSOADelete() {\n\t\treturn handleDeleteRR(activeZone.SOA, -1)\n\t}\n\n\tasync function handleSOASave() {\n\t\t_rr = activeZone.SOA\n\t\treturn handleCreateRR()\n\t}\n\n\tasync function handleCreateRR() {\n\t\tswitch (_rr.Type) {\n\t\tcase 15:\n\t\t\t_rr.Value = _rrMX;\n\t\t\tbreak;\n\t\t}\n\n\t\tlet api = apiMasterd + activeZone.Name +\"/rr/\"+ _rr.Type;\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n \t\t\t'Content-Type': 'application/json',\n\t\t\t},\n\t\t\tbody: JSON.stringify(_rr),\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleCreateRR: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tlet resRR = await res.json()\n\n\t\tif (resRR.Type === 6) {\n\t\t\tactiveZone.SOA = resRR\n\t\t\tWuiPushNotif.Info(\"SOA record has been saved\")\n\t\t} else {\n\t\t\tlet listRR = activeZone.Records[resRR.Name]\n\t\t\tif (typeof listRR === \"undefined\") {\n\t\t\t\tlistRR = [];\n\t\t\t}\n\t\t\tlistRR.push(resRR);\n\t\t\tactiveZone.Records[resRR.Name] = listRR\n\t\t\tWuiPushNotif.Info(\"The new record '\"+ resRR.Name +\"' has been created\")\n\t\t}\n\t}\n\n\tasync function handleDeleteRR(rr, idx) {\n\t\tlet api = apiMasterd + activeZone.Name +\"/rr/\"+ rr.Type\n\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"DELETE\",\n\t\t\theaders: {\n \t\t\t'Content-Type': 'application/json',\n\t\t\t},\n\t\t\tbody: JSON.stringify(rr),\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleDeleteRR: \"+ resError.message)\n\t\t\treturn\n\t\t}\n\n\t\tWuiPushNotif.Info(\"The record '\"+ rr.Name +\"' has been deleted\")\n\n\t\tif (rr.Type == 6) { // SOA.\n\t\t\tactiveZone.SOA = newSOA()\n\t\t} else {\n\t\t\tlet listRR = activeZone.Records[rr.Name]\n\t\t\tlistRR.splice(idx, 1)\n\t\t\tactiveZone.Records[rr.Name] = listRR\n\t\t}\n\n\t\tlet resbody = await res.json()\n\t}\n\n\tfunction getTypeName(k) {\n\t\tlet v = RRTypes[k];\n\t\tif (v === \"\") {\n\t\t\treturn k;\n\t\t}\n\t\treturn v;\n\t}\n\n\tfunction newRR() {\n\t\treturn {\n\t\t\tName: \"\",\n\t\t\tType: 1,\n\t\t\tValue: \"\",\n\t\t};\n\t}\n\n\tfunction newMX() {\n\t\treturn {\n\t\t\tPreference: 1,\n\t\t\tExchange: \"\",\n\t\t}\n\t}\n\n\tfunction newSOA() {\n\t\treturn {\n\t\t\tName: \"\",\n\t\t\tType: 6,\n\t\t\tValue: {\n\t\t\t\tMName: \"\",\n\t\t\t\tRName: \"\",\n\t\t\t\tSerial: 0,\n\t\t\t\tRefresh: 0,\n\t\t\t\tRetry: 0,\n\t\t\t\tExpire: 0,\n\t\t\t\tMinimum: 0\n\t\t\t}\n\t\t};\n\t}\n</script>\n\n<style>\n\th4 {\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.nav-left {\n\t\tpadding: 0px;\n\t\twidth: 280px;\n\t\tfloat: left;\n\t}\n\t.nav-left .item {\n\t\tmargin: 4px 0px;\n\t\tcursor: pointer;\n\t\tcolor: rgb(0, 100, 200);\n\t}\n\t.content {\n\t\tfloat: left;\n\t\twidth: calc(100% - 300px);\n\t}\n\t.action-delete {\n\t\tmargin-left: 1em;\n\t}\n\t.actions {\n\t\tpadding: 1em;\n\t}\n\t.actions button {\n\t\twidth: 100%;\n\t}\n\n\tform {\n\t\tmargin: 1em 0px;\n\t\tpadding: 10px 10px 0px 10px;\n\t\tborder: 1px solid silver;\n\t}\n\tform > label > span {\n\t\twidth: 7em;\n\t\tdisplay: inline-block;\n\t}\n\tform > label > input {\n\t\twidth: calc(100% - 8em);\n\t}\n\tform > label > input.name {\n\t\twidth: 12em;\n\t}\n\n\t.rr {\n\t\tfont-family: monospace;\n\t\twidth: 100%;\n\t\tpadding: 1em 0px;\n\t}\n\t.rr.header {\n\t\tfont-weight: bold;\n\t}\n\t.rr > .name {\n\t\twidth: 20em;\n\t\tdisplay: inline-block;\n\t\tword-wrap: break-word;\n\t}\n\t.rr > .type {\n\t\twidth: 4em;\n\t\tdisplay: inline-block;\n\t}\n\t.rr > .ttl {\n\t\twidth: 6em;\n\t\tdisplay: inline-block;\n\t}\n\t.rr > .value {\n\t\tdisplay: inline-block;\n\t\tword-wrap: break-word;\n\t}\n</style>\n\n<div class=\"master_d\">\n\t<div class=\"nav-left\">\n\t\t{#each Object.entries(env.ZoneFiles) as [name, zoneFile]}\n\t\t\t<div class=\"item\">\n\t\t\t\t<span on:click={setActiveZone(zoneFile)}>\n\t\t\t\t\t{zoneFile.Name}\n\t\t\t\t</span>\n\t\t\t</div>\n\t\t{/each}\n\t\t<br/>\n\n\t\t<label>\n\t\t\t<span>New zone file:</span>\n\t\t\t<br/>\n\t\t\t<input bind:value={newZoneFile}>\n\t\t</label>\n\t\t<button on:click={handleZoneFileCreate}>\n\t\t\tCreate\n\t\t</button>\n\t</div>\n\n\t<div class=\"content\">\n\t\t{#if activeZone.Name === \"\"}\n\t\t\t<p>\n\t\t\t\tSelect one of the zone file to manage.\n\t\t\t</p>\n\t\t{:else}\n\t\t\t<h3>\n\t\t\t\t{activeZone.Name}\n\t\t\t\t<button\n\t\t\t\t\tclass=\"action-delete\"\n\t\t\t\t\ton:click={handleZoneFileDelete}\n\t\t\t\t>\n\t\t\t\t\tDelete\n\t\t\t\t</button>\n\t\t\t</h3>\n\n\t\t\t<h4>\n\t\t\t\tSOA record\n\t\t\t\t<button\n\t\t\t\t\tclass=\"action-delete\"\n\t\t\t\t\ton:click={handleSOADelete}\n\t\t\t\t>\n\t\t\t\t\tDelete\n\t\t\t\t</button>\n\t\t\t</h4>\n\t\t\t<div class=\"rr-soa\">\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Name server\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"The domain-name of the name server that was the\noriginal or primary source of data for this zone.\nIt should be domain-name where the rescached run.\"\n\t\t\t\t>\n\t\t\t\t\t<input bind:value={activeZone.SOA.Value.MName}>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Admin email\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo='Email address of the administrator responsible for\nthis zone.\nThe \"@\" on email address is replaced with dot, and if there is a dot before\n\"@\" it should be escaped with \"\\\".\nFor example, \"dns.admin@domain.tld\" would be written as\n\"dns\\.admin.domain.tld\".'\n\t\t\t\t>\n\t\t\t\t\t<input bind:value={activeZone.SOA.Value.RName}>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Serial\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Serial number for this zone. If a secondary name\nserver observes an increase in this number, the server will assume that the\nzone has been updated and initiate a zone transfer.\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Serial}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Refresh\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Number of seconds after which secondary name servers\nshould query the master for the SOA record, to detect zone changes.\nRecommendation for small and stable zones is 86400 seconds (24 hours).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Refresh}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Retry\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Number of seconds after which secondary name servers\nshould retry to request the serial number from the master if the master does\nnot respond.\nIt must be less than Refresh.\nRecommendation for small and stable zones is 7200 seconds (2 hours).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Retry}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Expire\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Number of seconds after which secondary name servers\nshould stop answering request for this zone if the master does not respond.\nThis value must be bigger than the sum of Refresh and Retry.\nRecommendation for small and stable zones is 3600000 seconds (1000 hours).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Expire}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Minimum\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Time to live for purposes of negative caching.\nRecommendation for small and stable zones is 1800 seconds (30 min).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Minimum}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<div class=\"actions\">\n\t\t\t\t\t<button on:click={handleSOASave}>\n\t\t\t\t\t\tSave\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t\t<h4> List records </h4>\n\t\t\t<div class=\"rr header\">\n\t\t\t\t<span class=\"name\">\n\t\t\t\t\tName\n\t\t\t\t</span>\n\t\t\t\t<span class=\"type\">\n\t\t\t\t\tType\n\t\t\t\t</span>\n\t\t\t\t<span class=\"value\">\n\t\t\t\t\tValue\n\t\t\t\t</span>\n\t\t\t</div>\n\n\t\t\t{#each Object.entries(activeZone.Records) as [dname, listRR]}\n\t\t\t\t{#each listRR as rr, idx}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"name\">\n\t\t\t\t\t\t\t{rr.Name}\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<span class=\"type\">\n\t\t\t\t\t\t\t{getTypeName(rr.Type)}\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<span class=\"value\">\n\t\t\t\t\t\t\t{rr.Value}\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<button on:click={handleDeleteRR(rr, idx)}>\n\t\t\t\t\t\t\tX\n\t\t\t\t\t\t</button>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/each}\n\n\t\t\t<form on:submit|preventDefault={handleCreateRR}>\n\t\t\t\t<label>\n\t\t\t\t\t<span>\n\t\t\t\t\t\tType:\n\t\t\t\t\t</span>\n\t\t\t\t\t<select\n\t\t\t\t\t\tbind:value={_rr.Type}\n\t\t\t\t\t\ton:blur={onSelectRRType}\n\t\t\t\t\t>\n\t\t\t\t\t\t{#each Object.entries(RRTypes) as [k, v]}\n\t\t\t\t\t\t\t<option value={parseInt(k)}>\n\t\t\t\t\t\t\t\t{v}\n\t\t\t\t\t\t\t</option>\n\t\t\t\t\t\t{/each}\n\t\t\t\t\t</select>\n\t\t\t\t</label>\n\n\t\t\t\t{#if _rr.Type === 1 || _rr.Type === 2 || _rr.Type === 5 ||\n\t\t\t\t\t_rr.Type === 16 || _rr.Type === 28\n\t\t\t\t}\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tName:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input class=\"name\" bind:value={_rr.Name}>\n\t\t\t\t\t\t.{activeZone.Name}\n\t\t\t\t\t</label>\n\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tValue:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rr.Value}>\n\t\t\t\t\t</label>\n\t\t\t\t{:else if _rr.Type === 12} <!-- PTR -->\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tName:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rr.Name}>\n\t\t\t\t\t</label>\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tValue:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input class=\"name\" bind:value={_rr.Value}>\n\t\t\t\t\t\t.{activeZone.Name}\n\t\t\t\t\t</label>\n\t\t\t\t{:else if _rr.Type === 15} <!-- MX -->\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tName:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input class=\"name\" bind:value={_rr.Name}>\n\t\t\t\t\t\t.{activeZone.Name}\n\t\t\t\t\t</label>\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tPreference:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rrMX.Preference} type=number min=1 max=65535>\n\t\t\t\t\t</label>\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tExchange:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rrMX.Exchange}>\n\t\t\t\t\t</label>\n\t\t\t\t{/if}\n\n\t\t\t\t<div class=\"actions\">\n\t\t\t\t\t<button class=\"create\" type=submit>\n\t\t\t\t\t\tCreate\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t</form>\n\t\t{/if}\n\t</div> <!-- content -->\n</div> <!-- master_d -->\n"
- ],
- "names": [],
- "mappings": "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"
-} \ No newline at end of file
diff --git a/_www/public/build/bundle.js b/_www/public/build/bundle.js
deleted file mode 100644
index 7e3c5f6..0000000
--- a/_www/public/build/bundle.js
+++ /dev/null
@@ -1,8001 +0,0 @@
-
-(function(l, r) { if (l.getElementById('livereloadscript')) return; r = l.createElement('script'); r.async = 1; r.src = '//' + (window.location.host || 'localhost').split(':')[0] + ':35729/livereload.js?snipver=1'; r.id = 'livereloadscript'; l.getElementsByTagName('head')[0].appendChild(r) })(window.document);
-var app = (function () {
- 'use strict';
-
- function noop() { }
- const identity = x => x;
- function assign(tar, src) {
- // @ts-ignore
- for (const k in src)
- tar[k] = src[k];
- return tar;
- }
- function add_location(element, file, line, column, char) {
- element.__svelte_meta = {
- loc: { file, line, column, char }
- };
- }
- function run(fn) {
- return fn();
- }
- function blank_object() {
- return Object.create(null);
- }
- function run_all(fns) {
- fns.forEach(run);
- }
- function is_function(thing) {
- return typeof thing === 'function';
- }
- function safe_not_equal(a, b) {
- return a != a ? b == b : a !== b || ((a && typeof a === 'object') || typeof a === 'function');
- }
- function is_empty(obj) {
- return Object.keys(obj).length === 0;
- }
- function validate_store(store, name) {
- if (store != null && typeof store.subscribe !== 'function') {
- throw new Error(`'${name}' is not a store with a 'subscribe' method`);
- }
- }
- function subscribe(store, ...callbacks) {
- if (store == null) {
- return noop;
- }
- const unsub = store.subscribe(...callbacks);
- return unsub.unsubscribe ? () => unsub.unsubscribe() : unsub;
- }
- function component_subscribe(component, store, callback) {
- component.$$.on_destroy.push(subscribe(store, callback));
- }
- function create_slot(definition, ctx, $$scope, fn) {
- if (definition) {
- const slot_ctx = get_slot_context(definition, ctx, $$scope, fn);
- return definition[0](slot_ctx);
- }
- }
- function get_slot_context(definition, ctx, $$scope, fn) {
- return definition[1] && fn
- ? assign($$scope.ctx.slice(), definition[1](fn(ctx)))
- : $$scope.ctx;
- }
- function get_slot_changes(definition, $$scope, dirty, fn) {
- if (definition[2] && fn) {
- const lets = definition[2](fn(dirty));
- if ($$scope.dirty === undefined) {
- return lets;
- }
- if (typeof lets === 'object') {
- const merged = [];
- const len = Math.max($$scope.dirty.length, lets.length);
- for (let i = 0; i < len; i += 1) {
- merged[i] = $$scope.dirty[i] | lets[i];
- }
- return merged;
- }
- return $$scope.dirty | lets;
- }
- return $$scope.dirty;
- }
- function update_slot(slot, slot_definition, ctx, $$scope, dirty, get_slot_changes_fn, get_slot_context_fn) {
- const slot_changes = get_slot_changes(slot_definition, $$scope, dirty, get_slot_changes_fn);
- if (slot_changes) {
- const slot_context = get_slot_context(slot_definition, ctx, $$scope, get_slot_context_fn);
- slot.p(slot_context, slot_changes);
- }
- }
-
- const is_client = typeof window !== 'undefined';
- let now = is_client
- ? () => window.performance.now()
- : () => Date.now();
- let raf = is_client ? cb => requestAnimationFrame(cb) : noop;
-
- const tasks = new Set();
- function run_tasks(now) {
- tasks.forEach(task => {
- if (!task.c(now)) {
- tasks.delete(task);
- task.f();
- }
- });
- if (tasks.size !== 0)
- raf(run_tasks);
- }
- /**
- * Creates a new task that runs on each raf frame
- * until it returns a falsy value or is aborted
- */
- function loop(callback) {
- let task;
- if (tasks.size === 0)
- raf(run_tasks);
- return {
- promise: new Promise(fulfill => {
- tasks.add(task = { c: callback, f: fulfill });
- }),
- abort() {
- tasks.delete(task);
- }
- };
- }
-
- function append(target, node) {
- target.appendChild(node);
- }
- function insert(target, node, anchor) {
- target.insertBefore(node, anchor || null);
- }
- function detach(node) {
- node.parentNode.removeChild(node);
- }
- function destroy_each(iterations, detaching) {
- for (let i = 0; i < iterations.length; i += 1) {
- if (iterations[i])
- iterations[i].d(detaching);
- }
- }
- function element(name) {
- return document.createElement(name);
- }
- function text(data) {
- return document.createTextNode(data);
- }
- function space() {
- return text(' ');
- }
- function empty() {
- return text('');
- }
- function listen(node, event, handler, options) {
- node.addEventListener(event, handler, options);
- return () => node.removeEventListener(event, handler, options);
- }
- function prevent_default(fn) {
- return function (event) {
- event.preventDefault();
- // @ts-ignore
- return fn.call(this, event);
- };
- }
- function attr(node, attribute, value) {
- if (value == null)
- node.removeAttribute(attribute);
- else if (node.getAttribute(attribute) !== value)
- node.setAttribute(attribute, value);
- }
- function to_number(value) {
- return value === '' ? undefined : +value;
- }
- function children(element) {
- return Array.from(element.childNodes);
- }
- function set_input_value(input, value) {
- input.value = value == null ? '' : value;
- }
- function set_style(node, key, value, important) {
- node.style.setProperty(key, value, important ? 'important' : '');
- }
- function select_option(select, value) {
- for (let i = 0; i < select.options.length; i += 1) {
- const option = select.options[i];
- if (option.__value === value) {
- option.selected = true;
- return;
- }
- }
- }
- function select_value(select) {
- const selected_option = select.querySelector(':checked') || select.options[0];
- return selected_option && selected_option.__value;
- }
- function toggle_class(element, name, toggle) {
- element.classList[toggle ? 'add' : 'remove'](name);
- }
- function custom_event(type, detail) {
- const e = document.createEvent('CustomEvent');
- e.initCustomEvent(type, false, false, detail);
- return e;
- }
-
- const active_docs = new Set();
- let active = 0;
- // https://github.com/darkskyapp/string-hash/blob/master/index.js
- function hash(str) {
- let hash = 5381;
- let i = str.length;
- while (i--)
- hash = ((hash << 5) - hash) ^ str.charCodeAt(i);
- return hash >>> 0;
- }
- function create_rule(node, a, b, duration, delay, ease, fn, uid = 0) {
- const step = 16.666 / duration;
- let keyframes = '{\n';
- for (let p = 0; p <= 1; p += step) {
- const t = a + (b - a) * ease(p);
- keyframes += p * 100 + `%{${fn(t, 1 - t)}}\n`;
- }
- const rule = keyframes + `100% {${fn(b, 1 - b)}}\n}`;
- const name = `__svelte_${hash(rule)}_${uid}`;
- const doc = node.ownerDocument;
- active_docs.add(doc);
- const stylesheet = doc.__svelte_stylesheet || (doc.__svelte_stylesheet = doc.head.appendChild(element('style')).sheet);
- const current_rules = doc.__svelte_rules || (doc.__svelte_rules = {});
- if (!current_rules[name]) {
- current_rules[name] = true;
- stylesheet.insertRule(`@keyframes ${name} ${rule}`, stylesheet.cssRules.length);
- }
- const animation = node.style.animation || '';
- node.style.animation = `${animation ? `${animation}, ` : ``}${name} ${duration}ms linear ${delay}ms 1 both`;
- active += 1;
- return name;
- }
- function delete_rule(node, name) {
- const previous = (node.style.animation || '').split(', ');
- const next = previous.filter(name
- ? anim => anim.indexOf(name) < 0 // remove specific animation
- : anim => anim.indexOf('__svelte') === -1 // remove all Svelte animations
- );
- const deleted = previous.length - next.length;
- if (deleted) {
- node.style.animation = next.join(', ');
- active -= deleted;
- if (!active)
- clear_rules();
- }
- }
- function clear_rules() {
- raf(() => {
- if (active)
- return;
- active_docs.forEach(doc => {
- const stylesheet = doc.__svelte_stylesheet;
- let i = stylesheet.cssRules.length;
- while (i--)
- stylesheet.deleteRule(i);
- doc.__svelte_rules = {};
- });
- active_docs.clear();
- });
- }
-
- let current_component;
- function set_current_component(component) {
- current_component = component;
- }
- function get_current_component() {
- if (!current_component)
- throw new Error(`Function called outside component initialization`);
- return current_component;
- }
- function onMount(fn) {
- get_current_component().$$.on_mount.push(fn);
- }
- function onDestroy(fn) {
- get_current_component().$$.on_destroy.push(fn);
- }
-
- const dirty_components = [];
- const binding_callbacks = [];
- const render_callbacks = [];
- const flush_callbacks = [];
- const resolved_promise = Promise.resolve();
- let update_scheduled = false;
- function schedule_update() {
- if (!update_scheduled) {
- update_scheduled = true;
- resolved_promise.then(flush);
- }
- }
- function add_render_callback(fn) {
- render_callbacks.push(fn);
- }
- function add_flush_callback(fn) {
- flush_callbacks.push(fn);
- }
- let flushing = false;
- const seen_callbacks = new Set();
- function flush() {
- if (flushing)
- return;
- flushing = true;
- do {
- // first, call beforeUpdate functions
- // and update components
- for (let i = 0; i < dirty_components.length; i += 1) {
- const component = dirty_components[i];
- set_current_component(component);
- update(component.$$);
- }
- dirty_components.length = 0;
- while (binding_callbacks.length)
- binding_callbacks.pop()();
- // then, once components are updated, call
- // afterUpdate functions. This may cause
- // subsequent updates...
- for (let i = 0; i < render_callbacks.length; i += 1) {
- const callback = render_callbacks[i];
- if (!seen_callbacks.has(callback)) {
- // ...so guard against infinite loops
- seen_callbacks.add(callback);
- callback();
- }
- }
- render_callbacks.length = 0;
- } while (dirty_components.length);
- while (flush_callbacks.length) {
- flush_callbacks.pop()();
- }
- update_scheduled = false;
- flushing = false;
- seen_callbacks.clear();
- }
- function update($$) {
- if ($$.fragment !== null) {
- $$.update();
- run_all($$.before_update);
- const dirty = $$.dirty;
- $$.dirty = [-1];
- $$.fragment && $$.fragment.p($$.ctx, dirty);
- $$.after_update.forEach(add_render_callback);
- }
- }
-
- let promise;
- function wait() {
- if (!promise) {
- promise = Promise.resolve();
- promise.then(() => {
- promise = null;
- });
- }
- return promise;
- }
- function dispatch(node, direction, kind) {
- node.dispatchEvent(custom_event(`${direction ? 'intro' : 'outro'}${kind}`));
- }
- const outroing = new Set();
- let outros;
- function group_outros() {
- outros = {
- r: 0,
- c: [],
- p: outros // parent group
- };
- }
- function check_outros() {
- if (!outros.r) {
- run_all(outros.c);
- }
- outros = outros.p;
- }
- function transition_in(block, local) {
- if (block && block.i) {
- outroing.delete(block);
- block.i(local);
- }
- }
- function transition_out(block, local, detach, callback) {
- if (block && block.o) {
- if (outroing.has(block))
- return;
- outroing.add(block);
- outros.c.push(() => {
- outroing.delete(block);
- if (callback) {
- if (detach)
- block.d(1);
- callback();
- }
- });
- block.o(local);
- }
- }
- const null_transition = { duration: 0 };
- function create_bidirectional_transition(node, fn, params, intro) {
- let config = fn(node, params);
- let t = intro ? 0 : 1;
- let running_program = null;
- let pending_program = null;
- let animation_name = null;
- function clear_animation() {
- if (animation_name)
- delete_rule(node, animation_name);
- }
- function init(program, duration) {
- const d = program.b - t;
- duration *= Math.abs(d);
- return {
- a: t,
- b: program.b,
- d,
- duration,
- start: program.start,
- end: program.start + duration,
- group: program.group
- };
- }
- function go(b) {
- const { delay = 0, duration = 300, easing = identity, tick = noop, css } = config || null_transition;
- const program = {
- start: now() + delay,
- b
- };
- if (!b) {
- // @ts-ignore todo: improve typings
- program.group = outros;
- outros.r += 1;
- }
- if (running_program) {
- pending_program = program;
- }
- else {
- // if this is an intro, and there's a delay, we need to do
- // an initial tick and/or apply CSS animation immediately
- if (css) {
- clear_animation();
- animation_name = create_rule(node, t, b, duration, delay, easing, css);
- }
- if (b)
- tick(0, 1);
- running_program = init(program, duration);
- add_render_callback(() => dispatch(node, b, 'start'));
- loop(now => {
- if (pending_program && now > pending_program.start) {
- running_program = init(pending_program, duration);
- pending_program = null;
- dispatch(node, running_program.b, 'start');
- if (css) {
- clear_animation();
- animation_name = create_rule(node, t, running_program.b, running_program.duration, 0, easing, config.css);
- }
- }
- if (running_program) {
- if (now >= running_program.end) {
- tick(t = running_program.b, 1 - t);
- dispatch(node, running_program.b, 'end');
- if (!pending_program) {
- // we're done
- if (running_program.b) {
- // intro — we can tidy up immediately
- clear_animation();
- }
- else {
- // outro — needs to be coordinated
- if (!--running_program.group.r)
- run_all(running_program.group.c);
- }
- }
- running_program = null;
- }
- else if (now >= running_program.start) {
- const p = now - running_program.start;
- t = running_program.a + running_program.d * easing(p / running_program.duration);
- tick(t, 1 - t);
- }
- }
- return !!(running_program || pending_program);
- });
- }
- }
- return {
- run(b) {
- if (is_function(config)) {
- wait().then(() => {
- // @ts-ignore
- config = config();
- go(b);
- });
- }
- else {
- go(b);
- }
- },
- end() {
- clear_animation();
- running_program = pending_program = null;
- }
- };
- }
-
- const globals = (typeof window !== 'undefined'
- ? window
- : typeof globalThis !== 'undefined'
- ? globalThis
- : global);
-
- function destroy_block(block, lookup) {
- block.d(1);
- lookup.delete(block.key);
- }
- function outro_and_destroy_block(block, lookup) {
- transition_out(block, 1, 1, () => {
- lookup.delete(block.key);
- });
- }
- function update_keyed_each(old_blocks, dirty, get_key, dynamic, ctx, list, lookup, node, destroy, create_each_block, next, get_context) {
- let o = old_blocks.length;
- let n = list.length;
- let i = o;
- const old_indexes = {};
- while (i--)
- old_indexes[old_blocks[i].key] = i;
- const new_blocks = [];
- const new_lookup = new Map();
- const deltas = new Map();
- i = n;
- while (i--) {
- const child_ctx = get_context(ctx, list, i);
- const key = get_key(child_ctx);
- let block = lookup.get(key);
- if (!block) {
- block = create_each_block(key, child_ctx);
- block.c();
- }
- else if (dynamic) {
- block.p(child_ctx, dirty);
- }
- new_lookup.set(key, new_blocks[i] = block);
- if (key in old_indexes)
- deltas.set(key, Math.abs(i - old_indexes[key]));
- }
- const will_move = new Set();
- const did_move = new Set();
- function insert(block) {
- transition_in(block, 1);
- block.m(node, next);
- lookup.set(block.key, block);
- next = block.first;
- n--;
- }
- while (o && n) {
- const new_block = new_blocks[n - 1];
- const old_block = old_blocks[o - 1];
- const new_key = new_block.key;
- const old_key = old_block.key;
- if (new_block === old_block) {
- // do nothing
- next = new_block.first;
- o--;
- n--;
- }
- else if (!new_lookup.has(old_key)) {
- // remove old block
- destroy(old_block, lookup);
- o--;
- }
- else if (!lookup.has(new_key) || will_move.has(new_key)) {
- insert(new_block);
- }
- else if (did_move.has(old_key)) {
- o--;
- }
- else if (deltas.get(new_key) > deltas.get(old_key)) {
- did_move.add(new_key);
- insert(new_block);
- }
- else {
- will_move.add(old_key);
- o--;
- }
- }
- while (o--) {
- const old_block = old_blocks[o];
- if (!new_lookup.has(old_block.key))
- destroy(old_block, lookup);
- }
- while (n)
- insert(new_blocks[n - 1]);
- return new_blocks;
- }
- function validate_each_keys(ctx, list, get_context, get_key) {
- const keys = new Set();
- for (let i = 0; i < list.length; i++) {
- const key = get_key(get_context(ctx, list, i));
- if (keys.has(key)) {
- throw new Error(`Cannot have duplicate keys in a keyed each`);
- }
- keys.add(key);
- }
- }
-
- function bind(component, name, callback) {
- const index = component.$$.props[name];
- if (index !== undefined) {
- component.$$.bound[index] = callback;
- callback(component.$$.ctx[index]);
- }
- }
- function create_component(block) {
- block && block.c();
- }
- function mount_component(component, target, anchor) {
- const { fragment, on_mount, on_destroy, after_update } = component.$$;
- fragment && fragment.m(target, anchor);
- // onMount happens before the initial afterUpdate
- add_render_callback(() => {
- const new_on_destroy = on_mount.map(run).filter(is_function);
- if (on_destroy) {
- on_destroy.push(...new_on_destroy);
- }
- else {
- // Edge case - component was destroyed immediately,
- // most likely as a result of a binding initialising
- run_all(new_on_destroy);
- }
- component.$$.on_mount = [];
- });
- after_update.forEach(add_render_callback);
- }
- function destroy_component(component, detaching) {
- const $$ = component.$$;
- if ($$.fragment !== null) {
- run_all($$.on_destroy);
- $$.fragment && $$.fragment.d(detaching);
- // TODO null out other refs, including component.$$ (but need to
- // preserve final state?)
- $$.on_destroy = $$.fragment = null;
- $$.ctx = [];
- }
- }
- function make_dirty(component, i) {
- if (component.$$.dirty[0] === -1) {
- dirty_components.push(component);
- schedule_update();
- component.$$.dirty.fill(0);
- }
- component.$$.dirty[(i / 31) | 0] |= (1 << (i % 31));
- }
- function init(component, options, instance, create_fragment, not_equal, props, dirty = [-1]) {
- const parent_component = current_component;
- set_current_component(component);
- const prop_values = options.props || {};
- const $$ = component.$$ = {
- fragment: null,
- ctx: null,
- // state
- props,
- update: noop,
- not_equal,
- bound: blank_object(),
- // lifecycle
- on_mount: [],
- on_destroy: [],
- before_update: [],
- after_update: [],
- context: new Map(parent_component ? parent_component.$$.context : []),
- // everything else
- callbacks: blank_object(),
- dirty,
- skip_bound: false
- };
- let ready = false;
- $$.ctx = instance
- ? instance(component, prop_values, (i, ret, ...rest) => {
- const value = rest.length ? rest[0] : ret;
- if ($$.ctx && not_equal($$.ctx[i], $$.ctx[i] = value)) {
- if (!$$.skip_bound && $$.bound[i])
- $$.bound[i](value);
- if (ready)
- make_dirty(component, i);
- }
- return ret;
- })
- : [];
- $$.update();
- ready = true;
- run_all($$.before_update);
- // `false` as a special case of no DOM component
- $$.fragment = create_fragment ? create_fragment($$.ctx) : false;
- if (options.target) {
- if (options.hydrate) {
- const nodes = children(options.target);
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
- $$.fragment && $$.fragment.l(nodes);
- nodes.forEach(detach);
- }
- else {
- // eslint-disable-next-line @typescript-eslint/no-non-null-assertion
- $$.fragment && $$.fragment.c();
- }
- if (options.intro)
- transition_in(component.$$.fragment);
- mount_component(component, options.target, options.anchor);
- flush();
- }
- set_current_component(parent_component);
- }
- class SvelteComponent {
- $destroy() {
- destroy_component(this, 1);
- this.$destroy = noop;
- }
- $on(type, callback) {
- const callbacks = (this.$$.callbacks[type] || (this.$$.callbacks[type] = []));
- callbacks.push(callback);
- return () => {
- const index = callbacks.indexOf(callback);
- if (index !== -1)
- callbacks.splice(index, 1);
- };
- }
- $set($$props) {
- if (this.$$set && !is_empty($$props)) {
- this.$$.skip_bound = true;
- this.$$set($$props);
- this.$$.skip_bound = false;
- }
- }
- }
-
- function dispatch_dev(type, detail) {
- document.dispatchEvent(custom_event(type, Object.assign({ version: '3.24.1' }, detail)));
- }
- function append_dev(target, node) {
- dispatch_dev("SvelteDOMInsert", { target, node });
- append(target, node);
- }
- function insert_dev(target, node, anchor) {
- dispatch_dev("SvelteDOMInsert", { target, node, anchor });
- insert(target, node, anchor);
- }
- function detach_dev(node) {
- dispatch_dev("SvelteDOMRemove", { node });
- detach(node);
- }
- function listen_dev(node, event, handler, options, has_prevent_default, has_stop_propagation) {
- const modifiers = options === true ? ["capture"] : options ? Array.from(Object.keys(options)) : [];
- if (has_prevent_default)
- modifiers.push('preventDefault');
- if (has_stop_propagation)
- modifiers.push('stopPropagation');
- dispatch_dev("SvelteDOMAddEventListener", { node, event, handler, modifiers });
- const dispose = listen(node, event, handler, options);
- return () => {
- dispatch_dev("SvelteDOMRemoveEventListener", { node, event, handler, modifiers });
- dispose();
- };
- }
- function attr_dev(node, attribute, value) {
- attr(node, attribute, value);
- if (value == null)
- dispatch_dev("SvelteDOMRemoveAttribute", { node, attribute });
- else
- dispatch_dev("SvelteDOMSetAttribute", { node, attribute, value });
- }
- function set_data_dev(text, data) {
- data = '' + data;
- if (text.wholeText === data)
- return;
- dispatch_dev("SvelteDOMSetData", { node: text, data });
- text.data = data;
- }
- function validate_each_argument(arg) {
- if (typeof arg !== 'string' && !(arg && typeof arg === 'object' && 'length' in arg)) {
- let msg = '{#each} only iterates over array-like objects.';
- if (typeof Symbol === 'function' && arg && Symbol.iterator in arg) {
- msg += ' You can use a spread to convert this iterable into an array.';
- }
- throw new Error(msg);
- }
- }
- function validate_slots(name, slot, keys) {
- for (const slot_key of Object.keys(slot)) {
- if (!~keys.indexOf(slot_key)) {
- console.warn(`<${name}> received an unexpected slot "${slot_key}".`);
- }
- }
- }
- class SvelteComponentDev extends SvelteComponent {
- constructor(options) {
- if (!options || (!options.target && !options.$$inline)) {
- throw new Error(`'target' is a required option`);
- }
- super();
- }
- $destroy() {
- super.$destroy();
- this.$destroy = () => {
- console.warn(`Component was already destroyed`); // eslint-disable-line no-console
- };
- }
- $capture_state() { }
- $inject_state() { }
- }
-
- /* node_modules/wui.svelte/src/components/InputIPPort.svelte generated by Svelte v3.24.1 */
-
- const file = "node_modules/wui.svelte/src/components/InputIPPort.svelte";
-
- // (54:1) {#if isInvalid}
- function create_if_block(ctx) {
- let div;
- let t;
-
- const block = {
- c: function create() {
- div = element("div");
- t = text(/*error*/ ctx[2]);
- attr_dev(div, "class", "invalid svelte-sy39ke");
- add_location(div, file, 54, 1, 911);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, t);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*error*/ 4) set_data_dev(t, /*error*/ ctx[2]);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block.name,
- type: "if",
- source: "(54:1) {#if isInvalid}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment(ctx) {
- let div;
- let input;
- let t;
- let mounted;
- let dispose;
- let if_block = /*isInvalid*/ ctx[1] && create_if_block(ctx);
-
- const block = {
- c: function create() {
- div = element("div");
- input = element("input");
- t = space();
- if (if_block) if_block.c();
- attr_dev(input, "type", "text");
- attr_dev(input, "placeholder", "127.0.0.1:8080");
- attr_dev(input, "class", "svelte-sy39ke");
- toggle_class(input, "invalid", /*isInvalid*/ ctx[1]);
- add_location(input, file, 46, 1, 770);
- attr_dev(div, "class", "wui-input-ipport svelte-sy39ke");
- add_location(div, file, 45, 0, 738);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, input);
- set_input_value(input, /*value*/ ctx[0]);
- append_dev(div, t);
- if (if_block) if_block.m(div, null);
-
- if (!mounted) {
- dispose = [
- listen_dev(input, "input", /*input_input_handler*/ ctx[4]),
- listen_dev(input, "blur", /*onBlur*/ ctx[3], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*value*/ 1 && input.value !== /*value*/ ctx[0]) {
- set_input_value(input, /*value*/ ctx[0]);
- }
-
- if (dirty & /*isInvalid*/ 2) {
- toggle_class(input, "invalid", /*isInvalid*/ ctx[1]);
- }
-
- if (/*isInvalid*/ ctx[1]) {
- if (if_block) {
- if_block.p(ctx, dirty);
- } else {
- if_block = create_if_block(ctx);
- if_block.c();
- if_block.m(div, null);
- }
- } else if (if_block) {
- if_block.d(1);
- if_block = null;
- }
- },
- i: noop,
- o: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- if (if_block) if_block.d();
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- function instance($$self, $$props, $$invalidate) {
- let { value = "" } = $$props;
- let isInvalid = false;
- let error = "";
-
- function onBlur() {
- if (value === "") {
- $$invalidate(1, isInvalid = false);
- $$invalidate(2, error = "");
- return;
- }
-
- const ipport = value.split(":");
-
- if (ipport.length !== 2) {
- $$invalidate(1, isInvalid = true);
- $$invalidate(2, error = "missing port number");
- return;
- }
-
- const ip = ipport[0];
- const port = parseInt(ipport[1]);
-
- if (isNaN(port) || port <= 0 || port >= 65535) {
- $$invalidate(1, isInvalid = true);
- $$invalidate(2, error = "invalid port number");
- return;
- }
-
- $$invalidate(1, isInvalid = false);
- $$invalidate(0, value = ip + ":" + port);
- }
-
- const writable_props = ["value"];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<InputIPPort> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("InputIPPort", $$slots, []);
-
- function input_input_handler() {
- value = this.value;
- $$invalidate(0, value);
- }
-
- $$self.$$set = $$props => {
- if ("value" in $$props) $$invalidate(0, value = $$props.value);
- };
-
- $$self.$capture_state = () => ({ value, isInvalid, error, onBlur });
-
- $$self.$inject_state = $$props => {
- if ("value" in $$props) $$invalidate(0, value = $$props.value);
- if ("isInvalid" in $$props) $$invalidate(1, isInvalid = $$props.isInvalid);
- if ("error" in $$props) $$invalidate(2, error = $$props.error);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [value, isInvalid, error, onBlur, input_input_handler];
- }
-
- class InputIPPort extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance, create_fragment, safe_not_equal, { value: 0 });
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "InputIPPort",
- options,
- id: create_fragment.name
- });
- }
-
- get value() {
- throw new Error("<InputIPPort>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set value(value) {
- throw new Error("<InputIPPort>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
- }
-
- /* node_modules/wui.svelte/src/components/InputNumber.svelte generated by Svelte v3.24.1 */
-
- const file$1 = "node_modules/wui.svelte/src/components/InputNumber.svelte";
-
- // (34:1) {#if unit !== ''}
- function create_if_block$1(ctx) {
- let span;
- let t;
-
- const block = {
- c: function create() {
- span = element("span");
- t = text(/*unit*/ ctx[1]);
- attr_dev(span, "class", "suffix svelte-1qrd8wr");
- add_location(span, file$1, 34, 2, 547);
- },
- m: function mount(target, anchor) {
- insert_dev(target, span, anchor);
- append_dev(span, t);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*unit*/ 2) set_data_dev(t, /*unit*/ ctx[1]);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(span);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block$1.name,
- type: "if",
- source: "(34:1) {#if unit !== ''}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$1(ctx) {
- let div;
- let input;
- let t;
- let mounted;
- let dispose;
- let if_block = /*unit*/ ctx[1] !== "" && create_if_block$1(ctx);
-
- const block = {
- c: function create() {
- div = element("div");
- input = element("input");
- t = space();
- if (if_block) if_block.c();
- attr_dev(input, "type", "number");
- attr_dev(input, "class", "svelte-1qrd8wr");
- add_location(input, file$1, 32, 1, 468);
- attr_dev(div, "class", "wui-input-number svelte-1qrd8wr");
- add_location(div, file$1, 31, 0, 436);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, input);
- set_input_value(input, /*value*/ ctx[0]);
- append_dev(div, t);
- if (if_block) if_block.m(div, null);
-
- if (!mounted) {
- dispose = [
- listen_dev(input, "blur", /*onBlur*/ ctx[2], false, false, false),
- listen_dev(input, "input", /*input_input_handler*/ ctx[5])
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*value*/ 1 && to_number(input.value) !== /*value*/ ctx[0]) {
- set_input_value(input, /*value*/ ctx[0]);
- }
-
- if (/*unit*/ ctx[1] !== "") {
- if (if_block) {
- if_block.p(ctx, dirty);
- } else {
- if_block = create_if_block$1(ctx);
- if_block.c();
- if_block.m(div, null);
- }
- } else if (if_block) {
- if_block.d(1);
- if_block = null;
- }
- },
- i: noop,
- o: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- if (if_block) if_block.d();
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$1.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- function instance$1($$self, $$props, $$invalidate) {
- let { min } = $$props;
- let { max } = $$props;
- let { value = 0 } = $$props;
- let { unit } = $$props;
-
- function onBlur() {
- $$invalidate(0, value = +value);
-
- if (isNaN(value)) {
- $$invalidate(0, value = max);
- } else if (value < min) {
- $$invalidate(0, value = min);
- } else if (value > max) {
- $$invalidate(0, value = max);
- }
- }
-
- const writable_props = ["min", "max", "value", "unit"];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<InputNumber> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("InputNumber", $$slots, []);
-
- function input_input_handler() {
- value = to_number(this.value);
- $$invalidate(0, value);
- }
-
- $$self.$$set = $$props => {
- if ("min" in $$props) $$invalidate(3, min = $$props.min);
- if ("max" in $$props) $$invalidate(4, max = $$props.max);
- if ("value" in $$props) $$invalidate(0, value = $$props.value);
- if ("unit" in $$props) $$invalidate(1, unit = $$props.unit);
- };
-
- $$self.$capture_state = () => ({ min, max, value, unit, onBlur });
-
- $$self.$inject_state = $$props => {
- if ("min" in $$props) $$invalidate(3, min = $$props.min);
- if ("max" in $$props) $$invalidate(4, max = $$props.max);
- if ("value" in $$props) $$invalidate(0, value = $$props.value);
- if ("unit" in $$props) $$invalidate(1, unit = $$props.unit);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [value, unit, onBlur, min, max, input_input_handler];
- }
-
- class InputNumber extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$1, create_fragment$1, safe_not_equal, { min: 3, max: 4, value: 0, unit: 1 });
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "InputNumber",
- options,
- id: create_fragment$1.name
- });
-
- const { ctx } = this.$$;
- const props = options.props || {};
-
- if (/*min*/ ctx[3] === undefined && !("min" in props)) {
- console.warn("<InputNumber> was created without expected prop 'min'");
- }
-
- if (/*max*/ ctx[4] === undefined && !("max" in props)) {
- console.warn("<InputNumber> was created without expected prop 'max'");
- }
-
- if (/*unit*/ ctx[1] === undefined && !("unit" in props)) {
- console.warn("<InputNumber> was created without expected prop 'unit'");
- }
- }
-
- get min() {
- throw new Error("<InputNumber>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set min(value) {
- throw new Error("<InputNumber>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get max() {
- throw new Error("<InputNumber>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set max(value) {
- throw new Error("<InputNumber>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get value() {
- throw new Error("<InputNumber>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set value(value) {
- throw new Error("<InputNumber>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get unit() {
- throw new Error("<InputNumber>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set unit(value) {
- throw new Error("<InputNumber>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
- }
-
- /* node_modules/wui.svelte/src/components/LabelHint.svelte generated by Svelte v3.24.1 */
-
- const file$2 = "node_modules/wui.svelte/src/components/LabelHint.svelte";
-
- // (57:0) {#if showInfo}
- function create_if_block$2(ctx) {
- let div;
-
- const block = {
- c: function create() {
- div = element("div");
- attr_dev(div, "class", "info svelte-1weevo5");
- add_location(div, file$2, 57, 0, 859);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- div.innerHTML = /*info*/ ctx[1];
- },
- p: function update(ctx, dirty) {
- if (dirty & /*info*/ 2) div.innerHTML = /*info*/ ctx[1]; },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block$2.name,
- type: "if",
- source: "(57:0) {#if showInfo}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$2(ctx) {
- let label;
- let span1;
- let t0;
- let t1;
- let span0;
- let t3;
- let t4;
- let if_block_anchor;
- let current;
- let mounted;
- let dispose;
- const default_slot_template = /*$$slots*/ ctx[5].default;
- const default_slot = create_slot(default_slot_template, ctx, /*$$scope*/ ctx[4], null);
- let if_block = /*showInfo*/ ctx[3] && create_if_block$2(ctx);
-
- const block = {
- c: function create() {
- label = element("label");
- span1 = element("span");
- t0 = text(/*title*/ ctx[0]);
- t1 = space();
- span0 = element("span");
- span0.textContent = "?";
- t3 = space();
- if (default_slot) default_slot.c();
- t4 = space();
- if (if_block) if_block.c();
- if_block_anchor = empty();
- attr_dev(span0, "class", "toggle svelte-1weevo5");
- add_location(span0, file$2, 46, 2, 731);
- attr_dev(span1, "class", "title svelte-1weevo5");
- set_style(span1, "width", /*title_width*/ ctx[2]);
- add_location(span1, file$2, 44, 1, 669);
- attr_dev(label, "class", "label-hint svelte-1weevo5");
- add_location(label, file$2, 43, 0, 641);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, label, anchor);
- append_dev(label, span1);
- append_dev(span1, t0);
- append_dev(span1, t1);
- append_dev(span1, span0);
- append_dev(label, t3);
-
- if (default_slot) {
- default_slot.m(label, null);
- }
-
- insert_dev(target, t4, anchor);
- if (if_block) if_block.m(target, anchor);
- insert_dev(target, if_block_anchor, anchor);
- current = true;
-
- if (!mounted) {
- dispose = listen_dev(span0, "click", /*click_handler*/ ctx[6], false, false, false);
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (!current || dirty & /*title*/ 1) set_data_dev(t0, /*title*/ ctx[0]);
-
- if (!current || dirty & /*title_width*/ 4) {
- set_style(span1, "width", /*title_width*/ ctx[2]);
- }
-
- if (default_slot) {
- if (default_slot.p && dirty & /*$$scope*/ 16) {
- update_slot(default_slot, default_slot_template, ctx, /*$$scope*/ ctx[4], dirty, null, null);
- }
- }
-
- if (/*showInfo*/ ctx[3]) {
- if (if_block) {
- if_block.p(ctx, dirty);
- } else {
- if_block = create_if_block$2(ctx);
- if_block.c();
- if_block.m(if_block_anchor.parentNode, if_block_anchor);
- }
- } else if (if_block) {
- if_block.d(1);
- if_block = null;
- }
- },
- i: function intro(local) {
- if (current) return;
- transition_in(default_slot, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(default_slot, local);
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(label);
- if (default_slot) default_slot.d(detaching);
- if (detaching) detach_dev(t4);
- if (if_block) if_block.d(detaching);
- if (detaching) detach_dev(if_block_anchor);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$2.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- function instance$2($$self, $$props, $$invalidate) {
- let { title } = $$props;
- let { info } = $$props;
- let { title_width = "300px" } = $$props;
- let showInfo = false;
- const writable_props = ["title", "info", "title_width"];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<LabelHint> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("LabelHint", $$slots, ['default']);
- const click_handler = () => $$invalidate(3, showInfo = !showInfo);
-
- $$self.$$set = $$props => {
- if ("title" in $$props) $$invalidate(0, title = $$props.title);
- if ("info" in $$props) $$invalidate(1, info = $$props.info);
- if ("title_width" in $$props) $$invalidate(2, title_width = $$props.title_width);
- if ("$$scope" in $$props) $$invalidate(4, $$scope = $$props.$$scope);
- };
-
- $$self.$capture_state = () => ({ title, info, title_width, showInfo });
-
- $$self.$inject_state = $$props => {
- if ("title" in $$props) $$invalidate(0, title = $$props.title);
- if ("info" in $$props) $$invalidate(1, info = $$props.info);
- if ("title_width" in $$props) $$invalidate(2, title_width = $$props.title_width);
- if ("showInfo" in $$props) $$invalidate(3, showInfo = $$props.showInfo);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [title, info, title_width, showInfo, $$scope, $$slots, click_handler];
- }
-
- class LabelHint extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$2, create_fragment$2, safe_not_equal, { title: 0, info: 1, title_width: 2 });
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "LabelHint",
- options,
- id: create_fragment$2.name
- });
-
- const { ctx } = this.$$;
- const props = options.props || {};
-
- if (/*title*/ ctx[0] === undefined && !("title" in props)) {
- console.warn("<LabelHint> was created without expected prop 'title'");
- }
-
- if (/*info*/ ctx[1] === undefined && !("info" in props)) {
- console.warn("<LabelHint> was created without expected prop 'info'");
- }
- }
-
- get title() {
- throw new Error("<LabelHint>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set title(value) {
- throw new Error("<LabelHint>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get info() {
- throw new Error("<LabelHint>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set info(value) {
- throw new Error("<LabelHint>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get title_width() {
- throw new Error("<LabelHint>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set title_width(value) {
- throw new Error("<LabelHint>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
- }
-
- function fade(node, { delay = 0, duration = 400, easing = identity }) {
- const o = +getComputedStyle(node).opacity;
- return {
- delay,
- duration,
- easing,
- css: t => `opacity: ${t * o}`
- };
- }
-
- const subscriber_queue = [];
- /**
- * Create a `Writable` store that allows both updating and reading by subscription.
- * @param {*=}value initial value
- * @param {StartStopNotifier=}start start and stop notifications for subscriptions
- */
- function writable(value, start = noop) {
- let stop;
- const subscribers = [];
- function set(new_value) {
- if (safe_not_equal(value, new_value)) {
- value = new_value;
- if (stop) { // store is ready
- const run_queue = !subscriber_queue.length;
- for (let i = 0; i < subscribers.length; i += 1) {
- const s = subscribers[i];
- s[1]();
- subscriber_queue.push(s, value);
- }
- if (run_queue) {
- for (let i = 0; i < subscriber_queue.length; i += 2) {
- subscriber_queue[i][0](subscriber_queue[i + 1]);
- }
- subscriber_queue.length = 0;
- }
- }
- }
- }
- function update(fn) {
- set(fn(value));
- }
- function subscribe(run, invalidate = noop) {
- const subscriber = [run, invalidate];
- subscribers.push(subscriber);
- if (subscribers.length === 1) {
- stop = start(set) || noop;
- }
- run(value);
- return () => {
- const index = subscribers.indexOf(subscriber);
- if (index !== -1) {
- subscribers.splice(index, 1);
- }
- if (subscribers.length === 0) {
- stop();
- stop = null;
- }
- };
- }
- return { set, update, subscribe };
- }
-
- const messages = writable([]);
-
- /* node_modules/wui.svelte/src/components/NotifItem.svelte generated by Svelte v3.24.1 */
- const file$3 = "node_modules/wui.svelte/src/components/NotifItem.svelte";
-
- function create_fragment$3(ctx) {
- let div;
- let t;
- let div_class_value;
- let div_transition;
- let current;
-
- const block = {
- c: function create() {
- div = element("div");
- t = text(/*text*/ ctx[0]);
- attr_dev(div, "class", div_class_value = "wui-notif-item " + /*kind*/ ctx[1] + " svelte-1n99njq");
- add_location(div, file$3, 35, 0, 625);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, t);
- current = true;
- },
- p: function update(ctx, [dirty]) {
- if (!current || dirty & /*text*/ 1) set_data_dev(t, /*text*/ ctx[0]);
-
- if (!current || dirty & /*kind*/ 2 && div_class_value !== (div_class_value = "wui-notif-item " + /*kind*/ ctx[1] + " svelte-1n99njq")) {
- attr_dev(div, "class", div_class_value);
- }
- },
- i: function intro(local) {
- if (current) return;
-
- add_render_callback(() => {
- if (!div_transition) div_transition = create_bidirectional_transition(div, fade, {}, true);
- div_transition.run(1);
- });
-
- current = true;
- },
- o: function outro(local) {
- if (!div_transition) div_transition = create_bidirectional_transition(div, fade, {}, false);
- div_transition.run(0);
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- if (detaching && div_transition) div_transition.end();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$3.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- function instance$3($$self, $$props, $$invalidate) {
- let { text = "" } = $$props;
- let { kind = "" } = $$props;
- let { timeout = 5000 } = $$props;
-
- onMount(() => {
- let timerID = setTimeout(
- () => {
- messages.update(msgs => {
- msgs.splice(0, 1);
- msgs = msgs;
- return msgs;
- });
- },
- timeout
- );
- });
-
- const writable_props = ["text", "kind", "timeout"];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<NotifItem> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("NotifItem", $$slots, []);
-
- $$self.$$set = $$props => {
- if ("text" in $$props) $$invalidate(0, text = $$props.text);
- if ("kind" in $$props) $$invalidate(1, kind = $$props.kind);
- if ("timeout" in $$props) $$invalidate(2, timeout = $$props.timeout);
- };
-
- $$self.$capture_state = () => ({
- onMount,
- fade,
- messages,
- text,
- kind,
- timeout
- });
-
- $$self.$inject_state = $$props => {
- if ("text" in $$props) $$invalidate(0, text = $$props.text);
- if ("kind" in $$props) $$invalidate(1, kind = $$props.kind);
- if ("timeout" in $$props) $$invalidate(2, timeout = $$props.timeout);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [text, kind, timeout];
- }
-
- class NotifItem extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$3, create_fragment$3, safe_not_equal, { text: 0, kind: 1, timeout: 2 });
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "NotifItem",
- options,
- id: create_fragment$3.name
- });
- }
-
- get text() {
- throw new Error("<NotifItem>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set text(value) {
- throw new Error("<NotifItem>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get kind() {
- throw new Error("<NotifItem>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set kind(value) {
- throw new Error("<NotifItem>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- get timeout() {
- throw new Error("<NotifItem>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set timeout(value) {
- throw new Error("<NotifItem>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
- }
-
- /* node_modules/wui.svelte/src/components/Notif.svelte generated by Svelte v3.24.1 */
- const file$4 = "node_modules/wui.svelte/src/components/Notif.svelte";
-
- function get_each_context(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[2] = list[i];
- return child_ctx;
- }
-
- // (41:1) {#each $messages as msg (msg)}
- function create_each_block(key_1, ctx) {
- let first;
- let notifitem;
- let current;
-
- notifitem = new NotifItem({
- props: {
- text: /*msg*/ ctx[2].text,
- kind: /*msg*/ ctx[2].kind,
- timeout: /*timeout*/ ctx[0]
- },
- $$inline: true
- });
-
- const block = {
- key: key_1,
- first: null,
- c: function create() {
- first = empty();
- create_component(notifitem.$$.fragment);
- this.first = first;
- },
- m: function mount(target, anchor) {
- insert_dev(target, first, anchor);
- mount_component(notifitem, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const notifitem_changes = {};
- if (dirty & /*$messages*/ 2) notifitem_changes.text = /*msg*/ ctx[2].text;
- if (dirty & /*$messages*/ 2) notifitem_changes.kind = /*msg*/ ctx[2].kind;
- if (dirty & /*timeout*/ 1) notifitem_changes.timeout = /*timeout*/ ctx[0];
- notifitem.$set(notifitem_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(notifitem.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(notifitem.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(first);
- destroy_component(notifitem, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block.name,
- type: "each",
- source: "(41:1) {#each $messages as msg (msg)}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$4(ctx) {
- let div;
- let each_blocks = [];
- let each_1_lookup = new Map();
- let current;
- let each_value = /*$messages*/ ctx[1];
- validate_each_argument(each_value);
- const get_key = ctx => /*msg*/ ctx[2];
- validate_each_keys(ctx, each_value, get_each_context, get_key);
-
- for (let i = 0; i < each_value.length; i += 1) {
- let child_ctx = get_each_context(ctx, each_value, i);
- let key = get_key(child_ctx);
- each_1_lookup.set(key, each_blocks[i] = create_each_block(key, child_ctx));
- }
-
- const block = {
- c: function create() {
- div = element("div");
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- attr_dev(div, "class", "wui-notif svelte-xdooa2");
- add_location(div, file$4, 39, 0, 670);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(div, null);
- }
-
- current = true;
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*$messages, timeout*/ 3) {
- const each_value = /*$messages*/ ctx[1];
- validate_each_argument(each_value);
- group_outros();
- validate_each_keys(ctx, each_value, get_each_context, get_key);
- each_blocks = update_keyed_each(each_blocks, dirty, get_key, 1, ctx, each_value, each_1_lookup, div, outro_and_destroy_block, create_each_block, null, get_each_context);
- check_outros();
- }
- },
- i: function intro(local) {
- if (current) return;
-
- for (let i = 0; i < each_value.length; i += 1) {
- transition_in(each_blocks[i]);
- }
-
- current = true;
- },
- o: function outro(local) {
- for (let i = 0; i < each_blocks.length; i += 1) {
- transition_out(each_blocks[i]);
- }
-
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].d();
- }
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$4.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const WuiPushNotif = {
- Info(text) {
- const msg = { text };
- messages.update(msgs => msgs = [...msgs, msg]);
- },
- Error(text) {
- const msg = { text, kind: "error" };
- messages.update(msgs => msgs = [...msgs, msg]);
- }
- };
-
- function instance$4($$self, $$props, $$invalidate) {
- let $messages;
- validate_store(messages, "messages");
- component_subscribe($$self, messages, $$value => $$invalidate(1, $messages = $$value));
- let { timeout = 5000 } = $$props;
- const writable_props = ["timeout"];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<Notif> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("Notif", $$slots, []);
-
- $$self.$$set = $$props => {
- if ("timeout" in $$props) $$invalidate(0, timeout = $$props.timeout);
- };
-
- $$self.$capture_state = () => ({
- messages,
- NotifItem,
- WuiPushNotif,
- timeout,
- $messages
- });
-
- $$self.$inject_state = $$props => {
- if ("timeout" in $$props) $$invalidate(0, timeout = $$props.timeout);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [timeout, $messages];
- }
-
- class Notif extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$4, create_fragment$4, safe_not_equal, { timeout: 0 });
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "Notif",
- options,
- id: create_fragment$4.name
- });
- }
-
- get timeout() {
- throw new Error("<Notif>: Props cannot be read directly from the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
-
- set timeout(value) {
- throw new Error("<Notif>: Props cannot be set directly on the component instance unless compiling with 'accessors: true' or '<svelte:options accessors/>'");
- }
- }
-
- const apiEnvironment = "/api/environment";
- const environment = writable({
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {},
- ZoneFiles: [],
- });
- const nanoSeconds = 1000000000;
-
- async function setEnvironment(got) {
- got.PruneDelay = got.PruneDelay / nanoSeconds;
- got.PruneThreshold = got.PruneThreshold / nanoSeconds;
- for (const [key, value] of Object.entries(got.HostsFiles)) {
- got.HostsFiles[key].Records = [];
- }
- environment.set(got);
- }
-
- const RRTypes = {
- 1: "A",
- 2: "NS",
- 3: "MD",
- 4: "MF",
- 5: "CNAME",
- 6: "SOA",
- 7: "MB",
- 8: "MG",
- 9: "MR",
- 10: "NULL",
- 11: "WKS",
- 12: "PTR",
- 13: "HINFO",
- 14: "MINFO",
- 15: "MX",
- 16: "TXT",
- 28: "AAAA",
- 33: "SRV",
- 41: "OPT",
- };
-
- function getRRTypeName(k) {
- let v = RRTypes[k];
- if (v === "") {
- return k
- }
- return v
- }
-
- /* src/Dashboard.svelte generated by Svelte v3.24.1 */
- const file$5 = "src/Dashboard.svelte";
-
- function get_each_context_1(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[8] = list[i];
- return child_ctx;
- }
-
- function get_each_context_2(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[8] = list[i];
- return child_ctx;
- }
-
- function get_each_context_3(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[8] = list[i];
- return child_ctx;
- }
-
- function get_each_context$1(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[5] = list[i];
- return child_ctx;
- }
-
- // (104:3) {#if msg.Answer !== null && msg.Answer.length > 0}
- function create_if_block_2(ctx) {
- let each_1_anchor;
- let each_value_3 = /*msg*/ ctx[5].Answer;
- validate_each_argument(each_value_3);
- let each_blocks = [];
-
- for (let i = 0; i < each_value_3.length; i += 1) {
- each_blocks[i] = create_each_block_3(get_each_context_3(ctx, each_value_3, i));
- }
-
- const block = {
- c: function create() {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- each_1_anchor = empty();
- },
- m: function mount(target, anchor) {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(target, anchor);
- }
-
- insert_dev(target, each_1_anchor, anchor);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*JSON, listMsg, getRRTypeName*/ 2) {
- each_value_3 = /*msg*/ ctx[5].Answer;
- validate_each_argument(each_value_3);
- let i;
-
- for (i = 0; i < each_value_3.length; i += 1) {
- const child_ctx = get_each_context_3(ctx, each_value_3, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block_3(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(each_1_anchor.parentNode, each_1_anchor);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value_3.length;
- }
- },
- d: function destroy(detaching) {
- destroy_each(each_blocks, detaching);
- if (detaching) detach_dev(each_1_anchor);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_2.name,
- type: "if",
- source: "(104:3) {#if msg.Answer !== null && msg.Answer.length > 0}",
- ctx
- });
-
- return block;
- }
-
- // (105:4) {#each msg.Answer as rr}
- function create_each_block_3(ctx) {
- let div;
- let span0;
- let t1;
- let span1;
- let t2_value = getRRTypeName(/*rr*/ ctx[8].Type) + "";
- let t2;
- let t3;
- let span2;
- let t4_value = /*rr*/ ctx[8].TTL + "";
- let t4;
- let t5;
- let span3;
- let t6_value = JSON.stringify(/*rr*/ ctx[8].Value, null, 2) + "";
- let t6;
- let t7;
-
- const block = {
- c: function create() {
- div = element("div");
- span0 = element("span");
- span0.textContent = "Answer";
- t1 = space();
- span1 = element("span");
- t2 = text(t2_value);
- t3 = space();
- span2 = element("span");
- t4 = text(t4_value);
- t5 = space();
- span3 = element("span");
- t6 = text(t6_value);
- t7 = space();
- attr_dev(span0, "class", "kind svelte-1d4jf0i");
- add_location(span0, file$5, 106, 6, 1952);
- attr_dev(span1, "class", "type svelte-1d4jf0i");
- add_location(span1, file$5, 107, 6, 1993);
- attr_dev(span2, "class", "ttl svelte-1d4jf0i");
- add_location(span2, file$5, 108, 6, 2052);
- attr_dev(span3, "class", "value svelte-1d4jf0i");
- add_location(span3, file$5, 109, 6, 2094);
- attr_dev(div, "class", "rr svelte-1d4jf0i");
- add_location(div, file$5, 105, 5, 1929);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span0);
- append_dev(div, t1);
- append_dev(div, span1);
- append_dev(span1, t2);
- append_dev(div, t3);
- append_dev(div, span2);
- append_dev(span2, t4);
- append_dev(div, t5);
- append_dev(div, span3);
- append_dev(span3, t6);
- append_dev(div, t7);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*listMsg*/ 2 && t2_value !== (t2_value = getRRTypeName(/*rr*/ ctx[8].Type) + "")) set_data_dev(t2, t2_value);
- if (dirty & /*listMsg*/ 2 && t4_value !== (t4_value = /*rr*/ ctx[8].TTL + "")) set_data_dev(t4, t4_value);
- if (dirty & /*listMsg*/ 2 && t6_value !== (t6_value = JSON.stringify(/*rr*/ ctx[8].Value, null, 2) + "")) set_data_dev(t6, t6_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_3.name,
- type: "each",
- source: "(105:4) {#each msg.Answer as rr}",
- ctx
- });
-
- return block;
- }
-
- // (114:3) {#if msg.Authority !== null && msg.Authority.length > 0}
- function create_if_block_1(ctx) {
- let each_1_anchor;
- let each_value_2 = /*msg*/ ctx[5].Authority;
- validate_each_argument(each_value_2);
- let each_blocks = [];
-
- for (let i = 0; i < each_value_2.length; i += 1) {
- each_blocks[i] = create_each_block_2(get_each_context_2(ctx, each_value_2, i));
- }
-
- const block = {
- c: function create() {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- each_1_anchor = empty();
- },
- m: function mount(target, anchor) {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(target, anchor);
- }
-
- insert_dev(target, each_1_anchor, anchor);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*JSON, listMsg, getRRTypeName*/ 2) {
- each_value_2 = /*msg*/ ctx[5].Authority;
- validate_each_argument(each_value_2);
- let i;
-
- for (i = 0; i < each_value_2.length; i += 1) {
- const child_ctx = get_each_context_2(ctx, each_value_2, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block_2(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(each_1_anchor.parentNode, each_1_anchor);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value_2.length;
- }
- },
- d: function destroy(detaching) {
- destroy_each(each_blocks, detaching);
- if (detaching) detach_dev(each_1_anchor);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_1.name,
- type: "if",
- source: "(114:3) {#if msg.Authority !== null && msg.Authority.length > 0}",
- ctx
- });
-
- return block;
- }
-
- // (115:4) {#each msg.Authority as rr}
- function create_each_block_2(ctx) {
- let div;
- let span0;
- let t1;
- let span1;
- let t2_value = getRRTypeName(/*rr*/ ctx[8].Type) + "";
- let t2;
- let t3;
- let span2;
- let t4_value = /*rr*/ ctx[8].TTL + "";
- let t4;
- let t5;
- let span3;
- let t6_value = JSON.stringify(/*rr*/ ctx[8].Value, null, 2) + "";
- let t6;
- let t7;
-
- const block = {
- c: function create() {
- div = element("div");
- span0 = element("span");
- span0.textContent = "Authority";
- t1 = space();
- span1 = element("span");
- t2 = text(t2_value);
- t3 = space();
- span2 = element("span");
- t4 = text(t4_value);
- t5 = space();
- span3 = element("span");
- t6 = text(t6_value);
- t7 = space();
- attr_dev(span0, "class", "kind svelte-1d4jf0i");
- add_location(span0, file$5, 116, 6, 2312);
- attr_dev(span1, "class", "type svelte-1d4jf0i");
- add_location(span1, file$5, 117, 6, 2356);
- attr_dev(span2, "class", "ttl svelte-1d4jf0i");
- add_location(span2, file$5, 118, 6, 2415);
- attr_dev(span3, "class", "value svelte-1d4jf0i");
- add_location(span3, file$5, 119, 6, 2457);
- attr_dev(div, "class", "rr svelte-1d4jf0i");
- add_location(div, file$5, 115, 5, 2289);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span0);
- append_dev(div, t1);
- append_dev(div, span1);
- append_dev(span1, t2);
- append_dev(div, t3);
- append_dev(div, span2);
- append_dev(span2, t4);
- append_dev(div, t5);
- append_dev(div, span3);
- append_dev(span3, t6);
- append_dev(div, t7);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*listMsg*/ 2 && t2_value !== (t2_value = getRRTypeName(/*rr*/ ctx[8].Type) + "")) set_data_dev(t2, t2_value);
- if (dirty & /*listMsg*/ 2 && t4_value !== (t4_value = /*rr*/ ctx[8].TTL + "")) set_data_dev(t4, t4_value);
- if (dirty & /*listMsg*/ 2 && t6_value !== (t6_value = JSON.stringify(/*rr*/ ctx[8].Value, null, 2) + "")) set_data_dev(t6, t6_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_2.name,
- type: "each",
- source: "(115:4) {#each msg.Authority as rr}",
- ctx
- });
-
- return block;
- }
-
- // (124:3) {#if msg.Additional !== null && msg.Additional.length > 0}
- function create_if_block$3(ctx) {
- let each_1_anchor;
- let each_value_1 = /*msg*/ ctx[5].Additional;
- validate_each_argument(each_value_1);
- let each_blocks = [];
-
- for (let i = 0; i < each_value_1.length; i += 1) {
- each_blocks[i] = create_each_block_1(get_each_context_1(ctx, each_value_1, i));
- }
-
- const block = {
- c: function create() {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- each_1_anchor = empty();
- },
- m: function mount(target, anchor) {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(target, anchor);
- }
-
- insert_dev(target, each_1_anchor, anchor);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*JSON, listMsg, getRRTypeName*/ 2) {
- each_value_1 = /*msg*/ ctx[5].Additional;
- validate_each_argument(each_value_1);
- let i;
-
- for (i = 0; i < each_value_1.length; i += 1) {
- const child_ctx = get_each_context_1(ctx, each_value_1, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block_1(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(each_1_anchor.parentNode, each_1_anchor);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value_1.length;
- }
- },
- d: function destroy(detaching) {
- destroy_each(each_blocks, detaching);
- if (detaching) detach_dev(each_1_anchor);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block$3.name,
- type: "if",
- source: "(124:3) {#if msg.Additional !== null && msg.Additional.length > 0}",
- ctx
- });
-
- return block;
- }
-
- // (125:4) {#each msg.Additional as rr}
- function create_each_block_1(ctx) {
- let div;
- let span0;
- let t1;
- let span1;
- let t2_value = getRRTypeName(/*rr*/ ctx[8].Type) + "";
- let t2;
- let t3;
- let span2;
- let t4_value = /*rr*/ ctx[8].TTL + "";
- let t4;
- let t5;
- let span3;
- let t6_value = JSON.stringify(/*rr*/ ctx[8].Value, null, 2) + "";
- let t6;
- let t7;
-
- const block = {
- c: function create() {
- div = element("div");
- span0 = element("span");
- span0.textContent = "Additional";
- t1 = space();
- span1 = element("span");
- t2 = text(t2_value);
- t3 = space();
- span2 = element("span");
- t4 = text(t4_value);
- t5 = space();
- span3 = element("span");
- t6 = text(t6_value);
- t7 = space();
- attr_dev(span0, "class", "kind svelte-1d4jf0i");
- add_location(span0, file$5, 126, 6, 2678);
- attr_dev(span1, "class", "type svelte-1d4jf0i");
- add_location(span1, file$5, 127, 6, 2723);
- attr_dev(span2, "class", "ttl svelte-1d4jf0i");
- add_location(span2, file$5, 128, 6, 2782);
- attr_dev(span3, "class", "value svelte-1d4jf0i");
- add_location(span3, file$5, 129, 6, 2824);
- attr_dev(div, "class", "rr svelte-1d4jf0i");
- add_location(div, file$5, 125, 5, 2655);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span0);
- append_dev(div, t1);
- append_dev(div, span1);
- append_dev(span1, t2);
- append_dev(div, t3);
- append_dev(div, span2);
- append_dev(span2, t4);
- append_dev(div, t5);
- append_dev(div, span3);
- append_dev(span3, t6);
- append_dev(div, t7);
- },
- p: function update(ctx, dirty) {
- if (dirty & /*listMsg*/ 2 && t2_value !== (t2_value = getRRTypeName(/*rr*/ ctx[8].Type) + "")) set_data_dev(t2, t2_value);
- if (dirty & /*listMsg*/ 2 && t4_value !== (t4_value = /*rr*/ ctx[8].TTL + "")) set_data_dev(t4, t4_value);
- if (dirty & /*listMsg*/ 2 && t6_value !== (t6_value = JSON.stringify(/*rr*/ ctx[8].Value, null, 2) + "")) set_data_dev(t6, t6_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_1.name,
- type: "each",
- source: "(125:4) {#each msg.Additional as rr}",
- ctx
- });
-
- return block;
- }
-
- // (85:1) {#each listMsg as msg (msg)}
- function create_each_block$1(key_1, ctx) {
- let div2;
- let div0;
- let t0_value = /*msg*/ ctx[5].Question.Name + "";
- let t0;
- let t1;
- let button;
- let t3;
- let div1;
- let span0;
- let t4;
- let span1;
- let t6;
- let span2;
- let t8;
- let span3;
- let t10;
- let t11;
- let t12;
- let t13;
- let mounted;
- let dispose;
- let if_block0 = /*msg*/ ctx[5].Answer !== null && /*msg*/ ctx[5].Answer.length > 0 && create_if_block_2(ctx);
- let if_block1 = /*msg*/ ctx[5].Authority !== null && /*msg*/ ctx[5].Authority.length > 0 && create_if_block_1(ctx);
- let if_block2 = /*msg*/ ctx[5].Additional !== null && /*msg*/ ctx[5].Additional.length > 0 && create_if_block$3(ctx);
-
- const block = {
- key: key_1,
- first: null,
- c: function create() {
- div2 = element("div");
- div0 = element("div");
- t0 = text(t0_value);
- t1 = space();
- button = element("button");
- button.textContent = "Remove from caches";
- t3 = space();
- div1 = element("div");
- span0 = element("span");
- t4 = space();
- span1 = element("span");
- span1.textContent = "Type";
- t6 = space();
- span2 = element("span");
- span2.textContent = "TTL";
- t8 = space();
- span3 = element("span");
- span3.textContent = "Value";
- t10 = space();
- if (if_block0) if_block0.c();
- t11 = space();
- if (if_block1) if_block1.c();
- t12 = space();
- if (if_block2) if_block2.c();
- t13 = space();
- attr_dev(button, "class", "b-remove");
- add_location(button, file$5, 89, 4, 1518);
- attr_dev(div0, "class", "qname");
- add_location(div0, file$5, 86, 3, 1469);
- attr_dev(span0, "class", "kind svelte-1d4jf0i");
- add_location(span0, file$5, 97, 4, 1691);
- attr_dev(span1, "class", "type svelte-1d4jf0i");
- add_location(span1, file$5, 98, 4, 1723);
- attr_dev(span2, "class", "ttl svelte-1d4jf0i");
- add_location(span2, file$5, 99, 4, 1760);
- attr_dev(span3, "class", "value svelte-1d4jf0i");
- add_location(span3, file$5, 100, 4, 1795);
- attr_dev(div1, "class", "rr header svelte-1d4jf0i");
- add_location(div1, file$5, 96, 3, 1663);
- attr_dev(div2, "class", "message svelte-1d4jf0i");
- add_location(div2, file$5, 85, 2, 1444);
- this.first = div2;
- },
- m: function mount(target, anchor) {
- insert_dev(target, div2, anchor);
- append_dev(div2, div0);
- append_dev(div0, t0);
- append_dev(div0, t1);
- append_dev(div0, button);
- append_dev(div2, t3);
- append_dev(div2, div1);
- append_dev(div1, span0);
- append_dev(div1, t4);
- append_dev(div1, span1);
- append_dev(div1, t6);
- append_dev(div1, span2);
- append_dev(div1, t8);
- append_dev(div1, span3);
- append_dev(div2, t10);
- if (if_block0) if_block0.m(div2, null);
- append_dev(div2, t11);
- if (if_block1) if_block1.m(div2, null);
- append_dev(div2, t12);
- if (if_block2) if_block2.m(div2, null);
- append_dev(div2, t13);
-
- if (!mounted) {
- dispose = listen_dev(
- button,
- "click",
- function () {
- if (is_function(/*handleRemoveFromCaches*/ ctx[3](/*msg*/ ctx[5].Question.Name))) /*handleRemoveFromCaches*/ ctx[3](/*msg*/ ctx[5].Question.Name).apply(this, arguments);
- },
- false,
- false,
- false
- );
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
- if (dirty & /*listMsg*/ 2 && t0_value !== (t0_value = /*msg*/ ctx[5].Question.Name + "")) set_data_dev(t0, t0_value);
-
- if (/*msg*/ ctx[5].Answer !== null && /*msg*/ ctx[5].Answer.length > 0) {
- if (if_block0) {
- if_block0.p(ctx, dirty);
- } else {
- if_block0 = create_if_block_2(ctx);
- if_block0.c();
- if_block0.m(div2, t11);
- }
- } else if (if_block0) {
- if_block0.d(1);
- if_block0 = null;
- }
-
- if (/*msg*/ ctx[5].Authority !== null && /*msg*/ ctx[5].Authority.length > 0) {
- if (if_block1) {
- if_block1.p(ctx, dirty);
- } else {
- if_block1 = create_if_block_1(ctx);
- if_block1.c();
- if_block1.m(div2, t12);
- }
- } else if (if_block1) {
- if_block1.d(1);
- if_block1 = null;
- }
-
- if (/*msg*/ ctx[5].Additional !== null && /*msg*/ ctx[5].Additional.length > 0) {
- if (if_block2) {
- if_block2.p(ctx, dirty);
- } else {
- if_block2 = create_if_block$3(ctx);
- if_block2.c();
- if_block2.m(div2, t13);
- }
- } else if (if_block2) {
- if_block2.d(1);
- if_block2 = null;
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div2);
- if (if_block0) if_block0.d();
- if (if_block1) if_block1.d();
- if (if_block2) if_block2.d();
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block$1.name,
- type: "each",
- source: "(85:1) {#each listMsg as msg (msg)}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$5(ctx) {
- let div1;
- let div0;
- let t0;
- let input;
- let t1;
- let button;
- let t3;
- let each_blocks = [];
- let each_1_lookup = new Map();
- let mounted;
- let dispose;
- let each_value = /*listMsg*/ ctx[1];
- validate_each_argument(each_value);
- const get_key = ctx => /*msg*/ ctx[5];
- validate_each_keys(ctx, each_value, get_each_context$1, get_key);
-
- for (let i = 0; i < each_value.length; i += 1) {
- let child_ctx = get_each_context$1(ctx, each_value, i);
- let key = get_key(child_ctx);
- each_1_lookup.set(key, each_blocks[i] = create_each_block$1(key, child_ctx));
- }
-
- const block = {
- c: function create() {
- div1 = element("div");
- div0 = element("div");
- t0 = text("Caches:\n\t\t");
- input = element("input");
- t1 = space();
- button = element("button");
- button.textContent = "Search";
- t3 = space();
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- add_location(input, file$5, 79, 2, 1320);
- add_location(button, file$5, 80, 2, 1349);
- attr_dev(div0, "class", "search");
- add_location(div0, file$5, 77, 1, 1287);
- attr_dev(div1, "class", "dashboard");
- add_location(div1, file$5, 76, 0, 1262);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div1, anchor);
- append_dev(div1, div0);
- append_dev(div0, t0);
- append_dev(div0, input);
- set_input_value(input, /*query*/ ctx[0]);
- append_dev(div0, t1);
- append_dev(div0, button);
- append_dev(div1, t3);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(div1, null);
- }
-
- if (!mounted) {
- dispose = [
- listen_dev(input, "input", /*input_input_handler*/ ctx[4]),
- listen_dev(button, "click", /*handleSearch*/ ctx[2], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*query*/ 1 && input.value !== /*query*/ ctx[0]) {
- set_input_value(input, /*query*/ ctx[0]);
- }
-
- if (dirty & /*listMsg, JSON, getRRTypeName, handleRemoveFromCaches*/ 10) {
- const each_value = /*listMsg*/ ctx[1];
- validate_each_argument(each_value);
- validate_each_keys(ctx, each_value, get_each_context$1, get_key);
- each_blocks = update_keyed_each(each_blocks, dirty, get_key, 1, ctx, each_value, each_1_lookup, div1, destroy_block, create_each_block$1, null, get_each_context$1);
- }
- },
- i: noop,
- o: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(div1);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].d();
- }
-
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$5.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const apiCaches = "/api/caches";
-
- function instance$5($$self, $$props, $$invalidate) {
- let query = "";
- let listMsg = [];
-
- async function handleSearch() {
- const res = await fetch(apiCaches + "?query=" + query);
-
- if (res.status >= 400) {
- const resbody = await res.json();
- WuiPushNotif.Error("ERROR: " + resbody.message);
- return;
- }
-
- $$invalidate(1, listMsg = await res.json());
- }
-
- async function handleRemoveFromCaches(name) {
- const res = await fetch(apiCaches + "?name=" + name, { method: "DELETE" });
-
- if (res.status >= 400) {
- const resbody = await res.json();
- WuiPushNotif.Error("ERROR: " + resbody.message);
- return;
- }
-
- for (let x = 0; x < listMsg.length; x++) {
- if (listMsg[x].Question.Name === name) {
- listMsg.splice(x, 1);
- $$invalidate(1, listMsg);
- break;
- }
- }
-
- const msg = await res.json();
- WuiPushNotif.Info(msg.message);
- }
-
- const writable_props = [];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<Dashboard> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("Dashboard", $$slots, []);
-
- function input_input_handler() {
- query = this.value;
- $$invalidate(0, query);
- }
-
- $$self.$capture_state = () => ({
- WuiPushNotif,
- getRRTypeName,
- apiCaches,
- query,
- listMsg,
- handleSearch,
- handleRemoveFromCaches
- });
-
- $$self.$inject_state = $$props => {
- if ("query" in $$props) $$invalidate(0, query = $$props.query);
- if ("listMsg" in $$props) $$invalidate(1, listMsg = $$props.listMsg);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [query, listMsg, handleSearch, handleRemoveFromCaches, input_input_handler];
- }
-
- class Dashboard extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$5, create_fragment$5, safe_not_equal, {});
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "Dashboard",
- options,
- id: create_fragment$5.name
- });
- }
- }
-
- /* src/Environment.svelte generated by Svelte v3.24.1 */
-
- const { Object: Object_1 } = globals;
- const file$6 = "src/Environment.svelte";
-
- function get_each_context$2(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[17] = list[i];
- child_ctx[18] = list;
- child_ctx[19] = i;
- return child_ctx;
- }
-
- // (97:1) <WuiLabelHint title="System resolv.conf" title_width="{defTitleWidth}" info="A path to dynamically generated resolv.conf(5) by resolvconf(8). If set, the nameserver values in referenced file will replace 'parent' value and 'parent' will become a fallback in case the referenced file being deleted or can not be parsed." >
- function create_default_slot_11(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "class", "svelte-ivqrh9");
- add_location(input, file$6, 104, 2, 2224);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*env*/ ctx[0].FileResolvConf);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler*/ ctx[4]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty & /*env*/ 1 && input.value !== /*env*/ ctx[0].FileResolvConf) {
- set_input_value(input, /*env*/ ctx[0].FileResolvConf);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_11.name,
- type: "slot",
- source: "(97:1) <WuiLabelHint title=\\\"System resolv.conf\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"A path to dynamically generated resolv.conf(5) by resolvconf(8). If set, the nameserver values in referenced file will replace 'parent' value and 'parent' will become a fallback in case the referenced file being deleted or can not be parsed.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (110:1) <WuiLabelHint title="Debug level" title_width="{defTitleWidth}" info="This option only used for debugging program or if user want to monitor what kind of traffic goes in and out of rescached." >
- function create_default_slot_10(ctx) {
- let wuiinputnumber;
- let updating_value;
- let current;
-
- function wuiinputnumber_value_binding(value) {
- /*wuiinputnumber_value_binding*/ ctx[5].call(null, value);
- }
-
- let wuiinputnumber_props = { min: "0", max: "3", unit: "" };
-
- if (/*env*/ ctx[0].Debug !== void 0) {
- wuiinputnumber_props.value = /*env*/ ctx[0].Debug;
- }
-
- wuiinputnumber = new InputNumber({
- props: wuiinputnumber_props,
- $$inline: true
- });
-
- binding_callbacks.push(() => bind(wuiinputnumber, "value", wuiinputnumber_value_binding));
-
- const block = {
- c: function create() {
- create_component(wuiinputnumber.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(wuiinputnumber, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const wuiinputnumber_changes = {};
-
- if (!updating_value && dirty & /*env*/ 1) {
- updating_value = true;
- wuiinputnumber_changes.value = /*env*/ ctx[0].Debug;
- add_flush_callback(() => updating_value = false);
- }
-
- wuiinputnumber.$set(wuiinputnumber_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuiinputnumber.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuiinputnumber.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuiinputnumber, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_10.name,
- type: "slot",
- source: "(110:1) <WuiLabelHint title=\\\"Debug level\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"This option only used for debugging program or if user want to monitor what kind of traffic goes in and out of rescached.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (134:1) {#each env.NameServers as ns}
- function create_each_block$2(ctx) {
- let div;
- let input;
- let t0;
- let button;
- let mounted;
- let dispose;
-
- function input_input_handler_1() {
- /*input_input_handler_1*/ ctx[6].call(input, /*each_value*/ ctx[18], /*ns_index*/ ctx[19]);
- }
-
- const block = {
- c: function create() {
- div = element("div");
- input = element("input");
- t0 = space();
- button = element("button");
- button.textContent = "Delete";
- attr_dev(input, "class", "svelte-ivqrh9");
- add_location(input, file$6, 135, 2, 2820);
- attr_dev(button, "class", "svelte-ivqrh9");
- add_location(button, file$6, 136, 2, 2846);
- attr_dev(div, "class", "input-deletable svelte-ivqrh9");
- add_location(div, file$6, 134, 1, 2788);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, input);
- set_input_value(input, /*ns*/ ctx[17]);
- append_dev(div, t0);
- append_dev(div, button);
-
- if (!mounted) {
- dispose = [
- listen_dev(input, "input", input_input_handler_1),
- listen_dev(
- button,
- "click",
- function () {
- if (is_function(/*deleteNameServer*/ ctx[2](/*ns*/ ctx[17]))) /*deleteNameServer*/ ctx[2](/*ns*/ ctx[17]).apply(this, arguments);
- },
- false,
- false,
- false
- )
- ];
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
-
- if (dirty & /*env*/ 1 && input.value !== /*ns*/ ctx[17]) {
- set_input_value(input, /*ns*/ ctx[17]);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block$2.name,
- type: "each",
- source: "(134:1) {#each env.NameServers as ns}",
- ctx
- });
-
- return block;
- }
-
- // (149:1) <WuiLabelHint title="Listen address" title_width="{defTitleWidth}" info="Address in local network where rescached will listening for query from client through UDP and TCP. <br/> If you want rescached to serve a query from another host in your local network, change this value to <tt>0.0.0.0:53</tt>." >
- function create_default_slot_8(ctx) {
- let wuiinputipport;
- let updating_value;
- let current;
-
- function wuiinputipport_value_binding(value) {
- /*wuiinputipport_value_binding*/ ctx[7].call(null, value);
- }
-
- let wuiinputipport_props = {};
-
- if (/*env*/ ctx[0].ListenAddress !== void 0) {
- wuiinputipport_props.value = /*env*/ ctx[0].ListenAddress;
- }
-
- wuiinputipport = new InputIPPort({
- props: wuiinputipport_props,
- $$inline: true
- });
-
- binding_callbacks.push(() => bind(wuiinputipport, "value", wuiinputipport_value_binding));
-
- const block = {
- c: function create() {
- create_component(wuiinputipport.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(wuiinputipport, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const wuiinputipport_changes = {};
-
- if (!updating_value && dirty & /*env*/ 1) {
- updating_value = true;
- wuiinputipport_changes.value = /*env*/ ctx[0].ListenAddress;
- add_flush_callback(() => updating_value = false);
- }
-
- wuiinputipport.$set(wuiinputipport_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuiinputipport.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuiinputipport.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuiinputipport, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_8.name,
- type: "slot",
- source: "(149:1) <WuiLabelHint title=\\\"Listen address\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"Address in local network where rescached will listening for query from client through UDP and TCP. <br/> If you want rescached to serve a query from another host in your local network, change this value to <tt>0.0.0.0:53</tt>.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (163:1) <WuiLabelHint title="HTTP listen port" title_width="{defTitleWidth}" info="Port to serve DNS over HTTP" >
- function create_default_slot_7(ctx) {
- let wuiinputnumber;
- let updating_value;
- let current;
-
- function wuiinputnumber_value_binding_1(value) {
- /*wuiinputnumber_value_binding_1*/ ctx[8].call(null, value);
- }
-
- let wuiinputnumber_props = { min: "0", max: "65535", unit: "" };
-
- if (/*env*/ ctx[0].HTTPPort !== void 0) {
- wuiinputnumber_props.value = /*env*/ ctx[0].HTTPPort;
- }
-
- wuiinputnumber = new InputNumber({
- props: wuiinputnumber_props,
- $$inline: true
- });
-
- binding_callbacks.push(() => bind(wuiinputnumber, "value", wuiinputnumber_value_binding_1));
-
- const block = {
- c: function create() {
- create_component(wuiinputnumber.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(wuiinputnumber, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const wuiinputnumber_changes = {};
-
- if (!updating_value && dirty & /*env*/ 1) {
- updating_value = true;
- wuiinputnumber_changes.value = /*env*/ ctx[0].HTTPPort;
- add_flush_callback(() => updating_value = false);
- }
-
- wuiinputnumber.$set(wuiinputnumber_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuiinputnumber.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuiinputnumber.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuiinputnumber, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_7.name,
- type: "slot",
- source: "(163:1) <WuiLabelHint title=\\\"HTTP listen port\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"Port to serve DNS over HTTP\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (176:1) <WuiLabelHint title="TLS listen port" title_width="{defTitleWidth}" info="Port to listen for DNS over TLS" >
- function create_default_slot_6(ctx) {
- let wuiinputnumber;
- let updating_value;
- let current;
-
- function wuiinputnumber_value_binding_2(value) {
- /*wuiinputnumber_value_binding_2*/ ctx[9].call(null, value);
- }
-
- let wuiinputnumber_props = { min: "0", max: "65535", unit: "" };
-
- if (/*env*/ ctx[0].TLSPort !== void 0) {
- wuiinputnumber_props.value = /*env*/ ctx[0].TLSPort;
- }
-
- wuiinputnumber = new InputNumber({
- props: wuiinputnumber_props,
- $$inline: true
- });
-
- binding_callbacks.push(() => bind(wuiinputnumber, "value", wuiinputnumber_value_binding_2));
-
- const block = {
- c: function create() {
- create_component(wuiinputnumber.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(wuiinputnumber, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const wuiinputnumber_changes = {};
-
- if (!updating_value && dirty & /*env*/ 1) {
- updating_value = true;
- wuiinputnumber_changes.value = /*env*/ ctx[0].TLSPort;
- add_flush_callback(() => updating_value = false);
- }
-
- wuiinputnumber.$set(wuiinputnumber_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuiinputnumber.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuiinputnumber.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuiinputnumber, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_6.name,
- type: "slot",
- source: "(176:1) <WuiLabelHint title=\\\"TLS listen port\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"Port to listen for DNS over TLS\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (189:1) <WuiLabelHint title="TLS certificate" title_width="{defTitleWidth}" info="Path to certificate file to serve DNS over TLS and HTTPS">
- function create_default_slot_5(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "placeholder", "/path/to/certificate");
- attr_dev(input, "class", "svelte-ivqrh9");
- add_location(input, file$6, 193, 2, 3948);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*env*/ ctx[0].TLSCertFile);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_2*/ ctx[10]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty & /*env*/ 1 && input.value !== /*env*/ ctx[0].TLSCertFile) {
- set_input_value(input, /*env*/ ctx[0].TLSCertFile);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_5.name,
- type: "slot",
- source: "(189:1) <WuiLabelHint title=\\\"TLS certificate\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"Path to certificate file to serve DNS over TLS and HTTPS\\\">",
- ctx
- });
-
- return block;
- }
-
- // (200:1) <WuiLabelHint title="TLS private key" title_width="{defTitleWidth}" info="Path to certificate private key file to serve DNS over TLS and HTTPS." >
- function create_default_slot_4(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "placeholder", "/path/to/certificate/private.key");
- attr_dev(input, "class", "svelte-ivqrh9");
- add_location(input, file$6, 205, 2, 4205);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*env*/ ctx[0].TLSPrivateKey);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_3*/ ctx[11]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty & /*env*/ 1 && input.value !== /*env*/ ctx[0].TLSPrivateKey) {
- set_input_value(input, /*env*/ ctx[0].TLSPrivateKey);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_4.name,
- type: "slot",
- source: "(200:1) <WuiLabelHint title=\\\"TLS private key\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"Path to certificate private key file to serve DNS over TLS and HTTPS.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (212:1) <WuiLabelHint title="TLS allow insecure" title_width="{defTitleWidth}" info="If its true, allow serving DoH and DoT with self signed certificate." >
- function create_default_slot_3(ctx) {
- let div;
- let input;
- let t0;
- let span;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- div = element("div");
- input = element("input");
- t0 = space();
- span = element("span");
- span.textContent = "Yes";
- attr_dev(input, "type", "checkbox");
- attr_dev(input, "class", "svelte-ivqrh9");
- add_location(input, file$6, 218, 3, 4510);
- attr_dev(span, "class", "suffix");
- add_location(span, file$6, 222, 3, 4583);
- attr_dev(div, "class", "input-checkbox svelte-ivqrh9");
- add_location(div, file$6, 217, 2, 4478);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, input);
- input.checked = /*env*/ ctx[0].TLSAllowInsecure;
- append_dev(div, t0);
- append_dev(div, span);
-
- if (!mounted) {
- dispose = listen_dev(input, "change", /*input_change_handler*/ ctx[12]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty & /*env*/ 1) {
- input.checked = /*env*/ ctx[0].TLSAllowInsecure;
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_3.name,
- type: "slot",
- source: "(212:1) <WuiLabelHint title=\\\"TLS allow insecure\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"If its true, allow serving DoH and DoT with self signed certificate.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (229:1) <WuiLabelHint title="DoH behind proxy" title_width="{defTitleWidth}" info="If its true, serve DNS over HTTP only, even if certificate files is defined. This allow serving DNS request forwarded by another proxy server." >
- function create_default_slot_2(ctx) {
- let div;
- let input;
- let t0;
- let span;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- div = element("div");
- input = element("input");
- t0 = space();
- span = element("span");
- span.textContent = "Yes";
- attr_dev(input, "type", "checkbox");
- attr_dev(input, "class", "svelte-ivqrh9");
- add_location(input, file$6, 236, 3, 4914);
- attr_dev(span, "class", "suffix");
- add_location(span, file$6, 240, 3, 4985);
- attr_dev(div, "class", "input-checkbox svelte-ivqrh9");
- add_location(div, file$6, 235, 2, 4882);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, input);
- input.checked = /*env*/ ctx[0].DoHBehindProxy;
- append_dev(div, t0);
- append_dev(div, span);
-
- if (!mounted) {
- dispose = listen_dev(input, "change", /*input_change_handler_1*/ ctx[13]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty & /*env*/ 1) {
- input.checked = /*env*/ ctx[0].DoHBehindProxy;
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_2.name,
- type: "slot",
- source: "(229:1) <WuiLabelHint title=\\\"DoH behind proxy\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"If its true, serve DNS over HTTP only, even if certificate files is defined. This allow serving DNS request forwarded by another proxy server.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (247:1) <WuiLabelHint title="Prune delay" title_width="{defTitleWidth}" info="Delay for pruning caches. Every N seconds, rescached will traverse all caches and remove response that has not been accessed less than cache.prune_threshold. Its value must be equal or greater than 1 hour (3600 seconds). " >
- function create_default_slot_1(ctx) {
- let wuiinputnumber;
- let updating_value;
- let current;
-
- function wuiinputnumber_value_binding_3(value) {
- /*wuiinputnumber_value_binding_3*/ ctx[14].call(null, value);
- }
-
- let wuiinputnumber_props = {
- min: "3600",
- max: "36000",
- unit: "seconds"
- };
-
- if (/*env*/ ctx[0].PruneDelay !== void 0) {
- wuiinputnumber_props.value = /*env*/ ctx[0].PruneDelay;
- }
-
- wuiinputnumber = new InputNumber({
- props: wuiinputnumber_props,
- $$inline: true
- });
-
- binding_callbacks.push(() => bind(wuiinputnumber, "value", wuiinputnumber_value_binding_3));
-
- const block = {
- c: function create() {
- create_component(wuiinputnumber.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(wuiinputnumber, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const wuiinputnumber_changes = {};
-
- if (!updating_value && dirty & /*env*/ 1) {
- updating_value = true;
- wuiinputnumber_changes.value = /*env*/ ctx[0].PruneDelay;
- add_flush_callback(() => updating_value = false);
- }
-
- wuiinputnumber.$set(wuiinputnumber_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuiinputnumber.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuiinputnumber.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuiinputnumber, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_1.name,
- type: "slot",
- source: "(247:1) <WuiLabelHint title=\\\"Prune delay\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"Delay for pruning caches. Every N seconds, rescached will traverse all caches and remove response that has not been accessed less than cache.prune_threshold. Its value must be equal or greater than 1 hour (3600 seconds). \\\" >",
- ctx
- });
-
- return block;
- }
-
- // (264:1) <WuiLabelHint title="Prune threshold" title_width="{defTitleWidth}" info="The duration when the cache will be considered expired. Its value must be negative and greater or equal than -1 hour (-3600 seconds)." >
- function create_default_slot(ctx) {
- let wuiinputnumber;
- let updating_value;
- let current;
-
- function wuiinputnumber_value_binding_4(value) {
- /*wuiinputnumber_value_binding_4*/ ctx[15].call(null, value);
- }
-
- let wuiinputnumber_props = {
- min: "-36000",
- max: "-3600",
- unit: "seconds"
- };
-
- if (/*env*/ ctx[0].PruneThreshold !== void 0) {
- wuiinputnumber_props.value = /*env*/ ctx[0].PruneThreshold;
- }
-
- wuiinputnumber = new InputNumber({
- props: wuiinputnumber_props,
- $$inline: true
- });
-
- binding_callbacks.push(() => bind(wuiinputnumber, "value", wuiinputnumber_value_binding_4));
-
- const block = {
- c: function create() {
- create_component(wuiinputnumber.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(wuiinputnumber, target, anchor);
- current = true;
- },
- p: function update(ctx, dirty) {
- const wuiinputnumber_changes = {};
-
- if (!updating_value && dirty & /*env*/ 1) {
- updating_value = true;
- wuiinputnumber_changes.value = /*env*/ ctx[0].PruneThreshold;
- add_flush_callback(() => updating_value = false);
- }
-
- wuiinputnumber.$set(wuiinputnumber_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuiinputnumber.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuiinputnumber.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuiinputnumber, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot.name,
- type: "slot",
- source: "(264:1) <WuiLabelHint title=\\\"Prune threshold\\\" title_width=\\\"{defTitleWidth}\\\" info=\\\"The duration when the cache will be considered expired. Its value must be negative and greater or equal than -1 hour (-3600 seconds).\\\" >",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$6(ctx) {
- let div4;
- let p;
- let t1;
- let h30;
- let t3;
- let div0;
- let wuilabelhint0;
- let t4;
- let wuilabelhint1;
- let t5;
- let h31;
- let t7;
- let div1;
- let wuilabelhint2;
- let t8;
- let t9;
- let button0;
- let t11;
- let wuilabelhint3;
- let t12;
- let wuilabelhint4;
- let t13;
- let wuilabelhint5;
- let t14;
- let wuilabelhint6;
- let t15;
- let wuilabelhint7;
- let t16;
- let wuilabelhint8;
- let t17;
- let wuilabelhint9;
- let t18;
- let wuilabelhint10;
- let t19;
- let wuilabelhint11;
- let t20;
- let div3;
- let div2;
- let button1;
- let current;
- let mounted;
- let dispose;
-
- wuilabelhint0 = new LabelHint({
- props: {
- title: "System resolv.conf",
- title_width: defTitleWidth,
- info: "A path to dynamically generated resolv.conf(5) by\nresolvconf(8). If set, the nameserver values in referenced file will\nreplace 'parent' value and 'parent' will become a fallback in\ncase the referenced file being deleted or can not be parsed.",
- $$slots: { default: [create_default_slot_11] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint1 = new LabelHint({
- props: {
- title: "Debug level",
- title_width: defTitleWidth,
- info: "This option only used for debugging program or if user\nwant to monitor what kind of traffic goes in and out of rescached.",
- $$slots: { default: [create_default_slot_10] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint2 = new LabelHint({
- props: {
- title: "Parent name servers",
- title_width: defTitleWidth,
- info: "List of parent DNS servers."
- },
- $$inline: true
- });
-
- let each_value = /*env*/ ctx[0].NameServers;
- validate_each_argument(each_value);
- let each_blocks = [];
-
- for (let i = 0; i < each_value.length; i += 1) {
- each_blocks[i] = create_each_block$2(get_each_context$2(ctx, each_value, i));
- }
-
- wuilabelhint3 = new LabelHint({
- props: {
- title: "Listen address",
- title_width: defTitleWidth,
- info: "Address in local network where rescached will\nlistening for query from client through UDP and TCP.\n<br/>\nIf you want rescached to serve a query from another host in your local\nnetwork, change this value to <tt>0.0.0.0:53</tt>.",
- $$slots: { default: [create_default_slot_8] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint4 = new LabelHint({
- props: {
- title: "HTTP listen port",
- title_width: defTitleWidth,
- info: "Port to serve DNS over HTTP",
- $$slots: { default: [create_default_slot_7] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint5 = new LabelHint({
- props: {
- title: "TLS listen port",
- title_width: defTitleWidth,
- info: "Port to listen for DNS over TLS",
- $$slots: { default: [create_default_slot_6] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint6 = new LabelHint({
- props: {
- title: "TLS certificate",
- title_width: defTitleWidth,
- info: "Path to certificate file to serve DNS over TLS and\nHTTPS",
- $$slots: { default: [create_default_slot_5] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint7 = new LabelHint({
- props: {
- title: "TLS private key",
- title_width: defTitleWidth,
- info: "Path to certificate private key file to serve DNS over TLS and\nHTTPS.",
- $$slots: { default: [create_default_slot_4] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint8 = new LabelHint({
- props: {
- title: "TLS allow insecure",
- title_width: defTitleWidth,
- info: "If its true, allow serving DoH and DoT with self signed\ncertificate.",
- $$slots: { default: [create_default_slot_3] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint9 = new LabelHint({
- props: {
- title: "DoH behind proxy",
- title_width: defTitleWidth,
- info: "If its true, serve DNS over HTTP only, even if\ncertificate files is defined.\nThis allow serving DNS request forwarded by another proxy server.",
- $$slots: { default: [create_default_slot_2] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint10 = new LabelHint({
- props: {
- title: "Prune delay",
- title_width: defTitleWidth,
- info: "Delay for pruning caches.\nEvery N seconds, rescached will traverse all caches and remove response that\nhas not been accessed less than cache.prune_threshold.\nIts value must be equal or greater than 1 hour (3600 seconds).\n",
- $$slots: { default: [create_default_slot_1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint11 = new LabelHint({
- props: {
- title: "Prune threshold",
- title_width: defTitleWidth,
- info: "The duration when the cache will be considered expired.\nIts value must be negative and greater or equal than -1 hour (-3600 seconds).",
- $$slots: { default: [create_default_slot] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- const block = {
- c: function create() {
- div4 = element("div");
- p = element("p");
- p.textContent = "This page allow you to change the rescached environment.\nUpon save, the rescached service will be restarted.";
- t1 = space();
- h30 = element("h3");
- h30.textContent = "rescached";
- t3 = space();
- div0 = element("div");
- create_component(wuilabelhint0.$$.fragment);
- t4 = space();
- create_component(wuilabelhint1.$$.fragment);
- t5 = space();
- h31 = element("h3");
- h31.textContent = "DNS server";
- t7 = space();
- div1 = element("div");
- create_component(wuilabelhint2.$$.fragment);
- t8 = space();
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- t9 = space();
- button0 = element("button");
- button0.textContent = "Add";
- t11 = space();
- create_component(wuilabelhint3.$$.fragment);
- t12 = space();
- create_component(wuilabelhint4.$$.fragment);
- t13 = space();
- create_component(wuilabelhint5.$$.fragment);
- t14 = space();
- create_component(wuilabelhint6.$$.fragment);
- t15 = space();
- create_component(wuilabelhint7.$$.fragment);
- t16 = space();
- create_component(wuilabelhint8.$$.fragment);
- t17 = space();
- create_component(wuilabelhint9.$$.fragment);
- t18 = space();
- create_component(wuilabelhint10.$$.fragment);
- t19 = space();
- create_component(wuilabelhint11.$$.fragment);
- t20 = space();
- div3 = element("div");
- div2 = element("div");
- button1 = element("button");
- button1.textContent = "Save";
- add_location(p, file$6, 89, 0, 1747);
- add_location(h30, file$6, 94, 0, 1866);
- add_location(div0, file$6, 95, 0, 1885);
- add_location(h31, file$6, 124, 0, 2595);
- add_location(button0, file$6, 142, 1, 2928);
- add_location(div1, file$6, 125, 0, 2615);
- add_location(button1, file$6, 280, 3, 5859);
- add_location(div2, file$6, 279, 2, 5850);
- attr_dev(div3, "class", "section-bottom svelte-ivqrh9");
- add_location(div3, file$6, 278, 1, 5819);
- attr_dev(div4, "class", "environment");
- add_location(div4, file$6, 88, 0, 1721);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div4, anchor);
- append_dev(div4, p);
- append_dev(div4, t1);
- append_dev(div4, h30);
- append_dev(div4, t3);
- append_dev(div4, div0);
- mount_component(wuilabelhint0, div0, null);
- append_dev(div0, t4);
- mount_component(wuilabelhint1, div0, null);
- append_dev(div4, t5);
- append_dev(div4, h31);
- append_dev(div4, t7);
- append_dev(div4, div1);
- mount_component(wuilabelhint2, div1, null);
- append_dev(div1, t8);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(div1, null);
- }
-
- append_dev(div1, t9);
- append_dev(div1, button0);
- append_dev(div1, t11);
- mount_component(wuilabelhint3, div1, null);
- append_dev(div1, t12);
- mount_component(wuilabelhint4, div1, null);
- append_dev(div1, t13);
- mount_component(wuilabelhint5, div1, null);
- append_dev(div1, t14);
- mount_component(wuilabelhint6, div1, null);
- append_dev(div1, t15);
- mount_component(wuilabelhint7, div1, null);
- append_dev(div1, t16);
- mount_component(wuilabelhint8, div1, null);
- append_dev(div1, t17);
- mount_component(wuilabelhint9, div1, null);
- append_dev(div1, t18);
- mount_component(wuilabelhint10, div1, null);
- append_dev(div1, t19);
- mount_component(wuilabelhint11, div1, null);
- append_dev(div4, t20);
- append_dev(div4, div3);
- append_dev(div3, div2);
- append_dev(div2, button1);
- current = true;
-
- if (!mounted) {
- dispose = [
- listen_dev(button0, "click", /*addNameServer*/ ctx[1], false, false, false),
- listen_dev(button1, "click", /*updateEnvironment*/ ctx[3], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- const wuilabelhint0_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint0_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint0.$set(wuilabelhint0_changes);
- const wuilabelhint1_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint1_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint1.$set(wuilabelhint1_changes);
- const wuilabelhint2_changes = {};
-
- if (dirty & /*$$scope*/ 1048576) {
- wuilabelhint2_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint2.$set(wuilabelhint2_changes);
-
- if (dirty & /*deleteNameServer, env*/ 5) {
- each_value = /*env*/ ctx[0].NameServers;
- validate_each_argument(each_value);
- let i;
-
- for (i = 0; i < each_value.length; i += 1) {
- const child_ctx = get_each_context$2(ctx, each_value, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block$2(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(div1, t9);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value.length;
- }
-
- const wuilabelhint3_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint3_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint3.$set(wuilabelhint3_changes);
- const wuilabelhint4_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint4_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint4.$set(wuilabelhint4_changes);
- const wuilabelhint5_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint5_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint5.$set(wuilabelhint5_changes);
- const wuilabelhint6_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint6_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint6.$set(wuilabelhint6_changes);
- const wuilabelhint7_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint7_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint7.$set(wuilabelhint7_changes);
- const wuilabelhint8_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint8_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint8.$set(wuilabelhint8_changes);
- const wuilabelhint9_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint9_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint9.$set(wuilabelhint9_changes);
- const wuilabelhint10_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint10_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint10.$set(wuilabelhint10_changes);
- const wuilabelhint11_changes = {};
-
- if (dirty & /*$$scope, env*/ 1048577) {
- wuilabelhint11_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint11.$set(wuilabelhint11_changes);
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuilabelhint0.$$.fragment, local);
- transition_in(wuilabelhint1.$$.fragment, local);
- transition_in(wuilabelhint2.$$.fragment, local);
- transition_in(wuilabelhint3.$$.fragment, local);
- transition_in(wuilabelhint4.$$.fragment, local);
- transition_in(wuilabelhint5.$$.fragment, local);
- transition_in(wuilabelhint6.$$.fragment, local);
- transition_in(wuilabelhint7.$$.fragment, local);
- transition_in(wuilabelhint8.$$.fragment, local);
- transition_in(wuilabelhint9.$$.fragment, local);
- transition_in(wuilabelhint10.$$.fragment, local);
- transition_in(wuilabelhint11.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuilabelhint0.$$.fragment, local);
- transition_out(wuilabelhint1.$$.fragment, local);
- transition_out(wuilabelhint2.$$.fragment, local);
- transition_out(wuilabelhint3.$$.fragment, local);
- transition_out(wuilabelhint4.$$.fragment, local);
- transition_out(wuilabelhint5.$$.fragment, local);
- transition_out(wuilabelhint6.$$.fragment, local);
- transition_out(wuilabelhint7.$$.fragment, local);
- transition_out(wuilabelhint8.$$.fragment, local);
- transition_out(wuilabelhint9.$$.fragment, local);
- transition_out(wuilabelhint10.$$.fragment, local);
- transition_out(wuilabelhint11.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div4);
- destroy_component(wuilabelhint0);
- destroy_component(wuilabelhint1);
- destroy_component(wuilabelhint2);
- destroy_each(each_blocks, detaching);
- destroy_component(wuilabelhint3);
- destroy_component(wuilabelhint4);
- destroy_component(wuilabelhint5);
- destroy_component(wuilabelhint6);
- destroy_component(wuilabelhint7);
- destroy_component(wuilabelhint8);
- destroy_component(wuilabelhint9);
- destroy_component(wuilabelhint10);
- destroy_component(wuilabelhint11);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$6.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const defTitleWidth = "300px";
-
- function instance$6($$self, $$props, $$invalidate) {
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {}
- };
-
- const envUnsubscribe = environment.subscribe(value => {
- $$invalidate(0, env = value);
- });
-
- onDestroy(envUnsubscribe);
-
- function addNameServer() {
- $$invalidate(0, env.NameServers = [...env.NameServers, ""], env);
- }
-
- function deleteNameServer(ns) {
- for (let x = 0; x < env.NameServers.length; x++) {
- if (env.NameServers[x] === ns) {
- env.NameServers.splice(x, 1);
- $$invalidate(0, env);
- break;
- }
- }
- }
-
- async function updateEnvironment() {
- let got = {};
- Object.assign(got, env);
- environment.set(env);
- got.PruneDelay = got.PruneDelay * nanoSeconds;
- got.PruneThreshold = got.PruneThreshold * nanoSeconds;
-
- const res = await fetch(apiEnvironment, {
- method: "POST",
- headers: { "Content-Type": "application/json" },
- body: JSON.stringify(got)
- });
-
- if (res.status >= 400) {
- const resbody = await res.json();
- WuiPushNotif.Error("ERROR: ", resbody.message);
- return;
- }
-
- WuiPushNotif.Info("The environment succesfully updated ...");
- }
-
- const writable_props = [];
-
- Object_1.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<Environment> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("Environment", $$slots, []);
-
- function input_input_handler() {
- env.FileResolvConf = this.value;
- $$invalidate(0, env);
- }
-
- function wuiinputnumber_value_binding(value) {
- env.Debug = value;
- $$invalidate(0, env);
- }
-
- function input_input_handler_1(each_value, ns_index) {
- each_value[ns_index] = this.value;
- $$invalidate(0, env);
- }
-
- function wuiinputipport_value_binding(value) {
- env.ListenAddress = value;
- $$invalidate(0, env);
- }
-
- function wuiinputnumber_value_binding_1(value) {
- env.HTTPPort = value;
- $$invalidate(0, env);
- }
-
- function wuiinputnumber_value_binding_2(value) {
- env.TLSPort = value;
- $$invalidate(0, env);
- }
-
- function input_input_handler_2() {
- env.TLSCertFile = this.value;
- $$invalidate(0, env);
- }
-
- function input_input_handler_3() {
- env.TLSPrivateKey = this.value;
- $$invalidate(0, env);
- }
-
- function input_change_handler() {
- env.TLSAllowInsecure = this.checked;
- $$invalidate(0, env);
- }
-
- function input_change_handler_1() {
- env.DoHBehindProxy = this.checked;
- $$invalidate(0, env);
- }
-
- function wuiinputnumber_value_binding_3(value) {
- env.PruneDelay = value;
- $$invalidate(0, env);
- }
-
- function wuiinputnumber_value_binding_4(value) {
- env.PruneThreshold = value;
- $$invalidate(0, env);
- }
-
- $$self.$capture_state = () => ({
- onDestroy,
- apiEnvironment,
- environment,
- nanoSeconds,
- WuiPushNotif,
- WuiLabelHint: LabelHint,
- WuiInputNumber: InputNumber,
- WuiInputIPPort: InputIPPort,
- env,
- envUnsubscribe,
- defTitleWidth,
- addNameServer,
- deleteNameServer,
- updateEnvironment
- });
-
- $$self.$inject_state = $$props => {
- if ("env" in $$props) $$invalidate(0, env = $$props.env);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [
- env,
- addNameServer,
- deleteNameServer,
- updateEnvironment,
- input_input_handler,
- wuiinputnumber_value_binding,
- input_input_handler_1,
- wuiinputipport_value_binding,
- wuiinputnumber_value_binding_1,
- wuiinputnumber_value_binding_2,
- input_input_handler_2,
- input_input_handler_3,
- input_change_handler,
- input_change_handler_1,
- wuiinputnumber_value_binding_3,
- wuiinputnumber_value_binding_4
- ];
- }
-
- class Environment extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$6, create_fragment$6, safe_not_equal, {});
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "Environment",
- options,
- id: create_fragment$6.name
- });
- }
- }
-
- /* src/HostsBlock.svelte generated by Svelte v3.24.1 */
- const file$7 = "src/HostsBlock.svelte";
-
- function get_each_context$3(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[5] = list[i];
- child_ctx[6] = list;
- child_ctx[7] = i;
- return child_ctx;
- }
-
- // (83:2) {#each env.HostsBlocks as hostsBlock}
- function create_each_block$3(ctx) {
- let div;
- let span0;
- let input0;
- let t0;
- let span1;
- let t1_value = /*hostsBlock*/ ctx[5].Name + "";
- let t1;
- let t2;
- let span2;
- let input1;
- let t3;
- let span3;
- let t4_value = /*hostsBlock*/ ctx[5].LastUpdated + "";
- let t4;
- let t5;
- let mounted;
- let dispose;
-
- function input0_change_handler() {
- /*input0_change_handler*/ ctx[2].call(input0, /*each_value*/ ctx[6], /*hostsBlock_index*/ ctx[7]);
- }
-
- function input1_input_handler() {
- /*input1_input_handler*/ ctx[3].call(input1, /*each_value*/ ctx[6], /*hostsBlock_index*/ ctx[7]);
- }
-
- const block = {
- c: function create() {
- div = element("div");
- span0 = element("span");
- input0 = element("input");
- t0 = space();
- span1 = element("span");
- t1 = text(t1_value);
- t2 = space();
- span2 = element("span");
- input1 = element("input");
- t3 = space();
- span3 = element("span");
- t4 = text(t4_value);
- t5 = space();
- attr_dev(input0, "type", "checkbox");
- attr_dev(input0, "class", "svelte-ze2due");
- add_location(input0, file$7, 85, 4, 1632);
- attr_dev(span0, "class", "svelte-ze2due");
- add_location(span0, file$7, 84, 3, 1621);
- attr_dev(span1, "class", "svelte-ze2due");
- add_location(span1, file$7, 90, 3, 1719);
- input1.disabled = true;
- attr_dev(input1, "class", "svelte-ze2due");
- add_location(input1, file$7, 94, 4, 1773);
- attr_dev(span2, "class", "svelte-ze2due");
- add_location(span2, file$7, 93, 3, 1762);
- attr_dev(span3, "class", "svelte-ze2due");
- add_location(span3, file$7, 99, 3, 1847);
- attr_dev(div, "class", "item svelte-ze2due");
- add_location(div, file$7, 83, 2, 1599);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span0);
- append_dev(span0, input0);
- input0.checked = /*hostsBlock*/ ctx[5].IsEnabled;
- append_dev(div, t0);
- append_dev(div, span1);
- append_dev(span1, t1);
- append_dev(div, t2);
- append_dev(div, span2);
- append_dev(span2, input1);
- set_input_value(input1, /*hostsBlock*/ ctx[5].URL);
- append_dev(div, t3);
- append_dev(div, span3);
- append_dev(span3, t4);
- append_dev(div, t5);
-
- if (!mounted) {
- dispose = [
- listen_dev(input0, "change", input0_change_handler),
- listen_dev(input1, "input", input1_input_handler)
- ];
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
-
- if (dirty & /*env*/ 1) {
- input0.checked = /*hostsBlock*/ ctx[5].IsEnabled;
- }
-
- if (dirty & /*env*/ 1 && t1_value !== (t1_value = /*hostsBlock*/ ctx[5].Name + "")) set_data_dev(t1, t1_value);
-
- if (dirty & /*env*/ 1 && input1.value !== /*hostsBlock*/ ctx[5].URL) {
- set_input_value(input1, /*hostsBlock*/ ctx[5].URL);
- }
-
- if (dirty & /*env*/ 1 && t4_value !== (t4_value = /*hostsBlock*/ ctx[5].LastUpdated + "")) set_data_dev(t4, t4_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block$3.name,
- type: "each",
- source: "(83:2) {#each env.HostsBlocks as hostsBlock}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$7(ctx) {
- let div3;
- let p;
- let t1;
- let div1;
- let div0;
- let span0;
- let t3;
- let span1;
- let t5;
- let span2;
- let t7;
- let span3;
- let t9;
- let t10;
- let div2;
- let button;
- let mounted;
- let dispose;
- let each_value = /*env*/ ctx[0].HostsBlocks;
- validate_each_argument(each_value);
- let each_blocks = [];
-
- for (let i = 0; i < each_value.length; i += 1) {
- each_blocks[i] = create_each_block$3(get_each_context$3(ctx, each_value, i));
- }
-
- const block = {
- c: function create() {
- div3 = element("div");
- p = element("p");
- p.textContent = "Configure the source of blocked hosts file.";
- t1 = space();
- div1 = element("div");
- div0 = element("div");
- span0 = element("span");
- span0.textContent = "Enabled";
- t3 = space();
- span1 = element("span");
- span1.textContent = "Name";
- t5 = space();
- span2 = element("span");
- span2.textContent = "URL";
- t7 = space();
- span3 = element("span");
- span3.textContent = "Last updated";
- t9 = space();
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- t10 = space();
- div2 = element("div");
- button = element("button");
- button.textContent = "Save";
- add_location(p, file$7, 71, 1, 1334);
- attr_dev(span0, "class", "svelte-ze2due");
- add_location(span0, file$7, 77, 3, 1449);
- attr_dev(span1, "class", "svelte-ze2due");
- add_location(span1, file$7, 78, 3, 1475);
- attr_dev(span2, "class", "svelte-ze2due");
- add_location(span2, file$7, 79, 3, 1498);
- attr_dev(span3, "class", "svelte-ze2due");
- add_location(span3, file$7, 80, 3, 1520);
- attr_dev(div0, "class", "item header svelte-ze2due");
- add_location(div0, file$7, 76, 2, 1420);
- attr_dev(div1, "class", "block_source svelte-ze2due");
- add_location(div1, file$7, 75, 1, 1391);
- add_location(button, file$7, 107, 2, 1931);
- add_location(div2, file$7, 106, 1, 1923);
- attr_dev(div3, "class", "hosts-block");
- add_location(div3, file$7, 70, 0, 1307);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div3, anchor);
- append_dev(div3, p);
- append_dev(div3, t1);
- append_dev(div3, div1);
- append_dev(div1, div0);
- append_dev(div0, span0);
- append_dev(div0, t3);
- append_dev(div0, span1);
- append_dev(div0, t5);
- append_dev(div0, span2);
- append_dev(div0, t7);
- append_dev(div0, span3);
- append_dev(div1, t9);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(div1, null);
- }
-
- append_dev(div3, t10);
- append_dev(div3, div2);
- append_dev(div2, button);
-
- if (!mounted) {
- dispose = listen_dev(button, "click", /*updateHostsBlocks*/ ctx[1], false, false, false);
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*env*/ 1) {
- each_value = /*env*/ ctx[0].HostsBlocks;
- validate_each_argument(each_value);
- let i;
-
- for (i = 0; i < each_value.length; i += 1) {
- const child_ctx = get_each_context$3(ctx, each_value, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block$3(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(div1, null);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value.length;
- }
- },
- i: noop,
- o: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(div3);
- destroy_each(each_blocks, detaching);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$7.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const apiHostsBlock = "/api/hosts_block";
-
- function instance$7($$self, $$props, $$invalidate) {
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {}
- };
-
- const envUnsubscribe = environment.subscribe(value => {
- $$invalidate(0, env = value);
- });
-
- onDestroy(envUnsubscribe);
-
- async function updateHostsBlocks() {
- const res = await fetch(apiHostsBlock, {
- method: "POST",
- headers: { "Content-Type": "application/json" },
- body: JSON.stringify(env.HostsBlocks)
- });
-
- if (res.status >= 400) {
- WuiPushNotif.Error("ERROR: ", res.status, res.statusText);
- return;
- }
-
- setEnvironment(await res.json());
- }
-
- const writable_props = [];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<HostsBlock> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("HostsBlock", $$slots, []);
-
- function input0_change_handler(each_value, hostsBlock_index) {
- each_value[hostsBlock_index].IsEnabled = this.checked;
- $$invalidate(0, env);
- }
-
- function input1_input_handler(each_value, hostsBlock_index) {
- each_value[hostsBlock_index].URL = this.value;
- $$invalidate(0, env);
- }
-
- $$self.$capture_state = () => ({
- onDestroy,
- WuiPushNotif,
- environment,
- nanoSeconds,
- setEnvironment,
- apiHostsBlock,
- env,
- envUnsubscribe,
- updateHostsBlocks
- });
-
- $$self.$inject_state = $$props => {
- if ("env" in $$props) $$invalidate(0, env = $$props.env);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [env, updateHostsBlocks, input0_change_handler, input1_input_handler];
- }
-
- class HostsBlock extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$7, create_fragment$7, safe_not_equal, {});
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "HostsBlock",
- options,
- id: create_fragment$7.name
- });
- }
- }
-
- /* src/HostsDir.svelte generated by Svelte v3.24.1 */
-
- const { Object: Object_1$1 } = globals;
- const file$8 = "src/HostsDir.svelte";
-
- function get_each_context$4(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[15] = list[i];
- child_ctx[17] = i;
- return child_ctx;
- }
-
- function get_each_context_1$1(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[18] = list[i][0];
- child_ctx[19] = list[i][1];
- child_ctx[18] = i;
- return child_ctx;
- }
-
- // (178:2) {#each Object.entries(env.HostsFiles) as [name,hf], name }
- function create_each_block_1$1(ctx) {
- let div;
- let a;
- let t_value = /*hf*/ ctx[19].Name + "";
- let t;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- div = element("div");
- a = element("a");
- t = text(t_value);
- attr_dev(a, "href", "#");
- add_location(a, file$8, 179, 3, 3810);
- attr_dev(div, "class", "item svelte-1w15y7z");
- add_location(div, file$8, 178, 2, 3788);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, a);
- append_dev(a, t);
-
- if (!mounted) {
- dispose = listen_dev(
- a,
- "click",
- function () {
- if (is_function(/*getHostsFile*/ ctx[4](/*hf*/ ctx[19]))) /*getHostsFile*/ ctx[4](/*hf*/ ctx[19]).apply(this, arguments);
- },
- false,
- false,
- false
- );
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
- if (dirty & /*env*/ 1 && t_value !== (t_value = /*hf*/ ctx[19].Name + "")) set_data_dev(t, t_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_1$1.name,
- type: "each",
- source: "(178:2) {#each Object.entries(env.HostsFiles) as [name,hf], name }",
- ctx
- });
-
- return block;
- }
-
- // (203:2) {:else}
- function create_else_block(ctx) {
- let p;
- let t0_value = /*hostsFile*/ ctx[1].Name + "";
- let t0;
- let t1;
- let t2_value = /*hostsFile*/ ctx[1].Records.length + "";
- let t2;
- let t3;
- let button0;
- let t5;
- let div0;
- let button1;
- let t7;
- let t8;
- let div1;
- let span0;
- let t10;
- let span1;
- let t12;
- let each_blocks = [];
- let each_1_lookup = new Map();
- let each_1_anchor;
- let mounted;
- let dispose;
- let if_block = /*newRecord*/ ctx[2] !== null && create_if_block_1$1(ctx);
- let each_value = /*hostsFile*/ ctx[1].Records;
- validate_each_argument(each_value);
- const get_key = ctx => /*idx*/ ctx[17];
- validate_each_keys(ctx, each_value, get_each_context$4, get_key);
-
- for (let i = 0; i < each_value.length; i += 1) {
- let child_ctx = get_each_context$4(ctx, each_value, i);
- let key = get_key(child_ctx);
- each_1_lookup.set(key, each_blocks[i] = create_each_block$4(key, child_ctx));
- }
-
- const block = {
- c: function create() {
- p = element("p");
- t0 = text(t0_value);
- t1 = text(" (");
- t2 = text(t2_value);
- t3 = text(" records)\n\t\t\t\t");
- button0 = element("button");
- button0.textContent = "Delete";
- t5 = space();
- div0 = element("div");
- button1 = element("button");
- button1.textContent = "Add";
- t7 = space();
- if (if_block) if_block.c();
- t8 = space();
- div1 = element("div");
- span0 = element("span");
- span0.textContent = "Domain name";
- t10 = space();
- span1 = element("span");
- span1.textContent = "IP address";
- t12 = space();
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- each_1_anchor = empty();
- add_location(button0, file$8, 205, 4, 4265);
- add_location(p, file$8, 203, 3, 4199);
- add_location(button1, file$8, 210, 4, 4359);
- add_location(div0, file$8, 209, 3, 4349);
- attr_dev(span0, "class", "host_name svelte-1w15y7z");
- add_location(span0, file$8, 234, 4, 4824);
- attr_dev(span1, "class", "host_value svelte-1w15y7z");
- add_location(span1, file$8, 235, 4, 4873);
- attr_dev(div1, "class", "host header svelte-1w15y7z");
- add_location(div1, file$8, 233, 3, 4794);
- },
- m: function mount(target, anchor) {
- insert_dev(target, p, anchor);
- append_dev(p, t0);
- append_dev(p, t1);
- append_dev(p, t2);
- append_dev(p, t3);
- append_dev(p, button0);
- insert_dev(target, t5, anchor);
- insert_dev(target, div0, anchor);
- append_dev(div0, button1);
- insert_dev(target, t7, anchor);
- if (if_block) if_block.m(target, anchor);
- insert_dev(target, t8, anchor);
- insert_dev(target, div1, anchor);
- append_dev(div1, span0);
- append_dev(div1, t10);
- append_dev(div1, span1);
- insert_dev(target, t12, anchor);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(target, anchor);
- }
-
- insert_dev(target, each_1_anchor, anchor);
-
- if (!mounted) {
- dispose = [
- listen_dev(
- button0,
- "click",
- function () {
- if (is_function(/*deleteHostsFile*/ ctx[9](/*hostsFile*/ ctx[1]))) /*deleteHostsFile*/ ctx[9](/*hostsFile*/ ctx[1]).apply(this, arguments);
- },
- false,
- false,
- false
- ),
- listen_dev(button1, "click", /*addRecord*/ ctx[6], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
- if (dirty & /*hostsFile*/ 2 && t0_value !== (t0_value = /*hostsFile*/ ctx[1].Name + "")) set_data_dev(t0, t0_value);
- if (dirty & /*hostsFile*/ 2 && t2_value !== (t2_value = /*hostsFile*/ ctx[1].Records.length + "")) set_data_dev(t2, t2_value);
-
- if (/*newRecord*/ ctx[2] !== null) {
- if (if_block) {
- if_block.p(ctx, dirty);
- } else {
- if_block = create_if_block_1$1(ctx);
- if_block.c();
- if_block.m(t8.parentNode, t8);
- }
- } else if (if_block) {
- if_block.d(1);
- if_block = null;
- }
-
- if (dirty & /*handleHostsRecordDelete, hostsFile*/ 258) {
- const each_value = /*hostsFile*/ ctx[1].Records;
- validate_each_argument(each_value);
- validate_each_keys(ctx, each_value, get_each_context$4, get_key);
- each_blocks = update_keyed_each(each_blocks, dirty, get_key, 1, ctx, each_value, each_1_lookup, each_1_anchor.parentNode, destroy_block, create_each_block$4, each_1_anchor, get_each_context$4);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(p);
- if (detaching) detach_dev(t5);
- if (detaching) detach_dev(div0);
- if (detaching) detach_dev(t7);
- if (if_block) if_block.d(detaching);
- if (detaching) detach_dev(t8);
- if (detaching) detach_dev(div1);
- if (detaching) detach_dev(t12);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].d(detaching);
- }
-
- if (detaching) detach_dev(each_1_anchor);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_else_block.name,
- type: "else",
- source: "(203:2) {:else}",
- ctx
- });
-
- return block;
- }
-
- // (199:2) {#if hostsFile.Name === ""}
- function create_if_block$4(ctx) {
- let div;
-
- const block = {
- c: function create() {
- div = element("div");
- div.textContent = "Select one of the hosts file to manage.";
- add_location(div, file$8, 199, 3, 4126);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- },
- p: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block$4.name,
- type: "if",
- source: "(199:2) {#if hostsFile.Name === \\\"\\\"}",
- ctx
- });
-
- return block;
- }
-
- // (216:3) {#if newRecord !== null}
- function create_if_block_1$1(ctx) {
- let div;
- let input0;
- let t0;
- let input1;
- let t1;
- let button;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- div = element("div");
- input0 = element("input");
- t0 = space();
- input1 = element("input");
- t1 = space();
- button = element("button");
- button.textContent = "Create";
- attr_dev(input0, "class", "host_name svelte-1w15y7z");
- attr_dev(input0, "placeholder", "Domain name");
- add_location(input0, file$8, 217, 5, 4479);
- attr_dev(input1, "class", "host_value svelte-1w15y7z");
- attr_dev(input1, "placeholder", "IP address");
- add_location(input1, file$8, 222, 5, 4588);
- add_location(button, file$8, 227, 5, 4698);
- attr_dev(div, "class", "host svelte-1w15y7z");
- add_location(div, file$8, 216, 4, 4455);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, input0);
- set_input_value(input0, /*newRecord*/ ctx[2].Name);
- append_dev(div, t0);
- append_dev(div, input1);
- set_input_value(input1, /*newRecord*/ ctx[2].Value);
- append_dev(div, t1);
- append_dev(div, button);
-
- if (!mounted) {
- dispose = [
- listen_dev(input0, "input", /*input0_input_handler*/ ctx[11]),
- listen_dev(input1, "input", /*input1_input_handler*/ ctx[12]),
- listen_dev(button, "click", /*handleHostsRecordCreate*/ ctx[7], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty & /*newRecord*/ 4 && input0.value !== /*newRecord*/ ctx[2].Name) {
- set_input_value(input0, /*newRecord*/ ctx[2].Name);
- }
-
- if (dirty & /*newRecord*/ 4 && input1.value !== /*newRecord*/ ctx[2].Value) {
- set_input_value(input1, /*newRecord*/ ctx[2].Value);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_1$1.name,
- type: "if",
- source: "(216:3) {#if newRecord !== null}",
- ctx
- });
-
- return block;
- }
-
- // (239:3) {#each hostsFile.Records as rr, idx (idx)}
- function create_each_block$4(key_1, ctx) {
- let div;
- let span0;
- let t0_value = /*rr*/ ctx[15].Name + "";
- let t0;
- let t1;
- let span1;
- let t2_value = /*rr*/ ctx[15].Value + "";
- let t2;
- let t3;
- let button;
- let t5;
- let mounted;
- let dispose;
-
- const block = {
- key: key_1,
- first: null,
- c: function create() {
- div = element("div");
- span0 = element("span");
- t0 = text(t0_value);
- t1 = space();
- span1 = element("span");
- t2 = text(t2_value);
- t3 = space();
- button = element("button");
- button.textContent = "X";
- t5 = space();
- attr_dev(span0, "class", "host_name svelte-1w15y7z");
- add_location(span0, file$8, 240, 5, 5003);
- attr_dev(span1, "class", "host_value svelte-1w15y7z");
- add_location(span1, file$8, 241, 5, 5051);
- add_location(button, file$8, 242, 5, 5101);
- attr_dev(div, "class", "host svelte-1w15y7z");
- add_location(div, file$8, 239, 4, 4979);
- this.first = div;
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span0);
- append_dev(span0, t0);
- append_dev(div, t1);
- append_dev(div, span1);
- append_dev(span1, t2);
- append_dev(div, t3);
- append_dev(div, button);
- append_dev(div, t5);
-
- if (!mounted) {
- dispose = listen_dev(
- button,
- "click",
- function () {
- if (is_function(/*handleHostsRecordDelete*/ ctx[8](/*rr*/ ctx[15], /*idx*/ ctx[17]))) /*handleHostsRecordDelete*/ ctx[8](/*rr*/ ctx[15], /*idx*/ ctx[17]).apply(this, arguments);
- },
- false,
- false,
- false
- );
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
- if (dirty & /*hostsFile*/ 2 && t0_value !== (t0_value = /*rr*/ ctx[15].Name + "")) set_data_dev(t0, t0_value);
- if (dirty & /*hostsFile*/ 2 && t2_value !== (t2_value = /*rr*/ ctx[15].Value + "")) set_data_dev(t2, t2_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block$4.name,
- type: "each",
- source: "(239:3) {#each hostsFile.Records as rr, idx (idx)}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$8(ctx) {
- let div2;
- let div0;
- let t0;
- let br0;
- let t1;
- let label;
- let span;
- let t3;
- let br1;
- let t4;
- let input;
- let t5;
- let button;
- let t7;
- let div1;
- let mounted;
- let dispose;
- let each_value_1 = Object.entries(/*env*/ ctx[0].HostsFiles);
- validate_each_argument(each_value_1);
- let each_blocks = [];
-
- for (let i = 0; i < each_value_1.length; i += 1) {
- each_blocks[i] = create_each_block_1$1(get_each_context_1$1(ctx, each_value_1, i));
- }
-
- function select_block_type(ctx, dirty) {
- if (/*hostsFile*/ ctx[1].Name === "") return create_if_block$4;
- return create_else_block;
- }
-
- let current_block_type = select_block_type(ctx);
- let if_block = current_block_type(ctx);
-
- const block = {
- c: function create() {
- div2 = element("div");
- div0 = element("div");
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- t0 = space();
- br0 = element("br");
- t1 = space();
- label = element("label");
- span = element("span");
- span.textContent = "New hosts file:";
- t3 = space();
- br1 = element("br");
- t4 = space();
- input = element("input");
- t5 = space();
- button = element("button");
- button.textContent = "Create";
- t7 = space();
- div1 = element("div");
- if_block.c();
- add_location(br0, file$8, 185, 2, 3895);
- add_location(span, file$8, 188, 3, 3915);
- add_location(br1, file$8, 189, 3, 3947);
- add_location(input, file$8, 190, 3, 3956);
- add_location(label, file$8, 187, 2, 3904);
- add_location(button, file$8, 192, 2, 4003);
- attr_dev(div0, "class", "nav-left svelte-1w15y7z");
- add_location(div0, file$8, 176, 1, 3702);
- attr_dev(div1, "class", "content svelte-1w15y7z");
- add_location(div1, file$8, 197, 1, 4071);
- attr_dev(div2, "class", "hosts_d");
- add_location(div2, file$8, 175, 0, 3679);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div2, anchor);
- append_dev(div2, div0);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(div0, null);
- }
-
- append_dev(div0, t0);
- append_dev(div0, br0);
- append_dev(div0, t1);
- append_dev(div0, label);
- append_dev(label, span);
- append_dev(label, t3);
- append_dev(label, br1);
- append_dev(label, t4);
- append_dev(label, input);
- set_input_value(input, /*newHostsFile*/ ctx[3]);
- append_dev(div0, t5);
- append_dev(div0, button);
- append_dev(div2, t7);
- append_dev(div2, div1);
- if_block.m(div1, null);
-
- if (!mounted) {
- dispose = [
- listen_dev(input, "input", /*input_input_handler*/ ctx[10]),
- listen_dev(button, "click", /*createHostsFile*/ ctx[5], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*getHostsFile, Object, env*/ 17) {
- each_value_1 = Object.entries(/*env*/ ctx[0].HostsFiles);
- validate_each_argument(each_value_1);
- let i;
-
- for (i = 0; i < each_value_1.length; i += 1) {
- const child_ctx = get_each_context_1$1(ctx, each_value_1, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block_1$1(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(div0, t0);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value_1.length;
- }
-
- if (dirty & /*newHostsFile*/ 8 && input.value !== /*newHostsFile*/ ctx[3]) {
- set_input_value(input, /*newHostsFile*/ ctx[3]);
- }
-
- if (current_block_type === (current_block_type = select_block_type(ctx)) && if_block) {
- if_block.p(ctx, dirty);
- } else {
- if_block.d(1);
- if_block = current_block_type(ctx);
-
- if (if_block) {
- if_block.c();
- if_block.m(div1, null);
- }
- }
- },
- i: noop,
- o: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(div2);
- destroy_each(each_blocks, detaching);
- if_block.d();
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$8.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const apiHostsDir = "/api/hosts.d";
-
- function instance$8($$self, $$props, $$invalidate) {
- let env = { HostsFiles: {} };
- let hostsFile = { Name: "", Records: [] };
- let newRecord = null;
- let newHostsFile = "";
-
- const envUnsubscribe = environment.subscribe(value => {
- $$invalidate(0, env = value);
- });
-
- onDestroy(envUnsubscribe);
-
- async function getHostsFile(hf) {
- if (hf.Records === null) {
- hf.Records = [];
- }
-
- if (hf.Records.length > 0) {
- $$invalidate(1, hostsFile = hf);
- return;
- }
-
- const res = await fetch(apiHostsDir + "/" + hf.Name);
- hf.Records = await res.json();
- $$invalidate(1, hostsFile = hf);
- }
-
- async function createHostsFile() {
- if (newHostsFile === "") {
- return;
- }
-
- const res = await fetch(apiHostsDir + "/" + newHostsFile, { method: "PUT" });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: createHostsFile: ", resError.message);
- return;
- }
-
- const hf = { Name: newHostsFile, Records: [] };
- $$invalidate(0, env.HostsFiles[newHostsFile] = hf, env);
- $$invalidate(0, env);
- WuiPushNotif.Info("The new host file '" + newHostsFile + "' has been created");
- }
-
- async function updateHostsFile() {
- const res = await fetch(apiHostsDir + "/" + hostsFile.Name, {
- method: "POST",
- body: JSON.stringify(hostsFile.Records)
- });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: updateHostsFile: ", resError.message);
- return;
- }
-
- $$invalidate(1, hostsFile.Records = await res.json(), hostsFile);
- WuiPushNotif.Info("The host file '" + hostsFile.Name + "' has been updated");
- }
-
- function addRecord() {
- if (newRecord !== null) {
- return;
- }
-
- $$invalidate(2, newRecord = { Name: "", Value: "" });
- }
-
- async function handleHostsRecordCreate() {
- const api = apiHostsDir + "/" + hostsFile.Name + "/rr" + "?domain=" + newRecord.Name + "&value=" + newRecord.Value;
- const res = await fetch(api, { method: "POST" });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: " + resError.message);
- return;
- }
-
- const rr = await res.json();
- hostsFile.Records.push(rr);
- $$invalidate(1, hostsFile);
- $$invalidate(2, newRecord = null);
- WuiPushNotif.Info("Record '" + rr.Name + "' has been created");
- }
-
- async function handleHostsRecordDelete(rr, idx) {
- const api = apiHostsDir + "/" + hostsFile.Name + "/rr" + "?domain=" + rr.Name;
- const res = await fetch(api, { method: "DELETE" });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: " + resError.message);
- return;
- }
-
- hostsFile.Records.splice(idx, 1);
- $$invalidate(1, hostsFile);
- WuiPushNotif.Info("Record '" + rr.Name + "' has been deleted");
- }
-
- async function deleteHostsFile(hfile) {
- const res = await fetch(apiHostsDir + "/" + hfile.Name, { method: "DELETE" });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: deleteHostsFile: ", resError.message);
- return;
- }
-
- delete env.HostsFiles[hfile.Name];
- $$invalidate(0, env);
- $$invalidate(1, hostsFile = { Name: "", Records: [] });
- WuiPushNotif.Info("The host file '" + hfile.Name + "' has been deleted");
- }
-
- const writable_props = [];
-
- Object_1$1.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<HostsDir> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("HostsDir", $$slots, []);
-
- function input_input_handler() {
- newHostsFile = this.value;
- $$invalidate(3, newHostsFile);
- }
-
- function input0_input_handler() {
- newRecord.Name = this.value;
- $$invalidate(2, newRecord);
- }
-
- function input1_input_handler() {
- newRecord.Value = this.value;
- $$invalidate(2, newRecord);
- }
-
- $$self.$capture_state = () => ({
- onDestroy,
- WuiPushNotif,
- apiEnvironment,
- environment,
- nanoSeconds,
- apiHostsDir,
- env,
- hostsFile,
- newRecord,
- newHostsFile,
- envUnsubscribe,
- getHostsFile,
- createHostsFile,
- updateHostsFile,
- addRecord,
- handleHostsRecordCreate,
- handleHostsRecordDelete,
- deleteHostsFile
- });
-
- $$self.$inject_state = $$props => {
- if ("env" in $$props) $$invalidate(0, env = $$props.env);
- if ("hostsFile" in $$props) $$invalidate(1, hostsFile = $$props.hostsFile);
- if ("newRecord" in $$props) $$invalidate(2, newRecord = $$props.newRecord);
- if ("newHostsFile" in $$props) $$invalidate(3, newHostsFile = $$props.newHostsFile);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [
- env,
- hostsFile,
- newRecord,
- newHostsFile,
- getHostsFile,
- createHostsFile,
- addRecord,
- handleHostsRecordCreate,
- handleHostsRecordDelete,
- deleteHostsFile,
- input_input_handler,
- input0_input_handler,
- input1_input_handler
- ];
- }
-
- class HostsDir extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$8, create_fragment$8, safe_not_equal, {});
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "HostsDir",
- options,
- id: create_fragment$8.name
- });
- }
- }
-
- /* src/MasterDir.svelte generated by Svelte v3.24.1 */
-
- const { Object: Object_1$2 } = globals;
- const file$9 = "src/MasterDir.svelte";
-
- function get_each_context$5(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[32] = list[i][0];
- child_ctx[33] = list[i][1];
- return child_ctx;
- }
-
- function get_each_context_2$1(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[40] = list[i];
- child_ctx[42] = i;
- return child_ctx;
- }
-
- function get_each_context_1$2(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[36] = list[i][0];
- child_ctx[37] = list[i][1];
- return child_ctx;
- }
-
- function get_each_context_3$1(ctx, list, i) {
- const child_ctx = ctx.slice();
- child_ctx[43] = list[i][0];
- child_ctx[44] = list[i][1];
- return child_ctx;
- }
-
- // (282:2) {#each Object.entries(env.ZoneFiles) as [name, zoneFile]}
- function create_each_block_3$1(ctx) {
- let div;
- let span;
- let t_value = /*zoneFile*/ ctx[44].Name + "";
- let t;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- div = element("div");
- span = element("span");
- t = text(t_value);
- add_location(span, file$9, 283, 4, 5107);
- attr_dev(div, "class", "item svelte-yu4l71");
- add_location(div, file$9, 282, 3, 5084);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span);
- append_dev(span, t);
-
- if (!mounted) {
- dispose = listen_dev(
- span,
- "click",
- function () {
- if (is_function(/*setActiveZone*/ ctx[6](/*zoneFile*/ ctx[44]))) /*setActiveZone*/ ctx[6](/*zoneFile*/ ctx[44]).apply(this, arguments);
- },
- false,
- false,
- false
- );
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
- if (dirty[0] & /*env*/ 1 && t_value !== (t_value = /*zoneFile*/ ctx[44].Name + "")) set_data_dev(t, t_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_3$1.name,
- type: "each",
- source: "(282:2) {#each Object.entries(env.ZoneFiles) as [name, zoneFile]}",
- ctx
- });
-
- return block;
- }
-
- // (306:2) {:else}
- function create_else_block$1(ctx) {
- let h3;
- let t0_value = /*activeZone*/ ctx[2].Name + "";
- let t0;
- let t1;
- let button0;
- let t3;
- let h40;
- let t4;
- let button1;
- let t6;
- let div1;
- let wuilabelhint0;
- let t7;
- let wuilabelhint1;
- let t8;
- let wuilabelhint2;
- let t9;
- let wuilabelhint3;
- let t10;
- let wuilabelhint4;
- let t11;
- let wuilabelhint5;
- let t12;
- let wuilabelhint6;
- let t13;
- let div0;
- let button2;
- let t15;
- let h41;
- let t17;
- let div2;
- let span0;
- let t19;
- let span1;
- let t21;
- let span2;
- let t23;
- let t24;
- let form;
- let label;
- let span3;
- let t26;
- let select;
- let t27;
- let t28;
- let div3;
- let button3;
- let current;
- let mounted;
- let dispose;
-
- wuilabelhint0 = new LabelHint({
- props: {
- title: "Name server",
- title_width: "150px",
- info: "The domain-name of the name server that was the\noriginal or primary source of data for this zone.\nIt should be domain-name where the rescached run.",
- $$slots: { default: [create_default_slot_6$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint1 = new LabelHint({
- props: {
- title: "Admin email",
- title_width: "150px",
- info: "Email address of the administrator responsible for\nthis zone.\nThe \"@\" on email address is replaced with dot, and if there is a dot before\n\"@\" it should be escaped with \"\\\".\nFor example, \"dns.admin@domain.tld\" would be written as\n\"dns\\.admin.domain.tld\".",
- $$slots: { default: [create_default_slot_5$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint2 = new LabelHint({
- props: {
- title: "Serial",
- title_width: "150px",
- info: "Serial number for this zone. If a secondary name\nserver observes an increase in this number, the server will assume that the\nzone has been updated and initiate a zone transfer.",
- $$slots: { default: [create_default_slot_4$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint3 = new LabelHint({
- props: {
- title: "Refresh",
- title_width: "150px",
- info: "Number of seconds after which secondary name servers\nshould query the master for the SOA record, to detect zone changes.\nRecommendation for small and stable zones is 86400 seconds (24 hours).",
- $$slots: { default: [create_default_slot_3$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint4 = new LabelHint({
- props: {
- title: "Retry",
- title_width: "150px",
- info: "Number of seconds after which secondary name servers\nshould retry to request the serial number from the master if the master does\nnot respond.\nIt must be less than Refresh.\nRecommendation for small and stable zones is 7200 seconds (2 hours).",
- $$slots: { default: [create_default_slot_2$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint5 = new LabelHint({
- props: {
- title: "Expire",
- title_width: "150px",
- info: "Number of seconds after which secondary name servers\nshould stop answering request for this zone if the master does not respond.\nThis value must be bigger than the sum of Refresh and Retry.\nRecommendation for small and stable zones is 3600000 seconds (1000 hours).",
- $$slots: { default: [create_default_slot_1$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- wuilabelhint6 = new LabelHint({
- props: {
- title: "Minimum",
- title_width: "150px",
- info: "Time to live for purposes of negative caching.\nRecommendation for small and stable zones is 1800 seconds (30 min).",
- $$slots: { default: [create_default_slot$1] },
- $$scope: { ctx }
- },
- $$inline: true
- });
-
- let each_value_1 = Object.entries(/*activeZone*/ ctx[2].Records);
- validate_each_argument(each_value_1);
- let each_blocks_1 = [];
-
- for (let i = 0; i < each_value_1.length; i += 1) {
- each_blocks_1[i] = create_each_block_1$2(get_each_context_1$2(ctx, each_value_1, i));
- }
-
- let each_value = Object.entries(/*RRTypes*/ ctx[5]);
- validate_each_argument(each_value);
- let each_blocks = [];
-
- for (let i = 0; i < each_value.length; i += 1) {
- each_blocks[i] = create_each_block$5(get_each_context$5(ctx, each_value, i));
- }
-
- function select_block_type_1(ctx, dirty) {
- if (/*_rr*/ ctx[3].Type === 1 || /*_rr*/ ctx[3].Type === 2 || /*_rr*/ ctx[3].Type === 5 || /*_rr*/ ctx[3].Type === 16 || /*_rr*/ ctx[3].Type === 28) return create_if_block_1$2;
- if (/*_rr*/ ctx[3].Type === 12) return create_if_block_2$1;
- if (/*_rr*/ ctx[3].Type === 15) return create_if_block_3;
- }
-
- let current_block_type = select_block_type_1(ctx);
- let if_block = current_block_type && current_block_type(ctx);
-
- const block = {
- c: function create() {
- h3 = element("h3");
- t0 = text(t0_value);
- t1 = space();
- button0 = element("button");
- button0.textContent = "Delete";
- t3 = space();
- h40 = element("h4");
- t4 = text("SOA record\n\t\t\t\t");
- button1 = element("button");
- button1.textContent = "Delete";
- t6 = space();
- div1 = element("div");
- create_component(wuilabelhint0.$$.fragment);
- t7 = space();
- create_component(wuilabelhint1.$$.fragment);
- t8 = space();
- create_component(wuilabelhint2.$$.fragment);
- t9 = space();
- create_component(wuilabelhint3.$$.fragment);
- t10 = space();
- create_component(wuilabelhint4.$$.fragment);
- t11 = space();
- create_component(wuilabelhint5.$$.fragment);
- t12 = space();
- create_component(wuilabelhint6.$$.fragment);
- t13 = space();
- div0 = element("div");
- button2 = element("button");
- button2.textContent = "Save";
- t15 = space();
- h41 = element("h4");
- h41.textContent = "List records";
- t17 = space();
- div2 = element("div");
- span0 = element("span");
- span0.textContent = "Name";
- t19 = space();
- span1 = element("span");
- span1.textContent = "Type";
- t21 = space();
- span2 = element("span");
- span2.textContent = "Value";
- t23 = space();
-
- for (let i = 0; i < each_blocks_1.length; i += 1) {
- each_blocks_1[i].c();
- }
-
- t24 = space();
- form = element("form");
- label = element("label");
- span3 = element("span");
- span3.textContent = "Type:";
- t26 = space();
- select = element("select");
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- t27 = space();
- if (if_block) if_block.c();
- t28 = space();
- div3 = element("div");
- button3 = element("button");
- button3.textContent = "Create";
- attr_dev(button0, "class", "action-delete svelte-yu4l71");
- add_location(button0, file$9, 308, 4, 5538);
- add_location(h3, file$9, 306, 3, 5507);
- attr_dev(button1, "class", "action-delete svelte-yu4l71");
- add_location(button1, file$9, 318, 4, 5679);
- attr_dev(h40, "class", "svelte-yu4l71");
- add_location(h40, file$9, 316, 3, 5655);
- attr_dev(button2, "class", "svelte-yu4l71");
- add_location(button2, file$9, 415, 5, 8540);
- attr_dev(div0, "class", "actions svelte-yu4l71");
- add_location(div0, file$9, 414, 4, 8513);
- attr_dev(div1, "class", "rr-soa");
- add_location(div1, file$9, 325, 3, 5790);
- attr_dev(h41, "class", "svelte-yu4l71");
- add_location(h41, file$9, 421, 3, 8625);
- attr_dev(span0, "class", "name svelte-yu4l71");
- add_location(span0, file$9, 423, 4, 8680);
- attr_dev(span1, "class", "type svelte-yu4l71");
- add_location(span1, file$9, 426, 4, 8726);
- attr_dev(span2, "class", "value svelte-yu4l71");
- add_location(span2, file$9, 429, 4, 8772);
- attr_dev(div2, "class", "rr header svelte-yu4l71");
- add_location(div2, file$9, 422, 3, 8652);
- attr_dev(span3, "class", "svelte-yu4l71");
- add_location(span3, file$9, 455, 5, 9310);
- if (/*_rr*/ ctx[3].Type === void 0) add_render_callback(() => /*select_change_handler*/ ctx[23].call(select));
- add_location(select, file$9, 458, 5, 9347);
- attr_dev(label, "class", "svelte-yu4l71");
- add_location(label, file$9, 454, 4, 9297);
- attr_dev(button3, "class", "create svelte-yu4l71");
- attr_dev(button3, "type", "submit");
- add_location(button3, file$9, 524, 5, 10707);
- attr_dev(div3, "class", "actions svelte-yu4l71");
- add_location(div3, file$9, 523, 4, 10680);
- attr_dev(form, "class", "svelte-yu4l71");
- add_location(form, file$9, 453, 3, 9244);
- },
- m: function mount(target, anchor) {
- insert_dev(target, h3, anchor);
- append_dev(h3, t0);
- append_dev(h3, t1);
- append_dev(h3, button0);
- insert_dev(target, t3, anchor);
- insert_dev(target, h40, anchor);
- append_dev(h40, t4);
- append_dev(h40, button1);
- insert_dev(target, t6, anchor);
- insert_dev(target, div1, anchor);
- mount_component(wuilabelhint0, div1, null);
- append_dev(div1, t7);
- mount_component(wuilabelhint1, div1, null);
- append_dev(div1, t8);
- mount_component(wuilabelhint2, div1, null);
- append_dev(div1, t9);
- mount_component(wuilabelhint3, div1, null);
- append_dev(div1, t10);
- mount_component(wuilabelhint4, div1, null);
- append_dev(div1, t11);
- mount_component(wuilabelhint5, div1, null);
- append_dev(div1, t12);
- mount_component(wuilabelhint6, div1, null);
- append_dev(div1, t13);
- append_dev(div1, div0);
- append_dev(div0, button2);
- insert_dev(target, t15, anchor);
- insert_dev(target, h41, anchor);
- insert_dev(target, t17, anchor);
- insert_dev(target, div2, anchor);
- append_dev(div2, span0);
- append_dev(div2, t19);
- append_dev(div2, span1);
- append_dev(div2, t21);
- append_dev(div2, span2);
- insert_dev(target, t23, anchor);
-
- for (let i = 0; i < each_blocks_1.length; i += 1) {
- each_blocks_1[i].m(target, anchor);
- }
-
- insert_dev(target, t24, anchor);
- insert_dev(target, form, anchor);
- append_dev(form, label);
- append_dev(label, span3);
- append_dev(label, t26);
- append_dev(label, select);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(select, null);
- }
-
- select_option(select, /*_rr*/ ctx[3].Type);
- append_dev(form, t27);
- if (if_block) if_block.m(form, null);
- append_dev(form, t28);
- append_dev(form, div3);
- append_dev(div3, button3);
- current = true;
-
- if (!mounted) {
- dispose = [
- listen_dev(button0, "click", /*handleZoneFileDelete*/ ctx[8], false, false, false),
- listen_dev(button1, "click", /*handleSOADelete*/ ctx[10], false, false, false),
- listen_dev(button2, "click", /*handleSOASave*/ ctx[11], false, false, false),
- listen_dev(select, "change", /*select_change_handler*/ ctx[23]),
- listen_dev(select, "blur", /*onSelectRRType*/ ctx[9], false, false, false),
- listen_dev(form, "submit", prevent_default(/*handleCreateRR*/ ctx[12]), false, true, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if ((!current || dirty[0] & /*activeZone*/ 4) && t0_value !== (t0_value = /*activeZone*/ ctx[2].Name + "")) set_data_dev(t0, t0_value);
- const wuilabelhint0_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint0_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint0.$set(wuilabelhint0_changes);
- const wuilabelhint1_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint1_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint1.$set(wuilabelhint1_changes);
- const wuilabelhint2_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint2_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint2.$set(wuilabelhint2_changes);
- const wuilabelhint3_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint3_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint3.$set(wuilabelhint3_changes);
- const wuilabelhint4_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint4_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint4.$set(wuilabelhint4_changes);
- const wuilabelhint5_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint5_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint5.$set(wuilabelhint5_changes);
- const wuilabelhint6_changes = {};
-
- if (dirty[0] & /*activeZone*/ 4 | dirty[1] & /*$$scope*/ 65536) {
- wuilabelhint6_changes.$$scope = { dirty, ctx };
- }
-
- wuilabelhint6.$set(wuilabelhint6_changes);
-
- if (dirty[0] & /*activeZone, handleDeleteRR, getTypeName*/ 24580) {
- each_value_1 = Object.entries(/*activeZone*/ ctx[2].Records);
- validate_each_argument(each_value_1);
- let i;
-
- for (i = 0; i < each_value_1.length; i += 1) {
- const child_ctx = get_each_context_1$2(ctx, each_value_1, i);
-
- if (each_blocks_1[i]) {
- each_blocks_1[i].p(child_ctx, dirty);
- } else {
- each_blocks_1[i] = create_each_block_1$2(child_ctx);
- each_blocks_1[i].c();
- each_blocks_1[i].m(t24.parentNode, t24);
- }
- }
-
- for (; i < each_blocks_1.length; i += 1) {
- each_blocks_1[i].d(1);
- }
-
- each_blocks_1.length = each_value_1.length;
- }
-
- if (dirty[0] & /*RRTypes*/ 32) {
- each_value = Object.entries(/*RRTypes*/ ctx[5]);
- validate_each_argument(each_value);
- let i;
-
- for (i = 0; i < each_value.length; i += 1) {
- const child_ctx = get_each_context$5(ctx, each_value, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block$5(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(select, null);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value.length;
- }
-
- if (dirty[0] & /*_rr, RRTypes*/ 40) {
- select_option(select, /*_rr*/ ctx[3].Type);
- }
-
- if (current_block_type === (current_block_type = select_block_type_1(ctx)) && if_block) {
- if_block.p(ctx, dirty);
- } else {
- if (if_block) if_block.d(1);
- if_block = current_block_type && current_block_type(ctx);
-
- if (if_block) {
- if_block.c();
- if_block.m(form, t28);
- }
- }
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuilabelhint0.$$.fragment, local);
- transition_in(wuilabelhint1.$$.fragment, local);
- transition_in(wuilabelhint2.$$.fragment, local);
- transition_in(wuilabelhint3.$$.fragment, local);
- transition_in(wuilabelhint4.$$.fragment, local);
- transition_in(wuilabelhint5.$$.fragment, local);
- transition_in(wuilabelhint6.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuilabelhint0.$$.fragment, local);
- transition_out(wuilabelhint1.$$.fragment, local);
- transition_out(wuilabelhint2.$$.fragment, local);
- transition_out(wuilabelhint3.$$.fragment, local);
- transition_out(wuilabelhint4.$$.fragment, local);
- transition_out(wuilabelhint5.$$.fragment, local);
- transition_out(wuilabelhint6.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(h3);
- if (detaching) detach_dev(t3);
- if (detaching) detach_dev(h40);
- if (detaching) detach_dev(t6);
- if (detaching) detach_dev(div1);
- destroy_component(wuilabelhint0);
- destroy_component(wuilabelhint1);
- destroy_component(wuilabelhint2);
- destroy_component(wuilabelhint3);
- destroy_component(wuilabelhint4);
- destroy_component(wuilabelhint5);
- destroy_component(wuilabelhint6);
- if (detaching) detach_dev(t15);
- if (detaching) detach_dev(h41);
- if (detaching) detach_dev(t17);
- if (detaching) detach_dev(div2);
- if (detaching) detach_dev(t23);
- destroy_each(each_blocks_1, detaching);
- if (detaching) detach_dev(t24);
- if (detaching) detach_dev(form);
- destroy_each(each_blocks, detaching);
-
- if (if_block) {
- if_block.d();
- }
-
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_else_block$1.name,
- type: "else",
- source: "(306:2) {:else}",
- ctx
- });
-
- return block;
- }
-
- // (302:2) {#if activeZone.Name === ""}
- function create_if_block$5(ctx) {
- let p;
-
- const block = {
- c: function create() {
- p = element("p");
- p.textContent = "Select one of the zone file to manage.";
- add_location(p, file$9, 302, 3, 5439);
- },
- m: function mount(target, anchor) {
- insert_dev(target, p, anchor);
- },
- p: noop,
- i: noop,
- o: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(p);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block$5.name,
- type: "if",
- source: "(302:2) {#if activeZone.Name === \\\"\\\"}",
- ctx
- });
-
- return block;
- }
-
- // (327:4) <WuiLabelHint title="Name server" title_width="150px" info="The domain-name of the name server that was the original or primary source of data for this zone. It should be domain-name where the rescached run." >
- function create_default_slot_6$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- add_location(input, file$9, 333, 5, 6050);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.MName);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_1*/ ctx[16]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && input.value !== /*activeZone*/ ctx[2].SOA.Value.MName) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.MName);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_6$1.name,
- type: "slot",
- source: "(327:4) <WuiLabelHint title=\\\"Name server\\\" title_width=\\\"150px\\\" info=\\\"The domain-name of the name server that was the original or primary source of data for this zone. It should be domain-name where the rescached run.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (336:4) <WuiLabelHint title="Admin email" title_width="150px" info='Email address of the administrator responsible for this zone. The "@" on email address is replaced with dot, and if there is a dot before "@" it should be escaped with "\". For example, "dns.admin@domain.tld" would be written as "dns\.admin.domain.tld".' >
- function create_default_slot_5$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- add_location(input, file$9, 345, 5, 6463);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.RName);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_2*/ ctx[17]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && input.value !== /*activeZone*/ ctx[2].SOA.Value.RName) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.RName);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_5$1.name,
- type: "slot",
- source: "(336:4) <WuiLabelHint title=\\\"Admin email\\\" title_width=\\\"150px\\\" info='Email address of the administrator responsible for this zone. The \\\"@\\\" on email address is replaced with dot, and if there is a dot before \\\"@\\\" it should be escaped with \\\"\\\\\". For example, \\\"dns.admin@domain.tld\\\" would be written as \\\"dns\\.admin.domain.tld\\\".' >",
- ctx
- });
-
- return block;
- }
-
- // (348:4) <WuiLabelHint title="Serial" title_width="150px" info="Serial number for this zone. If a secondary name server observes an increase in this number, the server will assume that the zone has been updated and initiate a zone transfer." >
- function create_default_slot_4$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "type", "number");
- attr_dev(input, "min", "0");
- add_location(input, file$9, 354, 5, 6794);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Serial);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_3*/ ctx[18]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && to_number(input.value) !== /*activeZone*/ ctx[2].SOA.Value.Serial) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Serial);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_4$1.name,
- type: "slot",
- source: "(348:4) <WuiLabelHint title=\\\"Serial\\\" title_width=\\\"150px\\\" info=\\\"Serial number for this zone. If a secondary name server observes an increase in this number, the server will assume that the zone has been updated and initiate a zone transfer.\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (361:4) <WuiLabelHint title="Refresh" title_width="150px" info="Number of seconds after which secondary name servers should query the master for the SOA record, to detect zone changes. Recommendation for small and stable zones is 86400 seconds (24 hours)." >
- function create_default_slot_3$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "type", "number");
- attr_dev(input, "min", "0");
- add_location(input, file$9, 367, 5, 7184);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Refresh);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_4*/ ctx[19]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && to_number(input.value) !== /*activeZone*/ ctx[2].SOA.Value.Refresh) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Refresh);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_3$1.name,
- type: "slot",
- source: "(361:4) <WuiLabelHint title=\\\"Refresh\\\" title_width=\\\"150px\\\" info=\\\"Number of seconds after which secondary name servers should query the master for the SOA record, to detect zone changes. Recommendation for small and stable zones is 86400 seconds (24 hours).\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (374:4) <WuiLabelHint title="Retry" title_width="150px" info="Number of seconds after which secondary name servers should retry to request the serial number from the master if the master does not respond. It must be less than Refresh. Recommendation for small and stable zones is 7200 seconds (2 hours)." >
- function create_default_slot_2$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "type", "number");
- attr_dev(input, "min", "0");
- add_location(input, file$9, 382, 5, 7623);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Retry);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_5*/ ctx[20]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && to_number(input.value) !== /*activeZone*/ ctx[2].SOA.Value.Retry) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Retry);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_2$1.name,
- type: "slot",
- source: "(374:4) <WuiLabelHint title=\\\"Retry\\\" title_width=\\\"150px\\\" info=\\\"Number of seconds after which secondary name servers should retry to request the serial number from the master if the master does not respond. It must be less than Refresh. Recommendation for small and stable zones is 7200 seconds (2 hours).\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (389:4) <WuiLabelHint title="Expire" title_width="150px" info="Number of seconds after which secondary name servers should stop answering request for this zone if the master does not respond. This value must be bigger than the sum of Refresh and Retry. Recommendation for small and stable zones is 3600000 seconds (1000 hours)." >
- function create_default_slot_1$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "type", "number");
- attr_dev(input, "min", "0");
- add_location(input, file$9, 396, 5, 8084);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Expire);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_6*/ ctx[21]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && to_number(input.value) !== /*activeZone*/ ctx[2].SOA.Value.Expire) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Expire);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot_1$1.name,
- type: "slot",
- source: "(389:4) <WuiLabelHint title=\\\"Expire\\\" title_width=\\\"150px\\\" info=\\\"Number of seconds after which secondary name servers should stop answering request for this zone if the master does not respond. This value must be bigger than the sum of Refresh and Retry. Recommendation for small and stable zones is 3600000 seconds (1000 hours).\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (403:4) <WuiLabelHint title="Minimum" title_width="150px" info="Time to live for purposes of negative caching. Recommendation for small and stable zones is 1800 seconds (30 min)." >
- function create_default_slot$1(ctx) {
- let input;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- input = element("input");
- attr_dev(input, "type", "number");
- attr_dev(input, "min", "0");
- add_location(input, file$9, 408, 5, 8397);
- },
- m: function mount(target, anchor) {
- insert_dev(target, input, anchor);
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Minimum);
-
- if (!mounted) {
- dispose = listen_dev(input, "input", /*input_input_handler_7*/ ctx[22]);
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*activeZone*/ 4 && to_number(input.value) !== /*activeZone*/ ctx[2].SOA.Value.Minimum) {
- set_input_value(input, /*activeZone*/ ctx[2].SOA.Value.Minimum);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(input);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_default_slot$1.name,
- type: "slot",
- source: "(403:4) <WuiLabelHint title=\\\"Minimum\\\" title_width=\\\"150px\\\" info=\\\"Time to live for purposes of negative caching. Recommendation for small and stable zones is 1800 seconds (30 min).\\\" >",
- ctx
- });
-
- return block;
- }
-
- // (436:4) {#each listRR as rr, idx}
- function create_each_block_2$1(ctx) {
- let div;
- let span0;
- let t0_value = /*rr*/ ctx[40].Name + "";
- let t0;
- let t1;
- let span1;
- let t2_value = /*getTypeName*/ ctx[14](/*rr*/ ctx[40].Type) + "";
- let t2;
- let t3;
- let span2;
- let t4_value = /*rr*/ ctx[40].Value + "";
- let t4;
- let t5;
- let button;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- div = element("div");
- span0 = element("span");
- t0 = text(t0_value);
- t1 = space();
- span1 = element("span");
- t2 = text(t2_value);
- t3 = space();
- span2 = element("span");
- t4 = text(t4_value);
- t5 = space();
- button = element("button");
- button.textContent = "X";
- attr_dev(span0, "class", "name svelte-yu4l71");
- add_location(span0, file$9, 437, 6, 8950);
- attr_dev(span1, "class", "type svelte-yu4l71");
- add_location(span1, file$9, 440, 6, 9007);
- attr_dev(span2, "class", "value svelte-yu4l71");
- add_location(span2, file$9, 443, 6, 9077);
- add_location(button, file$9, 446, 6, 9136);
- attr_dev(div, "class", "rr svelte-yu4l71");
- add_location(div, file$9, 436, 5, 8927);
- },
- m: function mount(target, anchor) {
- insert_dev(target, div, anchor);
- append_dev(div, span0);
- append_dev(span0, t0);
- append_dev(div, t1);
- append_dev(div, span1);
- append_dev(span1, t2);
- append_dev(div, t3);
- append_dev(div, span2);
- append_dev(span2, t4);
- append_dev(div, t5);
- append_dev(div, button);
-
- if (!mounted) {
- dispose = listen_dev(
- button,
- "click",
- function () {
- if (is_function(/*handleDeleteRR*/ ctx[13](/*rr*/ ctx[40], /*idx*/ ctx[42]))) /*handleDeleteRR*/ ctx[13](/*rr*/ ctx[40], /*idx*/ ctx[42]).apply(this, arguments);
- },
- false,
- false,
- false
- );
-
- mounted = true;
- }
- },
- p: function update(new_ctx, dirty) {
- ctx = new_ctx;
- if (dirty[0] & /*activeZone*/ 4 && t0_value !== (t0_value = /*rr*/ ctx[40].Name + "")) set_data_dev(t0, t0_value);
- if (dirty[0] & /*activeZone*/ 4 && t2_value !== (t2_value = /*getTypeName*/ ctx[14](/*rr*/ ctx[40].Type) + "")) set_data_dev(t2, t2_value);
- if (dirty[0] & /*activeZone*/ 4 && t4_value !== (t4_value = /*rr*/ ctx[40].Value + "")) set_data_dev(t4, t4_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div);
- mounted = false;
- dispose();
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_2$1.name,
- type: "each",
- source: "(436:4) {#each listRR as rr, idx}",
- ctx
- });
-
- return block;
- }
-
- // (435:3) {#each Object.entries(activeZone.Records) as [dname, listRR]}
- function create_each_block_1$2(ctx) {
- let each_1_anchor;
- let each_value_2 = /*listRR*/ ctx[37];
- validate_each_argument(each_value_2);
- let each_blocks = [];
-
- for (let i = 0; i < each_value_2.length; i += 1) {
- each_blocks[i] = create_each_block_2$1(get_each_context_2$1(ctx, each_value_2, i));
- }
-
- const block = {
- c: function create() {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- each_1_anchor = empty();
- },
- m: function mount(target, anchor) {
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(target, anchor);
- }
-
- insert_dev(target, each_1_anchor, anchor);
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*handleDeleteRR, activeZone, getTypeName*/ 24580) {
- each_value_2 = /*listRR*/ ctx[37];
- validate_each_argument(each_value_2);
- let i;
-
- for (i = 0; i < each_value_2.length; i += 1) {
- const child_ctx = get_each_context_2$1(ctx, each_value_2, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block_2$1(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(each_1_anchor.parentNode, each_1_anchor);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value_2.length;
- }
- },
- d: function destroy(detaching) {
- destroy_each(each_blocks, detaching);
- if (detaching) detach_dev(each_1_anchor);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block_1$2.name,
- type: "each",
- source: "(435:3) {#each Object.entries(activeZone.Records) as [dname, listRR]}",
- ctx
- });
-
- return block;
- }
-
- // (463:6) {#each Object.entries(RRTypes) as [k, v]}
- function create_each_block$5(ctx) {
- let option;
- let t0_value = /*v*/ ctx[33] + "";
- let t0;
- let t1;
- let option_value_value;
-
- const block = {
- c: function create() {
- option = element("option");
- t0 = text(t0_value);
- t1 = space();
- option.__value = option_value_value = parseInt(/*k*/ ctx[32]);
- option.value = option.__value;
- add_location(option, file$9, 463, 7, 9476);
- },
- m: function mount(target, anchor) {
- insert_dev(target, option, anchor);
- append_dev(option, t0);
- append_dev(option, t1);
- },
- p: noop,
- d: function destroy(detaching) {
- if (detaching) detach_dev(option);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_each_block$5.name,
- type: "each",
- source: "(463:6) {#each Object.entries(RRTypes) as [k, v]}",
- ctx
- });
-
- return block;
- }
-
- // (502:30)
- function create_if_block_3(ctx) {
- let label0;
- let span0;
- let t1;
- let input0;
- let t2;
- let t3_value = /*activeZone*/ ctx[2].Name + "";
- let t3;
- let t4;
- let label1;
- let span1;
- let t6;
- let input1;
- let t7;
- let label2;
- let span2;
- let t9;
- let input2;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- label0 = element("label");
- span0 = element("span");
- span0.textContent = "Name:";
- t1 = space();
- input0 = element("input");
- t2 = text("\n\t\t\t\t\t\t.");
- t3 = text(t3_value);
- t4 = space();
- label1 = element("label");
- span1 = element("span");
- span1.textContent = "Preference:";
- t6 = space();
- input1 = element("input");
- t7 = space();
- label2 = element("label");
- span2 = element("span");
- span2.textContent = "Exchange:";
- t9 = space();
- input2 = element("input");
- attr_dev(span0, "class", "svelte-yu4l71");
- add_location(span0, file$9, 503, 6, 10285);
- attr_dev(input0, "class", "name svelte-yu4l71");
- add_location(input0, file$9, 506, 6, 10325);
- attr_dev(label0, "class", "svelte-yu4l71");
- add_location(label0, file$9, 502, 5, 10271);
- attr_dev(span1, "class", "svelte-yu4l71");
- add_location(span1, file$9, 510, 6, 10426);
- attr_dev(input1, "type", "number");
- attr_dev(input1, "min", "1");
- attr_dev(input1, "max", "65535");
- attr_dev(input1, "class", "svelte-yu4l71");
- add_location(input1, file$9, 513, 6, 10472);
- attr_dev(label1, "class", "svelte-yu4l71");
- add_location(label1, file$9, 509, 5, 10412);
- attr_dev(span2, "class", "svelte-yu4l71");
- add_location(span2, file$9, 516, 6, 10571);
- attr_dev(input2, "class", "svelte-yu4l71");
- add_location(input2, file$9, 519, 6, 10615);
- attr_dev(label2, "class", "svelte-yu4l71");
- add_location(label2, file$9, 515, 5, 10557);
- },
- m: function mount(target, anchor) {
- insert_dev(target, label0, anchor);
- append_dev(label0, span0);
- append_dev(label0, t1);
- append_dev(label0, input0);
- set_input_value(input0, /*_rr*/ ctx[3].Name);
- append_dev(label0, t2);
- append_dev(label0, t3);
- insert_dev(target, t4, anchor);
- insert_dev(target, label1, anchor);
- append_dev(label1, span1);
- append_dev(label1, t6);
- append_dev(label1, input1);
- set_input_value(input1, /*_rrMX*/ ctx[4].Preference);
- insert_dev(target, t7, anchor);
- insert_dev(target, label2, anchor);
- append_dev(label2, span2);
- append_dev(label2, t9);
- append_dev(label2, input2);
- set_input_value(input2, /*_rrMX*/ ctx[4].Exchange);
-
- if (!mounted) {
- dispose = [
- listen_dev(input0, "input", /*input0_input_handler_2*/ ctx[28]),
- listen_dev(input1, "input", /*input1_input_handler_2*/ ctx[29]),
- listen_dev(input2, "input", /*input2_input_handler*/ ctx[30])
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*_rr, RRTypes*/ 40 && input0.value !== /*_rr*/ ctx[3].Name) {
- set_input_value(input0, /*_rr*/ ctx[3].Name);
- }
-
- if (dirty[0] & /*activeZone*/ 4 && t3_value !== (t3_value = /*activeZone*/ ctx[2].Name + "")) set_data_dev(t3, t3_value);
-
- if (dirty[0] & /*_rrMX*/ 16 && to_number(input1.value) !== /*_rrMX*/ ctx[4].Preference) {
- set_input_value(input1, /*_rrMX*/ ctx[4].Preference);
- }
-
- if (dirty[0] & /*_rrMX*/ 16 && input2.value !== /*_rrMX*/ ctx[4].Exchange) {
- set_input_value(input2, /*_rrMX*/ ctx[4].Exchange);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(label0);
- if (detaching) detach_dev(t4);
- if (detaching) detach_dev(label1);
- if (detaching) detach_dev(t7);
- if (detaching) detach_dev(label2);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_3.name,
- type: "if",
- source: "(502:30) ",
- ctx
- });
-
- return block;
- }
-
- // (488:30)
- function create_if_block_2$1(ctx) {
- let label0;
- let span0;
- let t1;
- let input0;
- let t2;
- let label1;
- let span1;
- let t4;
- let input1;
- let t5;
- let t6_value = /*activeZone*/ ctx[2].Name + "";
- let t6;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- label0 = element("label");
- span0 = element("span");
- span0.textContent = "Name:";
- t1 = space();
- input0 = element("input");
- t2 = space();
- label1 = element("label");
- span1 = element("span");
- span1.textContent = "Value:";
- t4 = space();
- input1 = element("input");
- t5 = text("\n\t\t\t\t\t\t.");
- t6 = text(t6_value);
- attr_dev(span0, "class", "svelte-yu4l71");
- add_location(span0, file$9, 489, 6, 9996);
- attr_dev(input0, "class", "svelte-yu4l71");
- add_location(input0, file$9, 492, 6, 10036);
- attr_dev(label0, "class", "svelte-yu4l71");
- add_location(label0, file$9, 488, 5, 9982);
- attr_dev(span1, "class", "svelte-yu4l71");
- add_location(span1, file$9, 495, 6, 10099);
- attr_dev(input1, "class", "name svelte-yu4l71");
- add_location(input1, file$9, 498, 6, 10140);
- attr_dev(label1, "class", "svelte-yu4l71");
- add_location(label1, file$9, 494, 5, 10085);
- },
- m: function mount(target, anchor) {
- insert_dev(target, label0, anchor);
- append_dev(label0, span0);
- append_dev(label0, t1);
- append_dev(label0, input0);
- set_input_value(input0, /*_rr*/ ctx[3].Name);
- insert_dev(target, t2, anchor);
- insert_dev(target, label1, anchor);
- append_dev(label1, span1);
- append_dev(label1, t4);
- append_dev(label1, input1);
- set_input_value(input1, /*_rr*/ ctx[3].Value);
- append_dev(label1, t5);
- append_dev(label1, t6);
-
- if (!mounted) {
- dispose = [
- listen_dev(input0, "input", /*input0_input_handler_1*/ ctx[26]),
- listen_dev(input1, "input", /*input1_input_handler_1*/ ctx[27])
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*_rr, RRTypes*/ 40 && input0.value !== /*_rr*/ ctx[3].Name) {
- set_input_value(input0, /*_rr*/ ctx[3].Name);
- }
-
- if (dirty[0] & /*_rr, RRTypes*/ 40 && input1.value !== /*_rr*/ ctx[3].Value) {
- set_input_value(input1, /*_rr*/ ctx[3].Value);
- }
-
- if (dirty[0] & /*activeZone*/ 4 && t6_value !== (t6_value = /*activeZone*/ ctx[2].Name + "")) set_data_dev(t6, t6_value);
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(label0);
- if (detaching) detach_dev(t2);
- if (detaching) detach_dev(label1);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_2$1.name,
- type: "if",
- source: "(488:30) ",
- ctx
- });
-
- return block;
- }
-
- // (471:4) {#if _rr.Type === 1 || _rr.Type === 2 || _rr.Type === 5 || _rr.Type === 16 || _rr.Type === 28 }
- function create_if_block_1$2(ctx) {
- let label0;
- let span0;
- let t1;
- let input0;
- let t2;
- let t3_value = /*activeZone*/ ctx[2].Name + "";
- let t3;
- let t4;
- let label1;
- let span1;
- let t6;
- let input1;
- let mounted;
- let dispose;
-
- const block = {
- c: function create() {
- label0 = element("label");
- span0 = element("span");
- span0.textContent = "Name:";
- t1 = space();
- input0 = element("input");
- t2 = text("\n\t\t\t\t\t\t.");
- t3 = text(t3_value);
- t4 = space();
- label1 = element("label");
- span1 = element("span");
- span1.textContent = "Value:";
- t6 = space();
- input1 = element("input");
- attr_dev(span0, "class", "svelte-yu4l71");
- add_location(span0, file$9, 474, 6, 9705);
- attr_dev(input0, "class", "name svelte-yu4l71");
- add_location(input0, file$9, 477, 6, 9745);
- attr_dev(label0, "class", "svelte-yu4l71");
- add_location(label0, file$9, 473, 5, 9691);
- attr_dev(span1, "class", "svelte-yu4l71");
- add_location(span1, file$9, 482, 6, 9847);
- attr_dev(input1, "class", "svelte-yu4l71");
- add_location(input1, file$9, 485, 6, 9888);
- attr_dev(label1, "class", "svelte-yu4l71");
- add_location(label1, file$9, 481, 5, 9833);
- },
- m: function mount(target, anchor) {
- insert_dev(target, label0, anchor);
- append_dev(label0, span0);
- append_dev(label0, t1);
- append_dev(label0, input0);
- set_input_value(input0, /*_rr*/ ctx[3].Name);
- append_dev(label0, t2);
- append_dev(label0, t3);
- insert_dev(target, t4, anchor);
- insert_dev(target, label1, anchor);
- append_dev(label1, span1);
- append_dev(label1, t6);
- append_dev(label1, input1);
- set_input_value(input1, /*_rr*/ ctx[3].Value);
-
- if (!mounted) {
- dispose = [
- listen_dev(input0, "input", /*input0_input_handler*/ ctx[24]),
- listen_dev(input1, "input", /*input1_input_handler*/ ctx[25])
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*_rr, RRTypes*/ 40 && input0.value !== /*_rr*/ ctx[3].Name) {
- set_input_value(input0, /*_rr*/ ctx[3].Name);
- }
-
- if (dirty[0] & /*activeZone*/ 4 && t3_value !== (t3_value = /*activeZone*/ ctx[2].Name + "")) set_data_dev(t3, t3_value);
-
- if (dirty[0] & /*_rr, RRTypes*/ 40 && input1.value !== /*_rr*/ ctx[3].Value) {
- set_input_value(input1, /*_rr*/ ctx[3].Value);
- }
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(label0);
- if (detaching) detach_dev(t4);
- if (detaching) detach_dev(label1);
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_1$2.name,
- type: "if",
- source: "(471:4) {#if _rr.Type === 1 || _rr.Type === 2 || _rr.Type === 5 || _rr.Type === 16 || _rr.Type === 28 }",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$9(ctx) {
- let div2;
- let div0;
- let t0;
- let br0;
- let t1;
- let label;
- let span;
- let t3;
- let br1;
- let t4;
- let input;
- let t5;
- let button;
- let t7;
- let div1;
- let current_block_type_index;
- let if_block;
- let current;
- let mounted;
- let dispose;
- let each_value_3 = Object.entries(/*env*/ ctx[0].ZoneFiles);
- validate_each_argument(each_value_3);
- let each_blocks = [];
-
- for (let i = 0; i < each_value_3.length; i += 1) {
- each_blocks[i] = create_each_block_3$1(get_each_context_3$1(ctx, each_value_3, i));
- }
-
- const if_block_creators = [create_if_block$5, create_else_block$1];
- const if_blocks = [];
-
- function select_block_type(ctx, dirty) {
- if (/*activeZone*/ ctx[2].Name === "") return 0;
- return 1;
- }
-
- current_block_type_index = select_block_type(ctx);
- if_block = if_blocks[current_block_type_index] = if_block_creators[current_block_type_index](ctx);
-
- const block = {
- c: function create() {
- div2 = element("div");
- div0 = element("div");
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].c();
- }
-
- t0 = space();
- br0 = element("br");
- t1 = space();
- label = element("label");
- span = element("span");
- span.textContent = "New zone file:";
- t3 = space();
- br1 = element("br");
- t4 = space();
- input = element("input");
- t5 = space();
- button = element("button");
- button.textContent = "Create";
- t7 = space();
- div1 = element("div");
- if_block.c();
- add_location(br0, file$9, 288, 2, 5204);
- add_location(span, file$9, 291, 3, 5224);
- add_location(br1, file$9, 292, 3, 5255);
- add_location(input, file$9, 293, 3, 5264);
- add_location(label, file$9, 290, 2, 5213);
- add_location(button, file$9, 295, 2, 5310);
- attr_dev(div0, "class", "nav-left svelte-yu4l71");
- add_location(div0, file$9, 280, 1, 4998);
- attr_dev(div1, "class", "content svelte-yu4l71");
- add_location(div1, file$9, 300, 1, 5383);
- attr_dev(div2, "class", "master_d");
- add_location(div2, file$9, 279, 0, 4974);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- insert_dev(target, div2, anchor);
- append_dev(div2, div0);
-
- for (let i = 0; i < each_blocks.length; i += 1) {
- each_blocks[i].m(div0, null);
- }
-
- append_dev(div0, t0);
- append_dev(div0, br0);
- append_dev(div0, t1);
- append_dev(div0, label);
- append_dev(label, span);
- append_dev(label, t3);
- append_dev(label, br1);
- append_dev(label, t4);
- append_dev(label, input);
- set_input_value(input, /*newZoneFile*/ ctx[1]);
- append_dev(div0, t5);
- append_dev(div0, button);
- append_dev(div2, t7);
- append_dev(div2, div1);
- if_blocks[current_block_type_index].m(div1, null);
- current = true;
-
- if (!mounted) {
- dispose = [
- listen_dev(input, "input", /*input_input_handler*/ ctx[15]),
- listen_dev(button, "click", /*handleZoneFileCreate*/ ctx[7], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, dirty) {
- if (dirty[0] & /*setActiveZone, env*/ 65) {
- each_value_3 = Object.entries(/*env*/ ctx[0].ZoneFiles);
- validate_each_argument(each_value_3);
- let i;
-
- for (i = 0; i < each_value_3.length; i += 1) {
- const child_ctx = get_each_context_3$1(ctx, each_value_3, i);
-
- if (each_blocks[i]) {
- each_blocks[i].p(child_ctx, dirty);
- } else {
- each_blocks[i] = create_each_block_3$1(child_ctx);
- each_blocks[i].c();
- each_blocks[i].m(div0, t0);
- }
- }
-
- for (; i < each_blocks.length; i += 1) {
- each_blocks[i].d(1);
- }
-
- each_blocks.length = each_value_3.length;
- }
-
- if (dirty[0] & /*newZoneFile*/ 2 && input.value !== /*newZoneFile*/ ctx[1]) {
- set_input_value(input, /*newZoneFile*/ ctx[1]);
- }
-
- let previous_block_index = current_block_type_index;
- current_block_type_index = select_block_type(ctx);
-
- if (current_block_type_index === previous_block_index) {
- if_blocks[current_block_type_index].p(ctx, dirty);
- } else {
- group_outros();
-
- transition_out(if_blocks[previous_block_index], 1, 1, () => {
- if_blocks[previous_block_index] = null;
- });
-
- check_outros();
- if_block = if_blocks[current_block_type_index];
-
- if (!if_block) {
- if_block = if_blocks[current_block_type_index] = if_block_creators[current_block_type_index](ctx);
- if_block.c();
- }
-
- transition_in(if_block, 1);
- if_block.m(div1, null);
- }
- },
- i: function intro(local) {
- if (current) return;
- transition_in(if_block);
- current = true;
- },
- o: function outro(local) {
- transition_out(if_block);
- current = false;
- },
- d: function destroy(detaching) {
- if (detaching) detach_dev(div2);
- destroy_each(each_blocks, detaching);
- if_blocks[current_block_type_index].d();
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$9.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const apiMasterd = "/api/master.d/";
-
- function newRR() {
- return { Name: "", Type: 1, Value: "" };
- }
-
- function newMX() {
- return { Preference: 1, Exchange: "" };
- }
-
- function newSOA() {
- return {
- Name: "",
- Type: 6,
- Value: {
- MName: "",
- RName: "",
- Serial: 0,
- Refresh: 0,
- Retry: 0,
- Expire: 0,
- Minimum: 0
- }
- };
- }
-
- function instance$9($$self, $$props, $$invalidate) {
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: [],
- ZoneFiles: {}
- };
-
- let newZoneFile = "";
- let activeZone = { Name: "" };
-
- let RRTypes = {
- 1: "A",
- 2: "NS",
- 5: "CNAME",
- 12: "PTR",
- 15: "MX",
- 16: "TXT",
- 28: "AAAA"
- };
-
- let _rr = newRR();
- let _rrMX = newMX();
-
- const envUnsubscribe = environment.subscribe(value => {
- $$invalidate(0, env = value);
- });
-
- onDestroy(envUnsubscribe);
-
- function setActiveZone(zone) {
- if (zone.SOA === null) {
- zone.SOA = newSOA();
- }
-
- $$invalidate(2, activeZone = zone);
- }
-
- async function handleZoneFileCreate() {
- const res = await fetch(apiMasterd + newZoneFile, { method: "PUT" });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: handleZoneFileCreate: " + resError.message);
- return;
- }
-
- $$invalidate(2, activeZone = await res.json());
- $$invalidate(2, activeZone.SOA = newSOA(), activeZone);
- $$invalidate(0, env.ZoneFiles[activeZone.Name] = activeZone, env);
- WuiPushNotif.Info("The new zone file '" + newZoneFile + "' has been created");
- }
-
- async function handleZoneFileDelete() {
- let api = apiMasterd + activeZone.Name;
- const res = await fetch(api, { method: "DELETE" });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: handleZoneFileDelete: " + resError.message);
- return;
- }
-
- WuiPushNotif.Info("The zone file '" + activeZone.Name + "' has beed deleted");
- delete env.ZoneFiles[activeZone.Name];
- $$invalidate(2, activeZone = { Name: "" });
- $$invalidate(0, env);
- }
-
- function onSelectRRType() {
- switch (_rr.Type) {
- case 15:
- $$invalidate(4, _rrMX = newMX());
- break;
- }
- }
-
- async function handleSOADelete() {
- return handleDeleteRR(activeZone.SOA, -1);
- }
-
- async function handleSOASave() {
- $$invalidate(3, _rr = activeZone.SOA);
- return handleCreateRR();
- }
-
- async function handleCreateRR() {
- switch (_rr.Type) {
- case 15:
- $$invalidate(3, _rr.Value = _rrMX, _rr);
- break;
- }
-
- let api = apiMasterd + activeZone.Name + "/rr/" + _rr.Type;
-
- const res = await fetch(api, {
- method: "POST",
- headers: { "Content-Type": "application/json" },
- body: JSON.stringify(_rr)
- });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: handleCreateRR: " + resError.message);
- return;
- }
-
- let resRR = await res.json();
-
- if (resRR.Type === 6) {
- $$invalidate(2, activeZone.SOA = resRR, activeZone);
- WuiPushNotif.Info("SOA record has been saved");
- } else {
- let listRR = activeZone.Records[resRR.Name];
-
- if (typeof listRR === "undefined") {
- listRR = [];
- }
-
- listRR.push(resRR);
- $$invalidate(2, activeZone.Records[resRR.Name] = listRR, activeZone);
- WuiPushNotif.Info("The new record '" + resRR.Name + "' has been created");
- }
- }
-
- async function handleDeleteRR(rr, idx) {
- let api = apiMasterd + activeZone.Name + "/rr/" + rr.Type;
-
- const res = await fetch(api, {
- method: "DELETE",
- headers: { "Content-Type": "application/json" },
- body: JSON.stringify(rr)
- });
-
- if (res.status >= 400) {
- const resError = await res.json();
- WuiPushNotif.Error("ERROR: handleDeleteRR: " + resError.message);
- return;
- }
-
- WuiPushNotif.Info("The record '" + rr.Name + "' has been deleted");
-
- if (rr.Type == 6) {
- // SOA.
- $$invalidate(2, activeZone.SOA = newSOA(), activeZone);
- } else {
- let listRR = activeZone.Records[rr.Name];
- listRR.splice(idx, 1);
- $$invalidate(2, activeZone.Records[rr.Name] = listRR, activeZone);
- }
-
- let resbody = await res.json();
- }
-
- function getTypeName(k) {
- let v = RRTypes[k];
-
- if (v === "") {
- return k;
- }
-
- return v;
- }
-
- const writable_props = [];
-
- Object_1$2.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<MasterDir> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("MasterDir", $$slots, []);
-
- function input_input_handler() {
- newZoneFile = this.value;
- $$invalidate(1, newZoneFile);
- }
-
- function input_input_handler_1() {
- activeZone.SOA.Value.MName = this.value;
- $$invalidate(2, activeZone);
- }
-
- function input_input_handler_2() {
- activeZone.SOA.Value.RName = this.value;
- $$invalidate(2, activeZone);
- }
-
- function input_input_handler_3() {
- activeZone.SOA.Value.Serial = to_number(this.value);
- $$invalidate(2, activeZone);
- }
-
- function input_input_handler_4() {
- activeZone.SOA.Value.Refresh = to_number(this.value);
- $$invalidate(2, activeZone);
- }
-
- function input_input_handler_5() {
- activeZone.SOA.Value.Retry = to_number(this.value);
- $$invalidate(2, activeZone);
- }
-
- function input_input_handler_6() {
- activeZone.SOA.Value.Expire = to_number(this.value);
- $$invalidate(2, activeZone);
- }
-
- function input_input_handler_7() {
- activeZone.SOA.Value.Minimum = to_number(this.value);
- $$invalidate(2, activeZone);
- }
-
- function select_change_handler() {
- _rr.Type = select_value(this);
- $$invalidate(3, _rr);
- $$invalidate(5, RRTypes);
- }
-
- function input0_input_handler() {
- _rr.Name = this.value;
- $$invalidate(3, _rr);
- $$invalidate(5, RRTypes);
- }
-
- function input1_input_handler() {
- _rr.Value = this.value;
- $$invalidate(3, _rr);
- $$invalidate(5, RRTypes);
- }
-
- function input0_input_handler_1() {
- _rr.Name = this.value;
- $$invalidate(3, _rr);
- $$invalidate(5, RRTypes);
- }
-
- function input1_input_handler_1() {
- _rr.Value = this.value;
- $$invalidate(3, _rr);
- $$invalidate(5, RRTypes);
- }
-
- function input0_input_handler_2() {
- _rr.Name = this.value;
- $$invalidate(3, _rr);
- $$invalidate(5, RRTypes);
- }
-
- function input1_input_handler_2() {
- _rrMX.Preference = to_number(this.value);
- $$invalidate(4, _rrMX);
- }
-
- function input2_input_handler() {
- _rrMX.Exchange = this.value;
- $$invalidate(4, _rrMX);
- }
-
- $$self.$capture_state = () => ({
- onDestroy,
- WuiPushNotif,
- WuiLabelHint: LabelHint,
- environment,
- nanoSeconds,
- setEnvironment,
- apiMasterd,
- env,
- newZoneFile,
- activeZone,
- RRTypes,
- _rr,
- _rrMX,
- envUnsubscribe,
- setActiveZone,
- handleZoneFileCreate,
- handleZoneFileDelete,
- onSelectRRType,
- handleSOADelete,
- handleSOASave,
- handleCreateRR,
- handleDeleteRR,
- getTypeName,
- newRR,
- newMX,
- newSOA
- });
-
- $$self.$inject_state = $$props => {
- if ("env" in $$props) $$invalidate(0, env = $$props.env);
- if ("newZoneFile" in $$props) $$invalidate(1, newZoneFile = $$props.newZoneFile);
- if ("activeZone" in $$props) $$invalidate(2, activeZone = $$props.activeZone);
- if ("RRTypes" in $$props) $$invalidate(5, RRTypes = $$props.RRTypes);
- if ("_rr" in $$props) $$invalidate(3, _rr = $$props._rr);
- if ("_rrMX" in $$props) $$invalidate(4, _rrMX = $$props._rrMX);
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [
- env,
- newZoneFile,
- activeZone,
- _rr,
- _rrMX,
- RRTypes,
- setActiveZone,
- handleZoneFileCreate,
- handleZoneFileDelete,
- onSelectRRType,
- handleSOADelete,
- handleSOASave,
- handleCreateRR,
- handleDeleteRR,
- getTypeName,
- input_input_handler,
- input_input_handler_1,
- input_input_handler_2,
- input_input_handler_3,
- input_input_handler_4,
- input_input_handler_5,
- input_input_handler_6,
- input_input_handler_7,
- select_change_handler,
- input0_input_handler,
- input1_input_handler,
- input0_input_handler_1,
- input1_input_handler_1,
- input0_input_handler_2,
- input1_input_handler_2,
- input2_input_handler
- ];
- }
-
- class MasterDir extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$9, create_fragment$9, safe_not_equal, {}, [-1, -1]);
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "MasterDir",
- options,
- id: create_fragment$9.name
- });
- }
- }
-
- /* src/App.svelte generated by Svelte v3.24.1 */
- const file$a = "src/App.svelte";
-
- // (113:1) {:else}
- function create_else_block$2(ctx) {
- let dashboard;
- let current;
- dashboard = new Dashboard({ $$inline: true });
-
- const block = {
- c: function create() {
- create_component(dashboard.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(dashboard, target, anchor);
- current = true;
- },
- i: function intro(local) {
- if (current) return;
- transition_in(dashboard.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(dashboard.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(dashboard, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_else_block$2.name,
- type: "else",
- source: "(113:1) {:else}",
- ctx
- });
-
- return block;
- }
-
- // (111:36)
- function create_if_block_3$1(ctx) {
- let masterdir;
- let current;
- masterdir = new MasterDir({ $$inline: true });
-
- const block = {
- c: function create() {
- create_component(masterdir.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(masterdir, target, anchor);
- current = true;
- },
- i: function intro(local) {
- if (current) return;
- transition_in(masterdir.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(masterdir.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(masterdir, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_3$1.name,
- type: "if",
- source: "(111:36) ",
- ctx
- });
-
- return block;
- }
-
- // (109:35)
- function create_if_block_2$2(ctx) {
- let hostsdir;
- let current;
- hostsdir = new HostsDir({ $$inline: true });
-
- const block = {
- c: function create() {
- create_component(hostsdir.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(hostsdir, target, anchor);
- current = true;
- },
- i: function intro(local) {
- if (current) return;
- transition_in(hostsdir.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(hostsdir.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(hostsdir, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_2$2.name,
- type: "if",
- source: "(109:35) ",
- ctx
- });
-
- return block;
- }
-
- // (107:37)
- function create_if_block_1$3(ctx) {
- let hostsblock;
- let current;
- hostsblock = new HostsBlock({ $$inline: true });
-
- const block = {
- c: function create() {
- create_component(hostsblock.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(hostsblock, target, anchor);
- current = true;
- },
- i: function intro(local) {
- if (current) return;
- transition_in(hostsblock.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(hostsblock.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(hostsblock, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block_1$3.name,
- type: "if",
- source: "(107:37) ",
- ctx
- });
-
- return block;
- }
-
- // (105:1) {#if state === stateEnvironment}
- function create_if_block$6(ctx) {
- let environment_1;
- let current;
- environment_1 = new Environment({ $$inline: true });
-
- const block = {
- c: function create() {
- create_component(environment_1.$$.fragment);
- },
- m: function mount(target, anchor) {
- mount_component(environment_1, target, anchor);
- current = true;
- },
- i: function intro(local) {
- if (current) return;
- transition_in(environment_1.$$.fragment, local);
- current = true;
- },
- o: function outro(local) {
- transition_out(environment_1.$$.fragment, local);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(environment_1, detaching);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_if_block$6.name,
- type: "if",
- source: "(105:1) {#if state === stateEnvironment}",
- ctx
- });
-
- return block;
- }
-
- function create_fragment$a(ctx) {
- let wuinotif;
- let t0;
- let div;
- let nav;
- let a0;
- let t2;
- let a1;
- let t4;
- let a2;
- let t5;
- let a2_href_value;
- let t6;
- let a3;
- let t7;
- let a3_href_value;
- let t8;
- let a4;
- let t9;
- let a4_href_value;
- let t10;
- let current_block_type_index;
- let if_block;
- let current;
- let mounted;
- let dispose;
-
- wuinotif = new Notif({
- props: { timeout: "3000" },
- $$inline: true
- });
-
- const if_block_creators = [
- create_if_block$6,
- create_if_block_1$3,
- create_if_block_2$2,
- create_if_block_3$1,
- create_else_block$2
- ];
-
- const if_blocks = [];
-
- function select_block_type(ctx, dirty) {
- if (/*state*/ ctx[0] === stateEnvironment) return 0;
- if (/*state*/ ctx[0] === stateHostsBlock) return 1;
- if (/*state*/ ctx[0] === stateHostsDir) return 2;
- if (/*state*/ ctx[0] === stateMasterDir) return 3;
- return 4;
- }
-
- current_block_type_index = select_block_type(ctx);
- if_block = if_blocks[current_block_type_index] = if_block_creators[current_block_type_index](ctx);
-
- const block = {
- c: function create() {
- create_component(wuinotif.$$.fragment);
- t0 = space();
- div = element("div");
- nav = element("nav");
- a0 = element("a");
- a0.textContent = "rescached";
- t2 = text("\n\t\t/\n\t\t");
- a1 = element("a");
- a1.textContent = "Environment";
- t4 = text("\n\t\t/\n\t\t");
- a2 = element("a");
- t5 = text("Hosts blocks");
- t6 = text("\n\t\t/\n\t\t");
- a3 = element("a");
- t7 = text("hosts.d");
- t8 = text("\n\t\t/\n\t\t");
- a4 = element("a");
- t9 = text("master.d");
- t10 = space();
- if_block.c();
- attr_dev(a0, "href", "#home");
- attr_dev(a0, "class", "svelte-jyzzth");
- toggle_class(a0, "active", /*state*/ ctx[0] === "" || /*state*/ ctx[0] === "home");
- add_location(a0, file$a, 63, 2, 1357);
- attr_dev(a1, "href", "#environment");
- attr_dev(a1, "class", "svelte-jyzzth");
- toggle_class(a1, "active", /*state*/ ctx[0] === stateEnvironment);
- add_location(a1, file$a, 71, 2, 1480);
- attr_dev(a2, "href", a2_href_value = "#" + stateHostsBlock);
- attr_dev(a2, "class", "svelte-jyzzth");
- toggle_class(a2, "active", /*state*/ ctx[0] === stateHostsBlock);
- add_location(a2, file$a, 79, 2, 1624);
- attr_dev(a3, "href", a3_href_value = "#" + stateHostsDir);
- attr_dev(a3, "class", "svelte-jyzzth");
- toggle_class(a3, "active", /*state*/ ctx[0] === stateHostsDir);
- add_location(a3, file$a, 87, 2, 1773);
- attr_dev(a4, "href", a4_href_value = "#" + stateMasterDir);
- attr_dev(a4, "class", "svelte-jyzzth");
- toggle_class(a4, "active", /*state*/ ctx[0] === stateMasterDir);
- add_location(a4, file$a, 95, 2, 1913);
- attr_dev(nav, "class", "menu svelte-jyzzth");
- add_location(nav, file$a, 62, 1, 1336);
- attr_dev(div, "class", "main svelte-jyzzth");
- add_location(div, file$a, 61, 0, 1316);
- },
- l: function claim(nodes) {
- throw new Error("options.hydrate only works if the component was compiled with the `hydratable: true` option");
- },
- m: function mount(target, anchor) {
- mount_component(wuinotif, target, anchor);
- insert_dev(target, t0, anchor);
- insert_dev(target, div, anchor);
- append_dev(div, nav);
- append_dev(nav, a0);
- append_dev(nav, t2);
- append_dev(nav, a1);
- append_dev(nav, t4);
- append_dev(nav, a2);
- append_dev(a2, t5);
- append_dev(nav, t6);
- append_dev(nav, a3);
- append_dev(a3, t7);
- append_dev(nav, t8);
- append_dev(nav, a4);
- append_dev(a4, t9);
- append_dev(div, t10);
- if_blocks[current_block_type_index].m(div, null);
- current = true;
-
- if (!mounted) {
- dispose = [
- listen_dev(a0, "click", /*click_handler*/ ctx[1], false, false, false),
- listen_dev(a1, "click", /*click_handler_1*/ ctx[2], false, false, false),
- listen_dev(a2, "click", /*click_handler_2*/ ctx[3], false, false, false),
- listen_dev(a3, "click", /*click_handler_3*/ ctx[4], false, false, false),
- listen_dev(a4, "click", /*click_handler_4*/ ctx[5], false, false, false)
- ];
-
- mounted = true;
- }
- },
- p: function update(ctx, [dirty]) {
- if (dirty & /*state*/ 1) {
- toggle_class(a0, "active", /*state*/ ctx[0] === "" || /*state*/ ctx[0] === "home");
- }
-
- if (dirty & /*state, stateEnvironment*/ 1) {
- toggle_class(a1, "active", /*state*/ ctx[0] === stateEnvironment);
- }
-
- if (dirty & /*state, stateHostsBlock*/ 1) {
- toggle_class(a2, "active", /*state*/ ctx[0] === stateHostsBlock);
- }
-
- if (dirty & /*state, stateHostsDir*/ 1) {
- toggle_class(a3, "active", /*state*/ ctx[0] === stateHostsDir);
- }
-
- if (dirty & /*state, stateMasterDir*/ 1) {
- toggle_class(a4, "active", /*state*/ ctx[0] === stateMasterDir);
- }
-
- let previous_block_index = current_block_type_index;
- current_block_type_index = select_block_type(ctx);
-
- if (current_block_type_index !== previous_block_index) {
- group_outros();
-
- transition_out(if_blocks[previous_block_index], 1, 1, () => {
- if_blocks[previous_block_index] = null;
- });
-
- check_outros();
- if_block = if_blocks[current_block_type_index];
-
- if (!if_block) {
- if_block = if_blocks[current_block_type_index] = if_block_creators[current_block_type_index](ctx);
- if_block.c();
- }
-
- transition_in(if_block, 1);
- if_block.m(div, null);
- }
- },
- i: function intro(local) {
- if (current) return;
- transition_in(wuinotif.$$.fragment, local);
- transition_in(if_block);
- current = true;
- },
- o: function outro(local) {
- transition_out(wuinotif.$$.fragment, local);
- transition_out(if_block);
- current = false;
- },
- d: function destroy(detaching) {
- destroy_component(wuinotif, detaching);
- if (detaching) detach_dev(t0);
- if (detaching) detach_dev(div);
- if_blocks[current_block_type_index].d();
- mounted = false;
- run_all(dispose);
- }
- };
-
- dispatch_dev("SvelteRegisterBlock", {
- block,
- id: create_fragment$a.name,
- type: "component",
- source: "",
- ctx
- });
-
- return block;
- }
-
- const stateEnvironment = "environment";
- const stateHostsBlock = "hosts_block";
- const stateHostsDir = "hosts_d";
- const stateMasterDir = "master_d";
-
- function instance$a($$self, $$props, $$invalidate) {
- let state;
-
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {}
- };
-
- onMount(async () => {
- const res = await fetch(apiEnvironment);
-
- if (res.status >= 400) {
- WuiPushNotif.Error("ERROR: {apiEnvironment}: ", res.status, res.statusText);
- return;
- }
-
- setEnvironment(await res.json());
- $$invalidate(0, state = window.location.hash.slice(1));
- });
-
- const writable_props = [];
-
- Object.keys($$props).forEach(key => {
- if (!~writable_props.indexOf(key) && key.slice(0, 2) !== "$$") console.warn(`<App> was created with unknown prop '${key}'`);
- });
-
- let { $$slots = {}, $$scope } = $$props;
- validate_slots("App", $$slots, []);
- const click_handler = () => $$invalidate(0, state = "");
- const click_handler_1 = () => $$invalidate(0, state = stateEnvironment);
- const click_handler_2 = () => $$invalidate(0, state = stateHostsBlock);
- const click_handler_3 = () => $$invalidate(0, state = stateHostsDir);
- const click_handler_4 = () => $$invalidate(0, state = stateMasterDir);
-
- $$self.$capture_state = () => ({
- onMount,
- WuiNotif: Notif,
- WuiPushNotif,
- apiEnvironment,
- environment,
- nanoSeconds,
- setEnvironment,
- Dashboard,
- Environment,
- HostsBlock,
- HostsDir,
- MasterDir,
- stateEnvironment,
- stateHostsBlock,
- stateHostsDir,
- stateMasterDir,
- state,
- env
- });
-
- $$self.$inject_state = $$props => {
- if ("state" in $$props) $$invalidate(0, state = $$props.state);
- if ("env" in $$props) env = $$props.env;
- };
-
- if ($$props && "$$inject" in $$props) {
- $$self.$inject_state($$props.$$inject);
- }
-
- return [
- state,
- click_handler,
- click_handler_1,
- click_handler_2,
- click_handler_3,
- click_handler_4
- ];
- }
-
- class App extends SvelteComponentDev {
- constructor(options) {
- super(options);
- init(this, options, instance$a, create_fragment$a, safe_not_equal, {});
-
- dispatch_dev("SvelteRegisterComponent", {
- component: this,
- tagName: "App",
- options,
- id: create_fragment$a.name
- });
- }
- }
-
- const app = new App({
- target: document.body,
- });
-
- return app;
-
-}());
-//# sourceMappingURL=bundle.js.map
diff --git a/_www/public/build/bundle.js.map b/_www/public/build/bundle.js.map
deleted file mode 100644
index 408e83e..0000000
--- a/_www/public/build/bundle.js.map
+++ /dev/null
@@ -1 +0,0 @@
-{"version":3,"file":"bundle.js","sources":["../../node_modules/svelte/internal/index.mjs","../../node_modules/wui.svelte/src/components/InputIPPort.svelte","../../node_modules/wui.svelte/src/components/InputNumber.svelte","../../node_modules/wui.svelte/src/components/LabelHint.svelte","../../node_modules/svelte/transition/index.mjs","../../node_modules/svelte/store/index.mjs","../../node_modules/wui.svelte/src/components/Notif.store.js","../../node_modules/wui.svelte/src/components/NotifItem.svelte","../../node_modules/wui.svelte/src/components/Notif.svelte","../../src/environment.js","../../src/common.js","../../src/Dashboard.svelte","../../src/Environment.svelte","../../src/HostsBlock.svelte","../../src/HostsDir.svelte","../../src/MasterDir.svelte","../../src/App.svelte","../../src/main.js"],"sourcesContent":["function noop() { }\nconst identity = x => x;\nfunction assign(tar, src) {\n // @ts-ignore\n for (const k in src)\n tar[k] = src[k];\n return tar;\n}\nfunction is_promise(value) {\n return value && typeof value === 'object' && typeof value.then === 'function';\n}\nfunction add_location(element, file, line, column, char) {\n element.__svelte_meta = {\n loc: { file, line, column, char }\n };\n}\nfunction run(fn) {\n return fn();\n}\nfunction blank_object() {\n return Object.create(null);\n}\nfunction run_all(fns) {\n fns.forEach(run);\n}\nfunction is_function(thing) {\n return typeof thing === 'function';\n}\nfunction safe_not_equal(a, b) {\n return a != a ? b == b : a !== b || ((a && typeof a === 'object') || typeof a === 'function');\n}\nfunction not_equal(a, b) {\n return a != a ? b == b : a !== b;\n}\nfunction is_empty(obj) {\n return Object.keys(obj).length === 0;\n}\nfunction validate_store(store, name) {\n if (store != null && typeof store.subscribe !== 'function') {\n throw new Error(`'${name}' is not a store with a 'subscribe' method`);\n }\n}\nfunction subscribe(store, ...callbacks) {\n if (store == null) {\n return noop;\n }\n const unsub = store.subscribe(...callbacks);\n return unsub.unsubscribe ? () => unsub.unsubscribe() : unsub;\n}\nfunction get_store_value(store) {\n let value;\n subscribe(store, _ => value = _)();\n return value;\n}\nfunction component_subscribe(component, store, callback) {\n component.$$.on_destroy.push(subscribe(store, callback));\n}\nfunction create_slot(definition, ctx, $$scope, fn) {\n if (definition) {\n const slot_ctx = get_slot_context(definition, ctx, $$scope, fn);\n return definition[0](slot_ctx);\n }\n}\nfunction get_slot_context(definition, ctx, $$scope, fn) {\n return definition[1] && fn\n ? assign($$scope.ctx.slice(), definition[1](fn(ctx)))\n : $$scope.ctx;\n}\nfunction get_slot_changes(definition, $$scope, dirty, fn) {\n if (definition[2] && fn) {\n const lets = definition[2](fn(dirty));\n if ($$scope.dirty === undefined) {\n return lets;\n }\n if (typeof lets === 'object') {\n const merged = [];\n const len = Math.max($$scope.dirty.length, lets.length);\n for (let i = 0; i < len; i += 1) {\n merged[i] = $$scope.dirty[i] | lets[i];\n }\n return merged;\n }\n return $$scope.dirty | lets;\n }\n return $$scope.dirty;\n}\nfunction update_slot(slot, slot_definition, ctx, $$scope, dirty, get_slot_changes_fn, get_slot_context_fn) {\n const slot_changes = get_slot_changes(slot_definition, $$scope, dirty, get_slot_changes_fn);\n if (slot_changes) {\n const slot_context = get_slot_context(slot_definition, ctx, $$scope, get_slot_context_fn);\n slot.p(slot_context, slot_changes);\n }\n}\nfunction exclude_internal_props(props) {\n const result = {};\n for (const k in props)\n if (k[0] !== '$')\n result[k] = props[k];\n return result;\n}\nfunction compute_rest_props(props, keys) {\n const rest = {};\n keys = new Set(keys);\n for (const k in props)\n if (!keys.has(k) && k[0] !== '$')\n rest[k] = props[k];\n return rest;\n}\nfunction once(fn) {\n let ran = false;\n return function (...args) {\n if (ran)\n return;\n ran = true;\n fn.call(this, ...args);\n };\n}\nfunction null_to_empty(value) {\n return value == null ? '' : value;\n}\nfunction set_store_value(store, ret, value = ret) {\n store.set(value);\n return ret;\n}\nconst has_prop = (obj, prop) => Object.prototype.hasOwnProperty.call(obj, prop);\nfunction action_destroyer(action_result) {\n return action_result && is_function(action_result.destroy) ? action_result.destroy : noop;\n}\n\nconst is_client = typeof window !== 'undefined';\nlet now = is_client\n ? () => window.performance.now()\n : () => Date.now();\nlet raf = is_client ? cb => requestAnimationFrame(cb) : noop;\n// used internally for testing\nfunction set_now(fn) {\n now = fn;\n}\nfunction set_raf(fn) {\n raf = fn;\n}\n\nconst tasks = new Set();\nfunction run_tasks(now) {\n tasks.forEach(task => {\n if (!task.c(now)) {\n tasks.delete(task);\n task.f();\n }\n });\n if (tasks.size !== 0)\n raf(run_tasks);\n}\n/**\n * For testing purposes only!\n */\nfunction clear_loops() {\n tasks.clear();\n}\n/**\n * Creates a new task that runs on each raf frame\n * until it returns a falsy value or is aborted\n */\nfunction loop(callback) {\n let task;\n if (tasks.size === 0)\n raf(run_tasks);\n return {\n promise: new Promise(fulfill => {\n tasks.add(task = { c: callback, f: fulfill });\n }),\n abort() {\n tasks.delete(task);\n }\n };\n}\n\nfunction append(target, node) {\n target.appendChild(node);\n}\nfunction insert(target, node, anchor) {\n target.insertBefore(node, anchor || null);\n}\nfunction detach(node) {\n node.parentNode.removeChild(node);\n}\nfunction destroy_each(iterations, detaching) {\n for (let i = 0; i < iterations.length; i += 1) {\n if (iterations[i])\n iterations[i].d(detaching);\n }\n}\nfunction element(name) {\n return document.createElement(name);\n}\nfunction element_is(name, is) {\n return document.createElement(name, { is });\n}\nfunction object_without_properties(obj, exclude) {\n const target = {};\n for (const k in obj) {\n if (has_prop(obj, k)\n // @ts-ignore\n && exclude.indexOf(k) === -1) {\n // @ts-ignore\n target[k] = obj[k];\n }\n }\n return target;\n}\nfunction svg_element(name) {\n return document.createElementNS('http://www.w3.org/2000/svg', name);\n}\nfunction text(data) {\n return document.createTextNode(data);\n}\nfunction space() {\n return text(' ');\n}\nfunction empty() {\n return text('');\n}\nfunction listen(node, event, handler, options) {\n node.addEventListener(event, handler, options);\n return () => node.removeEventListener(event, handler, options);\n}\nfunction prevent_default(fn) {\n return function (event) {\n event.preventDefault();\n // @ts-ignore\n return fn.call(this, event);\n };\n}\nfunction stop_propagation(fn) {\n return function (event) {\n event.stopPropagation();\n // @ts-ignore\n return fn.call(this, event);\n };\n}\nfunction self(fn) {\n return function (event) {\n // @ts-ignore\n if (event.target === this)\n fn.call(this, event);\n };\n}\nfunction attr(node, attribute, value) {\n if (value == null)\n node.removeAttribute(attribute);\n else if (node.getAttribute(attribute) !== value)\n node.setAttribute(attribute, value);\n}\nfunction set_attributes(node, attributes) {\n // @ts-ignore\n const descriptors = Object.getOwnPropertyDescriptors(node.__proto__);\n for (const key in attributes) {\n if (attributes[key] == null) {\n node.removeAttribute(key);\n }\n else if (key === 'style') {\n node.style.cssText = attributes[key];\n }\n else if (key === '__value') {\n node.value = node[key] = attributes[key];\n }\n else if (descriptors[key] && descriptors[key].set) {\n node[key] = attributes[key];\n }\n else {\n attr(node, key, attributes[key]);\n }\n }\n}\nfunction set_svg_attributes(node, attributes) {\n for (const key in attributes) {\n attr(node, key, attributes[key]);\n }\n}\nfunction set_custom_element_data(node, prop, value) {\n if (prop in node) {\n node[prop] = value;\n }\n else {\n attr(node, prop, value);\n }\n}\nfunction xlink_attr(node, attribute, value) {\n node.setAttributeNS('http://www.w3.org/1999/xlink', attribute, value);\n}\nfunction get_binding_group_value(group, __value, checked) {\n const value = new Set();\n for (let i = 0; i < group.length; i += 1) {\n if (group[i].checked)\n value.add(group[i].__value);\n }\n if (!checked) {\n value.delete(__value);\n }\n return Array.from(value);\n}\nfunction to_number(value) {\n return value === '' ? undefined : +value;\n}\nfunction time_ranges_to_array(ranges) {\n const array = [];\n for (let i = 0; i < ranges.length; i += 1) {\n array.push({ start: ranges.start(i), end: ranges.end(i) });\n }\n return array;\n}\nfunction children(element) {\n return Array.from(element.childNodes);\n}\nfunction claim_element(nodes, name, attributes, svg) {\n for (let i = 0; i < nodes.length; i += 1) {\n const node = nodes[i];\n if (node.nodeName === name) {\n let j = 0;\n const remove = [];\n while (j < node.attributes.length) {\n const attribute = node.attributes[j++];\n if (!attributes[attribute.name]) {\n remove.push(attribute.name);\n }\n }\n for (let k = 0; k < remove.length; k++) {\n node.removeAttribute(remove[k]);\n }\n return nodes.splice(i, 1)[0];\n }\n }\n return svg ? svg_element(name) : element(name);\n}\nfunction claim_text(nodes, data) {\n for (let i = 0; i < nodes.length; i += 1) {\n const node = nodes[i];\n if (node.nodeType === 3) {\n node.data = '' + data;\n return nodes.splice(i, 1)[0];\n }\n }\n return text(data);\n}\nfunction claim_space(nodes) {\n return claim_text(nodes, ' ');\n}\nfunction set_data(text, data) {\n data = '' + data;\n if (text.wholeText !== data)\n text.data = data;\n}\nfunction set_input_value(input, value) {\n input.value = value == null ? '' : value;\n}\nfunction set_input_type(input, type) {\n try {\n input.type = type;\n }\n catch (e) {\n // do nothing\n }\n}\nfunction set_style(node, key, value, important) {\n node.style.setProperty(key, value, important ? 'important' : '');\n}\nfunction select_option(select, value) {\n for (let i = 0; i < select.options.length; i += 1) {\n const option = select.options[i];\n if (option.__value === value) {\n option.selected = true;\n return;\n }\n }\n}\nfunction select_options(select, value) {\n for (let i = 0; i < select.options.length; i += 1) {\n const option = select.options[i];\n option.selected = ~value.indexOf(option.__value);\n }\n}\nfunction select_value(select) {\n const selected_option = select.querySelector(':checked') || select.options[0];\n return selected_option && selected_option.__value;\n}\nfunction select_multiple_value(select) {\n return [].map.call(select.querySelectorAll(':checked'), option => option.__value);\n}\n// unfortunately this can't be a constant as that wouldn't be tree-shakeable\n// so we cache the result instead\nlet crossorigin;\nfunction is_crossorigin() {\n if (crossorigin === undefined) {\n crossorigin = false;\n try {\n if (typeof window !== 'undefined' && window.parent) {\n void window.parent.document;\n }\n }\n catch (error) {\n crossorigin = true;\n }\n }\n return crossorigin;\n}\nfunction add_resize_listener(node, fn) {\n const computed_style = getComputedStyle(node);\n const z_index = (parseInt(computed_style.zIndex) || 0) - 1;\n if (computed_style.position === 'static') {\n node.style.position = 'relative';\n }\n const iframe = element('iframe');\n iframe.setAttribute('style', `display: block; position: absolute; top: 0; left: 0; width: 100%; height: 100%; ` +\n `overflow: hidden; border: 0; opacity: 0; pointer-events: none; z-index: ${z_index};`);\n iframe.setAttribute('aria-hidden', 'true');\n iframe.tabIndex = -1;\n const crossorigin = is_crossorigin();\n let unsubscribe;\n if (crossorigin) {\n iframe.src = `data:text/html,<script>onresize=function(){parent.postMessage(0,'*')}</script>`;\n unsubscribe = listen(window, 'message', (event) => {\n if (event.source === iframe.contentWindow)\n fn();\n });\n }\n else {\n iframe.src = 'about:blank';\n iframe.onload = () => {\n unsubscribe = listen(iframe.contentWindow, 'resize', fn);\n };\n }\n append(node, iframe);\n return () => {\n if (crossorigin) {\n unsubscribe();\n }\n else if (unsubscribe && iframe.contentWindow) {\n unsubscribe();\n }\n detach(iframe);\n };\n}\nfunction toggle_class(element, name, toggle) {\n element.classList[toggle ? 'add' : 'remove'](name);\n}\nfunction custom_event(type, detail) {\n const e = document.createEvent('CustomEvent');\n e.initCustomEvent(type, false, false, detail);\n return e;\n}\nfunction query_selector_all(selector, parent = document.body) {\n return Array.from(parent.querySelectorAll(selector));\n}\nclass HtmlTag {\n constructor(anchor = null) {\n this.a = anchor;\n this.e = this.n = null;\n }\n m(html, target, anchor = null) {\n if (!this.e) {\n this.e = element(target.nodeName);\n this.t = target;\n this.h(html);\n }\n this.i(anchor);\n }\n h(html) {\n this.e.innerHTML = html;\n this.n = Array.from(this.e.childNodes);\n }\n i(anchor) {\n for (let i = 0; i < this.n.length; i += 1) {\n insert(this.t, this.n[i], anchor);\n }\n }\n p(html) {\n this.d();\n this.h(html);\n this.i(this.a);\n }\n d() {\n this.n.forEach(detach);\n }\n}\n\nconst active_docs = new Set();\nlet active = 0;\n// https://github.com/darkskyapp/string-hash/blob/master/index.js\nfunction hash(str) {\n let hash = 5381;\n let i = str.length;\n while (i--)\n hash = ((hash << 5) - hash) ^ str.charCodeAt(i);\n return hash >>> 0;\n}\nfunction create_rule(node, a, b, duration, delay, ease, fn, uid = 0) {\n const step = 16.666 / duration;\n let keyframes = '{\\n';\n for (let p = 0; p <= 1; p += step) {\n const t = a + (b - a) * ease(p);\n keyframes += p * 100 + `%{${fn(t, 1 - t)}}\\n`;\n }\n const rule = keyframes + `100% {${fn(b, 1 - b)}}\\n}`;\n const name = `__svelte_${hash(rule)}_${uid}`;\n const doc = node.ownerDocument;\n active_docs.add(doc);\n const stylesheet = doc.__svelte_stylesheet || (doc.__svelte_stylesheet = doc.head.appendChild(element('style')).sheet);\n const current_rules = doc.__svelte_rules || (doc.__svelte_rules = {});\n if (!current_rules[name]) {\n current_rules[name] = true;\n stylesheet.insertRule(`@keyframes ${name} ${rule}`, stylesheet.cssRules.length);\n }\n const animation = node.style.animation || '';\n node.style.animation = `${animation ? `${animation}, ` : ``}${name} ${duration}ms linear ${delay}ms 1 both`;\n active += 1;\n return name;\n}\nfunction delete_rule(node, name) {\n const previous = (node.style.animation || '').split(', ');\n const next = previous.filter(name\n ? anim => anim.indexOf(name) < 0 // remove specific animation\n : anim => anim.indexOf('__svelte') === -1 // remove all Svelte animations\n );\n const deleted = previous.length - next.length;\n if (deleted) {\n node.style.animation = next.join(', ');\n active -= deleted;\n if (!active)\n clear_rules();\n }\n}\nfunction clear_rules() {\n raf(() => {\n if (active)\n return;\n active_docs.forEach(doc => {\n const stylesheet = doc.__svelte_stylesheet;\n let i = stylesheet.cssRules.length;\n while (i--)\n stylesheet.deleteRule(i);\n doc.__svelte_rules = {};\n });\n active_docs.clear();\n });\n}\n\nfunction create_animation(node, from, fn, params) {\n if (!from)\n return noop;\n const to = node.getBoundingClientRect();\n if (from.left === to.left && from.right === to.right && from.top === to.top && from.bottom === to.bottom)\n return noop;\n const { delay = 0, duration = 300, easing = identity, \n // @ts-ignore todo: should this be separated from destructuring? Or start/end added to public api and documentation?\n start: start_time = now() + delay, \n // @ts-ignore todo:\n end = start_time + duration, tick = noop, css } = fn(node, { from, to }, params);\n let running = true;\n let started = false;\n let name;\n function start() {\n if (css) {\n name = create_rule(node, 0, 1, duration, delay, easing, css);\n }\n if (!delay) {\n started = true;\n }\n }\n function stop() {\n if (css)\n delete_rule(node, name);\n running = false;\n }\n loop(now => {\n if (!started && now >= start_time) {\n started = true;\n }\n if (started && now >= end) {\n tick(1, 0);\n stop();\n }\n if (!running) {\n return false;\n }\n if (started) {\n const p = now - start_time;\n const t = 0 + 1 * easing(p / duration);\n tick(t, 1 - t);\n }\n return true;\n });\n start();\n tick(0, 1);\n return stop;\n}\nfunction fix_position(node) {\n const style = getComputedStyle(node);\n if (style.position !== 'absolute' && style.position !== 'fixed') {\n const { width, height } = style;\n const a = node.getBoundingClientRect();\n node.style.position = 'absolute';\n node.style.width = width;\n node.style.height = height;\n add_transform(node, a);\n }\n}\nfunction add_transform(node, a) {\n const b = node.getBoundingClientRect();\n if (a.left !== b.left || a.top !== b.top) {\n const style = getComputedStyle(node);\n const transform = style.transform === 'none' ? '' : style.transform;\n node.style.transform = `${transform} translate(${a.left - b.left}px, ${a.top - b.top}px)`;\n }\n}\n\nlet current_component;\nfunction set_current_component(component) {\n current_component = component;\n}\nfunction get_current_component() {\n if (!current_component)\n throw new Error(`Function called outside component initialization`);\n return current_component;\n}\nfunction beforeUpdate(fn) {\n get_current_component().$$.before_update.push(fn);\n}\nfunction onMount(fn) {\n get_current_component().$$.on_mount.push(fn);\n}\nfunction afterUpdate(fn) {\n get_current_component().$$.after_update.push(fn);\n}\nfunction onDestroy(fn) {\n get_current_component().$$.on_destroy.push(fn);\n}\nfunction createEventDispatcher() {\n const component = get_current_component();\n return (type, detail) => {\n const callbacks = component.$$.callbacks[type];\n if (callbacks) {\n // TODO are there situations where events could be dispatched\n // in a server (non-DOM) environment?\n const event = custom_event(type, detail);\n callbacks.slice().forEach(fn => {\n fn.call(component, event);\n });\n }\n };\n}\nfunction setContext(key, context) {\n get_current_component().$$.context.set(key, context);\n}\nfunction getContext(key) {\n return get_current_component().$$.context.get(key);\n}\n// TODO figure out if we still want to support\n// shorthand events, or if we want to implement\n// a real bubbling mechanism\nfunction bubble(component, event) {\n const callbacks = component.$$.callbacks[event.type];\n if (callbacks) {\n callbacks.slice().forEach(fn => fn(event));\n }\n}\n\nconst dirty_components = [];\nconst intros = { enabled: false };\nconst binding_callbacks = [];\nconst render_callbacks = [];\nconst flush_callbacks = [];\nconst resolved_promise = Promise.resolve();\nlet update_scheduled = false;\nfunction schedule_update() {\n if (!update_scheduled) {\n update_scheduled = true;\n resolved_promise.then(flush);\n }\n}\nfunction tick() {\n schedule_update();\n return resolved_promise;\n}\nfunction add_render_callback(fn) {\n render_callbacks.push(fn);\n}\nfunction add_flush_callback(fn) {\n flush_callbacks.push(fn);\n}\nlet flushing = false;\nconst seen_callbacks = new Set();\nfunction flush() {\n if (flushing)\n return;\n flushing = true;\n do {\n // first, call beforeUpdate functions\n // and update components\n for (let i = 0; i < dirty_components.length; i += 1) {\n const component = dirty_components[i];\n set_current_component(component);\n update(component.$$);\n }\n dirty_components.length = 0;\n while (binding_callbacks.length)\n binding_callbacks.pop()();\n // then, once components are updated, call\n // afterUpdate functions. This may cause\n // subsequent updates...\n for (let i = 0; i < render_callbacks.length; i += 1) {\n const callback = render_callbacks[i];\n if (!seen_callbacks.has(callback)) {\n // ...so guard against infinite loops\n seen_callbacks.add(callback);\n callback();\n }\n }\n render_callbacks.length = 0;\n } while (dirty_components.length);\n while (flush_callbacks.length) {\n flush_callbacks.pop()();\n }\n update_scheduled = false;\n flushing = false;\n seen_callbacks.clear();\n}\nfunction update($$) {\n if ($$.fragment !== null) {\n $$.update();\n run_all($$.before_update);\n const dirty = $$.dirty;\n $$.dirty = [-1];\n $$.fragment && $$.fragment.p($$.ctx, dirty);\n $$.after_update.forEach(add_render_callback);\n }\n}\n\nlet promise;\nfunction wait() {\n if (!promise) {\n promise = Promise.resolve();\n promise.then(() => {\n promise = null;\n });\n }\n return promise;\n}\nfunction dispatch(node, direction, kind) {\n node.dispatchEvent(custom_event(`${direction ? 'intro' : 'outro'}${kind}`));\n}\nconst outroing = new Set();\nlet outros;\nfunction group_outros() {\n outros = {\n r: 0,\n c: [],\n p: outros // parent group\n };\n}\nfunction check_outros() {\n if (!outros.r) {\n run_all(outros.c);\n }\n outros = outros.p;\n}\nfunction transition_in(block, local) {\n if (block && block.i) {\n outroing.delete(block);\n block.i(local);\n }\n}\nfunction transition_out(block, local, detach, callback) {\n if (block && block.o) {\n if (outroing.has(block))\n return;\n outroing.add(block);\n outros.c.push(() => {\n outroing.delete(block);\n if (callback) {\n if (detach)\n block.d(1);\n callback();\n }\n });\n block.o(local);\n }\n}\nconst null_transition = { duration: 0 };\nfunction create_in_transition(node, fn, params) {\n let config = fn(node, params);\n let running = false;\n let animation_name;\n let task;\n let uid = 0;\n function cleanup() {\n if (animation_name)\n delete_rule(node, animation_name);\n }\n function go() {\n const { delay = 0, duration = 300, easing = identity, tick = noop, css } = config || null_transition;\n if (css)\n animation_name = create_rule(node, 0, 1, duration, delay, easing, css, uid++);\n tick(0, 1);\n const start_time = now() + delay;\n const end_time = start_time + duration;\n if (task)\n task.abort();\n running = true;\n add_render_callback(() => dispatch(node, true, 'start'));\n task = loop(now => {\n if (running) {\n if (now >= end_time) {\n tick(1, 0);\n dispatch(node, true, 'end');\n cleanup();\n return running = false;\n }\n if (now >= start_time) {\n const t = easing((now - start_time) / duration);\n tick(t, 1 - t);\n }\n }\n return running;\n });\n }\n let started = false;\n return {\n start() {\n if (started)\n return;\n delete_rule(node);\n if (is_function(config)) {\n config = config();\n wait().then(go);\n }\n else {\n go();\n }\n },\n invalidate() {\n started = false;\n },\n end() {\n if (running) {\n cleanup();\n running = false;\n }\n }\n };\n}\nfunction create_out_transition(node, fn, params) {\n let config = fn(node, params);\n let running = true;\n let animation_name;\n const group = outros;\n group.r += 1;\n function go() {\n const { delay = 0, duration = 300, easing = identity, tick = noop, css } = config || null_transition;\n if (css)\n animation_name = create_rule(node, 1, 0, duration, delay, easing, css);\n const start_time = now() + delay;\n const end_time = start_time + duration;\n add_render_callback(() => dispatch(node, false, 'start'));\n loop(now => {\n if (running) {\n if (now >= end_time) {\n tick(0, 1);\n dispatch(node, false, 'end');\n if (!--group.r) {\n // this will result in `end()` being called,\n // so we don't need to clean up here\n run_all(group.c);\n }\n return false;\n }\n if (now >= start_time) {\n const t = easing((now - start_time) / duration);\n tick(1 - t, t);\n }\n }\n return running;\n });\n }\n if (is_function(config)) {\n wait().then(() => {\n // @ts-ignore\n config = config();\n go();\n });\n }\n else {\n go();\n }\n return {\n end(reset) {\n if (reset && config.tick) {\n config.tick(1, 0);\n }\n if (running) {\n if (animation_name)\n delete_rule(node, animation_name);\n running = false;\n }\n }\n };\n}\nfunction create_bidirectional_transition(node, fn, params, intro) {\n let config = fn(node, params);\n let t = intro ? 0 : 1;\n let running_program = null;\n let pending_program = null;\n let animation_name = null;\n function clear_animation() {\n if (animation_name)\n delete_rule(node, animation_name);\n }\n function init(program, duration) {\n const d = program.b - t;\n duration *= Math.abs(d);\n return {\n a: t,\n b: program.b,\n d,\n duration,\n start: program.start,\n end: program.start + duration,\n group: program.group\n };\n }\n function go(b) {\n const { delay = 0, duration = 300, easing = identity, tick = noop, css } = config || null_transition;\n const program = {\n start: now() + delay,\n b\n };\n if (!b) {\n // @ts-ignore todo: improve typings\n program.group = outros;\n outros.r += 1;\n }\n if (running_program) {\n pending_program = program;\n }\n else {\n // if this is an intro, and there's a delay, we need to do\n // an initial tick and/or apply CSS animation immediately\n if (css) {\n clear_animation();\n animation_name = create_rule(node, t, b, duration, delay, easing, css);\n }\n if (b)\n tick(0, 1);\n running_program = init(program, duration);\n add_render_callback(() => dispatch(node, b, 'start'));\n loop(now => {\n if (pending_program && now > pending_program.start) {\n running_program = init(pending_program, duration);\n pending_program = null;\n dispatch(node, running_program.b, 'start');\n if (css) {\n clear_animation();\n animation_name = create_rule(node, t, running_program.b, running_program.duration, 0, easing, config.css);\n }\n }\n if (running_program) {\n if (now >= running_program.end) {\n tick(t = running_program.b, 1 - t);\n dispatch(node, running_program.b, 'end');\n if (!pending_program) {\n // we're done\n if (running_program.b) {\n // intro — we can tidy up immediately\n clear_animation();\n }\n else {\n // outro — needs to be coordinated\n if (!--running_program.group.r)\n run_all(running_program.group.c);\n }\n }\n running_program = null;\n }\n else if (now >= running_program.start) {\n const p = now - running_program.start;\n t = running_program.a + running_program.d * easing(p / running_program.duration);\n tick(t, 1 - t);\n }\n }\n return !!(running_program || pending_program);\n });\n }\n }\n return {\n run(b) {\n if (is_function(config)) {\n wait().then(() => {\n // @ts-ignore\n config = config();\n go(b);\n });\n }\n else {\n go(b);\n }\n },\n end() {\n clear_animation();\n running_program = pending_program = null;\n }\n };\n}\n\nfunction handle_promise(promise, info) {\n const token = info.token = {};\n function update(type, index, key, value) {\n if (info.token !== token)\n return;\n info.resolved = value;\n let child_ctx = info.ctx;\n if (key !== undefined) {\n child_ctx = child_ctx.slice();\n child_ctx[key] = value;\n }\n const block = type && (info.current = type)(child_ctx);\n let needs_flush = false;\n if (info.block) {\n if (info.blocks) {\n info.blocks.forEach((block, i) => {\n if (i !== index && block) {\n group_outros();\n transition_out(block, 1, 1, () => {\n info.blocks[i] = null;\n });\n check_outros();\n }\n });\n }\n else {\n info.block.d(1);\n }\n block.c();\n transition_in(block, 1);\n block.m(info.mount(), info.anchor);\n needs_flush = true;\n }\n info.block = block;\n if (info.blocks)\n info.blocks[index] = block;\n if (needs_flush) {\n flush();\n }\n }\n if (is_promise(promise)) {\n const current_component = get_current_component();\n promise.then(value => {\n set_current_component(current_component);\n update(info.then, 1, info.value, value);\n set_current_component(null);\n }, error => {\n set_current_component(current_component);\n update(info.catch, 2, info.error, error);\n set_current_component(null);\n });\n // if we previously had a then/catch block, destroy it\n if (info.current !== info.pending) {\n update(info.pending, 0);\n return true;\n }\n }\n else {\n if (info.current !== info.then) {\n update(info.then, 1, info.value, promise);\n return true;\n }\n info.resolved = promise;\n }\n}\n\nconst globals = (typeof window !== 'undefined'\n ? window\n : typeof globalThis !== 'undefined'\n ? globalThis\n : global);\n\nfunction destroy_block(block, lookup) {\n block.d(1);\n lookup.delete(block.key);\n}\nfunction outro_and_destroy_block(block, lookup) {\n transition_out(block, 1, 1, () => {\n lookup.delete(block.key);\n });\n}\nfunction fix_and_destroy_block(block, lookup) {\n block.f();\n destroy_block(block, lookup);\n}\nfunction fix_and_outro_and_destroy_block(block, lookup) {\n block.f();\n outro_and_destroy_block(block, lookup);\n}\nfunction update_keyed_each(old_blocks, dirty, get_key, dynamic, ctx, list, lookup, node, destroy, create_each_block, next, get_context) {\n let o = old_blocks.length;\n let n = list.length;\n let i = o;\n const old_indexes = {};\n while (i--)\n old_indexes[old_blocks[i].key] = i;\n const new_blocks = [];\n const new_lookup = new Map();\n const deltas = new Map();\n i = n;\n while (i--) {\n const child_ctx = get_context(ctx, list, i);\n const key = get_key(child_ctx);\n let block = lookup.get(key);\n if (!block) {\n block = create_each_block(key, child_ctx);\n block.c();\n }\n else if (dynamic) {\n block.p(child_ctx, dirty);\n }\n new_lookup.set(key, new_blocks[i] = block);\n if (key in old_indexes)\n deltas.set(key, Math.abs(i - old_indexes[key]));\n }\n const will_move = new Set();\n const did_move = new Set();\n function insert(block) {\n transition_in(block, 1);\n block.m(node, next);\n lookup.set(block.key, block);\n next = block.first;\n n--;\n }\n while (o && n) {\n const new_block = new_blocks[n - 1];\n const old_block = old_blocks[o - 1];\n const new_key = new_block.key;\n const old_key = old_block.key;\n if (new_block === old_block) {\n // do nothing\n next = new_block.first;\n o--;\n n--;\n }\n else if (!new_lookup.has(old_key)) {\n // remove old block\n destroy(old_block, lookup);\n o--;\n }\n else if (!lookup.has(new_key) || will_move.has(new_key)) {\n insert(new_block);\n }\n else if (did_move.has(old_key)) {\n o--;\n }\n else if (deltas.get(new_key) > deltas.get(old_key)) {\n did_move.add(new_key);\n insert(new_block);\n }\n else {\n will_move.add(old_key);\n o--;\n }\n }\n while (o--) {\n const old_block = old_blocks[o];\n if (!new_lookup.has(old_block.key))\n destroy(old_block, lookup);\n }\n while (n)\n insert(new_blocks[n - 1]);\n return new_blocks;\n}\nfunction validate_each_keys(ctx, list, get_context, get_key) {\n const keys = new Set();\n for (let i = 0; i < list.length; i++) {\n const key = get_key(get_context(ctx, list, i));\n if (keys.has(key)) {\n throw new Error(`Cannot have duplicate keys in a keyed each`);\n }\n keys.add(key);\n }\n}\n\nfunction get_spread_update(levels, updates) {\n const update = {};\n const to_null_out = {};\n const accounted_for = { $$scope: 1 };\n let i = levels.length;\n while (i--) {\n const o = levels[i];\n const n = updates[i];\n if (n) {\n for (const key in o) {\n if (!(key in n))\n to_null_out[key] = 1;\n }\n for (const key in n) {\n if (!accounted_for[key]) {\n update[key] = n[key];\n accounted_for[key] = 1;\n }\n }\n levels[i] = n;\n }\n else {\n for (const key in o) {\n accounted_for[key] = 1;\n }\n }\n }\n for (const key in to_null_out) {\n if (!(key in update))\n update[key] = undefined;\n }\n return update;\n}\nfunction get_spread_object(spread_props) {\n return typeof spread_props === 'object' && spread_props !== null ? spread_props : {};\n}\n\n// source: https://html.spec.whatwg.org/multipage/indices.html\nconst boolean_attributes = new Set([\n 'allowfullscreen',\n 'allowpaymentrequest',\n 'async',\n 'autofocus',\n 'autoplay',\n 'checked',\n 'controls',\n 'default',\n 'defer',\n 'disabled',\n 'formnovalidate',\n 'hidden',\n 'ismap',\n 'loop',\n 'multiple',\n 'muted',\n 'nomodule',\n 'novalidate',\n 'open',\n 'playsinline',\n 'readonly',\n 'required',\n 'reversed',\n 'selected'\n]);\n\nconst invalid_attribute_name_character = /[\\s'\">/=\\u{FDD0}-\\u{FDEF}\\u{FFFE}\\u{FFFF}\\u{1FFFE}\\u{1FFFF}\\u{2FFFE}\\u{2FFFF}\\u{3FFFE}\\u{3FFFF}\\u{4FFFE}\\u{4FFFF}\\u{5FFFE}\\u{5FFFF}\\u{6FFFE}\\u{6FFFF}\\u{7FFFE}\\u{7FFFF}\\u{8FFFE}\\u{8FFFF}\\u{9FFFE}\\u{9FFFF}\\u{AFFFE}\\u{AFFFF}\\u{BFFFE}\\u{BFFFF}\\u{CFFFE}\\u{CFFFF}\\u{DFFFE}\\u{DFFFF}\\u{EFFFE}\\u{EFFFF}\\u{FFFFE}\\u{FFFFF}\\u{10FFFE}\\u{10FFFF}]/u;\n// https://html.spec.whatwg.org/multipage/syntax.html#attributes-2\n// https://infra.spec.whatwg.org/#noncharacter\nfunction spread(args, classes_to_add) {\n const attributes = Object.assign({}, ...args);\n if (classes_to_add) {\n if (attributes.class == null) {\n attributes.class = classes_to_add;\n }\n else {\n attributes.class += ' ' + classes_to_add;\n }\n }\n let str = '';\n Object.keys(attributes).forEach(name => {\n if (invalid_attribute_name_character.test(name))\n return;\n const value = attributes[name];\n if (value === true)\n str += \" \" + name;\n else if (boolean_attributes.has(name.toLowerCase())) {\n if (value)\n str += \" \" + name;\n }\n else if (value != null) {\n str += ` ${name}=\"${String(value).replace(/\"/g, '&#34;').replace(/'/g, '&#39;')}\"`;\n }\n });\n return str;\n}\nconst escaped = {\n '\"': '&quot;',\n \"'\": '&#39;',\n '&': '&amp;',\n '<': '&lt;',\n '>': '&gt;'\n};\nfunction escape(html) {\n return String(html).replace(/[\"'&<>]/g, match => escaped[match]);\n}\nfunction each(items, fn) {\n let str = '';\n for (let i = 0; i < items.length; i += 1) {\n str += fn(items[i], i);\n }\n return str;\n}\nconst missing_component = {\n $$render: () => ''\n};\nfunction validate_component(component, name) {\n if (!component || !component.$$render) {\n if (name === 'svelte:component')\n name += ' this={...}';\n throw new Error(`<${name}> is not a valid SSR component. You may need to review your build config to ensure that dependencies are compiled, rather than imported as pre-compiled modules`);\n }\n return component;\n}\nfunction debug(file, line, column, values) {\n console.log(`{@debug} ${file ? file + ' ' : ''}(${line}:${column})`); // eslint-disable-line no-console\n console.log(values); // eslint-disable-line no-console\n return '';\n}\nlet on_destroy;\nfunction create_ssr_component(fn) {\n function $$render(result, props, bindings, slots) {\n const parent_component = current_component;\n const $$ = {\n on_destroy,\n context: new Map(parent_component ? parent_component.$$.context : []),\n // these will be immediately discarded\n on_mount: [],\n before_update: [],\n after_update: [],\n callbacks: blank_object()\n };\n set_current_component({ $$ });\n const html = fn(result, props, bindings, slots);\n set_current_component(parent_component);\n return html;\n }\n return {\n render: (props = {}, options = {}) => {\n on_destroy = [];\n const result = { title: '', head: '', css: new Set() };\n const html = $$render(result, props, {}, options);\n run_all(on_destroy);\n return {\n html,\n css: {\n code: Array.from(result.css).map(css => css.code).join('\\n'),\n map: null // TODO\n },\n head: result.title + result.head\n };\n },\n $$render\n };\n}\nfunction add_attribute(name, value, boolean) {\n if (value == null || (boolean && !value))\n return '';\n return ` ${name}${value === true ? '' : `=${typeof value === 'string' ? JSON.stringify(escape(value)) : `\"${value}\"`}`}`;\n}\nfunction add_classes(classes) {\n return classes ? ` class=\"${classes}\"` : ``;\n}\n\nfunction bind(component, name, callback) {\n const index = component.$$.props[name];\n if (index !== undefined) {\n component.$$.bound[index] = callback;\n callback(component.$$.ctx[index]);\n }\n}\nfunction create_component(block) {\n block && block.c();\n}\nfunction claim_component(block, parent_nodes) {\n block && block.l(parent_nodes);\n}\nfunction mount_component(component, target, anchor) {\n const { fragment, on_mount, on_destroy, after_update } = component.$$;\n fragment && fragment.m(target, anchor);\n // onMount happens before the initial afterUpdate\n add_render_callback(() => {\n const new_on_destroy = on_mount.map(run).filter(is_function);\n if (on_destroy) {\n on_destroy.push(...new_on_destroy);\n }\n else {\n // Edge case - component was destroyed immediately,\n // most likely as a result of a binding initialising\n run_all(new_on_destroy);\n }\n component.$$.on_mount = [];\n });\n after_update.forEach(add_render_callback);\n}\nfunction destroy_component(component, detaching) {\n const $$ = component.$$;\n if ($$.fragment !== null) {\n run_all($$.on_destroy);\n $$.fragment && $$.fragment.d(detaching);\n // TODO null out other refs, including component.$$ (but need to\n // preserve final state?)\n $$.on_destroy = $$.fragment = null;\n $$.ctx = [];\n }\n}\nfunction make_dirty(component, i) {\n if (component.$$.dirty[0] === -1) {\n dirty_components.push(component);\n schedule_update();\n component.$$.dirty.fill(0);\n }\n component.$$.dirty[(i / 31) | 0] |= (1 << (i % 31));\n}\nfunction init(component, options, instance, create_fragment, not_equal, props, dirty = [-1]) {\n const parent_component = current_component;\n set_current_component(component);\n const prop_values = options.props || {};\n const $$ = component.$$ = {\n fragment: null,\n ctx: null,\n // state\n props,\n update: noop,\n not_equal,\n bound: blank_object(),\n // lifecycle\n on_mount: [],\n on_destroy: [],\n before_update: [],\n after_update: [],\n context: new Map(parent_component ? parent_component.$$.context : []),\n // everything else\n callbacks: blank_object(),\n dirty,\n skip_bound: false\n };\n let ready = false;\n $$.ctx = instance\n ? instance(component, prop_values, (i, ret, ...rest) => {\n const value = rest.length ? rest[0] : ret;\n if ($$.ctx && not_equal($$.ctx[i], $$.ctx[i] = value)) {\n if (!$$.skip_bound && $$.bound[i])\n $$.bound[i](value);\n if (ready)\n make_dirty(component, i);\n }\n return ret;\n })\n : [];\n $$.update();\n ready = true;\n run_all($$.before_update);\n // `false` as a special case of no DOM component\n $$.fragment = create_fragment ? create_fragment($$.ctx) : false;\n if (options.target) {\n if (options.hydrate) {\n const nodes = children(options.target);\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n $$.fragment && $$.fragment.l(nodes);\n nodes.forEach(detach);\n }\n else {\n // eslint-disable-next-line @typescript-eslint/no-non-null-assertion\n $$.fragment && $$.fragment.c();\n }\n if (options.intro)\n transition_in(component.$$.fragment);\n mount_component(component, options.target, options.anchor);\n flush();\n }\n set_current_component(parent_component);\n}\nlet SvelteElement;\nif (typeof HTMLElement === 'function') {\n SvelteElement = class extends HTMLElement {\n constructor() {\n super();\n this.attachShadow({ mode: 'open' });\n }\n connectedCallback() {\n // @ts-ignore todo: improve typings\n for (const key in this.$$.slotted) {\n // @ts-ignore todo: improve typings\n this.appendChild(this.$$.slotted[key]);\n }\n }\n attributeChangedCallback(attr, _oldValue, newValue) {\n this[attr] = newValue;\n }\n $destroy() {\n destroy_component(this, 1);\n this.$destroy = noop;\n }\n $on(type, callback) {\n // TODO should this delegate to addEventListener?\n const callbacks = (this.$$.callbacks[type] || (this.$$.callbacks[type] = []));\n callbacks.push(callback);\n return () => {\n const index = callbacks.indexOf(callback);\n if (index !== -1)\n callbacks.splice(index, 1);\n };\n }\n $set($$props) {\n if (this.$$set && !is_empty($$props)) {\n this.$$.skip_bound = true;\n this.$$set($$props);\n this.$$.skip_bound = false;\n }\n }\n };\n}\nclass SvelteComponent {\n $destroy() {\n destroy_component(this, 1);\n this.$destroy = noop;\n }\n $on(type, callback) {\n const callbacks = (this.$$.callbacks[type] || (this.$$.callbacks[type] = []));\n callbacks.push(callback);\n return () => {\n const index = callbacks.indexOf(callback);\n if (index !== -1)\n callbacks.splice(index, 1);\n };\n }\n $set($$props) {\n if (this.$$set && !is_empty($$props)) {\n this.$$.skip_bound = true;\n this.$$set($$props);\n this.$$.skip_bound = false;\n }\n }\n}\n\nfunction dispatch_dev(type, detail) {\n document.dispatchEvent(custom_event(type, Object.assign({ version: '3.24.1' }, detail)));\n}\nfunction append_dev(target, node) {\n dispatch_dev(\"SvelteDOMInsert\", { target, node });\n append(target, node);\n}\nfunction insert_dev(target, node, anchor) {\n dispatch_dev(\"SvelteDOMInsert\", { target, node, anchor });\n insert(target, node, anchor);\n}\nfunction detach_dev(node) {\n dispatch_dev(\"SvelteDOMRemove\", { node });\n detach(node);\n}\nfunction detach_between_dev(before, after) {\n while (before.nextSibling && before.nextSibling !== after) {\n detach_dev(before.nextSibling);\n }\n}\nfunction detach_before_dev(after) {\n while (after.previousSibling) {\n detach_dev(after.previousSibling);\n }\n}\nfunction detach_after_dev(before) {\n while (before.nextSibling) {\n detach_dev(before.nextSibling);\n }\n}\nfunction listen_dev(node, event, handler, options, has_prevent_default, has_stop_propagation) {\n const modifiers = options === true ? [\"capture\"] : options ? Array.from(Object.keys(options)) : [];\n if (has_prevent_default)\n modifiers.push('preventDefault');\n if (has_stop_propagation)\n modifiers.push('stopPropagation');\n dispatch_dev(\"SvelteDOMAddEventListener\", { node, event, handler, modifiers });\n const dispose = listen(node, event, handler, options);\n return () => {\n dispatch_dev(\"SvelteDOMRemoveEventListener\", { node, event, handler, modifiers });\n dispose();\n };\n}\nfunction attr_dev(node, attribute, value) {\n attr(node, attribute, value);\n if (value == null)\n dispatch_dev(\"SvelteDOMRemoveAttribute\", { node, attribute });\n else\n dispatch_dev(\"SvelteDOMSetAttribute\", { node, attribute, value });\n}\nfunction prop_dev(node, property, value) {\n node[property] = value;\n dispatch_dev(\"SvelteDOMSetProperty\", { node, property, value });\n}\nfunction dataset_dev(node, property, value) {\n node.dataset[property] = value;\n dispatch_dev(\"SvelteDOMSetDataset\", { node, property, value });\n}\nfunction set_data_dev(text, data) {\n data = '' + data;\n if (text.wholeText === data)\n return;\n dispatch_dev(\"SvelteDOMSetData\", { node: text, data });\n text.data = data;\n}\nfunction validate_each_argument(arg) {\n if (typeof arg !== 'string' && !(arg && typeof arg === 'object' && 'length' in arg)) {\n let msg = '{#each} only iterates over array-like objects.';\n if (typeof Symbol === 'function' && arg && Symbol.iterator in arg) {\n msg += ' You can use a spread to convert this iterable into an array.';\n }\n throw new Error(msg);\n }\n}\nfunction validate_slots(name, slot, keys) {\n for (const slot_key of Object.keys(slot)) {\n if (!~keys.indexOf(slot_key)) {\n console.warn(`<${name}> received an unexpected slot \"${slot_key}\".`);\n }\n }\n}\nclass SvelteComponentDev extends SvelteComponent {\n constructor(options) {\n if (!options || (!options.target && !options.$$inline)) {\n throw new Error(`'target' is a required option`);\n }\n super();\n }\n $destroy() {\n super.$destroy();\n this.$destroy = () => {\n console.warn(`Component was already destroyed`); // eslint-disable-line no-console\n };\n }\n $capture_state() { }\n $inject_state() { }\n}\nfunction loop_guard(timeout) {\n const start = Date.now();\n return () => {\n if (Date.now() - start > timeout) {\n throw new Error(`Infinite loop detected`);\n }\n };\n}\n\nexport { HtmlTag, SvelteComponent, SvelteComponentDev, SvelteElement, action_destroyer, add_attribute, add_classes, add_flush_callback, add_location, add_render_callback, add_resize_listener, add_transform, afterUpdate, append, append_dev, assign, attr, attr_dev, beforeUpdate, bind, binding_callbacks, blank_object, bubble, check_outros, children, claim_component, claim_element, claim_space, claim_text, clear_loops, component_subscribe, compute_rest_props, createEventDispatcher, create_animation, create_bidirectional_transition, create_component, create_in_transition, create_out_transition, create_slot, create_ssr_component, current_component, custom_event, dataset_dev, debug, destroy_block, destroy_component, destroy_each, detach, detach_after_dev, detach_before_dev, detach_between_dev, detach_dev, dirty_components, dispatch_dev, each, element, element_is, empty, escape, escaped, exclude_internal_props, fix_and_destroy_block, fix_and_outro_and_destroy_block, fix_position, flush, getContext, get_binding_group_value, get_current_component, get_slot_changes, get_slot_context, get_spread_object, get_spread_update, get_store_value, globals, group_outros, handle_promise, has_prop, identity, init, insert, insert_dev, intros, invalid_attribute_name_character, is_client, is_crossorigin, is_empty, is_function, is_promise, listen, listen_dev, loop, loop_guard, missing_component, mount_component, noop, not_equal, now, null_to_empty, object_without_properties, onDestroy, onMount, once, outro_and_destroy_block, prevent_default, prop_dev, query_selector_all, raf, run, run_all, safe_not_equal, schedule_update, select_multiple_value, select_option, select_options, select_value, self, setContext, set_attributes, set_current_component, set_custom_element_data, set_data, set_data_dev, set_input_type, set_input_value, set_now, set_raf, set_store_value, set_style, set_svg_attributes, space, spread, stop_propagation, subscribe, svg_element, text, tick, time_ranges_to_array, to_number, toggle_class, transition_in, transition_out, update_keyed_each, update_slot, validate_component, validate_each_argument, validate_each_keys, validate_slots, validate_store, xlink_attr };\n","<script>\n\texport let value = \"\";\n\tlet isInvalid = false;\n\tlet error = \"\";\n\n\tfunction onBlur() {\n\t\tif (value === \"\") {\n\t\t\tisInvalid = false;\n\t\t\terror = \"\";\n\t\t\treturn;\n\t\t}\n\t\tconst ipport = value.split(\":\");\n\t\tif (ipport.length !== 2) {\n\t\t\tisInvalid = true;\n\t\t\terror = \"missing port number\";\n\t\t\treturn;\n\t\t}\n\t\tconst ip = ipport[0];\n\t\tconst port = parseInt(ipport[1]);\n\t\tif (isNaN(port) || port <= 0 || port >= 65535) {\n\t\t\tisInvalid = true;\n\t\t\terror = \"invalid port number\";\n\t\t\treturn;\n\t\t}\n\t\tisInvalid = false;\n\t\tvalue = ip +\":\"+ port;\n\t}\n</script>\n\n<style>\n\t.wui-input-ipport {\n\t\tdisplay: inline-block;\n\t\twidth: 100%;\n\t}\n\t.wui-input-ipport input {\n\t\twidth: 100%;\n\t}\n\t.invalid {\n\t\tcolor: red;\n\t}\n\tdiv.invalid {\n\t\tfont-size: 12px;\n\t}\n</style>\n\n<div class=\"wui-input-ipport\">\n\t<input\n\t\ttype=\"text\"\n\t\tbind:value={value}\n\t\ton:blur={onBlur}\n\t\tclass:invalid={isInvalid}\n\t\tplaceholder=\"127.0.0.1:8080\"\n\t>\n\t{#if isInvalid}\n\t<div class=\"invalid\">{error}</div>\n\t{/if}\n</div>\n","<script>\n\texport let min;\n\texport let max;\n\texport let value = 0;\n\texport let unit;\n\n\tfunction onBlur() {\n\t\tvalue = +value\n\t\tif (isNaN(value)) {\n\t\t\tvalue = max\n\t\t} else if (value < min) {\n\t\t\tvalue = min\n\t\t} else if (value > max) {\n\t\t\tvalue = max\n\t\t}\n\t}\n</script>\n\n<style>\n\t.wui-input-number {\n\t\twidth: 100%;\n\t}\n\t.wui-input-number input {\n\t\tdisplay: inline-block;\n\t\twidth: 70%;\n\t}\n\t.wui-input-number .suffix {\n\t\twidth: 30%;\n\t}\n</style>\n\n<div class=\"wui-input-number\">\n\t<input type=\"number\" on:blur={onBlur} bind:value={value}>\n\t{#if unit !== ''}\n\t\t<span class=\"suffix\">\n\t\t\t{unit}\n\t\t</span>\n\t{/if}\n</div>\n","<script>\n\texport let title;\n\texport let info;\n\texport let title_width = \"300px\";\n\tlet showInfo = false;\n</script>\n\n<style>\n\t.label-hint {\n\t\tdisplay: inline-flex;\n\t\tmargin-top: 1em;\n\t\twidth: 100%;\n\t}\n\t.title {\n\t\tmargin-bottom: 4px;\n\t}\n\t.title {\n\t\tdisplay: inline-block;\n\t}\n\t.toggle {\n\t\tborder-radius: 50%;\n\t\tborder: 1px solid grey;\n\t\tcursor: pointer;\n\t\tdisplay: inline-block;\n\t\tfont-size: 12px;\n\t\theight: 14px;\n\t\tline-height: 14px;\n\t\tpadding: 2px;\n\t\ttext-align: center;\n\t\twidth: 14px;\n\t}\n\t.info {\n\t\tbackground-color: #eee;\n\t\tmargin: 8px 0px;\n\t\tpadding: 1em;\n\t}\n\t@media (max-width: 720px) {\n\t\t.label-hint {\n\t\t\tdisplay: block;\n\t\t}\n\t}\n</style>\n\n<label class=\"label-hint\">\n\t<span class=\"title\" style=\"width:{title_width};\">\n\t\t{title}\n\t\t<span class=\"toggle\" on:click={() => showInfo = !showInfo}>\n\t\t\t?\n\t\t</span>\n\t</span>\n\n\t<slot>\n\t</slot>\n\n</label>\n\n{#if showInfo}\n<div class=\"info\">\n\t{@html info}\n</div>\n{/if}\n","import { cubicInOut, linear, cubicOut } from '../easing';\nimport { is_function, assign } from '../internal';\n\n/*! *****************************************************************************\r\nCopyright (c) Microsoft Corporation. All rights reserved.\r\nLicensed under the Apache License, Version 2.0 (the \"License\"); you may not use\r\nthis file except in compliance with the License. You may obtain a copy of the\r\nLicense at http://www.apache.org/licenses/LICENSE-2.0\r\n\r\nTHIS CODE IS PROVIDED ON AN *AS IS* BASIS, WITHOUT WARRANTIES OR CONDITIONS OF ANY\r\nKIND, EITHER EXPRESS OR IMPLIED, INCLUDING WITHOUT LIMITATION ANY IMPLIED\r\nWARRANTIES OR CONDITIONS OF TITLE, FITNESS FOR A PARTICULAR PURPOSE,\r\nMERCHANTABLITY OR NON-INFRINGEMENT.\r\n\r\nSee the Apache Version 2.0 License for specific language governing permissions\r\nand limitations under the License.\r\n***************************************************************************** */\r\n\r\nfunction __rest(s, e) {\r\n var t = {};\r\n for (var p in s) if (Object.prototype.hasOwnProperty.call(s, p) && e.indexOf(p) < 0)\r\n t[p] = s[p];\r\n if (s != null && typeof Object.getOwnPropertySymbols === \"function\")\r\n for (var i = 0, p = Object.getOwnPropertySymbols(s); i < p.length; i++) {\r\n if (e.indexOf(p[i]) < 0 && Object.prototype.propertyIsEnumerable.call(s, p[i]))\r\n t[p[i]] = s[p[i]];\r\n }\r\n return t;\r\n}\n\nfunction blur(node, { delay = 0, duration = 400, easing = cubicInOut, amount = 5, opacity = 0 }) {\n const style = getComputedStyle(node);\n const target_opacity = +style.opacity;\n const f = style.filter === 'none' ? '' : style.filter;\n const od = target_opacity * (1 - opacity);\n return {\n delay,\n duration,\n easing,\n css: (_t, u) => `opacity: ${target_opacity - (od * u)}; filter: ${f} blur(${u * amount}px);`\n };\n}\nfunction fade(node, { delay = 0, duration = 400, easing = linear }) {\n const o = +getComputedStyle(node).opacity;\n return {\n delay,\n duration,\n easing,\n css: t => `opacity: ${t * o}`\n };\n}\nfunction fly(node, { delay = 0, duration = 400, easing = cubicOut, x = 0, y = 0, opacity = 0 }) {\n const style = getComputedStyle(node);\n const target_opacity = +style.opacity;\n const transform = style.transform === 'none' ? '' : style.transform;\n const od = target_opacity * (1 - opacity);\n return {\n delay,\n duration,\n easing,\n css: (t, u) => `\n\t\t\ttransform: ${transform} translate(${(1 - t) * x}px, ${(1 - t) * y}px);\n\t\t\topacity: ${target_opacity - (od * u)}`\n };\n}\nfunction slide(node, { delay = 0, duration = 400, easing = cubicOut }) {\n const style = getComputedStyle(node);\n const opacity = +style.opacity;\n const height = parseFloat(style.height);\n const padding_top = parseFloat(style.paddingTop);\n const padding_bottom = parseFloat(style.paddingBottom);\n const margin_top = parseFloat(style.marginTop);\n const margin_bottom = parseFloat(style.marginBottom);\n const border_top_width = parseFloat(style.borderTopWidth);\n const border_bottom_width = parseFloat(style.borderBottomWidth);\n return {\n delay,\n duration,\n easing,\n css: t => `overflow: hidden;` +\n `opacity: ${Math.min(t * 20, 1) * opacity};` +\n `height: ${t * height}px;` +\n `padding-top: ${t * padding_top}px;` +\n `padding-bottom: ${t * padding_bottom}px;` +\n `margin-top: ${t * margin_top}px;` +\n `margin-bottom: ${t * margin_bottom}px;` +\n `border-top-width: ${t * border_top_width}px;` +\n `border-bottom-width: ${t * border_bottom_width}px;`\n };\n}\nfunction scale(node, { delay = 0, duration = 400, easing = cubicOut, start = 0, opacity = 0 }) {\n const style = getComputedStyle(node);\n const target_opacity = +style.opacity;\n const transform = style.transform === 'none' ? '' : style.transform;\n const sd = 1 - start;\n const od = target_opacity * (1 - opacity);\n return {\n delay,\n duration,\n easing,\n css: (_t, u) => `\n\t\t\ttransform: ${transform} scale(${1 - (sd * u)});\n\t\t\topacity: ${target_opacity - (od * u)}\n\t\t`\n };\n}\nfunction draw(node, { delay = 0, speed, duration, easing = cubicInOut }) {\n const len = node.getTotalLength();\n if (duration === undefined) {\n if (speed === undefined) {\n duration = 800;\n }\n else {\n duration = len / speed;\n }\n }\n else if (typeof duration === 'function') {\n duration = duration(len);\n }\n return {\n delay,\n duration,\n easing,\n css: (t, u) => `stroke-dasharray: ${t * len} ${u * len}`\n };\n}\nfunction crossfade(_a) {\n var { fallback } = _a, defaults = __rest(_a, [\"fallback\"]);\n const to_receive = new Map();\n const to_send = new Map();\n function crossfade(from, node, params) {\n const { delay = 0, duration = d => Math.sqrt(d) * 30, easing = cubicOut } = assign(assign({}, defaults), params);\n const to = node.getBoundingClientRect();\n const dx = from.left - to.left;\n const dy = from.top - to.top;\n const dw = from.width / to.width;\n const dh = from.height / to.height;\n const d = Math.sqrt(dx * dx + dy * dy);\n const style = getComputedStyle(node);\n const transform = style.transform === 'none' ? '' : style.transform;\n const opacity = +style.opacity;\n return {\n delay,\n duration: is_function(duration) ? duration(d) : duration,\n easing,\n css: (t, u) => `\n\t\t\t\topacity: ${t * opacity};\n\t\t\t\ttransform-origin: top left;\n\t\t\t\ttransform: ${transform} translate(${u * dx}px,${u * dy}px) scale(${t + (1 - t) * dw}, ${t + (1 - t) * dh});\n\t\t\t`\n };\n }\n function transition(items, counterparts, intro) {\n return (node, params) => {\n items.set(params.key, {\n rect: node.getBoundingClientRect()\n });\n return () => {\n if (counterparts.has(params.key)) {\n const { rect } = counterparts.get(params.key);\n counterparts.delete(params.key);\n return crossfade(rect, node, params);\n }\n // if the node is disappearing altogether\n // (i.e. wasn't claimed by the other list)\n // then we need to supply an outro\n items.delete(params.key);\n return fallback && fallback(node, params, intro);\n };\n };\n }\n return [\n transition(to_send, to_receive, false),\n transition(to_receive, to_send, true)\n ];\n}\n\nexport { blur, crossfade, draw, fade, fly, scale, slide };\n","import { noop, safe_not_equal, subscribe, run_all, is_function } from '../internal';\nexport { get_store_value as get } from '../internal';\n\nconst subscriber_queue = [];\n/**\n * Creates a `Readable` store that allows reading by subscription.\n * @param value initial value\n * @param {StartStopNotifier}start start and stop notifications for subscriptions\n */\nfunction readable(value, start) {\n return {\n subscribe: writable(value, start).subscribe,\n };\n}\n/**\n * Create a `Writable` store that allows both updating and reading by subscription.\n * @param {*=}value initial value\n * @param {StartStopNotifier=}start start and stop notifications for subscriptions\n */\nfunction writable(value, start = noop) {\n let stop;\n const subscribers = [];\n function set(new_value) {\n if (safe_not_equal(value, new_value)) {\n value = new_value;\n if (stop) { // store is ready\n const run_queue = !subscriber_queue.length;\n for (let i = 0; i < subscribers.length; i += 1) {\n const s = subscribers[i];\n s[1]();\n subscriber_queue.push(s, value);\n }\n if (run_queue) {\n for (let i = 0; i < subscriber_queue.length; i += 2) {\n subscriber_queue[i][0](subscriber_queue[i + 1]);\n }\n subscriber_queue.length = 0;\n }\n }\n }\n }\n function update(fn) {\n set(fn(value));\n }\n function subscribe(run, invalidate = noop) {\n const subscriber = [run, invalidate];\n subscribers.push(subscriber);\n if (subscribers.length === 1) {\n stop = start(set) || noop;\n }\n run(value);\n return () => {\n const index = subscribers.indexOf(subscriber);\n if (index !== -1) {\n subscribers.splice(index, 1);\n }\n if (subscribers.length === 0) {\n stop();\n stop = null;\n }\n };\n }\n return { set, update, subscribe };\n}\nfunction derived(stores, fn, initial_value) {\n const single = !Array.isArray(stores);\n const stores_array = single\n ? [stores]\n : stores;\n const auto = fn.length < 2;\n return readable(initial_value, (set) => {\n let inited = false;\n const values = [];\n let pending = 0;\n let cleanup = noop;\n const sync = () => {\n if (pending) {\n return;\n }\n cleanup();\n const result = fn(single ? values[0] : values, set);\n if (auto) {\n set(result);\n }\n else {\n cleanup = is_function(result) ? result : noop;\n }\n };\n const unsubscribers = stores_array.map((store, i) => subscribe(store, (value) => {\n values[i] = value;\n pending &= ~(1 << i);\n if (inited) {\n sync();\n }\n }, () => {\n pending |= (1 << i);\n }));\n inited = true;\n sync();\n return function stop() {\n run_all(unsubscribers);\n cleanup();\n };\n });\n}\n\nexport { derived, readable, writable };\n","import { writable } from 'svelte/store';\n\nexport const messages = writable([]);\n","<script>\n\timport { onMount } from 'svelte'\n\timport { fade } from 'svelte/transition'\n\timport { messages } from \"./Notif.store.js\"\n\n\texport let text = \"\"\n\texport let kind = \"\"\n\texport let timeout = 5000\n\n\tonMount(() => {\n\t\tlet timerID = setTimeout(()=> {\n\t\t\tmessages.update(msgs => {\n\t\t\t\tmsgs.splice(0, 1);\n\t\t\t\tmsgs = msgs\n\t\t\t\treturn msgs\n\t\t\t})\n\t\t}, timeout)\n\t})\n</script>\n\n<style>\n\t.wui-notif-item {\n\t\tbackground-color: white;\n\t\tborder: 1px solid black;\n\t\tbox-shadow: 3px 3px;\n\t\tpadding: 1em;\n\t\tmargin-bottom: 1em;\n\t\tz-index: 1000;\n\t}\n\t.wui-notif-item.error {\n\t\tborder: 1px solid red;\n\t\tbox-shadow: 3px 3px red;\n\t}\n</style>\n\n<div transition:fade class=\"wui-notif-item {kind}\">\n\t{text}\n</div>\n","<script context=\"module\">\n\timport { messages } from \"./Notif.store.js\"\n\timport NotifItem from \"./NotifItem.svelte\"\n\n\texport const WuiPushNotif = {\n\t\tInfo: function(text) {\n\t\t\tconst msg = {\n\t\t\t\ttext: text,\n\t\t\t}\n\t\t\tmessages.update(msgs => msgs = [...msgs, msg])\n\t\t},\n\t\tError: function(text) {\n\t\t\tconst msg = {\n\t\t\t\ttext: text,\n\t\t\t\tkind: \"error\",\n\t\t\t}\n\t\t\tmessages.update(msgs => msgs = [...msgs, msg])\n\t\t}\n\t}\n</script>\n<script>\n\texport let timeout = 5000;\n</script>\n\n<style>\n\t.wui-notif {\n\t\tposition: fixed;\n\t\ttop: 5px;\n\t\tleft: calc((100% - 400px)/2);\n\t\twidth: 400px;\n\t}\n\t@media (max-width: 500px) {\n\t\t.wui-notif {\n\t\t\tleft: 1em;\n\t\t\twidth: calc(100% - 2em);\n\t\t}\n\t}\n</style>\n\n<div class=\"wui-notif\">\n\t{#each $messages as msg (msg)}\n\t<NotifItem text={msg.text} kind=\"{msg.kind}\" {timeout}/>\n\t{/each}\n</div>\n","import { writable } from \"svelte/store\"\n\nexport const apiEnvironment = \"/api/environment\"\nexport const environment = writable({\n\tNameServers: [],\n\tHostsBlocks: [],\n\tHostsFiles: {},\n\tZoneFiles: [],\n})\nexport const nanoSeconds = 1000000000\n\nexport async function setEnvironment(got) {\n\tgot.PruneDelay = got.PruneDelay / nanoSeconds\n\tgot.PruneThreshold = got.PruneThreshold / nanoSeconds\n\tfor (const [key, value] of Object.entries(got.HostsFiles)) {\n\t\tgot.HostsFiles[key].Records = []\n\t}\n\tenvironment.set(got)\n}\n","const RRTypes = {\n\t1: \"A\",\n\t2: \"NS\",\n\t3: \"MD\",\n\t4: \"MF\",\n\t5: \"CNAME\",\n\t6: \"SOA\",\n\t7: \"MB\",\n\t8: \"MG\",\n\t9: \"MR\",\n\t10: \"NULL\",\n\t11: \"WKS\",\n\t12: \"PTR\",\n\t13: \"HINFO\",\n\t14: \"MINFO\",\n\t15: \"MX\",\n\t16: \"TXT\",\n\t28: \"AAAA\",\n\t33: \"SRV\",\n\t41: \"OPT\",\n}\n\nexport function getRRTypeName(k) {\n\tlet v = RRTypes[k]\n\tif (v === \"\") {\n\t\treturn k\n\t}\n\treturn v\n}\n","<script>\n\timport {WuiPushNotif} from 'wui.svelte';\n\timport {getRRTypeName} from './common.js';\n\n\tconst apiCaches = \"/api/caches\";\n\n\tlet query = \"\";\n\tlet listMsg = [];\n\n\tasync function handleSearch() {\n\t\tconst res = await fetch(apiCaches+\"?query=\"+ query)\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resbody = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resbody.message)\n\t\t\treturn;\n\t\t}\n\n\t\tlistMsg = await res.json()\n\t}\n\n\tasync function handleRemoveFromCaches(name) {\n\t\tconst res = await fetch(apiCaches+\"?name=\"+ name, {\n\t\t\tmethod: \"DELETE\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resbody = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resbody.message)\n\t\t\treturn;\n\t\t}\n\n\t\tfor (let x = 0; x < listMsg.length; x++) {\n\t\t\tif (listMsg[x].Question.Name === name) {\n\t\t\t\tlistMsg.splice(x, 1)\n\t\t\t\tlistMsg = listMsg\n\t\t\t\tbreak\n\t\t\t}\n\t\t}\n\n\t\tconst msg = await res.json()\n\n\t\tWuiPushNotif.Info(msg.message)\n\t}\n</script>\n\n<style>\n\t.message {\n\t\tpadding: 1em 0px;\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.rr {\n\t\tmargin-left: 1em;\n\t\twidth: 100%;\n\t}\n\t.rr.header {\n\t\tfont-weight: bold;\n\t}\n\t.rr span {\n\t\tdisplay: inline-block;\n\t}\n\t.kind {\n\t\twidth: 9em;\n\t}\n\t.type {\n\t\twidth: 5em;\n\t}\n\t.ttl {\n\t\twidth: 6em;\n\t}\n\t.value {\n\t\tword-wrap: anywhere;\n\t\twidth: calc(100% - 24em);\n\t}\n</style>\n\n<div class=\"dashboard\">\n\t<div class=\"search\">\n\t\tCaches:\n\t\t<input bind:value={query}>\n\t\t<button on:click={handleSearch}>\n\t\t\tSearch\n\t\t</button>\n\t</div>\n\t{#each listMsg as msg (msg)}\n\t\t<div class=\"message\">\n\t\t\t<div class=\"qname\">\n\t\t\t\t{msg.Question.Name}\n\n\t\t\t\t<button\n\t\t\t\t\tclass=\"b-remove\"\n\t\t\t\t\ton:click={handleRemoveFromCaches(msg.Question.Name)}\n\t\t\t\t>\n\t\t\t\t\tRemove from caches\n\t\t\t\t</button>\n\t\t\t</div>\n\t\t\t<div class=\"rr header\">\n\t\t\t\t<span class=\"kind\"> </span>\n\t\t\t\t<span class=\"type\"> Type </span>\n\t\t\t\t<span class=\"ttl\"> TTL </span>\n\t\t\t\t<span class=\"value\"> Value </span>\n\t\t\t</div>\n\n\t\t\t{#if msg.Answer !== null && msg.Answer.length > 0}\n\t\t\t\t{#each msg.Answer as rr}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"kind\"> Answer </span>\n\t\t\t\t\t\t<span class=\"type\"> {getRRTypeName(rr.Type)} </span>\n\t\t\t\t\t\t<span class=\"ttl\"> {rr.TTL} </span>\n\t\t\t\t\t\t<span class=\"value\"> {JSON.stringify(rr.Value, null, 2)} </span>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/if}\n\t\t\t{#if msg.Authority !== null && msg.Authority.length > 0}\n\t\t\t\t{#each msg.Authority as rr}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"kind\"> Authority </span>\n\t\t\t\t\t\t<span class=\"type\"> {getRRTypeName(rr.Type)} </span>\n\t\t\t\t\t\t<span class=\"ttl\"> {rr.TTL} </span>\n\t\t\t\t\t\t<span class=\"value\"> {JSON.stringify(rr.Value, null, 2)} </span>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/if}\n\t\t\t{#if msg.Additional !== null && msg.Additional.length > 0}\n\t\t\t\t{#each msg.Additional as rr}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"kind\"> Additional </span>\n\t\t\t\t\t\t<span class=\"type\"> {getRRTypeName(rr.Type)} </span>\n\t\t\t\t\t\t<span class=\"ttl\"> {rr.TTL} </span>\n\t\t\t\t\t\t<span class=\"value\"> {JSON.stringify(rr.Value, null, 2)} </span>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/if}\n\t\t</div>\n\t{/each}\n</div>\n","<script>\n\timport { onDestroy } from 'svelte';\n\n\timport { apiEnvironment, environment, nanoSeconds } from './environment.js';\n\timport { WuiPushNotif } from \"wui.svelte\";\n\timport { WuiLabelHint, WuiInputNumber, WuiInputIPPort } from \"wui.svelte\";\n\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: {},\n\t};\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\n\tonDestroy(envUnsubscribe);\n\n\tconst defTitleWidth = \"300px\";\n\n\tfunction addNameServer() {\n\t\tenv.NameServers = [...env.NameServers, '']\n\t}\n\n\tfunction deleteNameServer(ns) {\n\t\tfor (let x = 0; x < env.NameServers.length; x++) {\n\t\t\tif (env.NameServers[x] === ns) {\n\t\t\t\tenv.NameServers.splice(x, 1);\n\t\t\t\tenv.NameServers = env.NameServers;\n\t\t\t\tbreak;\n\t\t\t}\n\t\t}\n\t}\n\n\tasync function updateEnvironment() {\n\t\tlet got = {};\n\n\t\tObject.assign(got, env)\n\t\tenvironment.set(env)\n\n\t\tgot.PruneDelay = got.PruneDelay * nanoSeconds;\n\t\tgot.PruneThreshold = got.PruneThreshold * nanoSeconds;\n\n\t\tconst res = await fetch(apiEnvironment, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tbody: JSON.stringify(got),\n\t\t});\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resbody = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \", resbody.message)\n\t\t\treturn;\n\t\t}\n\n\t\tWuiPushNotif.Info(\"The environment succesfully updated ...\")\n\t}\n</script>\n\n<style>\n\tinput {\n\t\twidth: 100%;\n\t}\n\t.input-deletable {\n\t\twidth: 100%;\n\t}\n\t.input-deletable > input {\n\t\tmax-width: calc(100% - 100px);\n\t}\n\t.input-deletable > button {\n\t\twidth: 80px;\n\t}\n\t.input-checkbox {\n\t\twidth: 100%;\n\t}\n\t.input-checkbox input[type=\"checkbox\"] {\n\t\twidth: auto;\n\t}\n\t.section-bottom {\n\t\tmargin: 2em 0px 0px 0px;\n\t\tpadding: 1em;\n\t\tborder-top: 1px solid black;\n\t}\n</style>\n\n<div class=\"environment\">\n<p>\nThis page allow you to change the rescached environment.\nUpon save, the rescached service will be restarted.\n</p>\n\n<h3>rescached</h3>\n<div>\n\t<WuiLabelHint\n\t\ttitle=\"System resolv.conf\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"A path to dynamically generated resolv.conf(5) by\nresolvconf(8). If set, the nameserver values in referenced file will\nreplace 'parent' value and 'parent' will become a fallback in\ncase the referenced file being deleted or can not be parsed.\"\n\t>\n\t\t<input\n\t\t\tbind:value={env.FileResolvConf}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"Debug level\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"This option only used for debugging program or if user\nwant to monitor what kind of traffic goes in and out of rescached.\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=0\n\t\t\tmax=3\n\t\t\tbind:value={env.Debug}\n\t\t\tunit=\"\"\n\t\t/>\n\t</WuiLabelHint>\n</div>\n\n<h3>DNS server</h3>\n<div>\n\t<WuiLabelHint\n\t\ttitle=\"Parent name servers\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"List of parent DNS servers.\"\n\t>\n\t</WuiLabelHint>\n\n\t{#each env.NameServers as ns}\n\t<div class=\"input-deletable\">\n\t\t<input bind:value={ns}>\n\t\t<button on:click={deleteNameServer(ns)}>\n\t\t\tDelete\n\t\t</button>\n\t</div>\n\t{/each}\n\n\t<button\n\t\ton:click={addNameServer}\n\t>\n\t\tAdd\n\t</button>\n\n\t<WuiLabelHint\n\t\ttitle=\"Listen address\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Address in local network where rescached will\nlistening for query from client through UDP and TCP.\n<br/>\nIf you want rescached to serve a query from another host in your local\nnetwork, change this value to <tt>0.0.0.0:53</tt>.\"\n\t>\n\t\t<WuiInputIPPort\n\t\t\tbind:value={env.ListenAddress}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"HTTP listen port\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Port to serve DNS over HTTP\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=0\n\t\t\tmax=65535\n\t\t\tbind:value={env.HTTPPort}\n\t\t\tunit=\"\"\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS listen port\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Port to listen for DNS over TLS\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=0\n\t\t\tmax=65535\n\t\t\tbind:value={env.TLSPort}\n\t\t\tunit=\"\"\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS certificate\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Path to certificate file to serve DNS over TLS and\nHTTPS\">\n\t\t<input\n\t\t\tplaceholder=\"/path/to/certificate\"\n\t\t\tbind:value={env.TLSCertFile}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS private key\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Path to certificate private key file to serve DNS over TLS and\nHTTPS.\"\n\t>\n\t\t<input\n\t\t\tplaceholder=\"/path/to/certificate/private.key\"\n\t\t\tbind:value={env.TLSPrivateKey}\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"TLS allow insecure\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"If its true, allow serving DoH and DoT with self signed\ncertificate.\"\n\t>\n\t\t<div class=\"input-checkbox\">\n\t\t\t<input\n\t\t\t\ttype=checkbox\n\t\t\t\tbind:checked={env.TLSAllowInsecure}\n\t\t\t>\n\t\t\t<span class=\"suffix\">\n\t\t\t\tYes\n\t\t\t</span>\n\t\t</div>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"DoH behind proxy\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"If its true, serve DNS over HTTP only, even if\ncertificate files is defined.\nThis allow serving DNS request forwarded by another proxy server.\"\n\t>\n\t\t<div class=\"input-checkbox\">\n\t\t\t<input\n\t\t\t\ttype=checkbox\n\t\t\t\tbind:checked={env.DoHBehindProxy}\n\t\t\t>\n\t\t\t<span class=\"suffix\">\n\t\t\t\tYes\n\t\t\t</span>\n\t\t</div>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"Prune delay\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"Delay for pruning caches.\nEvery N seconds, rescached will traverse all caches and remove response that\nhas not been accessed less than cache.prune_threshold.\nIts value must be equal or greater than 1 hour (3600 seconds).\n\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=3600\n\t\t\tmax=36000\n\t\t\tbind:value={env.PruneDelay}\n\t\t\tunit=\"seconds\"\n\t\t/>\n\t</WuiLabelHint>\n\n\t<WuiLabelHint\n\t\ttitle=\"Prune threshold\"\n\t\ttitle_width=\"{defTitleWidth}\"\n\t\tinfo=\"The duration when the cache will be considered expired.\nIts value must be negative and greater or equal than -1 hour (-3600 seconds).\"\n\t>\n\t\t<WuiInputNumber\n\t\t\tmin=-36000\n\t\t\tmax=-3600\n\t\t\tbind:value={env.PruneThreshold}\n\t\t\tunit=\"seconds\"\n\t\t/>\n\t</WuiLabelHint>\n</div>\n\n\t<div class=\"section-bottom\">\n\t\t<div>\n\t\t\t<button on:click={updateEnvironment}>\n\t\t\t\tSave\n\t\t\t</button>\n\t\t</div>\n\t</div>\n</div>\n","<script>\n\timport { onDestroy } from 'svelte';\n\timport { WuiPushNotif } from 'wui.svelte';\n\timport { environment, nanoSeconds, setEnvironment } from './environment.js';\n\n\tconst apiHostsBlock = \"/api/hosts_block\"\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: {},\n\t};\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\tonDestroy(envUnsubscribe);\n\n\tasync function updateHostsBlocks() {\n\t\tconst res = await fetch(apiHostsBlock, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n\t\t\t\t\"Content-Type\": \"application/json\",\n\t\t\t},\n\t\t\tbody: JSON.stringify(env.HostsBlocks),\n\t\t});\n\n\t\tif (res.status >= 400) {\n\t\t\tWuiPushNotif.Error(\"ERROR: \", res.status, res.statusText)\n\t\t\treturn;\n\t\t}\n\n\t\tsetEnvironment(await res.json());\n\t}\n</script>\n\n<style>\n\t.block_source {\n\t\twidth: calc(100% - 2em);\n\t\toverflow: auto;\n\t\tfont-size: 12px;\n\t}\n\t.block_source input:disabled {\n\t\tcolor: black;\n\t}\n\t.item span {\n\t\tdisplay: inline-block;\n\t\tmargin-right: 1em;\n\t}\n\t.item.header {\n\t\tfont-weight: bold;\n\t\tmargin-bottom: 1em;\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.item > span:nth-child(1) {\n\t\twidth: 4em;\n\t}\n\t.item > span:nth-child(2) {\n\t\twidth: 15em;\n\t}\n\t.item > span:nth-child(3) {\n\t\twidth: 23em;\n\t}\n\t.item > span:nth-child(3) input {\n\t\twidth: 100%;\n\t}\n\t.item > span:nth-child(4) {\n\t\twidth: 16em;\n\t}\n</style>\n\n<div class=\"hosts-block\">\n\t<p>\n\tConfigure the source of blocked hosts file.\n\t</p>\n\n\t<div class=\"block_source\">\n\t\t<div class=\"item header\">\n\t\t\t<span> Enabled </span>\n\t\t\t<span> Name </span>\n\t\t\t<span> URL </span>\n\t\t\t<span> Last updated </span>\n\t\t</div>\n\t\t{#each env.HostsBlocks as hostsBlock}\n\t\t<div class=\"item\">\n\t\t\t<span>\n\t\t\t\t<input\n\t\t\t\t\ttype=checkbox\n\t\t\t\t\tbind:checked={hostsBlock.IsEnabled}\n\t\t\t\t>\n\t\t\t</span>\n\t\t\t<span>\n\t\t\t\t{hostsBlock.Name}\n\t\t\t</span>\n\t\t\t<span>\n\t\t\t\t<input\n\t\t\t\t\tbind:value={hostsBlock.URL}\n\t\t\t\t\tdisabled\n\t\t\t\t>\n\t\t\t</span>\n\t\t\t<span>\n\t\t\t\t{hostsBlock.LastUpdated}\n\t\t\t</span>\n\t\t</div>\n\t\t{/each}\n\t</div>\n\n\t<div>\n\t\t<button on:click={updateHostsBlocks}>\n\t\t\tSave\n\t\t</button>\n\t</div>\n</div>\n","<script> import { onDestroy } from 'svelte';\n\timport { WuiPushNotif } from 'wui.svelte';\n\timport { apiEnvironment, environment, nanoSeconds } from './environment.js';\n\n\tconst apiHostsDir = \"/api/hosts.d\"\n\n\tlet env = {\n\t\tHostsFiles: {},\n\t};\n\tlet hostsFile = {\n\t\tName: \"\",\n\t\tRecords: [],\n\t};\n\tlet newRecord = null;\n\tlet newHostsFile = \"\";\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\tonDestroy(envUnsubscribe);\n\n\tasync function getHostsFile(hf) {\n\t\tif (hf.Records === null) {\n\t\t\thf.Records = []\n\t\t}\n\t\tif (hf.Records.length > 0) {\n\t\t\thostsFile = hf;\n\t\t\treturn;\n\t\t}\n\t\tconst res = await fetch(apiHostsDir +\"/\"+ hf.Name);\n\t\thf.Records = await res.json();\n\t\thostsFile = hf;\n\t}\n\n\tasync function createHostsFile() {\n\t\tif (newHostsFile === \"\") {\n\t\t\treturn;\n\t\t}\n\n\t\tconst res = await fetch(apiHostsDir+ \"/\"+ newHostsFile, {\n\t\t\tmethod: \"PUT\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: createHostsFile: \", resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tconst hf = {\n\t\t\tName: newHostsFile,\n\t\t\tRecords: [],\n\t\t}\n\t\tenv.HostsFiles[newHostsFile] = hf\n\t\tenv = env\n\n\t\tWuiPushNotif.Info(\"The new host file '\"+ newHostsFile +\"' has been created\")\n\t}\n\n\tasync function updateHostsFile() {\n\t\tconst res = await fetch(apiHostsDir+\"/\"+ hostsFile.Name, {\n\t\t\tmethod: \"POST\",\n\t\t\tbody: JSON.stringify(hostsFile.Records),\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: updateHostsFile: \", resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\thostsFile.Records = await res.json()\n\n\t\tWuiPushNotif.Info(\"The host file '\"+ hostsFile.Name +\"' has been updated\")\n\t}\n\n\tfunction addRecord() {\n\t\tif (newRecord !== null) {\n\t\t\treturn\n\t\t}\n\t\tnewRecord = {\n\t\t\tName: \"\",\n\t\t\tValue: \"\",\n\t\t}\n\t}\n\n\tasync function handleHostsRecordCreate() {\n\t\tconst api = apiHostsDir +\"/\"+ hostsFile.Name +\"/rr\"\n\t\t\t+\"?domain=\"+ newRecord.Name\n\t\t\t+\"&value=\"+ newRecord.Value\n\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"POST\"\n\t\t})\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\t\tconst rr = await res.json()\n\t\thostsFile.Records.push(rr)\n\t\thostsFile.Records = hostsFile.Records\n\t\tnewRecord = null\n\t\tWuiPushNotif.Info(\"Record '\"+ rr.Name +\"' has been created\")\n\t}\n\n\tasync function handleHostsRecordDelete(rr, idx) {\n\t\tconst api = apiHostsDir +\"/\"+ hostsFile.Name +\"/rr\"+\n\t\t\t\"?domain=\"+rr.Name\n\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"DELETE\"\n\t\t})\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\t\thostsFile.Records.splice(idx, 1);\n\t\thostsFile.Records = hostsFile.Records;\n\t\tWuiPushNotif.Info(\"Record '\"+ rr.Name +\"' has been deleted\")\n\t}\n\n\tasync function deleteHostsFile(hfile) {\n\t\tconst res = await fetch(apiHostsDir+\"/\"+hfile.Name, {\n\t\t\tmethod: \"DELETE\",\n\t\t});\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: deleteHostsFile: \", resError.message)\n\t\t\treturn;\n\t\t}\n\t\tdelete env.HostsFiles[hfile.Name]\n\t\tenv = env;\n\t\thostsFile = {\n\t\t\tName: \"\",\n\t\t\tRecords: [],\n\t\t}\n\t\tWuiPushNotif.Info(\"The host file '\"+ hfile.Name +\"' has been deleted\")\n\t}\n</script>\n\n<style>\n\t.nav-left {\n\t\tpadding: 0px;\n\t\twidth: 280px;\n\t\tfloat: left;\n\t}\n\t.nav-left .item {\n\t\tmargin: 4px 0px;\n\t}\n\t.content {\n\t\tfloat: left;\n\t\twidth: calc(100% - 300px);\n\t}\n\t.host {\n\t\tfont-family: monospace;\n\t\twidth: 100%;\n\t}\n\t.host.header {\n\t\tmargin: 1em 0px;\n\t\tfont-weight: bold;\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.host_name {\n\t\tdisplay: inline-block;\n\t\twidth: 240px;\n\t\tword-wrap: break-word;\n\t}\n\t.host_value {\n\t\tdisplay: inline-block;\n\t\twidth: 140px;\n\t}\n</style>\n\n<div class=\"hosts_d\">\n\t<div class=\"nav-left\">\n\t\t{#each Object.entries(env.HostsFiles) as [name,hf], name }\n\t\t<div class=\"item\">\n\t\t\t<a href=\"#\" on:click={getHostsFile(hf)}>\n\t\t\t\t{hf.Name}\n\t\t\t</a>\n\t\t</div>\n\t\t{/each}\n\n\t\t<br/>\n\n\t\t<label>\n\t\t\t<span>New hosts file:</span>\n\t\t\t<br/>\n\t\t\t<input bind:value={newHostsFile}>\n\t\t</label>\n\t\t<button on:click={createHostsFile}>\n\t\t\tCreate\n\t\t</button>\n\t</div>\n\n\t<div class=\"content\">\n\t\t{#if hostsFile.Name === \"\"}\n\t\t\t<div>\n\t\t\t\tSelect one of the hosts file to manage.\n\t\t\t</div>\n\t\t{:else}\n\t\t\t<p>\n\t\t\t\t{hostsFile.Name} ({hostsFile.Records.length} records)\n\t\t\t\t<button on:click={deleteHostsFile(hostsFile)}>\n\t\t\t\t\tDelete\n\t\t\t\t</button>\n\t\t\t</p>\n\t\t\t<div>\n\t\t\t\t<button on:click={addRecord}>\n\t\t\t\t\tAdd\n\t\t\t\t</button>\n\t\t\t</div>\n\n\t\t\t{#if newRecord !== null}\n\t\t\t\t<div class=\"host\">\n\t\t\t\t\t<input\n\t\t\t\t\t\tclass=\"host_name\"\n\t\t\t\t\t\tplaceholder=\"Domain name\"\n\t\t\t\t\t\tbind:value={newRecord.Name}\n\t\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tclass=\"host_value\"\n\t\t\t\t\t\tplaceholder=\"IP address\"\n\t\t\t\t\t\tbind:value={newRecord.Value}\n\t\t\t\t\t>\n\t\t\t\t\t<button on:click={handleHostsRecordCreate}>\n\t\t\t\t\t\tCreate\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t{/if}\n\n\t\t\t<div class=\"host header\">\n\t\t\t\t<span class=\"host_name\"> Domain name </span>\n\t\t\t\t<span class=\"host_value\"> IP address </span>\n\t\t\t</div>\n\n\t\t\t{#each hostsFile.Records as rr, idx (idx)}\n\t\t\t\t<div class=\"host\">\n\t\t\t\t\t<span class=\"host_name\"> {rr.Name} </span>\n\t\t\t\t\t<span class=\"host_value\"> {rr.Value} </span>\n\t\t\t\t\t<button on:click={handleHostsRecordDelete(rr, idx)}>\n\t\t\t\t\t\tX\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t{/each}\n\t\t{/if}\n\t</div>\n</div>\n","<script>\n\timport { onDestroy } from 'svelte';\n\timport { WuiPushNotif, WuiLabelHint } from 'wui.svelte';\n\timport { environment, nanoSeconds, setEnvironment } from './environment.js';\n\n\tconst apiMasterd = \"/api/master.d/\"\n\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: [],\n\t\tZoneFiles: {},\n\t};\n\tlet newZoneFile = \"\";\n\tlet activeZone = {\n\t\tName: \"\",\n\t};\n\n\tlet RRTypes = {\n\t\t1: 'A',\n\t\t2: 'NS',\n\t\t5: 'CNAME',\n\t\t12: 'PTR',\n\t\t15: 'MX',\n\t\t16: 'TXT',\n\t\t28: 'AAAA',\n\t};\n\n\tlet _rr = newRR()\n\tlet _rrMX = newMX();\n\n\tconst envUnsubscribe = environment.subscribe(value => {\n\t\tenv = value;\n\t});\n\tonDestroy(envUnsubscribe);\n\n\tfunction setActiveZone(zone) {\n\t\tif (zone.SOA === null) {\n\t\t\tzone.SOA = newSOA()\n\t\t}\n\t\tactiveZone = zone\n\t}\n\n\tasync function handleZoneFileCreate() {\n\t\tconst res = await fetch(apiMasterd + newZoneFile, {\n\t\t\tmethod: \"PUT\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleZoneFileCreate: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tactiveZone = await res.json()\n\t\tactiveZone.SOA = newSOA()\n\t\tenv.ZoneFiles[activeZone.Name] = activeZone\n\n\t\tWuiPushNotif.Info(\"The new zone file '\"+ newZoneFile +\"' has been created\")\n\t}\n\n\tasync function handleZoneFileDelete() {\n\t\tlet api = apiMasterd + activeZone.Name\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"DELETE\",\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleZoneFileDelete: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tWuiPushNotif.Info(\"The zone file '\"+ activeZone.Name + \"' has beed deleted\")\n\n\t\tdelete env.ZoneFiles[activeZone.Name]\n\t\tactiveZone = {\n\t\t\tName: \"\",\n\t\t}\n\t\tenv.ZoneFiles = env.ZoneFiles\n\n\t}\n\n\tfunction onSelectRRType() {\n\t\tswitch (_rr.Type) {\n\t\tcase 15:\n\t\t\t_rrMX = newMX()\n\t\t\tbreak\n\t\t}\n\t}\n\n\tasync function handleSOADelete() {\n\t\treturn handleDeleteRR(activeZone.SOA, -1)\n\t}\n\n\tasync function handleSOASave() {\n\t\t_rr = activeZone.SOA\n\t\treturn handleCreateRR()\n\t}\n\n\tasync function handleCreateRR() {\n\t\tswitch (_rr.Type) {\n\t\tcase 15:\n\t\t\t_rr.Value = _rrMX;\n\t\t\tbreak;\n\t\t}\n\n\t\tlet api = apiMasterd + activeZone.Name +\"/rr/\"+ _rr.Type;\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"POST\",\n\t\t\theaders: {\n \t\t\t'Content-Type': 'application/json',\n\t\t\t},\n\t\t\tbody: JSON.stringify(_rr),\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleCreateRR: \"+ resError.message)\n\t\t\treturn;\n\t\t}\n\n\t\tlet resRR = await res.json()\n\n\t\tif (resRR.Type === 6) {\n\t\t\tactiveZone.SOA = resRR\n\t\t\tWuiPushNotif.Info(\"SOA record has been saved\")\n\t\t} else {\n\t\t\tlet listRR = activeZone.Records[resRR.Name]\n\t\t\tif (typeof listRR === \"undefined\") {\n\t\t\t\tlistRR = [];\n\t\t\t}\n\t\t\tlistRR.push(resRR);\n\t\t\tactiveZone.Records[resRR.Name] = listRR\n\t\t\tWuiPushNotif.Info(\"The new record '\"+ resRR.Name +\"' has been created\")\n\t\t}\n\t}\n\n\tasync function handleDeleteRR(rr, idx) {\n\t\tlet api = apiMasterd + activeZone.Name +\"/rr/\"+ rr.Type\n\n\t\tconst res = await fetch(api, {\n\t\t\tmethod: \"DELETE\",\n\t\t\theaders: {\n \t\t\t'Content-Type': 'application/json',\n\t\t\t},\n\t\t\tbody: JSON.stringify(rr),\n\t\t})\n\n\t\tif (res.status >= 400) {\n\t\t\tconst resError = await res.json()\n\t\t\tWuiPushNotif.Error(\"ERROR: handleDeleteRR: \"+ resError.message)\n\t\t\treturn\n\t\t}\n\n\t\tWuiPushNotif.Info(\"The record '\"+ rr.Name +\"' has been deleted\")\n\n\t\tif (rr.Type == 6) { // SOA.\n\t\t\tactiveZone.SOA = newSOA()\n\t\t} else {\n\t\t\tlet listRR = activeZone.Records[rr.Name]\n\t\t\tlistRR.splice(idx, 1)\n\t\t\tactiveZone.Records[rr.Name] = listRR\n\t\t}\n\n\t\tlet resbody = await res.json()\n\t}\n\n\tfunction getTypeName(k) {\n\t\tlet v = RRTypes[k];\n\t\tif (v === \"\") {\n\t\t\treturn k;\n\t\t}\n\t\treturn v;\n\t}\n\n\tfunction newRR() {\n\t\treturn {\n\t\t\tName: \"\",\n\t\t\tType: 1,\n\t\t\tValue: \"\",\n\t\t};\n\t}\n\n\tfunction newMX() {\n\t\treturn {\n\t\t\tPreference: 1,\n\t\t\tExchange: \"\",\n\t\t}\n\t}\n\n\tfunction newSOA() {\n\t\treturn {\n\t\t\tName: \"\",\n\t\t\tType: 6,\n\t\t\tValue: {\n\t\t\t\tMName: \"\",\n\t\t\t\tRName: \"\",\n\t\t\t\tSerial: 0,\n\t\t\t\tRefresh: 0,\n\t\t\t\tRetry: 0,\n\t\t\t\tExpire: 0,\n\t\t\t\tMinimum: 0\n\t\t\t}\n\t\t};\n\t}\n</script>\n\n<style>\n\th4 {\n\t\tborder-bottom: 1px solid silver;\n\t}\n\t.nav-left {\n\t\tpadding: 0px;\n\t\twidth: 280px;\n\t\tfloat: left;\n\t}\n\t.nav-left .item {\n\t\tmargin: 4px 0px;\n\t\tcursor: pointer;\n\t\tcolor: rgb(0, 100, 200);\n\t}\n\t.content {\n\t\tfloat: left;\n\t\twidth: calc(100% - 300px);\n\t}\n\t.action-delete {\n\t\tmargin-left: 1em;\n\t}\n\t.actions {\n\t\tpadding: 1em;\n\t}\n\t.actions button {\n\t\twidth: 100%;\n\t}\n\n\tform {\n\t\tmargin: 1em 0px;\n\t\tpadding: 10px 10px 0px 10px;\n\t\tborder: 1px solid silver;\n\t}\n\tform > label > span {\n\t\twidth: 7em;\n\t\tdisplay: inline-block;\n\t}\n\tform > label > input {\n\t\twidth: calc(100% - 8em);\n\t}\n\tform > label > input.name {\n\t\twidth: 12em;\n\t}\n\n\t.rr {\n\t\tfont-family: monospace;\n\t\twidth: 100%;\n\t\tpadding: 1em 0px;\n\t}\n\t.rr.header {\n\t\tfont-weight: bold;\n\t}\n\t.rr > .name {\n\t\twidth: 20em;\n\t\tdisplay: inline-block;\n\t\tword-wrap: break-word;\n\t}\n\t.rr > .type {\n\t\twidth: 4em;\n\t\tdisplay: inline-block;\n\t}\n\t.rr > .ttl {\n\t\twidth: 6em;\n\t\tdisplay: inline-block;\n\t}\n\t.rr > .value {\n\t\tdisplay: inline-block;\n\t\tword-wrap: break-word;\n\t}\n</style>\n\n<div class=\"master_d\">\n\t<div class=\"nav-left\">\n\t\t{#each Object.entries(env.ZoneFiles) as [name, zoneFile]}\n\t\t\t<div class=\"item\">\n\t\t\t\t<span on:click={setActiveZone(zoneFile)}>\n\t\t\t\t\t{zoneFile.Name}\n\t\t\t\t</span>\n\t\t\t</div>\n\t\t{/each}\n\t\t<br/>\n\n\t\t<label>\n\t\t\t<span>New zone file:</span>\n\t\t\t<br/>\n\t\t\t<input bind:value={newZoneFile}>\n\t\t</label>\n\t\t<button on:click={handleZoneFileCreate}>\n\t\t\tCreate\n\t\t</button>\n\t</div>\n\n\t<div class=\"content\">\n\t\t{#if activeZone.Name === \"\"}\n\t\t\t<p>\n\t\t\t\tSelect one of the zone file to manage.\n\t\t\t</p>\n\t\t{:else}\n\t\t\t<h3>\n\t\t\t\t{activeZone.Name}\n\t\t\t\t<button\n\t\t\t\t\tclass=\"action-delete\"\n\t\t\t\t\ton:click={handleZoneFileDelete}\n\t\t\t\t>\n\t\t\t\t\tDelete\n\t\t\t\t</button>\n\t\t\t</h3>\n\n\t\t\t<h4>\n\t\t\t\tSOA record\n\t\t\t\t<button\n\t\t\t\t\tclass=\"action-delete\"\n\t\t\t\t\ton:click={handleSOADelete}\n\t\t\t\t>\n\t\t\t\t\tDelete\n\t\t\t\t</button>\n\t\t\t</h4>\n\t\t\t<div class=\"rr-soa\">\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Name server\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"The domain-name of the name server that was the\noriginal or primary source of data for this zone.\nIt should be domain-name where the rescached run.\"\n\t\t\t\t>\n\t\t\t\t\t<input bind:value={activeZone.SOA.Value.MName}>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Admin email\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo='Email address of the administrator responsible for\nthis zone.\nThe \"@\" on email address is replaced with dot, and if there is a dot before\n\"@\" it should be escaped with \"\\\".\nFor example, \"dns.admin@domain.tld\" would be written as\n\"dns\\.admin.domain.tld\".'\n\t\t\t\t>\n\t\t\t\t\t<input bind:value={activeZone.SOA.Value.RName}>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Serial\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Serial number for this zone. If a secondary name\nserver observes an increase in this number, the server will assume that the\nzone has been updated and initiate a zone transfer.\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Serial}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Refresh\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Number of seconds after which secondary name servers\nshould query the master for the SOA record, to detect zone changes.\nRecommendation for small and stable zones is 86400 seconds (24 hours).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Refresh}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Retry\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Number of seconds after which secondary name servers\nshould retry to request the serial number from the master if the master does\nnot respond.\nIt must be less than Refresh.\nRecommendation for small and stable zones is 7200 seconds (2 hours).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Retry}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Expire\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Number of seconds after which secondary name servers\nshould stop answering request for this zone if the master does not respond.\nThis value must be bigger than the sum of Refresh and Retry.\nRecommendation for small and stable zones is 3600000 seconds (1000 hours).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Expire}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<WuiLabelHint\n\t\t\t\t\ttitle=\"Minimum\"\n\t\t\t\t\ttitle_width=\"150px\"\n\t\t\t\t\tinfo=\"Time to live for purposes of negative caching.\nRecommendation for small and stable zones is 1800 seconds (30 min).\"\n\t\t\t\t>\n\t\t\t\t\t<input\n\t\t\t\t\t\tbind:value={activeZone.SOA.Value.Minimum}\n\t\t\t\t\t\ttype=number\n\t\t\t\t\t\tmin=0\n\t\t\t\t\t>\n\t\t\t\t</WuiLabelHint>\n\t\t\t\t<div class=\"actions\">\n\t\t\t\t\t<button on:click={handleSOASave}>\n\t\t\t\t\t\tSave\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t</div>\n\n\t\t\t<h4> List records </h4>\n\t\t\t<div class=\"rr header\">\n\t\t\t\t<span class=\"name\">\n\t\t\t\t\tName\n\t\t\t\t</span>\n\t\t\t\t<span class=\"type\">\n\t\t\t\t\tType\n\t\t\t\t</span>\n\t\t\t\t<span class=\"value\">\n\t\t\t\t\tValue\n\t\t\t\t</span>\n\t\t\t</div>\n\n\t\t\t{#each Object.entries(activeZone.Records) as [dname, listRR]}\n\t\t\t\t{#each listRR as rr, idx}\n\t\t\t\t\t<div class=\"rr\">\n\t\t\t\t\t\t<span class=\"name\">\n\t\t\t\t\t\t\t{rr.Name}\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<span class=\"type\">\n\t\t\t\t\t\t\t{getTypeName(rr.Type)}\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<span class=\"value\">\n\t\t\t\t\t\t\t{rr.Value}\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<button on:click={handleDeleteRR(rr, idx)}>\n\t\t\t\t\t\t\tX\n\t\t\t\t\t\t</button>\n\t\t\t\t\t</div>\n\t\t\t\t{/each}\n\t\t\t{/each}\n\n\t\t\t<form on:submit|preventDefault={handleCreateRR}>\n\t\t\t\t<label>\n\t\t\t\t\t<span>\n\t\t\t\t\t\tType:\n\t\t\t\t\t</span>\n\t\t\t\t\t<select\n\t\t\t\t\t\tbind:value={_rr.Type}\n\t\t\t\t\t\ton:blur={onSelectRRType}\n\t\t\t\t\t>\n\t\t\t\t\t\t{#each Object.entries(RRTypes) as [k, v]}\n\t\t\t\t\t\t\t<option value={parseInt(k)}>\n\t\t\t\t\t\t\t\t{v}\n\t\t\t\t\t\t\t</option>\n\t\t\t\t\t\t{/each}\n\t\t\t\t\t</select>\n\t\t\t\t</label>\n\n\t\t\t\t{#if _rr.Type === 1 || _rr.Type === 2 || _rr.Type === 5 ||\n\t\t\t\t\t_rr.Type === 16 || _rr.Type === 28\n\t\t\t\t}\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tName:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input class=\"name\" bind:value={_rr.Name}>\n\t\t\t\t\t\t.{activeZone.Name}\n\t\t\t\t\t</label>\n\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tValue:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rr.Value}>\n\t\t\t\t\t</label>\n\t\t\t\t{:else if _rr.Type === 12} <!-- PTR -->\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tName:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rr.Name}>\n\t\t\t\t\t</label>\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tValue:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input class=\"name\" bind:value={_rr.Value}>\n\t\t\t\t\t\t.{activeZone.Name}\n\t\t\t\t\t</label>\n\t\t\t\t{:else if _rr.Type === 15} <!-- MX -->\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tName:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input class=\"name\" bind:value={_rr.Name}>\n\t\t\t\t\t\t.{activeZone.Name}\n\t\t\t\t\t</label>\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tPreference:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rrMX.Preference} type=number min=1 max=65535>\n\t\t\t\t\t</label>\n\t\t\t\t\t<label>\n\t\t\t\t\t\t<span>\n\t\t\t\t\t\t\tExchange:\n\t\t\t\t\t\t</span>\n\t\t\t\t\t\t<input bind:value={_rrMX.Exchange}>\n\t\t\t\t\t</label>\n\t\t\t\t{/if}\n\n\t\t\t\t<div class=\"actions\">\n\t\t\t\t\t<button class=\"create\" type=submit>\n\t\t\t\t\t\tCreate\n\t\t\t\t\t</button>\n\t\t\t\t</div>\n\t\t\t</form>\n\t\t{/if}\n\t</div> <!-- content -->\n</div> <!-- master_d -->\n","<script>\n\timport { onMount } from 'svelte';\n\timport { WuiNotif, WuiPushNotif } from 'wui.svelte';\n\n\timport { apiEnvironment, environment, nanoSeconds, setEnvironment } from './environment.js';\n\timport Dashboard from './Dashboard.svelte';\n\timport Environment from './Environment.svelte';\n\timport HostsBlock from './HostsBlock.svelte';\n\timport HostsDir from './HostsDir.svelte';\n\timport MasterDir from './MasterDir.svelte';\n\n\tconst stateEnvironment = \"environment\";\n\tconst stateHostsBlock = \"hosts_block\";\n\tconst stateHostsDir = \"hosts_d\";\n\tconst stateMasterDir = \"master_d\";\n\n\tlet state;\n\tlet env = {\n\t\tNameServers: [],\n\t\tHostsBlocks: [],\n\t\tHostsFiles: {},\n\t};\n\n\tonMount(async () => {\n\t\tconst res = await fetch(apiEnvironment);\n\t\tif (res.status >= 400) {\n\t\t\tWuiPushNotif.Error(\"ERROR: {apiEnvironment}: \",\n\t\t\t\tres.status, res.statusText);\n\t\t\treturn;\n\t\t}\n\n\t\tsetEnvironment(await res.json());\n \t\tstate = window.location.hash.slice(1);\n\t});\n</script>\n\n<style>\n\tdiv.main {\n\t\tmargin: 0 auto;\n\t\twidth: 800px;\n\t\tpadding: 0px 1em;\n\t}\n\tnav.menu {\n\t\tcolor: #ff3e00;\n\t\ttext-transform: uppercase;\n\t\tfont-weight: 100;\n\t\tmargin-bottom: 2em;\n\t}\n\t.active {\n\t\tpadding-bottom: 4px;\n\t\tborder-bottom: 4px solid #ff3e00;\n\t}\n\t@media (max-width: 900px) {\n\t\tdiv.main {\n\t\t\twidth: calc(100% - 2em);\n\t\t}\n\t}\n</style>\n\n<WuiNotif timeout=3000 />\n\n<div class=\"main\">\n\t<nav class=\"menu\">\n\t\t<a\n\t\t\thref=\"#home\"\n\t\t\ton:click={()=>state=\"\"}\n\t\t\tclass:active=\"{state===''||state==='home'}\"\n\t\t>\n\t\t\trescached\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#environment\"\n\t\t\ton:click={()=>state=stateEnvironment}\n\t\t\tclass:active=\"{state===stateEnvironment}\"\n\t\t>\n\t\t\tEnvironment\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#{stateHostsBlock}\"\n\t\t\ton:click={()=>state=stateHostsBlock}\n\t\t\tclass:active=\"{state===stateHostsBlock}\"\n\t\t>\n\t\t\tHosts blocks\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#{stateHostsDir}\"\n\t\t\ton:click={()=>state=stateHostsDir}\n\t\t\tclass:active=\"{state === stateHostsDir}\"\n\t\t>\n\t\t\thosts.d\n\t\t</a>\n\t\t/\n\t\t<a\n\t\t\thref=\"#{stateMasterDir}\"\n\t\t\ton:click={()=>state=stateMasterDir}\n\t\t\tclass:active=\"{state === stateMasterDir}\"\n\t\t>\n\t\t\tmaster.d\n\t\t</a>\n\t</nav>\n\n\t{#if state === stateEnvironment}\n\t\t<Environment/>\n\t{:else if state === stateHostsBlock}\n\t\t<HostsBlock/>\n\t{:else if state === stateHostsDir}\n\t\t<HostsDir/>\n\t{:else if state === stateMasterDir}\n\t\t<MasterDir/>\n\t{:else}\n\t\t<Dashboard/>\n\t{/if}\n</div>\n","import App from \"./App.svelte\"\n\nconst app = new App({\n\ttarget: document.body,\n})\n\nexport default app\n"],"names":["linear"],"mappings":"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"} \ No newline at end of file
diff --git a/_www/public/global.css b/_www/public/global.css
deleted file mode 100644
index 2546012..0000000
--- a/_www/public/global.css
+++ /dev/null
@@ -1,79 +0,0 @@
-html,
-body {
- position: relative;
- width: 100%;
- height: 100%;
-}
-
-body {
- color: #333;
- margin: 0;
- padding: 8px;
- box-sizing: border-box;
- font-family: -apple-system, BlinkMacSystemFont, "Segoe UI", Roboto,
- Oxygen-Sans, Ubuntu, Cantarell, "Helvetica Neue", sans-serif;
-}
-
-a {
- color: rgb(0, 100, 200);
- text-decoration: none;
-}
-
-a:hover {
- text-decoration: underline;
-}
-
-a:visited {
- color: rgb(0, 80, 160);
-}
-
-label {
- display: block;
-}
-
-input,
-button,
-select,
-textarea {
- font-family: inherit;
- font-size: inherit;
- padding: 0.4em;
- margin: 0 0 0.5em 0;
- box-sizing: border-box;
- border: 1px solid #ccc;
- border-radius: 2px;
-}
-
-input:disabled {
- color: #ccc;
-}
-
-input[type="range"] {
- height: 0;
-}
-
-button {
- color: #333;
- background-color: #f4f4f4;
- outline: none;
-}
-
-button:disabled {
- color: #999;
-}
-
-button:not(:disabled):active {
- background-color: #ddd;
-}
-
-button:focus {
- border-color: #666;
-}
-
-h1,
-h2 {
- color: #ff3e00;
- text-transform: uppercase;
- font-size: normal;
- font-weight: 100;
-}
diff --git a/_www/public/index.html b/_www/public/index.html
deleted file mode 100644
index e8e6f92..0000000
--- a/_www/public/index.html
+++ /dev/null
@@ -1,17 +0,0 @@
-<!DOCTYPE html>
-<html lang="en">
- <head>
- <meta charset="utf-8" />
- <meta name="viewport" content="width=device-width,initial-scale=1" />
-
- <title>rescached</title>
-
- <link rel="icon" type="image/png" href="/favicon.png" />
- <link rel="stylesheet" href="/global.css" />
- <link rel="stylesheet" href="/build/bundle.css" />
-
- <script defer src="/build/bundle.js"></script>
- </head>
-
- <body></body>
-</html>
diff --git a/_www/rescached.js b/_www/rescached.js
new file mode 100644
index 0000000..5610e4e
--- /dev/null
+++ b/_www/rescached.js
@@ -0,0 +1,289 @@
+const RRTypes = {
+ 1: "A",
+ 2: "NS",
+ 3: "MD",
+ 4: "MF",
+ 5: "CNAME",
+ 6: "SOA",
+ 7: "MB",
+ 8: "MG",
+ 9: "MR",
+ 10: "NULL",
+ 11: "WKS",
+ 12: "PTR",
+ 13: "HINFO",
+ 14: "MINFO",
+ 15: "MX",
+ 16: "TXT",
+ 28: "AAAA",
+ 33: "SRV",
+ 41: "OPT",
+}
+
+function getRRTypeName(k) {
+ let v = RRTypes[k]
+ if (v === "") {
+ return k
+ }
+ return v
+}
+
+class Rescached {
+ static nanoSeconds = 1000000000
+ static apiHostsd = "/api/hosts.d/"
+ static apiZoned = "/api/zone.d/"
+
+ constructor(server) {
+ this.server = server
+ this.env = {}
+ }
+
+ async HostsFileCreate(name) {
+ const httpRes = await fetch(
+ this.server + Rescached.apiHostsd + name,
+ {
+ method: "PUT",
+ },
+ )
+ let res = await httpRes.json()
+ if (res.code === 200) {
+ this.env.HostsFiles[name] = {
+ Name: name,
+ Records: [],
+ }
+ }
+ return res
+ }
+
+ async Search(query) {
+ console.log("Search: ", query)
+ const res = await fetch(
+ this.server + "/api/caches" + "?query=" + query,
+ )
+ return await res.json()
+ }
+
+ async getEnvironment() {
+ const httpRes = await fetch(this.server + "/api/environment")
+ const res = await httpRes.json()
+
+ if (httpRes.status === 200) {
+ res.data.PruneDelay =
+ res.data.PruneDelay / Rescached.nanoSeconds
+ res.data.PruneThreshold =
+ res.data.PruneThreshold /
+ Rescached.nanoSeconds
+
+ for (let k in res.data.HostsFiles) {
+ if (!res.data.HostsFiles.hasOwnProperty(k)) {
+ continue
+ }
+ let hf = res.data.HostsFiles[k]
+ if (typeof hf.Records === "undefined") {
+ hf.Records = []
+ }
+ }
+ this.env = res.data
+ }
+ return res
+ }
+
+ GetRRTypeName(k) {
+ let v = RRTypes[k]
+ if (v === "") {
+ return k
+ }
+ return v
+ }
+
+ async HostsFileDelete(name) {
+ const httpRes = await fetch(
+ this.server + Rescached.apiHostsd + name,
+ {
+ method: "DELETE",
+ },
+ )
+ const res = await httpRes.json()
+ if (httpRes.status === 200) {
+ delete this.env.HostsFiles[name]
+ }
+ return res
+ }
+
+ async HostsFileGet(name) {
+ const httpRes = await fetch(
+ this.server + Rescached.apiHostsd + name,
+ )
+ let res = await httpRes.json()
+ if (httpRes.Status === 200) {
+ this.env.HostsFiles[name] = {
+ Name: name,
+ Records: res.data,
+ }
+ }
+ return res
+ }
+
+ async HostsFileRecordAdd(hostsFile, domain, value) {
+ let params = new URLSearchParams()
+ params.set("domain", domain)
+ params.set("value", value)
+
+ const api =
+ this.server +
+ Rescached.apiHostsd +
+ hostsFile +
+ "/rr" +
+ "?" +
+ params.toString()
+
+ const httpRes = await fetch(api, {
+ method: "POST",
+ })
+ const res = await httpRes.json()
+ if (httpRes.Status === 200) {
+ let hf = this.env.HostsFiles[hostsFile]
+ hf.Records.push(res.data)
+ }
+ return res
+ }
+
+ async HostsFileRecordDelete(hostsFile, domain) {
+ let params = new URLSearchParams()
+ params.set("domain", domain)
+
+ const api =
+ this.server +
+ Rescached.apiHostsd +
+ hostsFile +
+ "/rr" +
+ "?" +
+ params.toString()
+
+ const httpRes = await fetch(api, {
+ method: "DELETE",
+ })
+ const res = await httpRes.json()
+ if (httpRes.Status === 200) {
+ let hf = this.env.HostsFiles[hostsFile]
+ for (let x = 0; x < hf.Records.length; x++) {
+ if (hf.Records[x].Name === domain) {
+ hf.Records.splice(x, 1)
+ }
+ }
+ }
+ return res
+ }
+
+ async updateEnvironment() {
+ let got = {}
+
+ Object.assign(got, this.env)
+
+ got.PruneDelay = got.PruneDelay * this.nanoSeconds
+ got.PruneThreshold = got.PruneThreshold * this.nanoSeconds
+
+ const httpRes = await fetch(
+ this.server + "/api/environment",
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify(got),
+ },
+ )
+
+ return await httpRes.json()
+ }
+
+ async updateHostsBlocks(hostsBlocks) {
+ const httpRes = await fetch(
+ this.server + "/api/hosts_block",
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify(hostsBlocks),
+ },
+ )
+ return await httpRes.json()
+ }
+
+ async ZoneFileCreate(name) {
+ const httpRes = await fetch(
+ this.server + Rescached.apiZoned + name,
+ {
+ method: "PUT",
+ },
+ )
+ let res = await httpRes.json()
+ if (res.code == 200) {
+ this.env.ZoneFiles[name] = res.data
+ }
+ return res
+ }
+
+ async ZoneFileDelete(name) {
+ const httpRes = await fetch(
+ this.server + Rescached.apiZoned + name,
+ {
+ method: "DELETE",
+ },
+ )
+ let res = await httpRes.json()
+ if (res.code == 200) {
+ delete this.env.ZoneFiles[name]
+ }
+ return res
+ }
+
+ async ZoneFileRecordCreate(name, rr) {
+ let api =
+ this.server +
+ Rescached.apiZoned +
+ name +
+ "/rr/" +
+ rr.Type
+ const httpRes = await fetch(api, {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify(rr),
+ })
+ let res = await httpRes.json()
+ if (httpRes.status === 200) {
+ let zf = this.env.ZoneFiles[name]
+ if (rr.Type == 6) {
+ // SOA.
+ zf.SOA = res.data
+ } else {
+ zf.Records = res.data
+ }
+ }
+ return res
+ }
+
+ async ZoneFileRecordDelete(name, rr) {
+ let api =
+ this.server +
+ Rescached.apiZoned +
+ name +
+ "/rr/" +
+ rr.Type
+ const httpRes = await fetch(api, {
+ method: "DELETE",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ body: JSON.stringify(rr),
+ })
+ let res = await httpRes.json()
+ if (httpRes.status === 200) {
+ this.env.ZoneFiles[name].Records = res.data
+ }
+ return res
+ }
+}
diff --git a/_www/rollup.config.js b/_www/rollup.config.js
deleted file mode 100644
index ce3c9eb..0000000
--- a/_www/rollup.config.js
+++ /dev/null
@@ -1,71 +0,0 @@
-import svelte from 'rollup-plugin-svelte';
-import resolve from '@rollup/plugin-node-resolve';
-import commonjs from '@rollup/plugin-commonjs';
-import livereload from 'rollup-plugin-livereload';
-import { terser } from 'rollup-plugin-terser';
-
-const production = !process.env.ROLLUP_WATCH;
-
-export default {
- input: 'src/main.js',
- output: {
- sourcemap: true,
- format: 'iife',
- name: 'app',
- file: 'public/build/bundle.js'
- },
- plugins: [
- svelte({
- // enable run-time checks when not in production
- dev: !production,
- // we'll extract any component CSS out into
- // a separate file - better for performance
- css: css => {
- css.write('public/build/bundle.css');
- }
- }),
-
- // If you have external dependencies installed from
- // npm, you'll most likely need these plugins. In
- // some cases you'll need additional configuration -
- // consult the documentation for details:
- // https://github.com/rollup/plugins/tree/master/packages/commonjs
- resolve({
- browser: true,
- dedupe: ['svelte']
- }),
- commonjs(),
-
- // In dev mode, call `npm run start` once
- // the bundle has been generated
- !production && serve(),
-
- // Watch the `public` directory and refresh the
- // browser on changes when not in production
- !production && livereload('public'),
-
- // If we're building for production (npm run build
- // instead of npm run dev), minify
- production && terser()
- ],
- watch: {
- clearScreen: false
- }
-};
-
-function serve() {
- let started = false;
-
- return {
- writeBundle() {
- if (!started) {
- started = true;
-
- require('child_process').spawn('npm', ['run', 'start', '--', '--dev'], {
- stdio: ['ignore', 'inherit', 'inherit'],
- shell: true
- });
- }
- }
- };
-}
diff --git a/_www/src/App.svelte b/_www/src/App.svelte
deleted file mode 100644
index caacc4d..0000000
--- a/_www/src/App.svelte
+++ /dev/null
@@ -1,116 +0,0 @@
-<script>
- import { onMount } from 'svelte';
- import { WuiNotif, WuiPushNotif } from 'wui.svelte';
-
- import { apiEnvironment, environment, nanoSeconds, setEnvironment } from './environment.js';
- import Dashboard from './Dashboard.svelte';
- import Environment from './Environment.svelte';
- import HostsBlock from './HostsBlock.svelte';
- import HostsDir from './HostsDir.svelte';
- import MasterDir from './MasterDir.svelte';
-
- const stateEnvironment = "environment";
- const stateHostsBlock = "hosts_block";
- const stateHostsDir = "hosts_d";
- const stateMasterDir = "master_d";
-
- let state;
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {},
- };
-
- onMount(async () => {
- const res = await fetch(apiEnvironment);
- if (res.status >= 400) {
- WuiPushNotif.Error("ERROR: {apiEnvironment}: ",
- res.status, res.statusText);
- return;
- }
-
- setEnvironment(await res.json());
- state = window.location.hash.slice(1);
- });
-</script>
-
-<style>
- div.main {
- margin: 0 auto;
- width: 800px;
- padding: 0px 1em;
- }
- nav.menu {
- color: #ff3e00;
- text-transform: uppercase;
- font-weight: 100;
- margin-bottom: 2em;
- }
- .active {
- padding-bottom: 4px;
- border-bottom: 4px solid #ff3e00;
- }
- @media (max-width: 900px) {
- div.main {
- width: calc(100% - 2em);
- }
- }
-</style>
-
-<WuiNotif timeout=3000 />
-
-<div class="main">
- <nav class="menu">
- <a
- href="#home"
- on:click={()=>state=""}
- class:active="{state===''||state==='home'}"
- >
- rescached
- </a>
- /
- <a
- href="#environment"
- on:click={()=>state=stateEnvironment}
- class:active="{state===stateEnvironment}"
- >
- Environment
- </a>
- /
- <a
- href="#{stateHostsBlock}"
- on:click={()=>state=stateHostsBlock}
- class:active="{state===stateHostsBlock}"
- >
- Hosts blocks
- </a>
- /
- <a
- href="#{stateHostsDir}"
- on:click={()=>state=stateHostsDir}
- class:active="{state === stateHostsDir}"
- >
- hosts.d
- </a>
- /
- <a
- href="#{stateMasterDir}"
- on:click={()=>state=stateMasterDir}
- class:active="{state === stateMasterDir}"
- >
- master.d
- </a>
- </nav>
-
- {#if state === stateEnvironment}
- <Environment/>
- {:else if state === stateHostsBlock}
- <HostsBlock/>
- {:else if state === stateHostsDir}
- <HostsDir/>
- {:else if state === stateMasterDir}
- <MasterDir/>
- {:else}
- <Dashboard/>
- {/if}
-</div>
diff --git a/_www/src/Dashboard.svelte b/_www/src/Dashboard.svelte
deleted file mode 100644
index 97db3a7..0000000
--- a/_www/src/Dashboard.svelte
+++ /dev/null
@@ -1,136 +0,0 @@
-<script>
- import {WuiPushNotif} from 'wui.svelte';
- import {getRRTypeName} from './common.js';
-
- const apiCaches = "/api/caches";
-
- let query = "";
- let listMsg = [];
-
- async function handleSearch() {
- const res = await fetch(apiCaches+"?query="+ query)
-
- if (res.status >= 400) {
- const resbody = await res.json()
- WuiPushNotif.Error("ERROR: "+ resbody.message)
- return;
- }
-
- listMsg = await res.json()
- }
-
- async function handleRemoveFromCaches(name) {
- const res = await fetch(apiCaches+"?name="+ name, {
- method: "DELETE",
- })
-
- if (res.status >= 400) {
- const resbody = await res.json()
- WuiPushNotif.Error("ERROR: "+ resbody.message)
- return;
- }
-
- for (let x = 0; x < listMsg.length; x++) {
- if (listMsg[x].Question.Name === name) {
- listMsg.splice(x, 1)
- listMsg = listMsg
- break
- }
- }
-
- const msg = await res.json()
-
- WuiPushNotif.Info(msg.message)
- }
-</script>
-
-<style>
- .message {
- padding: 1em 0px;
- border-bottom: 1px solid silver;
- }
- .rr {
- margin-left: 1em;
- width: 100%;
- }
- .rr.header {
- font-weight: bold;
- }
- .rr span {
- display: inline-block;
- }
- .kind {
- width: 9em;
- }
- .type {
- width: 5em;
- }
- .ttl {
- width: 6em;
- }
- .value {
- word-wrap: anywhere;
- width: calc(100% - 24em);
- }
-</style>
-
-<div class="dashboard">
- <div class="search">
- Caches:
- <input bind:value={query}>
- <button on:click={handleSearch}>
- Search
- </button>
- </div>
- {#each listMsg as msg (msg)}
- <div class="message">
- <div class="qname">
- {msg.Question.Name}
-
- <button
- class="b-remove"
- on:click={handleRemoveFromCaches(msg.Question.Name)}
- >
- Remove from caches
- </button>
- </div>
- <div class="rr header">
- <span class="kind"> </span>
- <span class="type"> Type </span>
- <span class="ttl"> TTL </span>
- <span class="value"> Value </span>
- </div>
-
- {#if msg.Answer !== null && msg.Answer.length > 0}
- {#each msg.Answer as rr}
- <div class="rr">
- <span class="kind"> Answer </span>
- <span class="type"> {getRRTypeName(rr.Type)} </span>
- <span class="ttl"> {rr.TTL} </span>
- <span class="value"> {JSON.stringify(rr.Value, null, 2)} </span>
- </div>
- {/each}
- {/if}
- {#if msg.Authority !== null && msg.Authority.length > 0}
- {#each msg.Authority as rr}
- <div class="rr">
- <span class="kind"> Authority </span>
- <span class="type"> {getRRTypeName(rr.Type)} </span>
- <span class="ttl"> {rr.TTL} </span>
- <span class="value"> {JSON.stringify(rr.Value, null, 2)} </span>
- </div>
- {/each}
- {/if}
- {#if msg.Additional !== null && msg.Additional.length > 0}
- {#each msg.Additional as rr}
- <div class="rr">
- <span class="kind"> Additional </span>
- <span class="type"> {getRRTypeName(rr.Type)} </span>
- <span class="ttl"> {rr.TTL} </span>
- <span class="value"> {JSON.stringify(rr.Value, null, 2)} </span>
- </div>
- {/each}
- {/if}
- </div>
- {/each}
-</div>
diff --git a/_www/src/Environment.svelte b/_www/src/Environment.svelte
deleted file mode 100644
index 1b284b5..0000000
--- a/_www/src/Environment.svelte
+++ /dev/null
@@ -1,286 +0,0 @@
-<script>
- import { onDestroy } from 'svelte';
-
- import { apiEnvironment, environment, nanoSeconds } from './environment.js';
- import { WuiPushNotif } from "wui.svelte";
- import { WuiLabelHint, WuiInputNumber, WuiInputIPPort } from "wui.svelte";
-
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {},
- };
-
- const envUnsubscribe = environment.subscribe(value => {
- env = value;
- });
-
- onDestroy(envUnsubscribe);
-
- const defTitleWidth = "300px";
-
- function addNameServer() {
- env.NameServers = [...env.NameServers, '']
- }
-
- function deleteNameServer(ns) {
- for (let x = 0; x < env.NameServers.length; x++) {
- if (env.NameServers[x] === ns) {
- env.NameServers.splice(x, 1);
- env.NameServers = env.NameServers;
- break;
- }
- }
- }
-
- async function updateEnvironment() {
- let got = {};
-
- Object.assign(got, env)
- environment.set(env)
-
- got.PruneDelay = got.PruneDelay * nanoSeconds;
- got.PruneThreshold = got.PruneThreshold * nanoSeconds;
-
- const res = await fetch(apiEnvironment, {
- method: "POST",
- headers: {
- "Content-Type": "application/json",
- },
- body: JSON.stringify(got),
- });
-
- if (res.status >= 400) {
- const resbody = await res.json()
- WuiPushNotif.Error("ERROR: ", resbody.message)
- return;
- }
-
- WuiPushNotif.Info("The environment succesfully updated ...")
- }
-</script>
-
-<style>
- input {
- width: 100%;
- }
- .input-deletable {
- width: 100%;
- }
- .input-deletable > input {
- max-width: calc(100% - 100px);
- }
- .input-deletable > button {
- width: 80px;
- }
- .input-checkbox {
- width: 100%;
- }
- .input-checkbox input[type="checkbox"] {
- width: auto;
- }
- .section-bottom {
- margin: 2em 0px 0px 0px;
- padding: 1em;
- border-top: 1px solid black;
- }
-</style>
-
-<div class="environment">
-<p>
-This page allow you to change the rescached environment.
-Upon save, the rescached service will be restarted.
-</p>
-
-<h3>rescached</h3>
-<div>
- <WuiLabelHint
- title="System resolv.conf"
- title_width="{defTitleWidth}"
- info="A path to dynamically generated resolv.conf(5) by
-resolvconf(8). If set, the nameserver values in referenced file will
-replace 'parent' value and 'parent' will become a fallback in
-case the referenced file being deleted or can not be parsed."
- >
- <input
- bind:value={env.FileResolvConf}
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="Debug level"
- title_width="{defTitleWidth}"
- info="This option only used for debugging program or if user
-want to monitor what kind of traffic goes in and out of rescached."
- >
- <WuiInputNumber
- min=0
- max=3
- bind:value={env.Debug}
- unit=""
- />
- </WuiLabelHint>
-</div>
-
-<h3>DNS server</h3>
-<div>
- <WuiLabelHint
- title="Parent name servers"
- title_width="{defTitleWidth}"
- info="List of parent DNS servers."
- >
- </WuiLabelHint>
-
- {#each env.NameServers as ns}
- <div class="input-deletable">
- <input bind:value={ns}>
- <button on:click={deleteNameServer(ns)}>
- Delete
- </button>
- </div>
- {/each}
-
- <button
- on:click={addNameServer}
- >
- Add
- </button>
-
- <WuiLabelHint
- title="Listen address"
- title_width="{defTitleWidth}"
- info="Address in local network where rescached will
-listening for query from client through UDP and TCP.
-<br/>
-If you want rescached to serve a query from another host in your local
-network, change this value to <tt>0.0.0.0:53</tt>."
- >
- <WuiInputIPPort
- bind:value={env.ListenAddress}
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="HTTP listen port"
- title_width="{defTitleWidth}"
- info="Port to serve DNS over HTTP"
- >
- <WuiInputNumber
- min=0
- max=65535
- bind:value={env.HTTPPort}
- unit=""
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="TLS listen port"
- title_width="{defTitleWidth}"
- info="Port to listen for DNS over TLS"
- >
- <WuiInputNumber
- min=0
- max=65535
- bind:value={env.TLSPort}
- unit=""
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="TLS certificate"
- title_width="{defTitleWidth}"
- info="Path to certificate file to serve DNS over TLS and
-HTTPS">
- <input
- placeholder="/path/to/certificate"
- bind:value={env.TLSCertFile}
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="TLS private key"
- title_width="{defTitleWidth}"
- info="Path to certificate private key file to serve DNS over TLS and
-HTTPS."
- >
- <input
- placeholder="/path/to/certificate/private.key"
- bind:value={env.TLSPrivateKey}
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="TLS allow insecure"
- title_width="{defTitleWidth}"
- info="If its true, allow serving DoH and DoT with self signed
-certificate."
- >
- <div class="input-checkbox">
- <input
- type=checkbox
- bind:checked={env.TLSAllowInsecure}
- >
- <span class="suffix">
- Yes
- </span>
- </div>
- </WuiLabelHint>
-
- <WuiLabelHint
- title="DoH behind proxy"
- title_width="{defTitleWidth}"
- info="If its true, serve DNS over HTTP only, even if
-certificate files is defined.
-This allow serving DNS request forwarded by another proxy server."
- >
- <div class="input-checkbox">
- <input
- type=checkbox
- bind:checked={env.DoHBehindProxy}
- >
- <span class="suffix">
- Yes
- </span>
- </div>
- </WuiLabelHint>
-
- <WuiLabelHint
- title="Prune delay"
- title_width="{defTitleWidth}"
- info="Delay for pruning caches.
-Every N seconds, rescached will traverse all caches and remove response that
-has not been accessed less than cache.prune_threshold.
-Its value must be equal or greater than 1 hour (3600 seconds).
-"
- >
- <WuiInputNumber
- min=3600
- max=36000
- bind:value={env.PruneDelay}
- unit="seconds"
- />
- </WuiLabelHint>
-
- <WuiLabelHint
- title="Prune threshold"
- title_width="{defTitleWidth}"
- info="The duration when the cache will be considered expired.
-Its value must be negative and greater or equal than -1 hour (-3600 seconds)."
- >
- <WuiInputNumber
- min=-36000
- max=-3600
- bind:value={env.PruneThreshold}
- unit="seconds"
- />
- </WuiLabelHint>
-</div>
-
- <div class="section-bottom">
- <div>
- <button on:click={updateEnvironment}>
- Save
- </button>
- </div>
- </div>
-</div>
diff --git a/_www/src/HostsBlock.svelte b/_www/src/HostsBlock.svelte
deleted file mode 100644
index 71f720a..0000000
--- a/_www/src/HostsBlock.svelte
+++ /dev/null
@@ -1,112 +0,0 @@
-<script>
- import { onDestroy } from 'svelte';
- import { WuiPushNotif } from 'wui.svelte';
- import { environment, nanoSeconds, setEnvironment } from './environment.js';
-
- const apiHostsBlock = "/api/hosts_block"
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {},
- };
-
- const envUnsubscribe = environment.subscribe(value => {
- env = value;
- });
- onDestroy(envUnsubscribe);
-
- async function updateHostsBlocks() {
- const res = await fetch(apiHostsBlock, {
- method: "POST",
- headers: {
- "Content-Type": "application/json",
- },
- body: JSON.stringify(env.HostsBlocks),
- });
-
- if (res.status >= 400) {
- WuiPushNotif.Error("ERROR: ", res.status, res.statusText)
- return;
- }
-
- setEnvironment(await res.json());
- }
-</script>
-
-<style>
- .block_source {
- width: calc(100% - 2em);
- overflow: auto;
- font-size: 12px;
- }
- .block_source input:disabled {
- color: black;
- }
- .item span {
- display: inline-block;
- margin-right: 1em;
- }
- .item.header {
- font-weight: bold;
- margin-bottom: 1em;
- border-bottom: 1px solid silver;
- }
- .item > span:nth-child(1) {
- width: 4em;
- }
- .item > span:nth-child(2) {
- width: 15em;
- }
- .item > span:nth-child(3) {
- width: 23em;
- }
- .item > span:nth-child(3) input {
- width: 100%;
- }
- .item > span:nth-child(4) {
- width: 16em;
- }
-</style>
-
-<div class="hosts-block">
- <p>
- Configure the source of blocked hosts file.
- </p>
-
- <div class="block_source">
- <div class="item header">
- <span> Enabled </span>
- <span> Name </span>
- <span> URL </span>
- <span> Last updated </span>
- </div>
- {#each env.HostsBlocks as hostsBlock}
- <div class="item">
- <span>
- <input
- type=checkbox
- bind:checked={hostsBlock.IsEnabled}
- >
- </span>
- <span>
- {hostsBlock.Name}
- </span>
- <span>
- <input
- bind:value={hostsBlock.URL}
- disabled
- >
- </span>
- <span>
- {hostsBlock.LastUpdated}
- </span>
- </div>
- {/each}
- </div>
-
- <div>
- <button on:click={updateHostsBlocks}>
- Save
- </button>
- </div>
-</div>
diff --git a/_www/src/HostsDir.svelte b/_www/src/HostsDir.svelte
deleted file mode 100644
index f45732d..0000000
--- a/_www/src/HostsDir.svelte
+++ /dev/null
@@ -1,250 +0,0 @@
-<script> import { onDestroy } from 'svelte';
- import { WuiPushNotif } from 'wui.svelte';
- import { apiEnvironment, environment, nanoSeconds } from './environment.js';
-
- const apiHostsDir = "/api/hosts.d"
-
- let env = {
- HostsFiles: {},
- };
- let hostsFile = {
- Name: "",
- Records: [],
- };
- let newRecord = null;
- let newHostsFile = "";
-
- const envUnsubscribe = environment.subscribe(value => {
- env = value;
- });
- onDestroy(envUnsubscribe);
-
- async function getHostsFile(hf) {
- if (hf.Records === null) {
- hf.Records = []
- }
- if (hf.Records.length > 0) {
- hostsFile = hf;
- return;
- }
- const res = await fetch(apiHostsDir +"/"+ hf.Name);
- hf.Records = await res.json();
- hostsFile = hf;
- }
-
- async function createHostsFile() {
- if (newHostsFile === "") {
- return;
- }
-
- const res = await fetch(apiHostsDir+ "/"+ newHostsFile, {
- method: "PUT",
- })
-
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: createHostsFile: ", resError.message)
- return;
- }
-
- const hf = {
- Name: newHostsFile,
- Records: [],
- }
- env.HostsFiles[newHostsFile] = hf
- env = env
-
- WuiPushNotif.Info("The new host file '"+ newHostsFile +"' has been created")
- }
-
- async function updateHostsFile() {
- const res = await fetch(apiHostsDir+"/"+ hostsFile.Name, {
- method: "POST",
- body: JSON.stringify(hostsFile.Records),
- })
-
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: updateHostsFile: ", resError.message)
- return;
- }
-
- hostsFile.Records = await res.json()
-
- WuiPushNotif.Info("The host file '"+ hostsFile.Name +"' has been updated")
- }
-
- function addRecord() {
- if (newRecord !== null) {
- return
- }
- newRecord = {
- Name: "",
- Value: "",
- }
- }
-
- async function handleHostsRecordCreate() {
- const api = apiHostsDir +"/"+ hostsFile.Name +"/rr"
- +"?domain="+ newRecord.Name
- +"&value="+ newRecord.Value
-
- const res = await fetch(api, {
- method: "POST"
- })
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: "+ resError.message)
- return;
- }
- const rr = await res.json()
- hostsFile.Records.push(rr)
- hostsFile.Records = hostsFile.Records
- newRecord = null
- WuiPushNotif.Info("Record '"+ rr.Name +"' has been created")
- }
-
- async function handleHostsRecordDelete(rr, idx) {
- const api = apiHostsDir +"/"+ hostsFile.Name +"/rr"+
- "?domain="+rr.Name
-
- const res = await fetch(api, {
- method: "DELETE"
- })
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: "+ resError.message)
- return;
- }
- hostsFile.Records.splice(idx, 1);
- hostsFile.Records = hostsFile.Records;
- WuiPushNotif.Info("Record '"+ rr.Name +"' has been deleted")
- }
-
- async function deleteHostsFile(hfile) {
- const res = await fetch(apiHostsDir+"/"+hfile.Name, {
- method: "DELETE",
- });
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: deleteHostsFile: ", resError.message)
- return;
- }
- delete env.HostsFiles[hfile.Name]
- env = env;
- hostsFile = {
- Name: "",
- Records: [],
- }
- WuiPushNotif.Info("The host file '"+ hfile.Name +"' has been deleted")
- }
-</script>
-
-<style>
- .nav-left {
- padding: 0px;
- width: 280px;
- float: left;
- }
- .nav-left .item {
- margin: 4px 0px;
- }
- .content {
- float: left;
- width: calc(100% - 300px);
- }
- .host {
- font-family: monospace;
- width: 100%;
- }
- .host.header {
- margin: 1em 0px;
- font-weight: bold;
- border-bottom: 1px solid silver;
- }
- .host_name {
- display: inline-block;
- width: 240px;
- word-wrap: break-word;
- }
- .host_value {
- display: inline-block;
- width: 140px;
- }
-</style>
-
-<div class="hosts_d">
- <div class="nav-left">
- {#each Object.entries(env.HostsFiles) as [name,hf], name }
- <div class="item">
- <a href="#" on:click={getHostsFile(hf)}>
- {hf.Name}
- </a>
- </div>
- {/each}
-
- <br/>
-
- <label>
- <span>New hosts file:</span>
- <br/>
- <input bind:value={newHostsFile}>
- </label>
- <button on:click={createHostsFile}>
- Create
- </button>
- </div>
-
- <div class="content">
- {#if hostsFile.Name === ""}
- <div>
- Select one of the hosts file to manage.
- </div>
- {:else}
- <p>
- {hostsFile.Name} ({hostsFile.Records.length} records)
- <button on:click={deleteHostsFile(hostsFile)}>
- Delete
- </button>
- </p>
- <div>
- <button on:click={addRecord}>
- Add
- </button>
- </div>
-
- {#if newRecord !== null}
- <div class="host">
- <input
- class="host_name"
- placeholder="Domain name"
- bind:value={newRecord.Name}
- >
- <input
- class="host_value"
- placeholder="IP address"
- bind:value={newRecord.Value}
- >
- <button on:click={handleHostsRecordCreate}>
- Create
- </button>
- </div>
- {/if}
-
- <div class="host header">
- <span class="host_name"> Domain name </span>
- <span class="host_value"> IP address </span>
- </div>
-
- {#each hostsFile.Records as rr, idx (idx)}
- <div class="host">
- <span class="host_name"> {rr.Name} </span>
- <span class="host_value"> {rr.Value} </span>
- <button on:click={handleHostsRecordDelete(rr, idx)}>
- X
- </button>
- </div>
- {/each}
- {/if}
- </div>
-</div>
diff --git a/_www/src/MasterDir.svelte b/_www/src/MasterDir.svelte
deleted file mode 100644
index 442f13f..0000000
--- a/_www/src/MasterDir.svelte
+++ /dev/null
@@ -1,532 +0,0 @@
-<script>
- import { onDestroy } from 'svelte';
- import { WuiPushNotif, WuiLabelHint } from 'wui.svelte';
- import { environment, nanoSeconds, setEnvironment } from './environment.js';
-
- const apiMasterd = "/api/master.d/"
-
- let env = {
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: [],
- ZoneFiles: {},
- };
- let newZoneFile = "";
- let activeZone = {
- Name: "",
- };
-
- let RRTypes = {
- 1: 'A',
- 2: 'NS',
- 5: 'CNAME',
- 12: 'PTR',
- 15: 'MX',
- 16: 'TXT',
- 28: 'AAAA',
- };
-
- let _rr = newRR()
- let _rrMX = newMX();
-
- const envUnsubscribe = environment.subscribe(value => {
- env = value;
- });
- onDestroy(envUnsubscribe);
-
- function setActiveZone(zone) {
- if (zone.SOA === null) {
- zone.SOA = newSOA()
- }
- activeZone = zone
- }
-
- async function handleZoneFileCreate() {
- const res = await fetch(apiMasterd + newZoneFile, {
- method: "PUT",
- })
-
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: handleZoneFileCreate: "+ resError.message)
- return;
- }
-
- activeZone = await res.json()
- activeZone.SOA = newSOA()
- env.ZoneFiles[activeZone.Name] = activeZone
-
- WuiPushNotif.Info("The new zone file '"+ newZoneFile +"' has been created")
- }
-
- async function handleZoneFileDelete() {
- let api = apiMasterd + activeZone.Name
- const res = await fetch(api, {
- method: "DELETE",
- })
-
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: handleZoneFileDelete: "+ resError.message)
- return;
- }
-
- WuiPushNotif.Info("The zone file '"+ activeZone.Name + "' has beed deleted")
-
- delete env.ZoneFiles[activeZone.Name]
- activeZone = {
- Name: "",
- }
- env.ZoneFiles = env.ZoneFiles
-
- }
-
- function onSelectRRType() {
- switch (_rr.Type) {
- case 15:
- _rrMX = newMX()
- break
- }
- }
-
- async function handleSOADelete() {
- return handleDeleteRR(activeZone.SOA, -1)
- }
-
- async function handleSOASave() {
- _rr = activeZone.SOA
- return handleCreateRR()
- }
-
- async function handleCreateRR() {
- switch (_rr.Type) {
- case 15:
- _rr.Value = _rrMX;
- break;
- }
-
- let api = apiMasterd + activeZone.Name +"/rr/"+ _rr.Type;
- const res = await fetch(api, {
- method: "POST",
- headers: {
- 'Content-Type': 'application/json',
- },
- body: JSON.stringify(_rr),
- })
-
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: handleCreateRR: "+ resError.message)
- return;
- }
-
- let resRR = await res.json()
-
- if (resRR.Type === 6) {
- activeZone.SOA = resRR
- WuiPushNotif.Info("SOA record has been saved")
- } else {
- let listRR = activeZone.Records[resRR.Name]
- if (typeof listRR === "undefined") {
- listRR = [];
- }
- listRR.push(resRR);
- activeZone.Records[resRR.Name] = listRR
- WuiPushNotif.Info("The new record '"+ resRR.Name +"' has been created")
- }
- }
-
- async function handleDeleteRR(rr, idx) {
- let api = apiMasterd + activeZone.Name +"/rr/"+ rr.Type
-
- const res = await fetch(api, {
- method: "DELETE",
- headers: {
- 'Content-Type': 'application/json',
- },
- body: JSON.stringify(rr),
- })
-
- if (res.status >= 400) {
- const resError = await res.json()
- WuiPushNotif.Error("ERROR: handleDeleteRR: "+ resError.message)
- return
- }
-
- WuiPushNotif.Info("The record '"+ rr.Name +"' has been deleted")
-
- if (rr.Type == 6) { // SOA.
- activeZone.SOA = newSOA()
- } else {
- let listRR = activeZone.Records[rr.Name]
- listRR.splice(idx, 1)
- activeZone.Records[rr.Name] = listRR
- }
-
- let resbody = await res.json()
- }
-
- function getTypeName(k) {
- let v = RRTypes[k];
- if (v === "") {
- return k;
- }
- return v;
- }
-
- function newRR() {
- return {
- Name: "",
- Type: 1,
- Value: "",
- };
- }
-
- function newMX() {
- return {
- Preference: 1,
- Exchange: "",
- }
- }
-
- function newSOA() {
- return {
- Name: "",
- Type: 6,
- Value: {
- MName: "",
- RName: "",
- Serial: 0,
- Refresh: 0,
- Retry: 0,
- Expire: 0,
- Minimum: 0
- }
- };
- }
-</script>
-
-<style>
- h4 {
- border-bottom: 1px solid silver;
- }
- .nav-left {
- padding: 0px;
- width: 280px;
- float: left;
- }
- .nav-left .item {
- margin: 4px 0px;
- cursor: pointer;
- color: rgb(0, 100, 200);
- }
- .content {
- float: left;
- width: calc(100% - 300px);
- }
- .action-delete {
- margin-left: 1em;
- }
- .actions {
- padding: 1em;
- }
- .actions button {
- width: 100%;
- }
-
- form {
- margin: 1em 0px;
- padding: 10px 10px 0px 10px;
- border: 1px solid silver;
- }
- form > label > span {
- width: 7em;
- display: inline-block;
- }
- form > label > input {
- width: calc(100% - 8em);
- }
- form > label > input.name {
- width: 12em;
- }
-
- .rr {
- font-family: monospace;
- width: 100%;
- padding: 1em 0px;
- }
- .rr.header {
- font-weight: bold;
- }
- .rr > .name {
- width: 20em;
- display: inline-block;
- word-wrap: break-word;
- }
- .rr > .type {
- width: 4em;
- display: inline-block;
- }
- .rr > .ttl {
- width: 6em;
- display: inline-block;
- }
- .rr > .value {
- display: inline-block;
- word-wrap: break-word;
- }
-</style>
-
-<div class="master_d">
- <div class="nav-left">
- {#each Object.entries(env.ZoneFiles) as [name, zoneFile]}
- <div class="item">
- <span on:click={setActiveZone(zoneFile)}>
- {zoneFile.Name}
- </span>
- </div>
- {/each}
- <br/>
-
- <label>
- <span>New zone file:</span>
- <br/>
- <input bind:value={newZoneFile}>
- </label>
- <button on:click={handleZoneFileCreate}>
- Create
- </button>
- </div>
-
- <div class="content">
- {#if activeZone.Name === ""}
- <p>
- Select one of the zone file to manage.
- </p>
- {:else}
- <h3>
- {activeZone.Name}
- <button
- class="action-delete"
- on:click={handleZoneFileDelete}
- >
- Delete
- </button>
- </h3>
-
- <h4>
- SOA record
- <button
- class="action-delete"
- on:click={handleSOADelete}
- >
- Delete
- </button>
- </h4>
- <div class="rr-soa">
- <WuiLabelHint
- title="Name server"
- title_width="150px"
- info="The domain-name of the name server that was the
-original or primary source of data for this zone.
-It should be domain-name where the rescached run."
- >
- <input bind:value={activeZone.SOA.Value.MName}>
- </WuiLabelHint>
- <WuiLabelHint
- title="Admin email"
- title_width="150px"
- info='Email address of the administrator responsible for
-this zone.
-The "@" on email address is replaced with dot, and if there is a dot before
-"@" it should be escaped with "\".
-For example, "dns.admin@domain.tld" would be written as
-"dns\.admin.domain.tld".'
- >
- <input bind:value={activeZone.SOA.Value.RName}>
- </WuiLabelHint>
- <WuiLabelHint
- title="Serial"
- title_width="150px"
- info="Serial number for this zone. If a secondary name
-server observes an increase in this number, the server will assume that the
-zone has been updated and initiate a zone transfer."
- >
- <input
- bind:value={activeZone.SOA.Value.Serial}
- type=number
- min=0
- >
- </WuiLabelHint>
- <WuiLabelHint
- title="Refresh"
- title_width="150px"
- info="Number of seconds after which secondary name servers
-should query the master for the SOA record, to detect zone changes.
-Recommendation for small and stable zones is 86400 seconds (24 hours)."
- >
- <input
- bind:value={activeZone.SOA.Value.Refresh}
- type=number
- min=0
- >
- </WuiLabelHint>
- <WuiLabelHint
- title="Retry"
- title_width="150px"
- info="Number of seconds after which secondary name servers
-should retry to request the serial number from the master if the master does
-not respond.
-It must be less than Refresh.
-Recommendation for small and stable zones is 7200 seconds (2 hours)."
- >
- <input
- bind:value={activeZone.SOA.Value.Retry}
- type=number
- min=0
- >
- </WuiLabelHint>
- <WuiLabelHint
- title="Expire"
- title_width="150px"
- info="Number of seconds after which secondary name servers
-should stop answering request for this zone if the master does not respond.
-This value must be bigger than the sum of Refresh and Retry.
-Recommendation for small and stable zones is 3600000 seconds (1000 hours)."
- >
- <input
- bind:value={activeZone.SOA.Value.Expire}
- type=number
- min=0
- >
- </WuiLabelHint>
- <WuiLabelHint
- title="Minimum"
- title_width="150px"
- info="Time to live for purposes of negative caching.
-Recommendation for small and stable zones is 1800 seconds (30 min)."
- >
- <input
- bind:value={activeZone.SOA.Value.Minimum}
- type=number
- min=0
- >
- </WuiLabelHint>
- <div class="actions">
- <button on:click={handleSOASave}>
- Save
- </button>
- </div>
- </div>
-
- <h4> List records </h4>
- <div class="rr header">
- <span class="name">
- Name
- </span>
- <span class="type">
- Type
- </span>
- <span class="value">
- Value
- </span>
- </div>
-
- {#each Object.entries(activeZone.Records) as [dname, listRR]}
- {#each listRR as rr, idx}
- <div class="rr">
- <span class="name">
- {rr.Name}
- </span>
- <span class="type">
- {getTypeName(rr.Type)}
- </span>
- <span class="value">
- {rr.Value}
- </span>
- <button on:click={handleDeleteRR(rr, idx)}>
- X
- </button>
- </div>
- {/each}
- {/each}
-
- <form on:submit|preventDefault={handleCreateRR}>
- <label>
- <span>
- Type:
- </span>
- <select
- bind:value={_rr.Type}
- on:blur={onSelectRRType}
- >
- {#each Object.entries(RRTypes) as [k, v]}
- <option value={parseInt(k)}>
- {v}
- </option>
- {/each}
- </select>
- </label>
-
- {#if _rr.Type === 1 || _rr.Type === 2 || _rr.Type === 5 ||
- _rr.Type === 16 || _rr.Type === 28
- }
- <label>
- <span>
- Name:
- </span>
- <input class="name" bind:value={_rr.Name}>
- .{activeZone.Name}
- </label>
-
- <label>
- <span>
- Value:
- </span>
- <input bind:value={_rr.Value}>
- </label>
- {:else if _rr.Type === 12} <!-- PTR -->
- <label>
- <span>
- Name:
- </span>
- <input bind:value={_rr.Name}>
- </label>
- <label>
- <span>
- Value:
- </span>
- <input class="name" bind:value={_rr.Value}>
- .{activeZone.Name}
- </label>
- {:else if _rr.Type === 15} <!-- MX -->
- <label>
- <span>
- Name:
- </span>
- <input class="name" bind:value={_rr.Name}>
- .{activeZone.Name}
- </label>
- <label>
- <span>
- Preference:
- </span>
- <input bind:value={_rrMX.Preference} type=number min=1 max=65535>
- </label>
- <label>
- <span>
- Exchange:
- </span>
- <input bind:value={_rrMX.Exchange}>
- </label>
- {/if}
-
- <div class="actions">
- <button class="create" type=submit>
- Create
- </button>
- </div>
- </form>
- {/if}
- </div> <!-- content -->
-</div> <!-- master_d -->
diff --git a/_www/src/common.js b/_www/src/common.js
deleted file mode 100644
index 2f49f69..0000000
--- a/_www/src/common.js
+++ /dev/null
@@ -1,29 +0,0 @@
-const RRTypes = {
- 1: "A",
- 2: "NS",
- 3: "MD",
- 4: "MF",
- 5: "CNAME",
- 6: "SOA",
- 7: "MB",
- 8: "MG",
- 9: "MR",
- 10: "NULL",
- 11: "WKS",
- 12: "PTR",
- 13: "HINFO",
- 14: "MINFO",
- 15: "MX",
- 16: "TXT",
- 28: "AAAA",
- 33: "SRV",
- 41: "OPT",
-}
-
-export function getRRTypeName(k) {
- let v = RRTypes[k]
- if (v === "") {
- return k
- }
- return v
-}
diff --git a/_www/src/environment.js b/_www/src/environment.js
deleted file mode 100644
index b85b587..0000000
--- a/_www/src/environment.js
+++ /dev/null
@@ -1,19 +0,0 @@
-import { writable } from "svelte/store"
-
-export const apiEnvironment = "/api/environment"
-export const environment = writable({
- NameServers: [],
- HostsBlocks: [],
- HostsFiles: {},
- ZoneFiles: [],
-})
-export const nanoSeconds = 1000000000
-
-export async function setEnvironment(got) {
- got.PruneDelay = got.PruneDelay / nanoSeconds
- got.PruneThreshold = got.PruneThreshold / nanoSeconds
- for (const [key, value] of Object.entries(got.HostsFiles)) {
- got.HostsFiles[key].Records = []
- }
- environment.set(got)
-}
diff --git a/_www/src/main.js b/_www/src/main.js
deleted file mode 100644
index ecbe53c..0000000
--- a/_www/src/main.js
+++ /dev/null
@@ -1,7 +0,0 @@
-import App from "./App.svelte"
-
-const app = new App({
- target: document.body,
-})
-
-export default app
diff --git a/_www/zone.d/index.html b/_www/zone.d/index.html
new file mode 100644
index 0000000..77bfb8c
--- /dev/null
+++ b/_www/zone.d/index.html
@@ -0,0 +1,513 @@
+<!DOCTYPE html>
+<html lang="en">
+ <head>
+ <meta charset="utf-8" />
+ <meta name="viewport" content="width=device-width, initial-scale=1, shrink-to-fit=no" />
+ <link rel="icon" type="image/png" href="/favicon.png" />
+ <link rel="stylesheet" href="/index.css" />
+ <title>rescached | zone.d</title>
+
+ <style>
+ h4 {
+ border-bottom: 1px solid silver;
+ }
+ .nav-left {
+ padding: 0;
+ width: 13em;
+ float: left;
+ }
+ .nav-left .item {
+ margin: 1em 0;
+ cursor: pointer;
+ color: rgb(0, 100, 200);
+ }
+ .nav-left > input {
+ width: 12em;
+ }
+ .content {
+ float: left;
+ width: calc(100% - 14em);
+ }
+ .action-delete {
+ margin-left: 1em;
+ }
+ .actions {
+ padding: 1em;
+ }
+ .actions button {
+ width: 100%;
+ }
+
+ .rr_form {
+ margin: 1em 0px;
+ padding: 10px 10px 0px 10px;
+ border: 1px solid silver;
+ }
+ .rr_form > * > .input > input.name {
+ width: auto;
+ }
+
+ .rr {
+ font-family: monospace;
+ width: 100%;
+ padding: 1em 0px;
+ }
+ .rr.header {
+ font-weight: bold;
+ }
+ .rr > .name {
+ width: 20em;
+ display: inline-block;
+ word-wrap: break-word;
+ }
+ .rr > .type {
+ width: 4em;
+ display: inline-block;
+ }
+ .rr > .ttl {
+ width: 6em;
+ display: inline-block;
+ }
+ .rr > .value {
+ display: inline-block;
+ word-wrap: break-word;
+ }
+ </style>
+ </head>
+ <body onload="main()">
+ <nav class="menu">
+ <a href="/"> rescached </a>
+ /
+ <a href="/environment/"> Environment </a>
+ /
+ <a href="/hosts_blocks/"> Hosts blocks </a>
+ /
+ <a href="/hosts.d/"> hosts.d </a>
+ /
+ <a href="/zone.d/" class="active"> zone.d </a>
+ </nav>
+ <div id="notif"></div>
+
+ <div class="nav-left">
+ <h3>Zone files</h3>
+ <div id="ZoneFiles"></div>
+
+ <label for="newZoneFile"> New zone file: </label>
+ <input id="newZoneFile" />
+ <button onclick="createZoneFile()">Create</button>
+ </div>
+
+ <div class="content">
+ <div id="activeZone"></div>
+
+ <div id="activeZone_soa" style="display: none">
+ <h4>SOA record</h4>
+
+ <div class="input">
+ <label for="soa_mname"> Name server </label>
+ <input id="soa_mname" oninput="updateSOA('MName', this.value)" />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_mname_info')">?</span>
+ <div id="soa_mname_info" class="input-info" style="display: none">
+ The domain-name of the name server that was the original or primary source
+ of data for this zone. It should be domain-name where the rescached run.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="soa_rname">Admin email</label>
+ <input id="soa_rname" oninput="updateSOA('RName', this.value)" />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_rname_info')">?</span>
+ <div id="soa_rname_info" class="input-info" style="display: none">
+ Email address of the administrator responsible for this zone. The "@" on
+ email address is replaced with dot, and if there is a dot before "@" it
+ should be escaped with "\". For example, "dns.admin@domain.tld" would be
+ written as "dns\.admin.domain.tld".'
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="soa_serial">Serial</label>
+ <input
+ id="soa_serial"
+ type="number"
+ min="0"
+ oninput="updateSOA('Serial', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_serial_info')">?</span>
+ <div id="soa_serial_info" class="input-info" style="display: none">
+ Serial number for this zone. If a secondary name server observes an increase
+ in this number, the server will assume that the zone has been updated and
+ initiate a zone transfer.
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="soa_refresh">Refresh</label>
+ <input
+ id="soa_refresh"
+ type="number"
+ min="0"
+ oninput="updateSOA('Refresh', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_refresh_info')">?</span>
+ <div id="soa_refresh_info" class="input-info" style="display: none">
+ Number of seconds after which secondary name servers should query the zone
+ for the SOA record, to detect zone changes. Recommendation for small and
+ stable zones is 86400 seconds (24 hours).
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="soa_retry">Retry</label>
+ <input
+ id="soa_retry"
+ type="number"
+ min="0"
+ oninput="updateSOA('Retry', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_retry_info')">?</span>
+ <div id="soa_retry_info" class="input-info" style="display: none">
+ Number of seconds after which secondary name servers should retry to request
+ the serial number from the zone if the zone does not respond. It must be
+ less than Refresh. Recommendation for small and stable zones is 7200 seconds
+ (2 hours).
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="soa_expire">Expire</label>
+ <input
+ id="soa_expire"
+ type="number"
+ min="0"
+ oninput="updateSOA('Expire', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_expire_info')">?</span>
+ <div class="soa_expire_info" class="input-info" style="display: none">
+ Number of seconds after which secondary name servers should stop answering
+ request for this zone if the zone does not respond. This value must be
+ bigger than the sum of Refresh and Retry. Recommendation for small and
+ stable zones is 3600000 seconds (1000 hours).
+ </div>
+ </div>
+
+ <div class="input">
+ <label for="soa_minimum"> Minimum </label>
+ <input
+ id="soa_minimum"
+ type="number"
+ min="0"
+ oninput="updateSOA('Minimum', parseInt(this.value))"
+ />
+ <span class="input-info-toggler" onclick="toggleInfo('soa_expire_info')">?</span>
+ <div id="soa_minimum_info" class="input-info" style="display: none">
+ Time to live for purposes of negative caching. Recommendation for small and
+ stable zones is 1800 seconds (30 min).
+ </div>
+ </div>
+
+ <div class="actions">
+ <button onclick="saveSOA()">Save</button>
+ </div>
+ </div>
+
+ <div id="activeZone_records" style="display: none">
+ <h4>List records</h4>
+ <div class="rr header">
+ <span class="name"> Name </span>
+ <span class="type"> Type </span>
+ <span class="value"> Value </span>
+ </div>
+ <div id="list_records"></div>
+ </div>
+
+ <div id="activeZone_form" class="rr_form" style="display: none">
+ <div class="input">
+ <label for="rr_type"> Type: </label>
+ <select id="rr_type" oninput="onSelectRRType(this.value)">
+ <option value="1">A</option>
+ <option value="2">NS</option>
+ <option value="5">CNAME</option>
+ <option value="12">PTR</option>
+ <option value="15">MX</option>
+ <option value="16">TXT</option>
+ <option value="28">AAAA</option>
+ </select>
+ </div>
+
+ <div id="activeZone_form_default">
+ <div class="input">
+ <label> Name: </label>
+ <input class="name" oninput="updateNewRR('Name', this.value)" />
+ <span></span>
+ </div>
+ <div class="input">
+ <label> Value: </label>
+ <input oninput="updateNewRR('Value', this.value)" />
+ </div>
+ </div>
+
+ <div id="activeZone_form_ptr" style="display: none">
+ <div class="input">
+ <label> Name: </label>
+ <input oninput="updateNewRR('Name', this.value)" />
+ </div>
+ <div class="input">
+ <label> Value: </label>
+ <input class="name" oninput="updateNewRR('Value', this.value)" />
+ <span></span>
+ </div>
+ </div>
+
+ <div id="activeZone_form_mx" style="display: none">
+ <div class="input">
+ <label> Name: </label>
+ <input class="name" oninput="updateNewRR('Name', this.value)" />
+ <span></span>
+ </div>
+ <div class="input">
+ <label> Preference: </label>
+ <input
+ type="number"
+ min="1"
+ max="65535"
+ oninput="updateNewRR('Value.Preference', this.value)"
+ />
+ </div>
+ <div class="input">
+ <label> Exchange: </label>
+ <input oninput="updateNewRR('Value.Exchange', this.value)" />
+ </div>
+ </div>
+
+ <div class="actions">
+ <button class="create" onclick="createRR()">Create</button>
+ </div>
+ </div>
+ </div>
+
+ <script src="/index.js"></script>
+ <script src="/rescached.js"></script>
+ <script>
+ let resc = null
+ let activeZone = null
+ let newRR = {
+ Name: "",
+ Value: "",
+ }
+
+ async function main() {
+ resc = new Rescached("")
+
+ let res = await resc.getEnvironment()
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ renderZoneFiles(resc.env.ZoneFiles)
+ resetActiveZone()
+ }
+
+ async function createZoneFile() {
+ let name = document.getElementById("newZoneFile").value
+ if (name === "") {
+ notifError("The zone file name must not be empty")
+ return
+ }
+ let res = await resc.ZoneFileCreate(name)
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ renderZoneFiles(resc.env.ZoneFiles)
+ }
+
+ async function deleteZoneFile() {
+ let res = await resc.ZoneFileDelete(activeZone.Name)
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ renderZoneFiles(resc.env.ZoneFiles)
+ resetActiveZone()
+ notifInfo(res.message)
+ }
+
+ async function createRR() {
+ let res = await resc.ZoneFileRecordCreate(activeZone.Name, newRR)
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ activeZone.Records = res.data
+ renderActiveZoneRecords()
+ notifInfo(res.message)
+ }
+
+ async function deleteRR(name, idx) {
+ let rr = activeZone.Records[name][idx]
+ console.log("deleteRR: ", rr)
+ let res = await resc.ZoneFileRecordDelete(activeZone.Name, rr)
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ activeZone.Records = res.data
+ renderActiveZoneRecords()
+ notifInfo(res.message)
+ }
+
+ function onSelectRRType(v) {
+ let formDefault = document.getElementById("activeZone_form_default")
+ let formPTR = document.getElementById("activeZone_form_ptr")
+ let formMX = document.getElementById("activeZone_form_mx")
+
+ newRR.Type = v
+ newRR.Value = ""
+
+ if (v == 12) {
+ // PTR
+ formDefault.style.display = "none"
+ formPTR.style.display = "block"
+ formPTR.children[1].children[2].innerText = "." + activeZone.Name
+ formMX.style.display = "none"
+ } else if (v == 15) {
+ formDefault.style.display = "none"
+ formPTR.style.display = "none"
+ formMX.style.display = "block"
+ formMX.children[0].children[2].innerText = "." + activeZone.Name
+ newRR.Value = {
+ Name: "",
+ Exchange: "",
+ Preference: 0,
+ }
+ } else {
+ formDefault.style.display = "block"
+ formDefault.children[0].children[2].innerText = "." + activeZone.Name
+ formPTR.style.display = "none"
+ formMX.style.display = "none"
+ }
+ }
+
+ function renderZoneFiles(zoneFiles) {
+ let wrapper = document.getElementById("ZoneFiles")
+ out = ""
+ for (let name in zoneFiles) {
+ if (!zoneFiles.hasOwnProperty(name)) {
+ continue
+ }
+ let zoneFile = zoneFiles[name]
+ out += `
+ <div class="item">
+ <span onclick="setActiveZone('${zoneFile.Name}')">
+ ${zoneFile.Name}
+ </span>
+ </div>`
+ }
+ wrapper.innerHTML = out
+ }
+
+ function renderActiveZone() {
+ let w = document.getElementById("activeZone")
+ w.innerHTML = `
+ <h3>
+ ${activeZone.Name}
+ <button
+ class="action-delete"
+ onclick="deleteZoneFile()"
+ >
+ Delete
+ </button>
+ </h3>
+ `
+ }
+
+ function renderActiveZoneSOA() {
+ const w = document.getElementById("activeZone_soa")
+ w.style.display = "block"
+ const soa = activeZone.SOA.Value
+ document.getElementById("soa_mname").value = soa.MName
+ document.getElementById("soa_rname").value = soa.RName
+ document.getElementById("soa_serial").value = soa.Serial
+ document.getElementById("soa_refresh").value = soa.Refresh
+ document.getElementById("soa_retry").value = soa.Retry
+ document.getElementById("soa_expire").value = soa.Expire
+ document.getElementById("soa_minimum").value = soa.Minimum
+ }
+
+ function renderActiveZoneRecords() {
+ let el = document.getElementById("activeZone_records")
+ el.style.display = "block"
+ let w = document.getElementById("list_records")
+ out = ""
+ for (const [name, listRR] of Object.entries(activeZone.Records)) {
+ listRR.forEach((rr, idx) => {
+ out += `
+ <div class="rr">
+ <span class="name">
+ ${rr.Name}
+ </span>
+ <span class="type">
+ ${resc.GetRRTypeName(rr.Type)}
+ </span>
+ <span class="value">
+ ${rr.Value}
+ </span>
+ <button onclick="deleteRR('${rr.Name}', ${idx})">
+ X
+ </button>
+ </div>
+ `
+ })
+ }
+ w.innerHTML = out
+ }
+
+ function renderActiveZoneForm() {
+ let form = document.getElementById("activeZone_form")
+ form.style.display = "block"
+ document.getElementById("rr_type").value = 1
+ onSelectRRType(1)
+ }
+
+ function resetActiveZone() {
+ document.getElementById("activeZone").innerHTML = `
+ <p>Select one of the zone file to manage.</p>
+ `
+ document.getElementById("activeZone_soa").style.display = "none"
+ document.getElementById("activeZone_records").style.display = "none"
+ document.getElementById("activeZone_form").style.display = "none"
+ activeZone = null
+ }
+
+ async function saveSOA() {
+ console.log("saveSOA: ", activeZone.SOA.Value)
+ let rr = activeZone.SOA
+ rr.Type = 6
+ let res = await resc.ZoneFileRecordCreate(activeZone.Name, rr)
+ if (res.code != 200) {
+ notifError(res.message)
+ return
+ }
+ notifInfo(res.message)
+ }
+
+ function setActiveZone(name) {
+ activeZone = resc.env.ZoneFiles[name]
+ console.log("setActiveZone: ", activeZone)
+ renderActiveZone()
+ renderActiveZoneSOA()
+ renderActiveZoneRecords()
+ renderActiveZoneForm()
+ }
+
+ function updateNewRR(k, v) {
+ newRR[k] = v
+ }
+
+ function updateSOA(k, v) {
+ activeZone.SOA.Value[k] = v
+ }
+ </script>
+ </body>
+</html>
diff --git a/environment.go b/environment.go
index f8b574e..d976702 100644
--- a/environment.go
+++ b/environment.go
@@ -46,8 +46,8 @@ const (
keyTLSPort = "tls.port"
keyTLSPrivateKey = "tls.private_key"
- dirHosts = "/etc/rescached/hosts.d"
- dirMaster = "/etc/rescached/master.d"
+ dirHosts = "/etc/rescached/hosts.d"
+ dirZone = "/etc/rescached/zone.d"
)
//
diff --git a/go.mod b/go.mod
index cdf139d..2c383a0 100644
--- a/go.mod
+++ b/go.mod
@@ -1,7 +1,7 @@
module github.com/shuLhan/rescached-go/v3
-go 1.13
+go 1.14
-require github.com/shuLhan/share v0.22.1-0.20210111194332-8df2d3ff4b02
+require github.com/shuLhan/share v0.22.1-0.20210119183627-da9256c28c9c
//replace github.com/shuLhan/share => ../share
diff --git a/go.sum b/go.sum
index b564bce..4bebf00 100644
--- a/go.sum
+++ b/go.sum
@@ -6,8 +6,8 @@ github.com/shuLhan/share v0.20.2-0.20201122173411-e8b3bf5ee6e9/go.mod h1:oBv+CGH
github.com/shuLhan/share v0.20.2-0.20201205202022-66069b9e49fe/go.mod h1:oBv+CGHG6u4Sa71+nJJJji8mCgPAadywjsB3I3k/b0o=
github.com/shuLhan/share v0.22.1-0.20210109185915-0490a19341d9 h1:Kqmfgkwq5dn68epHLBmPVkkLRDk3XFftqO/o3X84Mqg=
github.com/shuLhan/share v0.22.1-0.20210109185915-0490a19341d9/go.mod h1:u9caerexlcxmPVDttj7PnkxCBDY6yBRTZ+gGR+1tO98=
-github.com/shuLhan/share v0.22.1-0.20210111194332-8df2d3ff4b02 h1:rlaqzCiWaoJS1VqvK3YUAMtcRwS3c/jq7Egg++aA7x8=
-github.com/shuLhan/share v0.22.1-0.20210111194332-8df2d3ff4b02/go.mod h1:y4+p5vUmKNNhMMhU6yGgE6QxTgJxA4nv6OOq+cIf7wU=
+github.com/shuLhan/share v0.22.1-0.20210119183627-da9256c28c9c h1:LrwXdGkzOKcszqS+IX9SQ94l8p6eAzcaml5ZFfW/Mz4=
+github.com/shuLhan/share v0.22.1-0.20210119183627-da9256c28c9c/go.mod h1:y4+p5vUmKNNhMMhU6yGgE6QxTgJxA4nv6OOq+cIf7wU=
golang.org/x/crypto v0.0.0-20190308221718-c2843e01d9a2/go.mod h1:djNgcEr1/C05ACkg1iLfiJU5Ep61QUkGW8qpdssI0+w=
golang.org/x/crypto v0.0.0-20200622213623-75b288015ac9/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
golang.org/x/crypto v0.0.0-20201016220609-9e8e0b390897/go.mod h1:LzIPMQfyMNhhGPhUkYOs5KpL4U8rLKemX1yGLhDgUto=
diff --git a/httpd.go b/httpd.go
index 90d90cd..149326f 100644
--- a/httpd.go
+++ b/httpd.go
@@ -9,7 +9,7 @@ import (
"errors"
"fmt"
"log"
- stdhttp "net/http"
+ "net/http"
"os"
"path/filepath"
"regexp"
@@ -18,7 +18,7 @@ import (
"github.com/shuLhan/share/lib/dns"
liberrors "github.com/shuLhan/share/lib/errors"
- "github.com/shuLhan/share/lib/http"
+ libhttp "github.com/shuLhan/share/lib/http"
"github.com/shuLhan/share/lib/memfs"
libnet "github.com/shuLhan/share/lib/net"
)
@@ -26,23 +26,32 @@ import (
var memFS *memfs.MemFS
const (
- defHTTPDRootDir = "_www/public/"
- paramNameDomain = "domain"
- paramNameName = "name"
- paramNameQuery = "query"
- paramNameType = "type"
- paramNameValue = "value"
- apiCaches = "/api/caches"
- apiEnvironment = "/api/environment"
- apiHostsBlock = "/api/hosts_block"
- apiHostsDir = "/api/hosts.d/:name"
- apiHostsDirRR = "/api/hosts.d/:name/rr"
- apiMasterFile = "/api/master.d/:name"
- apiMasterFileRRType = "/api/master.d/:name/rr/:type"
+ defHTTPDRootDir = "_www"
+ paramNameDomain = "domain"
+ paramNameName = "name"
+ paramNameQuery = "query"
+ paramNameType = "type"
+ paramNameValue = "value"
+ apiCaches = "/api/caches"
+ apiEnvironment = "/api/environment"
+ apiHostsBlock = "/api/hosts_block"
+ apiHostsDir = "/api/hosts.d/:name"
+ apiHostsDirRR = "/api/hosts.d/:name/rr"
+ apiZoneFile = "/api/zone.d/:name"
+ apiZoneFileRRType = "/api/zone.d/:name/rr/:type"
)
+type response struct {
+ liberrors.E
+ Data interface{} `json:"data"`
+}
+
+func (r *response) Unwrap() error {
+ return &r.E
+}
+
func (srv *Server) httpdInit() (err error) {
- env := &http.ServerOptions{
+ env := &libhttp.ServerOptions{
Options: memfs.Options{
Root: defHTTPDRootDir,
Includes: []string{
@@ -59,11 +68,11 @@ func (srv *Server) httpdInit() (err error) {
"http://127.0.0.1:5000",
},
CORSAllowHeaders: []string{
- http.HeaderContentType,
+ libhttp.HeaderContentType,
},
}
- srv.httpd, err = http.NewServer(env)
+ srv.httpd, err = libhttp.NewServer(env)
if err != nil {
return fmt.Errorf("newHTTPServer: %w", err)
}
@@ -77,32 +86,32 @@ func (srv *Server) httpdInit() (err error) {
}
func (srv *Server) httpdRegisterEndpoints() (err error) {
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodGet,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodGet,
Path: apiCaches,
- RequestType: http.RequestTypeQuery,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeQuery,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.httpdAPIGetCaches,
})
if err != nil {
return err
}
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodDelete,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodDelete,
Path: apiCaches,
- RequestType: http.RequestTypeQuery,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeQuery,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.httpdAPIDeleteCaches,
})
if err != nil {
return err
}
- epAPIGetEnvironment := &http.Endpoint{
- Method: http.RequestMethodGet,
+ epAPIGetEnvironment := &libhttp.Endpoint{
+ Method: libhttp.RequestMethodGet,
Path: apiEnvironment,
- RequestType: http.RequestTypeJSON,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeJSON,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.httpdAPIGetEnvironment,
}
@@ -111,11 +120,11 @@ func (srv *Server) httpdRegisterEndpoints() (err error) {
return err
}
- epAPIPostEnvironment := &http.Endpoint{
- Method: http.RequestMethodPost,
+ epAPIPostEnvironment := &libhttp.Endpoint{
+ Method: libhttp.RequestMethodPost,
Path: apiEnvironment,
- RequestType: http.RequestTypeJSON,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeJSON,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.httpdAPIPostEnvironment,
}
@@ -124,11 +133,11 @@ func (srv *Server) httpdRegisterEndpoints() (err error) {
return err
}
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodPost,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodPost,
Path: apiHostsBlock,
- RequestType: http.RequestTypeJSON,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeJSON,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.apiHostsBlockUpdate,
})
if err != nil {
@@ -136,33 +145,33 @@ func (srv *Server) httpdRegisterEndpoints() (err error) {
}
// Register API to create new hosts file.
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodPut,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodPut,
Path: apiHostsDir,
- RequestType: http.RequestTypeNone,
- ResponseType: http.ResponseTypeNone,
+ RequestType: libhttp.RequestTypeNone,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.apiHostsFileCreate,
})
if err != nil {
return err
}
// Register API to get content of hosts file.
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodGet,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodGet,
Path: apiHostsDir,
- RequestType: http.RequestTypeNone,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeNone,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.apiHostsFileGet,
})
if err != nil {
return err
}
// Register API to delete hosts file.
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodDelete,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodDelete,
Path: apiHostsDir,
- RequestType: http.RequestTypeNone,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeNone,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.apiHostsFileDelete,
})
if err != nil {
@@ -170,64 +179,65 @@ func (srv *Server) httpdRegisterEndpoints() (err error) {
}
// Register API to create one record in hosts file.
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodPost,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodPost,
Path: apiHostsDirRR,
- RequestType: http.RequestTypeQuery,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeQuery,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.apiHostsFileRRCreate,
})
if err != nil {
return err
}
+
// Register API to delete a record from hosts file.
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodDelete,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodDelete,
Path: apiHostsDirRR,
- RequestType: http.RequestTypeQuery,
- ResponseType: http.ResponseTypeJSON,
+ RequestType: libhttp.RequestTypeQuery,
+ ResponseType: libhttp.ResponseTypeJSON,
Call: srv.apiHostsFileRRDelete,
})
if err != nil {
return err
}
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodPut,
- Path: apiMasterFile,
- RequestType: http.RequestTypeNone,
- ResponseType: http.ResponseTypeJSON,
- Call: srv.apiMasterFileCreate,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodPut,
+ Path: apiZoneFile,
+ RequestType: libhttp.RequestTypeNone,
+ ResponseType: libhttp.ResponseTypeJSON,
+ Call: srv.apiZoneFileCreate,
})
if err != nil {
return err
}
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodDelete,
- Path: apiMasterFile,
- RequestType: http.RequestTypeNone,
- ResponseType: http.ResponseTypeJSON,
- Call: srv.apiMasterFileDelete,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodDelete,
+ Path: apiZoneFile,
+ RequestType: libhttp.RequestTypeNone,
+ ResponseType: libhttp.ResponseTypeJSON,
+ Call: srv.apiZoneFileDelete,
})
if err != nil {
return err
}
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodPost,
- Path: apiMasterFileRRType,
- RequestType: http.RequestTypeJSON,
- ResponseType: http.ResponseTypeJSON,
- Call: srv.apiMasterFileCreateRR,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodPost,
+ Path: apiZoneFileRRType,
+ RequestType: libhttp.RequestTypeJSON,
+ ResponseType: libhttp.ResponseTypeJSON,
+ Call: srv.apiZoneFileRRCreate,
})
if err != nil {
return err
}
- err = srv.httpd.RegisterEndpoint(&http.Endpoint{
- Method: http.RequestMethodDelete,
- Path: apiMasterFileRRType,
- RequestType: http.RequestTypeJSON,
- ResponseType: http.ResponseTypeJSON,
- Call: srv.apiMasterFileDeleteRR,
+ err = srv.httpd.RegisterEndpoint(&libhttp.Endpoint{
+ Method: libhttp.RequestMethodDelete,
+ Path: apiZoneFileRRType,
+ RequestType: libhttp.RequestTypeJSON,
+ ResponseType: libhttp.ResponseTypeJSON,
+ Call: srv.apiZoneFileRRDelete,
})
if err != nil {
return err
@@ -253,12 +263,14 @@ func (srv *Server) httpdRun() {
}
func (srv *Server) httpdAPIGetCaches(
- _ stdhttp.ResponseWriter, req *stdhttp.Request, _ []byte,
+ _ http.ResponseWriter, req *http.Request, _ []byte,
) (
resBody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusInternalServerError,
+ res := response{
+ E: liberrors.E{
+ Code: http.StatusInternalServerError,
+ },
}
q := req.Form.Get(paramNameQuery)
@@ -266,7 +278,7 @@ func (srv *Server) httpdAPIGetCaches(
re, err := regexp.Compile(q)
if err != nil {
res.Message = err.Error()
- return nil, res
+ return nil, &res
}
listMsg := srv.dns.SearchCaches(re)
@@ -274,16 +286,19 @@ func (srv *Server) httpdAPIGetCaches(
listMsg = make([]*dns.Message, 0)
}
- return json.Marshal(listMsg)
+ res.Code = http.StatusOK
+ res.Data = listMsg
+
+ return json.Marshal(res)
}
func (srv *Server) httpdAPIDeleteCaches(
- _ stdhttp.ResponseWriter, req *stdhttp.Request, _ []byte,
+ _ http.ResponseWriter, req *http.Request, _ []byte,
) (
resBody []byte, err error,
) {
res := &liberrors.E{
- Code: stdhttp.StatusInternalServerError,
+ Code: http.StatusInternalServerError,
}
q := req.Form.Get(paramNameName)
@@ -294,28 +309,34 @@ func (srv *Server) httpdAPIDeleteCaches(
srv.dns.RemoveCachesByNames([]string{q})
- res.Code = stdhttp.StatusOK
+ res.Code = http.StatusOK
res.Message = fmt.Sprintf("%q has been removed from caches", q)
return json.Marshal(res)
}
func (srv *Server) httpdAPIGetEnvironment(
- httpRes stdhttp.ResponseWriter, req *stdhttp.Request, reqBody []byte,
+ httpRes http.ResponseWriter, req *http.Request, reqBody []byte,
) (
resBody []byte, err error,
) {
- return json.Marshal(srv.env)
+ res := &response{}
+ res.Code = http.StatusOK
+ res.Data = srv.env
+
+ return json.Marshal(res)
}
func (srv *Server) httpdAPIPostEnvironment(
- httpRes stdhttp.ResponseWriter, req *stdhttp.Request, reqBody []byte,
+ httpRes http.ResponseWriter, req *http.Request, reqBody []byte,
) (
resBody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusOK,
- Message: "Restarting DNS server",
+ res := &response{
+ E: liberrors.E{
+ Code: http.StatusOK,
+ Message: "Restarting DNS server",
+ },
}
newOpts := new(environment)
@@ -325,7 +346,7 @@ func (srv *Server) httpdAPIPostEnvironment(
}
if len(newOpts.NameServers) == 0 {
- res.Code = stdhttp.StatusBadRequest
+ res.Code = http.StatusBadRequest
res.Message = "at least one parent name servers must be defined"
return nil, res
}
@@ -336,7 +357,7 @@ func (srv *Server) httpdAPIPostEnvironment(
err = newOpts.write(srv.fileConfig)
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
@@ -346,7 +367,7 @@ func (srv *Server) httpdAPIPostEnvironment(
srv.Stop()
err = srv.Start()
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
@@ -357,14 +378,14 @@ func (srv *Server) httpdAPIPostEnvironment(
//
// apiHostsBlockUpdate set the HostsBlock to be enabled or disabled.
//
-// If its status changes to enabled, unhide it file, populate the hosts back
-// to caches, and add it to list of HostsFiles.
+// If its status changes to enabled, unhide the hosts block file, populate the
+// hosts back to caches, and add it to list of HostsFiles.
//
// If its status changes to disabled, remove the hosts from caches, hide it,
// and remove it from list of HostsFiles.
//
func (srv *Server) apiHostsBlockUpdate(
- httpRes stdhttp.ResponseWriter, req *stdhttp.Request, reqBody []byte,
+ httpRes http.ResponseWriter, req *http.Request, reqBody []byte,
) (
resBody []byte, err error,
) {
@@ -375,8 +396,10 @@ func (srv *Server) apiHostsBlockUpdate(
return nil, err
}
- res := &liberrors.E{
- Code: stdhttp.StatusInternalServerError,
+ res := &response{
+ E: liberrors.E{
+ Code: http.StatusInternalServerError,
+ },
}
for _, hbx := range hostsBlocks {
@@ -413,7 +436,10 @@ func (srv *Server) apiHostsBlockUpdate(
return nil, res
}
- return json.Marshal(srv.env)
+ res.Code = http.StatusOK
+ res.Data = hostsBlocks
+
+ return json.Marshal(res)
}
func (srv *Server) hostsBlockEnable(hb *hostsBlock) (err error) {
@@ -469,16 +495,17 @@ func (srv *Server) hostsBlockDisable(hb *hostsBlock) (err error) {
}
func (srv *Server) apiHostsFileCreate(
- httpres stdhttp.ResponseWriter, httpreq *stdhttp.Request, _ []byte,
+ httpres http.ResponseWriter, httpreq *http.Request, _ []byte,
) (
resbody []byte, err error,
) {
+ res := &response{}
+
name := httpreq.Form.Get(paramNameName)
if len(name) == 0 {
- return nil, &liberrors.E{
- Code: stdhttp.StatusBadRequest,
- Message: "hosts file name is invalid or empty",
- }
+ res.Code = http.StatusBadRequest
+ res.Message = "hosts file name is invalid or empty"
+ return nil, res
}
_, found := srv.env.HostsFiles[name]
@@ -486,51 +513,61 @@ func (srv *Server) apiHostsFileCreate(
path := filepath.Join(dirHosts, name)
hfile, err := dns.NewHostsFile(path, nil)
if err != nil {
- return nil, &liberrors.E{
- Code: stdhttp.StatusInternalServerError,
- Message: err.Error(),
- }
+ res.Code = http.StatusInternalServerError
+ res.Message = err.Error()
+ return nil, res
}
srv.env.HostsFiles[hfile.Name] = hfile
}
- httpres.WriteHeader(stdhttp.StatusCreated)
+ res.Code = http.StatusOK
+ res.Message = fmt.Sprintf("Hosts file %q has been created", name)
- return nil, nil
+ return json.Marshal(res)
}
func (srv *Server) apiHostsFileGet(
- _ stdhttp.ResponseWriter, httpreq *stdhttp.Request, _ []byte,
+ _ http.ResponseWriter, httpreq *http.Request, _ []byte,
) (
resbody []byte, err error,
) {
+ res := &response{}
+
name := httpreq.Form.Get(paramNameName)
- hfile, found := srv.env.HostsFiles[name]
- if !found || hfile.Records == nil {
- return []byte("[]"), nil
+ hf, found := srv.env.HostsFiles[name]
+ if !found {
+ res.Code = http.StatusNotFound
+ res.Message = "invalid or empty hosts file " + name
+ return nil, res
}
+ if hf.Records == nil || cap(hf.Records) == 0 {
+ hf.Records = make([]*dns.ResourceRecord, 0, 1)
+ }
+
+ res.Code = http.StatusOK
+ res.Data = hf.Records
- return json.Marshal(&hfile.Records)
+ return json.Marshal(res)
}
func (srv *Server) apiHostsFileDelete(
- _ stdhttp.ResponseWriter, httpreq *stdhttp.Request, reqbody []byte,
+ _ http.ResponseWriter, httpreq *http.Request, reqbody []byte,
) (
resbody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusOK,
- }
+ res := &response{}
name := httpreq.Form.Get(paramNameName)
if len(name) == 0 {
+ res.Code = http.StatusBadRequest
res.Message = "empty or invalid host file name"
return nil, res
}
hfile, found := srv.env.HostsFiles[name]
if !found {
+ res.Code = http.StatusBadRequest
res.Message = "apiDeleteHostsFile: " + name + " not found"
return nil, res
}
@@ -540,12 +577,14 @@ func (srv *Server) apiHostsFileDelete(
err = hfile.Delete()
if err != nil {
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
delete(srv.env.HostsFiles, name)
+ res.Code = http.StatusOK
res.Message = name + " has been deleted"
return json.Marshal(res)
}
@@ -554,13 +593,12 @@ func (srv *Server) apiHostsFileDelete(
// apiHostsFileRRCreate create new record and save it to the hosts file.
//
func (srv *Server) apiHostsFileRRCreate(
- _ stdhttp.ResponseWriter, httpreq *stdhttp.Request, _ []byte,
+ _ http.ResponseWriter, httpreq *http.Request, _ []byte,
) (
resbody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusBadRequest,
- }
+ res := &response{}
+ res.Code = http.StatusBadRequest
hostsFileName := httpreq.Form.Get(paramNameName)
if len(hostsFileName) == 0 {
@@ -597,28 +635,31 @@ func (srv *Server) apiHostsFileRRCreate(
err = hfile.AppendAndSaveRecord(rr)
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
err = srv.dns.PopulateCachesByRR([]*dns.ResourceRecord{rr}, hostsFileName)
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
- return json.Marshal(rr)
+ res.Code = http.StatusOK
+ res.Data = rr
+
+ return json.Marshal(res)
}
func (srv *Server) apiHostsFileRRDelete(
- _ stdhttp.ResponseWriter, httpreq *stdhttp.Request, reqbody []byte,
+ _ http.ResponseWriter, httpreq *http.Request, reqbody []byte,
) (
resbody []byte, err error,
) {
res := &liberrors.E{
- Code: stdhttp.StatusBadRequest,
+ Code: http.StatusBadRequest,
}
hostsFileName := httpreq.Form.Get(paramNameName)
@@ -646,29 +687,28 @@ func (srv *Server) apiHostsFileRRDelete(
}
err = hfile.Save()
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
srv.dns.RemoveLocalCachesByNames([]string{domainName})
- res.Code = stdhttp.StatusOK
+ res.Code = http.StatusOK
res.Message = "domain name '" + domainName + "' has been removed from hosts file"
return json.Marshal(res)
}
-func (srv *Server) apiMasterFileCreate(
- _ stdhttp.ResponseWriter,
- httpreq *stdhttp.Request,
+func (srv *Server) apiZoneFileCreate(
+ _ http.ResponseWriter,
+ httpreq *http.Request,
reqbody []byte,
) (
resbody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusBadRequest,
- }
+ res := &response{}
+ res.Code = http.StatusBadRequest
zoneName := httpreq.Form.Get(paramNameName)
if len(zoneName) == 0 {
@@ -683,33 +723,37 @@ func (srv *Server) apiMasterFileCreate(
mf, ok := srv.env.ZoneFiles[zoneName]
if ok {
- return json.Marshal(mf)
+ res.Code = http.StatusOK
+ res.Data = mf
+ return json.Marshal(res)
}
- zoneFile := filepath.Join(dirMaster, zoneName)
+ zoneFile := filepath.Join(dirZone, zoneName)
mf = dns.NewZoneFile(zoneFile, zoneName)
err = mf.Save()
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
srv.env.ZoneFiles[zoneName] = mf
- return json.Marshal(mf)
+ res.Code = http.StatusOK
+ res.Data = mf
+
+ return json.Marshal(res)
}
-func (srv *Server) apiMasterFileDelete(
- _ stdhttp.ResponseWriter,
- httpreq *stdhttp.Request,
+func (srv *Server) apiZoneFileDelete(
+ _ http.ResponseWriter,
+ httpreq *http.Request,
reqbody []byte,
) (
resbody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusBadRequest,
- }
+ res := &response{}
+ res.Code = http.StatusBadRequest
zoneName := httpreq.Form.Get(paramNameName)
if len(zoneName) == 0 {
@@ -733,40 +777,39 @@ func (srv *Server) apiMasterFileDelete(
err = mf.Delete()
if err != nil {
- res.Code = stdhttp.StatusInternalServerError
+ res.Code = http.StatusInternalServerError
res.Message = err.Error()
return nil, res
}
- res.Code = stdhttp.StatusOK
+ res.Code = http.StatusOK
res.Message = zoneName + " has been deleted"
return json.Marshal(res)
}
//
-// apiMasterFileCreateRR create new RR for the master file.
+// apiZoneFileRRCreate create new RR for the zone file.
//
-func (srv *Server) apiMasterFileCreateRR(
- _ stdhttp.ResponseWriter,
- httpreq *stdhttp.Request,
+func (srv *Server) apiZoneFileRRCreate(
+ _ http.ResponseWriter,
+ httpreq *http.Request,
reqbody []byte,
) (
resbody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusBadRequest,
- }
+ res := &response{}
+ res.Code = http.StatusBadRequest
zoneFileName := httpreq.Form.Get(paramNameName)
if len(zoneFileName) == 0 {
- res.Message = "empty or invalid master file name"
+ res.Message = "empty or invalid zone file name"
return nil, res
}
zoneFile := srv.env.ZoneFiles[zoneFileName]
if zoneFile == nil {
- res.Message = "unknown master file name " + zoneFileName
+ res.Message = "unknown zone file name " + zoneFileName
return nil, res
}
@@ -778,14 +821,14 @@ func (srv *Server) apiMasterFileCreateRR(
return nil, res
}
- rr := dns.ResourceRecord{}
+ rr := &dns.ResourceRecord{}
switch uint16(rrType) {
case dns.QueryTypeSOA:
rr.Value = &dns.RDataSOA{}
case dns.QueryTypeMX:
rr.Value = &dns.RDataMX{}
}
- err = json.Unmarshal(reqbody, &rr)
+ err = json.Unmarshal(reqbody, rr)
if err != nil {
res.Message = "json.Unmarshal:" + err.Error()
return nil, res
@@ -812,37 +855,45 @@ func (srv *Server) apiMasterFileCreateRR(
}
}
- listRR := []*dns.ResourceRecord{&rr}
+ listRR := []*dns.ResourceRecord{rr}
err = srv.dns.PopulateCachesByRR(listRR, zoneFile.Path)
if err != nil {
- res.Message = "UpsertCacheByRR: " + err.Error()
+ res.Code = http.StatusBadRequest
+ res.Message = "PopulateCacheByRR: " + err.Error()
return nil, res
}
- // Update the Master file.
- zoneFile.Add(&rr)
+ // Update the Zone file.
+ zoneFile.Add(rr)
err = zoneFile.Save()
if err != nil {
res.Message = err.Error()
return nil, res
}
- return json.Marshal(&rr)
+ res.Code = http.StatusOK
+ res.Message = fmt.Sprintf("%s record has been saved", dns.QueryTypeNames[rr.Type])
+ if rr.Type == dns.QueryTypeSOA {
+ res.Data = rr
+ } else {
+ res.Data = zoneFile.Records
+ }
+
+ return json.Marshal(res)
}
//
-// apiMasterFileDeleteRR delete RR from the master file.
+// apiZoneFileRRDelete delete RR from the zone file.
//
-func (srv *Server) apiMasterFileDeleteRR(
- _ stdhttp.ResponseWriter,
- httpreq *stdhttp.Request,
+func (srv *Server) apiZoneFileRRDelete(
+ _ http.ResponseWriter,
+ httpreq *http.Request,
reqbody []byte,
) (
resbody []byte, err error,
) {
- res := &liberrors.E{
- Code: stdhttp.StatusBadRequest,
- }
+ res := &response{}
+ res.Code = http.StatusBadRequest
zoneFileName := httpreq.Form.Get(paramNameName)
if len(zoneFileName) == 0 {
@@ -852,14 +903,15 @@ func (srv *Server) apiMasterFileDeleteRR(
mf := srv.env.ZoneFiles[zoneFileName]
if mf == nil {
- res.Message = "unknown master file name " + zoneFileName
+ res.Message = "unknown zone file name " + zoneFileName
return nil, res
}
v := httpreq.Form.Get(paramNameType)
rrType, err := strconv.Atoi(v)
if err != nil {
- res.Message = err.Error()
+ res.Message = fmt.Sprintf("invalid or empty param type %s: %s",
+ paramNameType, err)
return nil, res
}
@@ -895,7 +947,10 @@ func (srv *Server) apiMasterFileDeleteRR(
return nil, res
}
- res.Code = stdhttp.StatusOK
+ res.Code = http.StatusOK
+ res.Message = fmt.Sprintf("The RR type %d and name %s has been deleted",
+ rr.Type, rr.Name)
+ res.Data = mf.Records
return json.Marshal(res)
}
diff --git a/internal/generate_memfs.go b/internal/generate_memfs.go
index e51fbc3..80586f4 100644
--- a/internal/generate_memfs.go
+++ b/internal/generate_memfs.go
@@ -14,7 +14,7 @@ import (
func main() {
opts := memfs.Options{
- Root: "_www/public/",
+ Root: "_www",
Includes: []string{
`.*\.html`,
`.*\.js`,
diff --git a/rescached.1.gz b/rescached.1.gz
index e0485c9..a41425f 100644
--- a/rescached.1.gz
+++ b/rescached.1.gz
Binary files differ
diff --git a/rescached.go b/rescached.go
index 710f3e1..6d6464f 100644
--- a/rescached.go
+++ b/rescached.go
@@ -6,8 +6,10 @@
package rescached
import (
+ "errors"
"fmt"
"log"
+ "os"
"sync"
"github.com/shuLhan/share/lib/debug"
@@ -82,12 +84,19 @@ func (srv *Server) Start() (err error) {
}
}
- srv.env.ZoneFiles, err = dns.LoadMasterDir(dirMaster)
+ srv.env.ZoneFiles, err = dns.LoadZoneDir(dirZone)
if err != nil {
- return err
+ if !errors.Is(err, os.ErrNotExist) {
+ return err
+ }
+ err = os.MkdirAll(dirZone, 0700)
+ if err != nil {
+ return err
+ }
+ err = nil
}
- for _, masterFile := range srv.env.ZoneFiles {
- srv.dns.PopulateCaches(masterFile.Messages(), masterFile.Path)
+ for _, zoneFile := range srv.env.ZoneFiles {
+ srv.dns.PopulateCaches(zoneFile.Messages(), zoneFile.Path)
}
if len(srv.env.FileResolvConf) > 0 {