You can not select more than 25 topics
Topics must start with a letter or number, can include dashes ('-') and can be up to 35 characters long.
168 lines
3.7 KiB
168 lines
3.7 KiB
1 year ago
|
package wireguard
|
||
|
|
||
|
import (
|
||
|
"io/fs"
|
||
|
"strconv"
|
||
|
|
||
|
"asciigoat.org/ini/basic"
|
||
|
"darvaza.org/core"
|
||
|
)
|
||
|
|
||
|
var sectionMap = map[string]func(*Config, *basic.Section) error{
|
||
|
"Interface": loadInterfaceConfSection,
|
||
|
"Peer": loadPeerConfSection,
|
||
|
}
|
||
|
|
||
|
func loadConfSection(out *Config, src *basic.Section) error {
|
||
|
h, ok := sectionMap[src.Key]
|
||
|
if !ok {
|
||
|
return core.Wrap(fs.ErrInvalid, "unknown section %q", src.Key)
|
||
|
}
|
||
|
|
||
|
return h(out, src)
|
||
|
}
|
||
|
|
||
|
func loadInterfaceConfSection(out *Config, src *basic.Section) error {
|
||
|
var cfg InterfaceConfig
|
||
|
|
||
|
for _, field := range src.Fields {
|
||
|
if err := loadInterfaceConfField(&cfg, field); err != nil {
|
||
|
return core.Wrap(err, "Interface")
|
||
|
}
|
||
|
}
|
||
|
|
||
|
out.Interface = cfg
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func loadPeerConfSection(out *Config, src *basic.Section) error {
|
||
|
var cfg PeerConfig
|
||
|
|
||
|
for _, field := range src.Fields {
|
||
|
if err := loadPeerConfField(&cfg, field); err != nil {
|
||
|
return core.Wrap(err, "Peer[%v]", len(out.Peer))
|
||
|
}
|
||
|
}
|
||
|
|
||
|
out.Peer = append(out.Peer, cfg)
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
// revive:disable:cyclomatic
|
||
|
// revive:disable:cognitive-complexity
|
||
|
|
||
|
func loadInterfaceConfField(cfg *InterfaceConfig, field basic.Field) error {
|
||
|
// revive:enable:cyclomatic
|
||
|
// revive:enable:cognitive-complexity
|
||
|
|
||
|
// TODO: refactor when asciigoat's ini parser learns to do reflection
|
||
|
switch field.Key {
|
||
|
case "Address":
|
||
|
if !core.IsZero(cfg.Address) {
|
||
|
return core.Wrap(fs.ErrInvalid, "duplicate field %q", field.Key)
|
||
|
}
|
||
|
|
||
|
err := cfg.Address.UnmarshalText([]byte(field.Value))
|
||
|
switch {
|
||
|
case err != nil:
|
||
|
return core.Wrap(err, field.Key)
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
case "PrivateKey":
|
||
|
if !core.IsZero(cfg.PrivateKey) {
|
||
|
return core.Wrap(fs.ErrInvalid, "duplicate field %q", field.Key)
|
||
|
}
|
||
|
|
||
|
err := cfg.PrivateKey.UnmarshalText([]byte(field.Value))
|
||
|
switch {
|
||
|
case err != nil:
|
||
|
return core.Wrap(err, field.Key)
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
case "ListenPort":
|
||
|
if cfg.ListenPort > 0 {
|
||
|
return core.Wrap(fs.ErrInvalid, "duplicate field %q", field.Key)
|
||
|
}
|
||
|
|
||
|
u64, err := strconv.ParseUint(field.Value, 10, 16)
|
||
|
switch {
|
||
|
case err != nil:
|
||
|
return core.Wrap(err, field.Key)
|
||
|
case u64 == 0:
|
||
|
return core.Wrap(fs.ErrInvalid, "invalid %q value", field.Key)
|
||
|
default:
|
||
|
cfg.ListenPort = uint16(u64)
|
||
|
return nil
|
||
|
}
|
||
|
default:
|
||
|
return core.Wrap(fs.ErrInvalid, "unknown field %q", field.Key)
|
||
|
}
|
||
|
}
|
||
|
|
||
|
// revive:disable:cyclomatic
|
||
|
// revive:disable:cognitive-complexity
|
||
|
|
||
|
func loadPeerConfField(cfg *PeerConfig, field basic.Field) error {
|
||
|
// revive:enable:cyclomatic
|
||
|
// revive:enable:cognitive-complexity
|
||
|
|
||
|
switch field.Key {
|
||
|
case "PublicKey":
|
||
|
if !core.IsZero(cfg.PublicKey) {
|
||
|
return core.Wrap(fs.ErrInvalid, "duplicate field %q", field.Key)
|
||
|
}
|
||
|
|
||
|
err := cfg.PublicKey.UnmarshalText([]byte(field.Value))
|
||
|
switch {
|
||
|
case err != nil:
|
||
|
return core.Wrap(err, field.Key)
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
case "Endpoint":
|
||
|
if cfg.Endpoint.String() != "" {
|
||
|
return core.Wrap(fs.ErrInvalid, "duplicate field %q", field.Key)
|
||
|
}
|
||
|
|
||
|
err := cfg.Endpoint.UnmarshalText([]byte(field.Value))
|
||
|
switch {
|
||
|
case err != nil:
|
||
|
return core.Wrap(err, field.Key)
|
||
|
default:
|
||
|
return nil
|
||
|
}
|
||
|
case "AllowedIPs":
|
||
|
s, err := parseAllowedIPs(field.Value)
|
||
|
switch {
|
||
|
case err != nil:
|
||
|
return core.Wrap(err, field.Key)
|
||
|
case len(s) > 0:
|
||
|
cfg.AllowedIPs = append(cfg.AllowedIPs, s...)
|
||
|
return nil
|
||
|
}
|
||
|
default:
|
||
|
return core.Wrap(fs.ErrInvalid, "unknown field %q", field.Key)
|
||
|
}
|
||
|
return nil
|
||
|
}
|
||
|
|
||
|
func newConfigFromDocument(doc *basic.Document) (*Config, error) {
|
||
|
var out Config
|
||
|
|
||
|
if len(doc.Global) > 0 {
|
||
|
err := core.Wrap(fs.ErrInvalid, "fields before the first section")
|
||
|
return nil, err
|
||
|
}
|
||
|
|
||
|
for i := range doc.Sections {
|
||
|
src := &doc.Sections[i]
|
||
|
if err := loadConfSection(&out, src); err != nil {
|
||
|
return nil, err
|
||
|
}
|
||
|
}
|
||
|
|
||
|
return &out, nil
|
||
|
}
|