1. 程式人生 > 程式設計 >golang獲取網絡卡資訊操作

golang獲取網絡卡資訊操作

我就廢話不多說了,大家還是直接看程式碼吧~

package main
import (
	"fmt"
	"os/exec"
	"strings"
	"github.com/safchain/ethtool"
)
func main() {
	baseNicPath := "/sys/class/net/"
	cmd := exec.Command("ls",baseNicPath)
	buf,err := cmd.Output()
	if err != nil {
		//fmt.Println("Error:",err)
		return
	}
	output := string(buf)
	for _,device := range strings.Split(output,"\n") {
		if len(device) > 1 {
			fmt.Println(device)
			ethHandle,err := ethtool.NewEthtool()
			if err != nil {
				panic(err.Error())
			}
			defer ethHandle.Close()
			stats,err := ethHandle.LinkState(device)
			if err != nil {
				panic(err.Error())
			}
			fmt.Printf("LinkName: %s LinkState: %d\n",device,stats)
		}
	}
}
©

補充:GO 語言取得 Ethernet 型別的網絡卡地址

在 C# 中取得 Ethernet 型別的網絡卡地址

在 C# 中,因為有 NetworkInterface .NetworkInterfaceType == NetworkInterfaceType.Ethernet 所以,很容易在程式碼中進行判斷。

public static PhysicalAddress GetMacAddress()
{
  foreach (NetworkInterface nic in NetworkInterface.GetAllNetworkInterfaces())
  {
    // Only consider Ethernet network interfaces
    if (nic.NetworkInterfaceType == NetworkInterfaceType.Ethernet) // nic.OperationalStatus == OperationalStatus.Up
    {
      return nic.GetPhysicalAddress();
    }
  }
  return null;
}

在 GO 中的處理辦法

搜尋百度百十回,沒有找到答案,無奈,搜尋 net 包的原始碼,找到蛛絲馬跡,在未公開的方法中,找到一個型別的判斷語句。net/interface_windows.go 中,有 case windows.IF_TYPE_ETHERNET_CSMACD 的型別判斷。

再查詢 IF_TYPE_ETHERNET_CSMACD 的資料,正是這裡所需要的。

參考:IP_INTERFACE_NAME_INFO_W2KSP1 structure

golang獲取網絡卡資訊操作

於是,將需要的部分複製出來加以改造,得到 IsEthernet 方法,結合 net 包中已有的方法,引數使用 net.Interface 的 Index 來使用。

完整程式碼如下:

package machine 
import (
  "errors"
  "net"
  "os"
  "strings"
  "syscall"
  "unsafe" 
  "golang.org/x/sys/windows"
)
 
func GetMACAddress() (string,error) {
  netInterfaces,err := net.Interfaces()
  if err != nil {
    panic(err.Error())
  }
 
  mac,macerr := "",errors.New("no valid mac address")
  for i := 0; i < len(netInterfaces); i++ {
    if (netInterfaces[i].Flags&net.FlagLoopback) == 0 && strings.Contains(netInterfaces[i].Flags.String(),"broadcast") {
      index := netInterfaces[i].Index
 
      if isEthernet(index) {
        mac = netInterfaces[i].HardwareAddr.String()
        return mac,nil
      }
    }
  }
  return mac,macerr
}
 
// 根據網絡卡介面 Index 判斷其是否為 Ethernet 網絡卡
func isEthernet(ifindex int) bool {
  aas,err := adapterAddresses()
  if err != nil {
    return false
  }
  result := false
  for _,aa := range aas {
    index := aa.IfIndex
    if ifindex == int(index) {
      switch aa.IfType {
      case windows.IF_TYPE_ETHERNET_CSMACD:
        result = true
      }
 
      if result {
        break
      }
    }
  }
  return result
}
 
// 從 net/interface_windows.go 中複製過來
func adapterAddresses() ([]*windows.IpAdapterAddresses,error) {
  var b []byte
  l := uint32(15000) // recommended initial size
  for {
    b = make([]byte,l)
    err := windows.GetAdaptersAddresses(syscall.AF_UNSPEC,windows.GAA_FLAG_INCLUDE_PREFIX,(*windows.IpAdapterAddresses)(unsafe.Pointer(&b[0])),&l)
    if err == nil {
      if l == 0 {
        return nil,nil
      }
      break
    }
    if err.(syscall.Errno) != syscall.ERROR_BUFFER_OVERFLOW {
      return nil,os.NewSyscallError("getadaptersaddresses",err)
    }
    if l <= uint32(len(b)) {
      return nil,err)
    }
  }
  var aas []*windows.IpAdapterAddresses
  for aa := (*windows.IpAdapterAddresses)(unsafe.Pointer(&b[0])); aa != nil; aa = aa.Next {
    aas = append(aas,aa)
  }
  return aas,nil
}

以上為個人經驗,希望能給大家一個參考,也希望大家多多支援我們。如有錯誤或未考慮完全的地方,望不吝賜教。