0

0

使用 Go 语言进行原始套接字编程

聖光之護

聖光之護

发布时间:2025-10-25 12:30:30

|

678人浏览过

|

来源于php中文网

原创

使用 go 语言进行原始套接字编程

本文旨在指导开发者如何使用 Go 语言进行原始套接字编程,以实现自定义网络数据包的发送和接收。重点介绍使用 go.net/ipv4 库创建和操作原始套接字,以及如何构造自定义 IP 头部来实现源 IP 地址欺骗等高级网络功能。同时,也强调了使用原始套接字的安全风险和权限要求。

在某些网络编程场景下,标准库提供的套接字 API 可能无法满足需求,例如需要修改 IP 头部、实现自定义协议等。这时,原始套接字就派上了用场。Go 语言的标准 net 库并没有直接提供原始套接字的支持,但 go.net 子仓库提供了 ipv4 和 ipv6 包,可以用来进行原始套接字编程。

安全性考虑

使用原始套接字需要特别注意安全性问题。由于可以自定义 IP 头部,包括源 IP 地址,这可能被用于恶意攻击,例如 IP 地址欺骗。在 Linux 系统下,通常需要以 root 权限运行程序,或者赋予程序 CAP_NET_RAW capability 才能使用原始套接字。可以使用 setcap 命令赋予程序 capability:

sudo setcap cap_net_raw+ep 

使用 go.net/ipv4 进行原始套接字编程

go.net/ipv4 包提供了创建和操作 IPv4 原始套接字的 API。核心是 ipv4.RawConn 类型。

1. 创建原始套接字

首先,需要创建一个 ipv4.RawConn 实例。可以使用 ipv4.NewRawConn 函数:

import (
    "fmt"
    "log"
    "net"

    "golang.org/x/net/ipv4"
)

func main() {
    // 创建 IPv4 原始套接字
    conn, err := net.ListenIP("ip4:icmp", &net.IPAddr{IP: net.IPv4zero})
    if err != nil {
        log.Fatal(err)
    }
    defer conn.Close()

    rawConn, err := ipv4.NewRawConn(conn)
    if err != nil {
        log.Fatal(err)
    }
    defer rawConn.Close()

    fmt.Println("Raw socket created successfully!")
}

这段代码创建了一个监听 ICMP 协议的 IPv4 原始套接字。 net.ListenIP 用于创建一个底层的 IP 连接,然后 ipv4.NewRawConn 基于这个连接创建一个 ipv4.RawConn 实例。

2. 读取数据包

可以使用 ipv4.RawConn 的 ReadFrom 方法读取接收到的数据包:

    buf := make([]byte, 1500) // MTU 大小
    for {
        hdr, payload, peer, err := rawConn.ReadFrom(buf)
        if err != nil {
            log.Println("Error reading:", err)
            continue
        }

        fmt.Printf("Received packet from: %v\n", peer)
        fmt.Printf("Header: %+v\n", hdr)
        fmt.Printf("Payload: %v\n", payload)
    }

ReadFrom 方法返回 IP 头部、数据载荷以及发送方的地址。

网趣购物系统加强升级版
网趣购物系统加强升级版

新版本程序更新主要体现在:完美整合BBS论坛程序,用户只须注册一个帐号,即可全站通用!采用目前流行的Flash滚动切换广告 变换形式多样,受人喜爱!在原有提供的5种在线支付基础上增加北京云网支付!对留言本重新进行编排,加入留言验证码,后台有留言审核开关对购物系统的前台进行了一处安全更新。在原有文字友情链接基础上,增加LOGO友情链接功能强大的6种在线支付方式可选,自由切换。对新闻列表进行了调整,

下载

3. 构造和发送数据包

可以使用 ipv4.RawConn 的 WriteTo 方法发送自定义的数据包。需要手动构造 IP 头部。

    // 构造 IP 头部
    ipHeader := &ipv4.Header{
        Version:  ipv4.Version,
        Len:      ipv4.HeaderLen,
        TOS:      0,
        TotalLen: ipv4.HeaderLen + len(payload),
        ID:       0,
        Flags:    0,
        FragOff:  0,
        TTL:      64,
        Protocol: 1, // ICMP
        Checksum: 0,
        Src:      net.ParseIP("192.168.1.100").To4(), // 伪造的源 IP
        Dst:      net.ParseIP("8.8.8.8").To4(),      // 目标 IP
    }

    // 计算校验和 (需要自行实现)
    ipHeader.Checksum = checksum(ipHeader, payload)

    // 发送数据包
    err = rawConn.WriteTo(ipHeader, payload, &net.IPAddr{IP: ipHeader.Dst})
    if err != nil {
        log.Println("Error writing:", err)
    }

这段代码构造了一个包含伪造源 IP 地址的 IP 头部,并使用 WriteTo 方法发送出去。注意,checksum 函数需要自行实现,用于计算 IP 头部的校验和。

4. 校验和计算

IP 头部的校验和计算是一个常见的操作。以下是一个示例的校验和计算函数:

func checksum(hdr *ipv4.Header, payload []byte) uint16 {
    h := ipv4.Header{
        Version:  ipv4.Version,
        Len:      ipv4.HeaderLen,
        TOS:      hdr.TOS,
        TotalLen: ipv4.HeaderLen + len(payload),
        ID:       hdr.ID,
        Flags:    hdr.Flags,
        FragOff:  hdr.FragOff,
        TTL:      hdr.TTL,
        Protocol: hdr.Protocol,
        Checksum: 0,
        Src:      hdr.Src,
        Dst:      hdr.Dst,
    }

    headerBytes, err := h.Marshal()
    if err != nil {
        panic(err)
    }
    data := append(headerBytes, payload...)
    var sum uint32
    for i := 0; i < len(data)-1; i += 2 {
        sum += uint32(data[i])<<8 | uint32(data[i+1])
    }
    if len(data)%2 == 1 {
        sum += uint32(data[len(data)-1]) << 8
    }
    for sum>>16 != 0 {
        sum = (sum & 0xffff) + (sum >> 16)
    }
    return uint16(^sum)
}

完整示例

下面是一个完整的示例代码,演示了如何使用 go.net/ipv4 创建原始套接字,发送包含自定义 IP 头部和 ICMP 协议数据的数据包。

package main

import (
    "fmt"
    "log"
    "net"
    "time"

    "golang.org/x/net/icmp"
    "golang.org/x/net/ipv4"
)

func checksum(hdr *ipv4.Header, payload []byte) uint16 {
    h := ipv4.Header{
        Version:  ipv4.Version,
        Len:      ipv4.HeaderLen,
        TOS:      hdr.TOS,
        TotalLen: ipv4.HeaderLen + len(payload),
        ID:       hdr.ID,
        Flags:    hdr.Flags,
        FragOff:  hdr.FragOff,
        TTL:      hdr.TTL,
        Protocol: hdr.Protocol,
        Checksum: 0,
        Src:      hdr.Src,
        Dst:      hdr.Dst,
    }

    headerBytes, err := h.Marshal()
    if err != nil {
        panic(err)
    }
    data := append(headerBytes, payload...)
    var sum uint32
    for i := 0; i < len(data)-1; i += 2 {
        sum += uint32(data[i])<<8 | uint32(data[i+1])
    }
    if len(data)%2 == 1 {
        sum += uint32(data[len(data)-1]) << 8
    }
    for sum>>16 != 0 {
        sum = (sum & 0xffff) + (sum >> 16)
    }
    return uint16(^sum)
}

func main() {
    // 创建 IPv4 原始套接字
    conn, err := net.ListenIP("ip4:icmp", &net.IPAddr{IP: net.IPv4zero})
    if err != nil {
        log.Fatal(err)
    }
    defer conn.Close()

    rawConn, err := ipv4.NewRawConn(conn)
    if err != nil {
        log.Fatal(err)
    }
    defer rawConn.Close()

    fmt.Println("Raw socket created successfully!")

    // 构造 ICMP 数据
    icmpMessage := icmp.Message{
        Type: ipv4.ICMPTypeEcho,
        Code: 0,
        Body: &icmp.Echo{
            ID:   12345,
            Seq:  1,
            Data: []byte("Hello, Raw Socket!"),
        },
    }

    icmpBytes, err := icmpMessage.Marshal(nil)
    if err != nil {
        log.Fatal(err)
    }

    // 构造 IP 头部
    ipHeader := &ipv4.Header{
        Version:  ipv4.Version,
        Len:      ipv4.HeaderLen,
        TOS:      0,
        TotalLen: ipv4.HeaderLen + len(icmpBytes),
        ID:       0,
        Flags:    0,
        FragOff:  0,
        TTL:      64,
        Protocol: 1, // ICMP
        Checksum: 0,
        Src:      net.ParseIP("192.168.1.100").To4(), // 伪造的源 IP
        Dst:      net.ParseIP("8.8.8.8").To4(),      // 目标 IP
    }

    // 计算校验和
    ipHeader.Checksum = checksum(ipHeader, icmpBytes)

    // 发送数据包
    err = rawConn.WriteTo(ipHeader, icmpBytes, &net.IPAddr{IP: ipHeader.Dst})
    if err != nil {
        log.Println("Error writing:", err)
    } else {
        fmt.Println("Packet sent successfully!")
    }

    // 接收数据 (可选)
    buf := make([]byte, 1500)
    rawConn.SetReadDeadline(time.Now().Add(5 * time.Second)) // 设置超时
    hdr, payload, peer, err := rawConn.ReadFrom(buf)
    if err != nil {
        log.Println("Error reading:", err)
    } else {
        fmt.Printf("Received packet from: %v\n", peer)
        fmt.Printf("Header: %+v\n", hdr)
        fmt.Printf("Payload: %v\n", payload)
    }
}

注意事项:

  • 需要 root 权限或者 CAP_NET_RAW capability 才能运行此程序。
  • net.ParseIP("192.168.1.100").To4() 这里的 IP 地址可以修改为任意合法的 IPv4 地址,用于模拟源 IP 地址欺骗。
  • checksum 函数的实现必须正确,否则发送的数据包会被丢弃。
  • 在实际应用中,需要根据具体的协议和需求,构造相应的 IP 头部和数据载荷。

总结

通过 go.net/ipv4 包,Go 语言提供了强大的原始套接字编程能力。开发者可以利用原始套接字实现自定义的网络协议、进行网络安全研究等。但同时也需要注意安全性问题,避免滥用。 理解 IP 协议的细节以及正确计算校验和是使用原始套接字的关键。

相关专题

更多
Java 网络安全
Java 网络安全

本专题聚焦 Java 在网络安全与加密通信中的应用,系统讲解常见加密算法(MD5、SHA、AES、RSA)、数字签名、HTTPS证书配置、令牌认证(JWT、OAuth2)及常见安全漏洞防护(XSS、SQL注入、CSRF)。通过实战项目(如安全登录系统、加密文件传输工具),帮助学习者掌握 Java 安全开发与加密技术的实战能力。

706

2025.10.13

PHP 安全与防护
PHP 安全与防护

本专题聚焦于PHP开发中的安全问题与防御措施,详细讲解SQL注入、XSS攻击、CSRF攻击、文件包含漏洞等常见安全风险及其修复方法。通过结合实际案例,帮助开发者理解漏洞成因,掌握输入验证、会话安全、加密存储与安全编码规范,全面提升PHP网站的安全防护水平。

110

2025.11.04

磁盘配额是什么
磁盘配额是什么

磁盘配额是计算机中指定磁盘的储存限制,就是管理员可以为用户所能使用的磁盘空间进行配额限制,每一用户只能使用最大配额范围内的磁盘空间。php中文网为大家提供各种磁盘配额相关的内容,教程,供大家免费下载安装。

1345

2023.06.21

如何安装LINUX
如何安装LINUX

本站专题提供如何安装LINUX的相关教程文章,还有相关的下载、课程,大家可以免费体验。

700

2023.06.29

linux find
linux find

find是linux命令,它将档案系统内符合 expression 的档案列出来。可以指要档案的名称、类别、时间、大小、权限等不同资讯的组合,只有完全相符的才会被列出来。find根据下列规则判断 path 和 expression,在命令列上第一个 - ( ) , ! 之前的部分为 path,之后的是 expression。还有指DOS 命令 find,Excel 函数 find等。本站专题提供linux find相关教程文章,还有相关

294

2023.06.30

linux修改文件名
linux修改文件名

本专题为大家提供linux修改文件名相关的文章,这些文章可以帮助用户快速轻松地完成文件名的修改工作,大家可以免费体验。

773

2023.07.05

linux系统安装教程
linux系统安装教程

linux系统是一种可以免费使用,自由传播,多用户、多任务、多线程、多CPU的操作系统。本专题提供linux系统安装教程相关的文章,大家可以免费体验。

571

2023.07.06

linux查看文件夹大小
linux查看文件夹大小

Linux是一种自由和开放源码的类Unix操作系统,存在着许多不同的Linux版本,但它们都使用了Linux内核。Linux可安装在各种计算机硬件设备中,比如手机、平板电脑、路由器、视频游戏控制台、台式计算机、大型机和超级计算机。linux怎么查看文件夹大小呢?php中文网给大家带来了相关的教程以及文章,欢迎大家前来学习阅读。

544

2023.07.20

php源码安装教程大全
php源码安装教程大全

本专题整合了php源码安装教程,阅读专题下面的文章了解更多详细内容。

65

2025.12.31

热门下载

更多
网站特效
/
网站源码
/
网站素材
/
前端模板

精品课程

更多
相关推荐
/
热门推荐
/
最新课程
PostgreSQL 教程
PostgreSQL 教程

共48课时 | 6.4万人学习

Git 教程
Git 教程

共21课时 | 2.3万人学习

关于我们 免责申明 举报中心 意见反馈 讲师合作 广告合作 最新更新
php中文网:公益在线php培训,帮助PHP学习者快速成长!
关注服务号 技术交流群
PHP中文网订阅号
每天精选资源文章推送

Copyright 2014-2026 https://www.php.cn/ All Rights Reserved | php.cn | 湘ICP备2023035733号