2019-06-10 08:33:19 +00:00
|
|
|
package home
|
2018-08-30 14:25:33 +00:00
|
|
|
|
|
|
|
import (
|
|
|
|
"encoding/json"
|
|
|
|
"fmt"
|
|
|
|
"net/http"
|
|
|
|
|
2018-12-05 11:21:25 +00:00
|
|
|
"github.com/AdguardTeam/AdGuardHome/dnsforward"
|
2019-02-27 15:28:09 +00:00
|
|
|
"github.com/AdguardTeam/golibs/log"
|
2019-04-15 13:21:12 +00:00
|
|
|
"github.com/NYTimes/gziphandler"
|
2018-08-30 14:25:33 +00:00
|
|
|
)
|
|
|
|
|
2019-02-10 17:47:43 +00:00
|
|
|
// ----------------
|
|
|
|
// helper functions
|
|
|
|
// ----------------
|
|
|
|
|
|
|
|
func returnOK(w http.ResponseWriter) {
|
|
|
|
_, err := fmt.Fprintf(w, "OK\n")
|
|
|
|
if err != nil {
|
2019-02-27 14:28:10 +00:00
|
|
|
httpError(w, http.StatusInternalServerError, "Couldn't write body: %s", err)
|
2019-02-10 17:47:43 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-11-08 11:56:19 +00:00
|
|
|
func httpOK(r *http.Request, w http.ResponseWriter) {
|
|
|
|
}
|
|
|
|
|
2019-02-10 17:47:43 +00:00
|
|
|
func httpError(w http.ResponseWriter, code int, format string, args ...interface{}) {
|
|
|
|
text := fmt.Sprintf(format, args...)
|
2019-02-25 13:44:22 +00:00
|
|
|
log.Info(text)
|
2019-02-10 17:47:43 +00:00
|
|
|
http.Error(w, text, code)
|
|
|
|
}
|
|
|
|
|
|
|
|
// ---------------
|
2018-12-05 17:29:00 +00:00
|
|
|
// dns run control
|
2019-02-10 17:47:43 +00:00
|
|
|
// ---------------
|
2018-12-05 17:29:00 +00:00
|
|
|
func writeAllConfigsAndReloadDNS() error {
|
2018-08-30 14:25:33 +00:00
|
|
|
err := writeAllConfigs()
|
|
|
|
if err != nil {
|
2019-02-25 13:44:22 +00:00
|
|
|
log.Error("Couldn't write all configs: %s", err)
|
2018-08-30 14:25:33 +00:00
|
|
|
return err
|
|
|
|
}
|
2019-01-24 17:11:01 +00:00
|
|
|
return reconfigureDNSServer()
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
|
|
|
|
2019-07-02 09:19:36 +00:00
|
|
|
func addDNSAddress(dnsAddresses *[]string, addr string) {
|
|
|
|
if config.DNS.Port != 53 {
|
|
|
|
addr = fmt.Sprintf("%s:%d", addr, config.DNS.Port)
|
|
|
|
}
|
|
|
|
*dnsAddresses = append(*dnsAddresses, addr)
|
|
|
|
}
|
2019-03-19 11:14:58 +00:00
|
|
|
|
2019-07-02 09:19:36 +00:00
|
|
|
// Get the list of DNS addresses the server is listening on
|
|
|
|
func getDNSAddresses() []string {
|
2019-03-19 11:14:58 +00:00
|
|
|
dnsAddresses := []string{}
|
2019-07-02 09:19:36 +00:00
|
|
|
|
2019-03-19 11:14:58 +00:00
|
|
|
if config.DNS.BindHost == "0.0.0.0" {
|
2019-07-02 09:19:36 +00:00
|
|
|
|
2019-03-19 11:14:58 +00:00
|
|
|
ifaces, e := getValidNetInterfacesForWeb()
|
|
|
|
if e != nil {
|
|
|
|
log.Error("Couldn't get network interfaces: %v", e)
|
2019-07-02 09:19:36 +00:00
|
|
|
return []string{}
|
2019-03-19 11:14:58 +00:00
|
|
|
}
|
2019-07-02 09:19:36 +00:00
|
|
|
|
2019-03-19 11:14:58 +00:00
|
|
|
for _, iface := range ifaces {
|
|
|
|
for _, addr := range iface.Addresses {
|
2019-07-02 09:19:36 +00:00
|
|
|
addDNSAddress(&dnsAddresses, addr)
|
2019-03-19 11:14:58 +00:00
|
|
|
}
|
|
|
|
}
|
2019-07-02 09:19:36 +00:00
|
|
|
|
|
|
|
} else {
|
|
|
|
addDNSAddress(&dnsAddresses, config.DNS.BindHost)
|
2019-03-19 11:14:58 +00:00
|
|
|
}
|
2019-07-02 09:19:36 +00:00
|
|
|
|
2019-07-02 09:27:10 +00:00
|
|
|
if config.TLS.Enabled && len(config.TLS.ServerName) != 0 {
|
|
|
|
|
|
|
|
if config.TLS.PortHTTPS != 0 {
|
|
|
|
addr := config.TLS.ServerName
|
|
|
|
if config.TLS.PortHTTPS != 443 {
|
|
|
|
addr = fmt.Sprintf("%s:%d", addr, config.TLS.PortHTTPS)
|
|
|
|
}
|
|
|
|
addr = fmt.Sprintf("https://%s/dns-query", addr)
|
|
|
|
dnsAddresses = append(dnsAddresses, addr)
|
|
|
|
}
|
|
|
|
|
|
|
|
if config.TLS.PortDNSOverTLS != 0 {
|
|
|
|
addr := fmt.Sprintf("tls://%s:%d", config.TLS.ServerName, config.TLS.PortDNSOverTLS)
|
|
|
|
dnsAddresses = append(dnsAddresses, addr)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-07-02 09:19:36 +00:00
|
|
|
return dnsAddresses
|
|
|
|
}
|
|
|
|
|
|
|
|
func handleStatus(w http.ResponseWriter, r *http.Request) {
|
2019-10-30 08:52:58 +00:00
|
|
|
c := dnsforward.FilteringConfig{}
|
2019-12-11 09:38:58 +00:00
|
|
|
if Context.dnsServer != nil {
|
|
|
|
Context.dnsServer.WriteDiskConfig(&c)
|
2019-10-30 08:52:58 +00:00
|
|
|
}
|
2018-08-30 14:25:33 +00:00
|
|
|
data := map[string]interface{}{
|
2019-10-30 08:52:58 +00:00
|
|
|
"dns_addresses": getDNSAddresses(),
|
|
|
|
"http_port": config.BindPort,
|
|
|
|
"dns_port": config.DNS.Port,
|
|
|
|
"running": isRunning(),
|
|
|
|
"version": versionString,
|
|
|
|
"language": config.Language,
|
|
|
|
|
|
|
|
"protection_enabled": c.ProtectionEnabled,
|
|
|
|
"bootstrap_dns": c.BootstrapDNS,
|
|
|
|
"upstream_dns": c.UpstreamDNS,
|
|
|
|
"all_servers": c.AllServers,
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
|
|
|
|
2018-10-20 16:58:39 +00:00
|
|
|
jsonVal, err := json.Marshal(data)
|
2018-08-30 14:25:33 +00:00
|
|
|
if err != nil {
|
2019-02-27 14:28:10 +00:00
|
|
|
httpError(w, http.StatusInternalServerError, "Unable to marshal status json: %s", err)
|
2018-08-30 14:25:33 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "application/json")
|
2018-10-20 16:58:39 +00:00
|
|
|
_, err = w.Write(jsonVal)
|
2018-08-30 14:25:33 +00:00
|
|
|
if err != nil {
|
2019-02-27 14:28:10 +00:00
|
|
|
httpError(w, http.StatusInternalServerError, "Unable to write response json: %s", err)
|
2018-08-30 14:25:33 +00:00
|
|
|
return
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2019-10-21 14:44:07 +00:00
|
|
|
type profileJSON struct {
|
|
|
|
Name string `json:"name"`
|
|
|
|
}
|
|
|
|
|
|
|
|
func handleGetProfile(w http.ResponseWriter, r *http.Request) {
|
|
|
|
pj := profileJSON{}
|
|
|
|
u := config.auth.GetCurrentUser(r)
|
|
|
|
pj.Name = u.Name
|
|
|
|
|
|
|
|
data, err := json.Marshal(pj)
|
|
|
|
if err != nil {
|
|
|
|
httpError(w, http.StatusInternalServerError, "json.Marshal: %s", err)
|
|
|
|
return
|
|
|
|
}
|
|
|
|
_, _ = w.Write(data)
|
|
|
|
}
|
|
|
|
|
2019-02-22 12:52:12 +00:00
|
|
|
// --------------
|
|
|
|
// DNS-over-HTTPS
|
|
|
|
// --------------
|
|
|
|
func handleDOH(w http.ResponseWriter, r *http.Request) {
|
2019-12-13 12:59:36 +00:00
|
|
|
if !config.TLS.AllowUnencryptedDOH && r.TLS == nil {
|
2019-02-22 12:52:12 +00:00
|
|
|
httpError(w, http.StatusNotFound, "Not Found")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
|
|
|
if !isRunning() {
|
|
|
|
httpError(w, http.StatusInternalServerError, "DNS server is not running")
|
|
|
|
return
|
|
|
|
}
|
|
|
|
|
2019-12-11 09:38:58 +00:00
|
|
|
Context.dnsServer.ServeHTTP(w, r)
|
2019-02-22 12:52:12 +00:00
|
|
|
}
|
|
|
|
|
2019-02-12 17:02:52 +00:00
|
|
|
// ------------------------
|
|
|
|
// registration of handlers
|
|
|
|
// ------------------------
|
2018-08-30 14:25:33 +00:00
|
|
|
func registerControlHandlers() {
|
2019-08-21 11:39:37 +00:00
|
|
|
httpRegister(http.MethodGet, "/control/status", handleStatus)
|
|
|
|
httpRegister(http.MethodPost, "/control/i18n/change_language", handleI18nChangeLanguage)
|
|
|
|
httpRegister(http.MethodGet, "/control/i18n/current_language", handleI18nCurrentLanguage)
|
2019-01-29 17:41:57 +00:00
|
|
|
http.HandleFunc("/control/version.json", postInstall(optionalAuth(handleGetVersionJSON)))
|
2019-08-21 11:39:37 +00:00
|
|
|
httpRegister(http.MethodPost, "/control/update", handleUpdate)
|
|
|
|
|
2019-10-21 14:44:07 +00:00
|
|
|
httpRegister("GET", "/control/profile", handleGetProfile)
|
2019-05-24 14:15:19 +00:00
|
|
|
|
2019-09-04 11:12:00 +00:00
|
|
|
RegisterFilteringHandlers()
|
2019-02-27 15:53:16 +00:00
|
|
|
RegisterTLSHandlers()
|
2019-07-23 09:16:36 +00:00
|
|
|
RegisterBlockedServicesHandlers()
|
2019-08-29 09:34:07 +00:00
|
|
|
RegisterAuthHandlers()
|
2019-02-22 12:52:12 +00:00
|
|
|
|
|
|
|
http.HandleFunc("/dns-query", postInstall(handleDOH))
|
2018-08-30 14:25:33 +00:00
|
|
|
}
|
2019-08-21 11:39:37 +00:00
|
|
|
|
2019-09-25 12:36:09 +00:00
|
|
|
func httpRegister(method string, url string, handler func(http.ResponseWriter, *http.Request)) {
|
2019-08-21 11:39:37 +00:00
|
|
|
http.Handle(url, postInstallHandler(optionalAuthHandler(gziphandler.GzipHandler(ensureHandler(method, handler)))))
|
|
|
|
}
|