0
  • 聊天消息
  • 系統(tǒng)消息
  • 評(píng)論與回復(fù)
登錄后你可以
  • 下載海量資料
  • 學(xué)習(xí)在線課程
  • 觀看技術(shù)視頻
  • 寫文章/發(fā)帖/加入社區(qū)
會(huì)員中心
創(chuàng)作中心

完善資料讓更多小伙伴認(rèn)識(shí)你,還能領(lǐng)取20積分哦,立即完善>

3天內(nèi)不再提示

探究Go語(yǔ)言如何實(shí)現(xiàn)簡(jiǎn)易版netstat命令

阿銘linux ? 來源:杰哥的IT之旅 ? 作者: 杰哥的IT之旅 ? 2021-07-27 10:35 ? 次閱讀

netstat 使用 go 語(yǔ)言實(shí)現(xiàn)是什么操作?本文從 netstat 原理出發(fā)詳細(xì)解讀了這一實(shí)踐。

netstat 工作原理

netstat 命令是 linux 系統(tǒng)中查看網(wǎng)絡(luò)情況的一個(gè)命令。比如我們可以通過netstat -ntlp | grep 8080查看監(jiān)聽 8080 端口的進(jìn)程。

17f98660-e3ff-11eb-a97a-12bb97331649.jpg

netstat 工作原理如下:

通過讀取/proc/net/tcp 、/proc/net/tcp6 文件,獲取 socket 本地地址,本地端口,遠(yuǎn)程地址,遠(yuǎn)程端口,狀態(tài),inode 等信息

接著掃描所有/proc/[pid]/fd 目錄下的的 socket 文件描述符,建立 inode 到進(jìn)程 pid 映射

根據(jù) pid 讀取/proc/[pid]/cmdline 文件,獲取進(jìn)程命令和啟動(dòng)參數(shù)

根據(jù) 2,3 步驟,即可以獲得 1 中對(duì)應(yīng) socket 的相關(guān)進(jìn)程信息

我們可以做個(gè)測(cè)試驗(yàn)證整個(gè)流程。先使用 nc 命令監(jiān)聽 8090 端口:

nc -l 8090

找到上面 nc 進(jìn)程的 pid,查看該進(jìn)程所有打開的文件描述符:

vagrant@vagrant:/proc/25556/fd$ ls -alh

total 0

dr-x------ 2 vagrant vagrant 0 Nov 18 12:21 。

dr-xr-xr-x 9 vagrant vagrant 0 Nov 18 12:20 。。

lrwx------ 1 vagrant vagrant 64 Nov 18 12:21 0 -》 /dev/pts/1

lrwx------ 1 vagrant vagrant 64 Nov 18 12:21 1 -》 /dev/pts/1

lrwx------ 1 vagrant vagrant 64 Nov 18 12:21 2 -》 /dev/pts/1

lrwx------ 1 vagrant vagrant 64 Nov 18 12:21 3 -》 socket:[2226056]

上面列出的所有文件描述中,socket:[2226056]為 nc 命令監(jiān)聽 8090 端口所創(chuàng)建的 socket。其中2226056為該 socket 的 inode。

根據(jù)該 inode 號(hào),我們查看/proc/net/tcp對(duì)應(yīng)的記錄信息,其中1F9A為本地端口號(hào),轉(zhuǎn)換成十進(jìn)制恰好為 8090:

vagrant@vagrant:/proc/25556/fd$ cat /proc/net/tcp | grep 2226056

1: 00000000:1F9A 00000000:0000 0A 00000000:00000000 00:00000000 00000000 1000 0 2226056 1 0000000000000000 100 0 0 10 0

根據(jù)進(jìn)程 id,我們查看進(jìn)程名稱和啟動(dòng)參數(shù):

vagrant@vagrant:/proc/25556/fd$ cat /proc/25556/cmdline

nc-l8090

下面我們看下/proc/net/tcp文件格式。

/proc/net/tcp 文件格式

/proc/net/tcp文件首先會(huì)列出所有監(jiān)聽狀態(tài)的 TCP 套接字,然后列出所有已建立的 TCP 套接字。我們通過head -n 5 /proc/net/tcp命令查看該文件頭五行:

sl local_address rem_address st tx_queue rx_queue tr tm-》when retrnsmt uid timeout inode

0: 0100007F:0019 00000000:0000 0A 00000000:00000000 00:00000000 00000000 0 0 22279 1 0000000000000000 100 0 0 10 0

1: 00000000:1FBB 00000000:0000 0A 00000000:00000000 00:00000000 00000000 0 0 21205 1 0000000000000000 100 0 0 10 0

2: 00000000:26FB 00000000:0000 0A 00000000:00000000 00:00000000 00000000 0 0 21203 1 0000000000000000 100 0 0 10 0

3: 00000000:26FD 00000000:0000 0A 00000000:00000000 00:00000000 00000000 0 0 21201 1 0000000000000000 100 0 0 10 0

每一行各個(gè)字段解釋說明如下,由于太長(zhǎng)分為三部分說明:

第一部分:

46: 010310AC:9C4C 030310AC:1770 01

| | | | | |--》 連接狀態(tài),16 進(jìn)制表示,具體值見下面說明

| | | | |------》 遠(yuǎn)程 TCP 端口號(hào),主機(jī)字節(jié)序,16 進(jìn)制表示

| | | |-------------》 遠(yuǎn)程 IPv4 地址,網(wǎng)絡(luò)字節(jié)序,16 進(jìn)制表示

| | |--------------------》 本地 TCP 端口號(hào),主機(jī)字節(jié)序,16 進(jìn)制表示

| |---------------------------》 本地 IPv4 地址,網(wǎng)絡(luò)字節(jié)序,16 進(jìn)制表示

|----------------------------------》 條目編號(hào),從 0 開始

上面連接狀態(tài)所有值如下,具體參見 linux 源碼 tcp\_states.h[1]:

enum {

TCP_ESTABLISHED = 1,

TCP_SYN_SENT,

TCP_SYN_RECV,

TCP_FIN_WAIT1,

TCP_FIN_WAIT2,

TCP_TIME_WAIT,

TCP_CLOSE,

TCP_CLOSE_WAIT,

TCP_LAST_ACK,

TCP_LISTEN,

TCP_CLOSING, /* Now a valid state */

TCP_NEW_SYN_RECV,

TCP_MAX_STATES /* Leave at the end! */

};

第二部分:

00000150:00000000 01:00000019 00000000

| | | | |--》 number of unrecovered RTO timeouts

| | | |----------》 number of jiffies until timer expires

| | |----------------》 timer_active,具體值見下面說明

| |----------------------》 receive-queue,當(dāng)狀態(tài)是 ESTABLISHED,表示接收隊(duì)列中數(shù)據(jù)長(zhǎng)度;狀態(tài)是 LISTEN,表示已經(jīng)完成連接隊(duì)列的長(zhǎng)度

|-------------------------------》 transmit-queue,發(fā)送隊(duì)列中數(shù)據(jù)長(zhǎng)度

timer_active 所有值與說明如下:

0 no timer is pending

1 retransmit-timer is pending

2 another timer (e.g. delayed ack or keepalive) is pending

3 this is a socket in TIME_WAIT state. Not all fields will contain data (or even exist)

4 zero window probe timer is pending

第三部分:

1000 0 54165785 4 cd1e6040 25 4 27 3 -1

| | | | | | | | | |--》 slow start size threshold,

| | | | | | | | | or -1 if the threshold

| | | | | | | | | is 》= 0xFFFF

| | | | | | | | |----》 sending congestion window

| | | | | | | |-------》 (ack.quick《《1)|ack.pingpong

| | | | | | |---------》 Predicted tick of soft clock

| | | | | | (delayed ACK control data)

| | | | | |------------》 retransmit timeout

| | | | |------------------》 location of socket in memory

| | | |-----------------------》 socket reference count

| | |-----------------------------》 socket 的 inode 號(hào)

| |----------------------------------》 unanswered 0-window probes

|---------------------------------------------》 socket 所屬用戶的 uid

Go 實(shí)現(xiàn)簡(jiǎn)易版本 netstat 命令

netstat 工作原理和/proc/net/tcp文件結(jié)構(gòu),我們都已經(jīng)了解了,現(xiàn)在可以使用據(jù)此使用 Go 實(shí)現(xiàn)一個(gè)簡(jiǎn)單版本的 netstat 命令。

18041148-e3ff-11eb-a97a-12bb97331649.jpg

核心代碼如下,完整代碼參加 go-netstat[2]:

// 狀態(tài)碼值const (

TCP_ESTABLISHED = iota + 1

TCP_SYN_SENT

TCP_SYN_RECV

TCP_FIN_WAIT1

TCP_FIN_WAIT2

TCP_TIME_WAIT

TCP_CLOSE

TCP_CLOSE_WAIT

TCP_LAST_ACK

TCP_LISTEN

TCP_CLOSING

//TCP_NEW_SYN_RECV

//TCP_MAX_STATES

// 狀態(tài)碼var states = map[int]string{

TCP_ESTABLISHED: “ESTABLISHED”,

TCP_SYN_SENT: “SYN_SENT”,

TCP_SYN_RECV: “SYN_RECV”,

TCP_FIN_WAIT1: “FIN_WAIT1”,

TCP_FIN_WAIT2: “FIN_WAIT2”,

TCP_TIME_WAIT: “TIME_WAIT”,

TCP_CLOSE: “CLOSE”,

TCP_CLOSE_WAIT: “CLOSE_WAIT”,

TCP_LAST_ACK: “LAST_ACK”,

TCP_LISTEN: “LISTEN”,

TCP_CLOSING: “CLOSING”,

//TCP_NEW_SYN_RECV: “NEW_SYN_RECV”,

//TCP_MAX_STATES: “MAX_STATES”,

}

// socketEntry 結(jié)構(gòu)體,用來存儲(chǔ)/proc/net/tcp 每一行解析后數(shù)據(jù)信息type socketEntry struct {

id int

srcIP net.IP

srcPort int

dstIP net.IP

dstPort int

state string

txQueue int

rxQueue int

timer int8

timerDuration time.Duration

rto time.Duration // retransmission timeout

uid int

uname string

timeout time.Duration

inode string

}

// 解析/proc/net/tcp 行記錄func parseRawSocketEntry(entry string) (*socketEntry, error) {

se := &socketEntry{}

entrys := strings.Split(strings.TrimSpace(entry), “ ”)

entryItems := make([]string, 0, 17)

for _, ent := range entrys {

if ent == “” {

continue

}

entryItems = append(entryItems, ent)

}

id, err := strconv.Atoi(string(entryItems[0][:len(entryItems[0])-1]))

if err != nil {

return nil, err

}

se.id = id // sockect entry id

localAddr := strings.Split(entryItems[1], “:”) // 本地 ip

se.srcIP = parseHexBigEndianIPStr(localAddr[0])

port, err := strconv.ParseInt(localAddr[1], 16, 32) // 本地 port

if err != nil {

return nil, err

}

se.srcPort = int(port)

remoteAddr := strings.Split(entryItems[2], “:”) // 遠(yuǎn)程 ip

se.dstIP = parseHexBigEndianIPStr(remoteAddr[0])

port, err = strconv.ParseInt(remoteAddr[1], 16, 32) // 遠(yuǎn)程 port

if err != nil {

return nil, err

}

se.dstPort = int(port)

state, _ := strconv.ParseInt(entryItems[3], 16, 32) // socket 狀態(tài)

se.state = states[int(state)]

tcpQueue := strings.Split(entryItems[4], “:”)

tQueue, err := strconv.ParseInt(tcpQueue[0], 16, 32) // 發(fā)送隊(duì)列數(shù)據(jù)長(zhǎng)度

if err != nil {

return nil, err

}

se.txQueue = int(tQueue)

sQueue, err := strconv.ParseInt(tcpQueue[1], 16, 32) // 接收隊(duì)列數(shù)據(jù)長(zhǎng)度

if err != nil {

return nil, err

}

se.rxQueue = int(sQueue)

se.uid, err = strconv.Atoi(entryItems[7]) // socket uid

if err != nil {

return nil, err

}

se.uname = systemUsers[entryItems[7]] // socket user name

se.inode = entryItems[9] // socket inode

return se, nil

}

// hexIP 是網(wǎng)絡(luò)字節(jié)序/大端法轉(zhuǎn)換成的 16 進(jìn)制的字符串func parseHexBigEndianIPStr(hexIP string) net.IP {

b := []byte(hexIP)

for i, j := 1, len(b)-2; i 《 j; i, j = i+2, j-2 { // 反轉(zhuǎn)字節(jié),轉(zhuǎn)換成小端法

b[i], b[i-1], b[j], b[j+1] = b[j+1], b[j], b[i-1], b[i]

}

l, _ := strconv.ParseInt(string(b), 16, 64)

return net.IPv4(byte(l》》24), byte(l》》16), byte(l》》8), byte(l))

}

編輯:jq

聲明:本文內(nèi)容及配圖由入駐作者撰寫或者入駐合作網(wǎng)站授權(quán)轉(zhuǎn)載。文章觀點(diǎn)僅代表作者本人,不代表電子發(fā)燒友網(wǎng)立場(chǎng)。文章及其配圖僅供工程師學(xué)習(xí)之用,如有內(nèi)容侵權(quán)或者其他違規(guī)問題,請(qǐng)聯(lián)系本站處理。 舉報(bào)投訴
  • Linux
    +關(guān)注

    關(guān)注

    87

    文章

    11211

    瀏覽量

    208721

原文標(biāo)題:Go 語(yǔ)言實(shí)現(xiàn)簡(jiǎn)易版 netstat 命令

文章出處:【微信號(hào):aming_linux,微信公眾號(hào):阿銘linux】歡迎添加關(guān)注!文章轉(zhuǎn)載請(qǐng)注明出處。

收藏 人收藏

    評(píng)論

    相關(guān)推薦

    在學(xué)習(xí)go語(yǔ)言的過程踩過的坑

    作為一個(gè)5年的phper,這兩年公司和個(gè)人都在順應(yīng)技術(shù)趨勢(shì),新項(xiàng)目慢慢從php轉(zhuǎn)向了go語(yǔ)言,從2021年到現(xiàn)在,筆者手上也先后開發(fā)了兩個(gè)go項(xiàng)目。在學(xué)習(xí)go
    的頭像 發(fā)表于 11-11 09:22 ?33次閱讀

    go語(yǔ)言如何解決并發(fā)問題

    作為一個(gè)后端開發(fā),日常工作中接觸最多的兩門語(yǔ)言就是PHP和GO了。無可否認(rèn),PHP確實(shí)是最好的語(yǔ)言(手動(dòng)狗頭哈哈),寫起來真的很舒爽,沒有任何心智負(fù)擔(dān),字符串和整型壓根就不用區(qū)分,開發(fā)速度真的是比
    的頭像 發(fā)表于 10-23 13:38 ?93次閱讀
    <b class='flag-5'>go</b><b class='flag-5'>語(yǔ)言</b>如何解決并發(fā)問題

    verilog設(shè)計(jì)之基于basys3實(shí)現(xiàn)簡(jiǎn)易分秒數(shù)字鐘

    基于basys3實(shí)現(xiàn)簡(jiǎn)易分秒數(shù)字鐘
    發(fā)表于 09-03 14:15 ?0次下載

    三十分鐘入門基礎(chǔ)Go Java小子版

    前言 Go語(yǔ)言定義 Go(又稱 Golang)是 Google 的 Robert Griesemer,Rob Pike 及 Ken Thompson 開發(fā)的一種靜態(tài)、強(qiáng)類型、編譯型語(yǔ)言
    的頭像 發(fā)表于 08-12 14:32 ?661次閱讀
    三十分鐘入門基礎(chǔ)<b class='flag-5'>Go</b> Java小子版

    用ESP32,做了個(gè)siri?!開源了!

    觸碰交互3.通過修改代碼,可以設(shè)置AI的對(duì)話方式和回答風(fēng)格4.支持用于拓展設(shè)計(jì),例如:智能MP3、簡(jiǎn)易版siri/小愛同學(xué)、實(shí)時(shí)翻譯工具…想實(shí)現(xiàn)這些功能,軟硬件如
    的頭像 發(fā)表于 06-29 08:04 ?328次閱讀
    用ESP32,做了個(gè)siri?!開源了!

    AWTK 開源串口屏開發(fā)(18) - 用 C 語(yǔ)言自定義命令

    編寫代碼即可實(shí)現(xiàn)常見的應(yīng)用。但是,有時(shí)候我們需要自定義一些命令,以實(shí)現(xiàn)一些特殊的功能。本文檔介紹如何使用C語(yǔ)言自定義命令。1.
    的頭像 發(fā)表于 05-11 08:24 ?404次閱讀
    AWTK 開源串口屏開發(fā)(18) - 用 C <b class='flag-5'>語(yǔ)言</b>自定義<b class='flag-5'>命令</b>

    關(guān)于go中接口類型的表示方法

    go是一個(gè)靜態(tài)性語(yǔ)言,每個(gè)變量都有靜態(tài)的類型,因此每個(gè)變量在編譯階段中有明確的變量類型,比如像:int、float32、MyType。
    的頭像 發(fā)表于 04-28 10:13 ?323次閱讀

    Go語(yǔ)言中的函數(shù)、方法與接口詳解

    Go 沒有類,不過可以為結(jié)構(gòu)體類型定義方法。方法就是一類帶特殊的接收者參數(shù)的函數(shù)。方法接收者在它自己的參數(shù)列表內(nèi),位于 func 關(guān)鍵字和方法名之間。(非結(jié)構(gòu)體類型也可以定義方法)
    的頭像 發(fā)表于 04-23 16:21 ?717次閱讀

    學(xué)習(xí)筆記|如何用Go程序采集溫濕度傳感器數(shù)據(jù)

    整個(gè)利用Go語(yǔ)言從AHT20溫濕度傳感器獲取數(shù)據(jù)的完整步驟和實(shí)現(xiàn)方法。一、C程序獲取AHT20溫濕度1、參考ElfBoard官方例程:03-例程源碼\03-1命令
    的頭像 發(fā)表于 03-21 11:46 ?608次閱讀
    學(xué)習(xí)筆記|如何用<b class='flag-5'>Go</b>程序采集溫濕度傳感器數(shù)據(jù)

    使用go語(yǔ)言實(shí)現(xiàn)一個(gè)grpc攔截器

    在開發(fā)grpc服務(wù)時(shí),我們經(jīng)常會(huì)遇到一些通用的需求,比如:日志、鏈路追蹤、鑒權(quán)等。這些需求可以通過grpc攔截器來實(shí)現(xiàn)。本文使用go語(yǔ)言實(shí)現(xiàn)一個(gè) grpc一元模式(Unary)攔截器
    的頭像 發(fā)表于 12-18 10:13 ?616次閱讀
    使用<b class='flag-5'>go</b><b class='flag-5'>語(yǔ)言實(shí)現(xiàn)</b>一個(gè)grpc攔截器

    Go編程語(yǔ)言-你應(yīng)該知道的一切

    Go 編程語(yǔ)言的故事始于 Google,當(dāng)時(shí)三位工程師 Robert Griesemer、Rob Pike 和 Ken Thompson 對(duì) C++ 的復(fù)雜性以及缺乏提供高效編譯和執(zhí)行的簡(jiǎn)單語(yǔ)言感到厭倦。
    的頭像 發(fā)表于 12-11 17:37 ?598次閱讀

    idea如何輸入命令行參數(shù)

    在許多軟件開發(fā)和系統(tǒng)管理的任務(wù)中,我們經(jīng)常需要向應(yīng)用程序傳遞命令行參數(shù)。命令行參數(shù)是在運(yùn)行時(shí)傳遞給程序的值,用于指定程序的行為和配置選項(xiàng)。本文將詳細(xì)介紹如何在不同的編程語(yǔ)言和操作系統(tǒng)中輸入命令
    的頭像 發(fā)表于 12-06 15:01 ?1032次閱讀

    Uboot命令實(shí)現(xiàn)大致流程圖

    zImage內(nèi)核鏡像下載到開發(fā)板之后,可以使用u-boot的go命令進(jìn)行直接跳轉(zhuǎn),這個(gè)時(shí)候內(nèi)核直接解壓?jiǎn)?dòng)。 但是此時(shí)的內(nèi)核無法掛載文件系統(tǒng),因?yàn)?b class='flag-5'>go命令沒有將內(nèi)核需要的相關(guān)啟動(dòng)參數(shù)從
    的頭像 發(fā)表于 12-04 17:09 ?377次閱讀
    Uboot<b class='flag-5'>命令</b><b class='flag-5'>實(shí)現(xiàn)</b>大致流程圖

    python調(diào)用windows命令

    Python是一種強(qiáng)大的編程語(yǔ)言,可以用于開發(fā)各種不同類型的應(yīng)用程序。其中一個(gè)常見的用途是使用Python調(diào)用Windows命令來執(zhí)行特定的任務(wù)。在本文中,我們將詳細(xì)討論如何使用Python調(diào)用
    的頭像 發(fā)表于 11-29 14:34 ?965次閱讀

    Python自帶的命令窗口

    Python自帶的命令窗口,也稱為Python交互式解釋器,是Python編程語(yǔ)言的一個(gè)重要工具,它允許用戶在命令行界面中輸入和執(zhí)行Python代碼。不同于編寫腳本并保存為文件后再執(zhí)行,Python
    的頭像 發(fā)表于 11-22 14:02 ?859次閱讀