-
Notifications
You must be signed in to change notification settings - Fork 71
Commit
This commit does not belong to any branch on this repository, and may belong to a fork outside of the repository.
- Loading branch information
1 parent
2fa0399
commit eec2fc3
Showing
5 changed files
with
119 additions
and
24 deletions.
There are no files selected for viewing
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -1,30 +1,18 @@ | ||
package control | ||
|
||
import "net" | ||
import ( | ||
"net/netip" | ||
) | ||
|
||
type InterfaceFinder interface { | ||
InterfaceIndexByName(name string) (int, error) | ||
InterfaceNameByIndex(index int) (string, error) | ||
InterfaceByAddr(addr netip.Addr) (*Interface, error) | ||
} | ||
|
||
func DefaultInterfaceFinder() InterfaceFinder { | ||
return (*netInterfaceFinder)(nil) | ||
} | ||
|
||
type netInterfaceFinder struct{} | ||
|
||
func (w *netInterfaceFinder) InterfaceIndexByName(name string) (int, error) { | ||
netInterface, err := net.InterfaceByName(name) | ||
if err != nil { | ||
return 0, err | ||
} | ||
return netInterface.Index, nil | ||
} | ||
|
||
func (w *netInterfaceFinder) InterfaceNameByIndex(index int) (string, error) { | ||
netInterface, err := net.InterfaceByIndex(index) | ||
if err != nil { | ||
return "", err | ||
} | ||
return netInterface.Name, nil | ||
type Interface struct { | ||
Index int | ||
MTU int | ||
Name string | ||
Addresses []netip.Prefix | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
Original file line number | Diff line number | Diff line change |
---|---|---|
@@ -0,0 +1,97 @@ | ||
package control | ||
|
||
import ( | ||
"net" | ||
"net/netip" | ||
_ "unsafe" | ||
|
||
"github.com/sagernet/sing/common" | ||
M "github.com/sagernet/sing/common/metadata" | ||
) | ||
|
||
type DefaultInterfaceFinder struct { | ||
interfaces []Interface | ||
} | ||
|
||
func NewDefaultInterfaceFinder() *DefaultInterfaceFinder { | ||
return &DefaultInterfaceFinder{} | ||
} | ||
|
||
func (f *DefaultInterfaceFinder) Update() error { | ||
netIfs, err := net.Interfaces() | ||
if err != nil { | ||
return err | ||
} | ||
interfaces := make([]Interface, 0, len(netIfs)) | ||
for _, netIf := range netIfs { | ||
ifAddrs, err := netIf.Addrs() | ||
if err != nil { | ||
return err | ||
} | ||
interfaces = append(interfaces, Interface{ | ||
Index: netIf.Index, | ||
MTU: netIf.MTU, | ||
Name: netIf.Name, | ||
Addresses: common.Map(ifAddrs, M.PrefixFromNet), | ||
}) | ||
} | ||
f.interfaces = interfaces | ||
return nil | ||
} | ||
|
||
func (f *DefaultInterfaceFinder) UpdateInterfaces(interfaces []Interface) { | ||
f.interfaces = interfaces | ||
} | ||
|
||
func (f *DefaultInterfaceFinder) InterfaceIndexByName(name string) (int, error) { | ||
for _, netInterface := range f.interfaces { | ||
if netInterface.Name == name { | ||
return netInterface.Index, nil | ||
} | ||
} | ||
netInterface, err := net.InterfaceByName(name) | ||
if err != nil { | ||
return 0, err | ||
} | ||
f.Update() | ||
return netInterface.Index, nil | ||
} | ||
|
||
func (f *DefaultInterfaceFinder) InterfaceNameByIndex(index int) (string, error) { | ||
for _, netInterface := range f.interfaces { | ||
if netInterface.Index == index { | ||
return netInterface.Name, nil | ||
} | ||
} | ||
netInterface, err := net.InterfaceByIndex(index) | ||
if err != nil { | ||
return "", err | ||
} | ||
f.Update() | ||
return netInterface.Name, nil | ||
} | ||
|
||
//go:linkname errNoSuchInterface net.errNoSuchInterface | ||
var errNoSuchInterface error | ||
|
||
func (f *DefaultInterfaceFinder) InterfaceByAddr(addr netip.Addr) (*Interface, error) { | ||
for _, netInterface := range f.interfaces { | ||
for _, prefix := range netInterface.Addresses { | ||
if prefix.Contains(addr) { | ||
return &netInterface, nil | ||
} | ||
} | ||
} | ||
err := f.Update() | ||
if err != nil { | ||
return nil, err | ||
} | ||
for _, netInterface := range f.interfaces { | ||
for _, prefix := range netInterface.Addresses { | ||
if prefix.Contains(addr) { | ||
return &netInterface, nil | ||
} | ||
} | ||
} | ||
return nil, &net.OpError{Op: "route", Net: "ip+net", Source: nil, Addr: &net.IPAddr{IP: addr.AsSlice()}, Err: errNoSuchInterface} | ||
} |
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters
This file contains bidirectional Unicode text that may be interpreted or compiled differently than what appears below. To review, open the file in an editor that reveals hidden Unicode characters.
Learn more about bidirectional Unicode characters