Pull request: home: imp err handling, marshalling
Merge in DNS/adguard-home from imp-code to master Squashed commit of the following: commit 9433fb9b0154a1cfaf804edbfa8531efbbcbf68a Author: Ainar Garipov <A.Garipov@AdGuard.COM> Date: Fri May 14 19:24:32 2021 +0300 home: imp err handling, marshalling
This commit is contained in:
parent
9d788a2983
commit
a031cae447
|
@ -95,6 +95,9 @@ on GitHub and most other Markdown renderers. -->
|
||||||
* Constructors should validate their arguments and return meaningful errors.
|
* Constructors should validate their arguments and return meaningful errors.
|
||||||
As a corollary, avoid lazy initialization.
|
As a corollary, avoid lazy initialization.
|
||||||
|
|
||||||
|
* Define `MarshalFoo` methods on non-pointer receivers, as pointer receivers
|
||||||
|
[can have surprising results][staticcheck-911].
|
||||||
|
|
||||||
* Don't mix horizontal and vertical placement of arguments in function and
|
* Don't mix horizontal and vertical placement of arguments in function and
|
||||||
method calls. That is, either this:
|
method calls. That is, either this:
|
||||||
|
|
||||||
|
@ -286,9 +289,10 @@ on GitHub and most other Markdown renderers. -->
|
||||||
|
|
||||||
* <https://go-proverbs.github.io/>
|
* <https://go-proverbs.github.io/>
|
||||||
|
|
||||||
[constant errors]: https://dave.cheney.net/2016/04/07/constant-errors
|
|
||||||
[Linus said]: https://www.kernel.org/doc/html/v4.17/process/coding-style.html#indentation
|
[Linus said]: https://www.kernel.org/doc/html/v4.17/process/coding-style.html#indentation
|
||||||
[Text, Including Comments]: #text-including-comments
|
[Text, Including Comments]: #text-including-comments
|
||||||
|
[constant errors]: https://dave.cheney.net/2016/04/07/constant-errors
|
||||||
|
[staticcheck-911]: https://github.com/dominikh/go-tools/issues/911
|
||||||
|
|
||||||
|
|
||||||
|
|
||||||
|
|
|
@ -84,17 +84,17 @@ type NetInterface struct {
|
||||||
Subnets []*net.IPNet `json:"-"`
|
Subnets []*net.IPNet `json:"-"`
|
||||||
}
|
}
|
||||||
|
|
||||||
// MarshalJSON implements the json.Marshaler interface for *NetInterface.
|
// MarshalJSON implements the json.Marshaler interface for NetInterface.
|
||||||
func (iface *NetInterface) MarshalJSON() ([]byte, error) {
|
func (iface NetInterface) MarshalJSON() ([]byte, error) {
|
||||||
type netInterface NetInterface
|
type netInterface NetInterface
|
||||||
return json.Marshal(&struct {
|
return json.Marshal(&struct {
|
||||||
HardwareAddr string `json:"hardware_address"`
|
HardwareAddr string `json:"hardware_address"`
|
||||||
Flags string `json:"flags"`
|
Flags string `json:"flags"`
|
||||||
*netInterface
|
netInterface
|
||||||
}{
|
}{
|
||||||
HardwareAddr: iface.HardwareAddr.String(),
|
HardwareAddr: iface.HardwareAddr.String(),
|
||||||
Flags: iface.Flags.String(),
|
Flags: iface.Flags.String(),
|
||||||
netInterface: (*netInterface)(iface),
|
netInterface: netInterface(iface),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -43,8 +43,8 @@ func (l *Lease) IsStatic() (ok bool) {
|
||||||
return l != nil && l.Expiry.Unix() == leaseExpireStatic
|
return l != nil && l.Expiry.Unix() == leaseExpireStatic
|
||||||
}
|
}
|
||||||
|
|
||||||
// MarshalJSON implements the json.Marshaler interface for *Lease.
|
// MarshalJSON implements the json.Marshaler interface for Lease.
|
||||||
func (l *Lease) MarshalJSON() ([]byte, error) {
|
func (l Lease) MarshalJSON() ([]byte, error) {
|
||||||
var expiryStr string
|
var expiryStr string
|
||||||
if !l.IsStatic() {
|
if !l.IsStatic() {
|
||||||
// The front-end is waiting for RFC 3999 format of the time
|
// The front-end is waiting for RFC 3999 format of the time
|
||||||
|
@ -59,11 +59,11 @@ func (l *Lease) MarshalJSON() ([]byte, error) {
|
||||||
return json.Marshal(&struct {
|
return json.Marshal(&struct {
|
||||||
HWAddr string `json:"mac"`
|
HWAddr string `json:"mac"`
|
||||||
Expiry string `json:"expires,omitempty"`
|
Expiry string `json:"expires,omitempty"`
|
||||||
*lease
|
lease
|
||||||
}{
|
}{
|
||||||
HWAddr: l.HWAddr.String(),
|
HWAddr: l.HWAddr.String(),
|
||||||
Expiry: expiryStr,
|
Expiry: expiryStr,
|
||||||
lease: (*lease)(l),
|
lease: lease(l),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -71,8 +71,8 @@ func (l *Lease) MarshalJSON() ([]byte, error) {
|
||||||
func (l *Lease) UnmarshalJSON(data []byte) (err error) {
|
func (l *Lease) UnmarshalJSON(data []byte) (err error) {
|
||||||
type lease Lease
|
type lease Lease
|
||||||
aux := struct {
|
aux := struct {
|
||||||
HWAddr string `json:"mac"`
|
|
||||||
*lease
|
*lease
|
||||||
|
HWAddr string `json:"mac"`
|
||||||
}{
|
}{
|
||||||
lease: (*lease)(l),
|
lease: (*lease)(l),
|
||||||
}
|
}
|
||||||
|
|
|
@ -2,6 +2,7 @@ package home
|
||||||
|
|
||||||
import (
|
import (
|
||||||
"bufio"
|
"bufio"
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"hash/crc32"
|
"hash/crc32"
|
||||||
"io"
|
"io"
|
||||||
|
@ -622,29 +623,32 @@ func (f *Filtering) updateIntl(filter *filter) (updated bool, err error) {
|
||||||
}
|
}
|
||||||
|
|
||||||
// loads filter contents from the file in dataDir
|
// loads filter contents from the file in dataDir
|
||||||
func (f *Filtering) load(filter *filter) error {
|
func (f *Filtering) load(filter *filter) (err error) {
|
||||||
filterFilePath := filter.Path()
|
filterFilePath := filter.Path()
|
||||||
log.Tracef("Loading filter %d contents to: %s", filter.ID, filterFilePath)
|
|
||||||
|
|
||||||
if _, err := os.Stat(filterFilePath); os.IsNotExist(err) {
|
log.Tracef("filtering: loading filter %d contents to: %s", filter.ID, filterFilePath)
|
||||||
// do nothing, file doesn't exist
|
|
||||||
return err
|
|
||||||
}
|
|
||||||
|
|
||||||
file, err := os.Open(filterFilePath)
|
file, err := os.Open(filterFilePath)
|
||||||
if err != nil {
|
if errors.Is(err, os.ErrNotExist) {
|
||||||
return err
|
// Do nothing, file doesn't exist.
|
||||||
|
return nil
|
||||||
|
} else if err != nil {
|
||||||
|
return fmt.Errorf("opening filter file: %w", err)
|
||||||
}
|
}
|
||||||
defer file.Close()
|
defer file.Close()
|
||||||
st, _ := file.Stat()
|
|
||||||
|
|
||||||
log.Tracef("File %s, id %d, length %d",
|
st, err := file.Stat()
|
||||||
filterFilePath, filter.ID, st.Size())
|
if err != nil {
|
||||||
|
return fmt.Errorf("getting filter file stat: %w", err)
|
||||||
|
}
|
||||||
|
|
||||||
|
log.Tracef("filtering: File %s, id %d, length %d", filterFilePath, filter.ID, st.Size())
|
||||||
|
|
||||||
rulesCount, checksum, _ := f.parseFilterContents(file)
|
rulesCount, checksum, _ := f.parseFilterContents(file)
|
||||||
|
|
||||||
filter.RulesCount = rulesCount
|
filter.RulesCount = rulesCount
|
||||||
filter.checksum = checksum
|
filter.checksum = checksum
|
||||||
filter.LastUpdated = filter.LastTimeUpdated()
|
filter.LastUpdated = st.ModTime()
|
||||||
|
|
||||||
return nil
|
return nil
|
||||||
}
|
}
|
||||||
|
@ -660,24 +664,6 @@ func (filter *filter) Path() string {
|
||||||
return filepath.Join(Context.getDataDir(), filterDir, strconv.FormatInt(filter.ID, 10)+".txt")
|
return filepath.Join(Context.getDataDir(), filterDir, strconv.FormatInt(filter.ID, 10)+".txt")
|
||||||
}
|
}
|
||||||
|
|
||||||
// LastTimeUpdated returns the time when the filter was last time updated
|
|
||||||
func (filter *filter) LastTimeUpdated() time.Time {
|
|
||||||
filterFilePath := filter.Path()
|
|
||||||
s, err := os.Stat(filterFilePath)
|
|
||||||
if os.IsNotExist(err) {
|
|
||||||
// if the filter file does not exist, return 0001-01-01
|
|
||||||
return time.Time{}
|
|
||||||
}
|
|
||||||
|
|
||||||
if err != nil {
|
|
||||||
// if the filter file does not exist, return 0001-01-01
|
|
||||||
return time.Time{}
|
|
||||||
}
|
|
||||||
|
|
||||||
// filter file modified time
|
|
||||||
return s.ModTime()
|
|
||||||
}
|
|
||||||
|
|
||||||
func enableFilters(async bool) {
|
func enableFilters(async bool) {
|
||||||
var whiteFilters []dnsfilter.Filter
|
var whiteFilters []dnsfilter.Filter
|
||||||
filters := []dnsfilter.Filter{{
|
filters := []dnsfilter.Filter{{
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package home
|
package home
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"io/ioutil"
|
"io/ioutil"
|
||||||
"os"
|
"os"
|
||||||
|
@ -88,24 +89,27 @@ func svcAction(s service.Service, action string) (err error) {
|
||||||
// If pid-file doesn't exist, find our PID using 'ps' command
|
// If pid-file doesn't exist, find our PID using 'ps' command
|
||||||
func sendSigReload() {
|
func sendSigReload() {
|
||||||
if runtime.GOOS == "windows" {
|
if runtime.GOOS == "windows" {
|
||||||
log.Error("Not implemented on Windows")
|
log.Error("not implemented on windows")
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
pidfile := fmt.Sprintf("/var/run/%s.pid", serviceName)
|
pidfile := fmt.Sprintf("/var/run/%s.pid", serviceName)
|
||||||
data, err := ioutil.ReadFile(pidfile)
|
data, err := ioutil.ReadFile(pidfile)
|
||||||
if os.IsNotExist(err) {
|
if errors.Is(err, os.ErrNotExist) {
|
||||||
var code int
|
var code int
|
||||||
var psdata string
|
var psdata string
|
||||||
code, psdata, err = aghos.RunCommand("ps", "-C", serviceName, "-o", "pid=")
|
code, psdata, err = aghos.RunCommand("ps", "-C", serviceName, "-o", "pid=")
|
||||||
if err != nil || code != 0 {
|
if err != nil || code != 0 {
|
||||||
log.Error("Can't find AdGuardHome process: %s code:%d", err, code)
|
log.Error("finding AdGuardHome process: code: %d, error: %s", code, err)
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
data = []byte(psdata)
|
|
||||||
|
|
||||||
|
data = []byte(psdata)
|
||||||
} else if err != nil {
|
} else if err != nil {
|
||||||
log.Error("Can't read PID file %s: %s", pidfile, err)
|
log.Error("reading pid file %s: %s", pidfile, err)
|
||||||
|
|
||||||
return
|
return
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -258,12 +262,12 @@ func handleServiceUninstallCommand(s service.Service) {
|
||||||
if runtime.GOOS == "darwin" {
|
if runtime.GOOS == "darwin" {
|
||||||
// Remove log files on cleanup and log errors.
|
// Remove log files on cleanup and log errors.
|
||||||
err = os.Remove(launchdStdoutPath)
|
err = os.Remove(launchdStdoutPath)
|
||||||
if err != nil && !os.IsNotExist(err) {
|
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
||||||
log.Printf("removing stdout file: %s", err)
|
log.Printf("removing stdout file: %s", err)
|
||||||
}
|
}
|
||||||
|
|
||||||
err = os.Remove(launchdStderrPath)
|
err = os.Remove(launchdStderrPath)
|
||||||
if err != nil && !os.IsNotExist(err) {
|
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
||||||
log.Printf("removing stderr file: %s", err)
|
log.Printf("removing stderr file: %s", err)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,6 +1,7 @@
|
||||||
package home
|
package home
|
||||||
|
|
||||||
import (
|
import (
|
||||||
|
"errors"
|
||||||
"fmt"
|
"fmt"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
|
@ -115,17 +116,16 @@ func upgradeConfigSchema(oldVersion int, diskConf yobj) (err error) {
|
||||||
|
|
||||||
// The first schema upgrade:
|
// The first schema upgrade:
|
||||||
// No more "dnsfilter.txt", filters are now kept in data/filters/
|
// No more "dnsfilter.txt", filters are now kept in data/filters/
|
||||||
func upgradeSchema0to1(diskConf yobj) error {
|
func upgradeSchema0to1(diskConf yobj) (err error) {
|
||||||
log.Printf("%s(): called", funcName())
|
log.Printf("%s(): called", funcName())
|
||||||
|
|
||||||
dnsFilterPath := filepath.Join(Context.workDir, "dnsfilter.txt")
|
dnsFilterPath := filepath.Join(Context.workDir, "dnsfilter.txt")
|
||||||
if _, err := os.Stat(dnsFilterPath); !os.IsNotExist(err) {
|
log.Printf("deleting %s as we don't need it anymore", dnsFilterPath)
|
||||||
log.Printf("Deleting %s as we don't need it anymore", dnsFilterPath)
|
|
||||||
err = os.Remove(dnsFilterPath)
|
err = os.Remove(dnsFilterPath)
|
||||||
if err != nil {
|
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
||||||
log.Printf("Cannot remove %s due to %s", dnsFilterPath, err)
|
log.Info("warning: %s", err)
|
||||||
// not fatal, move on
|
|
||||||
}
|
// Go on.
|
||||||
}
|
}
|
||||||
|
|
||||||
diskConf["schema_version"] = 1
|
diskConf["schema_version"] = 1
|
||||||
|
@ -136,17 +136,16 @@ func upgradeSchema0to1(diskConf yobj) error {
|
||||||
// Second schema upgrade:
|
// Second schema upgrade:
|
||||||
// coredns is now dns in config
|
// coredns is now dns in config
|
||||||
// delete 'Corefile', since we don't use that anymore
|
// delete 'Corefile', since we don't use that anymore
|
||||||
func upgradeSchema1to2(diskConf yobj) error {
|
func upgradeSchema1to2(diskConf yobj) (err error) {
|
||||||
log.Printf("%s(): called", funcName())
|
log.Printf("%s(): called", funcName())
|
||||||
|
|
||||||
coreFilePath := filepath.Join(Context.workDir, "Corefile")
|
coreFilePath := filepath.Join(Context.workDir, "Corefile")
|
||||||
if _, err := os.Stat(coreFilePath); !os.IsNotExist(err) {
|
log.Printf("deleting %s as we don't need it anymore", coreFilePath)
|
||||||
log.Printf("Deleting %s as we don't need it anymore", coreFilePath)
|
|
||||||
err = os.Remove(coreFilePath)
|
err = os.Remove(coreFilePath)
|
||||||
if err != nil {
|
if err != nil && !errors.Is(err, os.ErrNotExist) {
|
||||||
log.Printf("Cannot remove %s due to %s", coreFilePath, err)
|
log.Info("warning: %s", err)
|
||||||
// not fatal, move on
|
|
||||||
}
|
// Go on.
|
||||||
}
|
}
|
||||||
|
|
||||||
if _, ok := diskConf["dns"]; !ok {
|
if _, ok := diskConf["dns"]; !ok {
|
||||||
|
|
Loading…
Reference in New Issue